ThemeModeContext.tsx 1.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. import React from 'react';
  2. import type { FC, PropsWithChildren } from 'react';
  3. import type { ThemeDropDownValue } from 'components/NavBar/NavBar';
  4. interface ThemeModeContextProps {
  5. isDarkMode: boolean;
  6. themeMode: ThemeDropDownValue;
  7. setThemeMode: (value: string | number) => void;
  8. }
  9. export const ThemeModeContext = React.createContext<ThemeModeContextProps>({
  10. isDarkMode: false,
  11. themeMode: 'auto_theme',
  12. setThemeMode: () => {},
  13. });
  14. export const ThemeModeProvider: FC<PropsWithChildren<unknown>> = ({
  15. children,
  16. }) => {
  17. const matchDark = window.matchMedia('(prefers-color-scheme: dark)');
  18. const [themeMode, setThemeModeState] =
  19. React.useState<ThemeDropDownValue>('auto_theme');
  20. React.useLayoutEffect(() => {
  21. const mode = localStorage.getItem('mode');
  22. setThemeModeState((mode as ThemeDropDownValue) ?? 'auto_theme');
  23. }, []);
  24. const isDarkMode = React.useMemo(() => {
  25. if (themeMode === 'auto_theme') {
  26. return matchDark.matches;
  27. }
  28. return themeMode === 'dark_theme';
  29. }, []);
  30. const setThemeMode = React.useCallback((value: string | number) => {
  31. setThemeMode(value as ThemeDropDownValue);
  32. localStorage.setItem('mode', value as string);
  33. }, []);
  34. return (
  35. <ThemeModeContext.Provider
  36. value={{
  37. isDarkMode,
  38. themeMode,
  39. setThemeMode,
  40. }}
  41. >
  42. {children}
  43. </ThemeModeContext.Provider>
  44. );
  45. };