diff options
Diffstat (limited to 'src/contexts')
| -rw-r--r-- | src/contexts/DarkMode.tsx | 41 | ||||
| -rw-r--r-- | src/contexts/Settings.tsx | 65 |
2 files changed, 65 insertions, 41 deletions
diff --git a/src/contexts/DarkMode.tsx b/src/contexts/DarkMode.tsx deleted file mode 100644 index fd536b2..0000000 --- a/src/contexts/DarkMode.tsx +++ /dev/null | |||
| @@ -1,41 +0,0 @@ | |||
| 1 | import * as React from "react"; | ||
| 2 | import { | ||
| 3 | Context, | ||
| 4 | createContext, | ||
| 5 | PropsWithChildren, | ||
| 6 | useEffect, | ||
| 7 | useMemo, | ||
| 8 | useState, | ||
| 9 | } from "react"; | ||
| 10 | import { ThemeConfig } from "bootstrap-darkmode"; | ||
| 11 | |||
| 12 | interface DarkModeApi { | ||
| 13 | darkMode: boolean; | ||
| 14 | toggle: () => void; | ||
| 15 | } | ||
| 16 | |||
| 17 | export const DarkModeContext = createContext<Partial<DarkModeApi>>( | ||
| 18 | {} | ||
| 19 | ) as Context<DarkModeApi>; | ||
| 20 | |||
| 21 | export default function DarkModeProvider({ children }: PropsWithChildren<any>) { | ||
| 22 | const themeConfig = useMemo(() => new ThemeConfig(), []); | ||
| 23 | const [darkMode, setDarkMode] = useState( | ||
| 24 | () => themeConfig.getTheme() === "dark" | ||
| 25 | ); | ||
| 26 | |||
| 27 | useEffect(() => { | ||
| 28 | themeConfig.setTheme(darkMode ? "dark" : "light"); | ||
| 29 | }, [darkMode]); | ||
| 30 | |||
| 31 | return ( | ||
| 32 | <DarkModeContext.Provider | ||
| 33 | value={{ | ||
| 34 | darkMode, | ||
| 35 | toggle: () => setDarkMode((dm) => !dm), | ||
| 36 | }} | ||
| 37 | > | ||
| 38 | {children} | ||
| 39 | </DarkModeContext.Provider> | ||
| 40 | ); | ||
| 41 | } | ||
diff --git a/src/contexts/Settings.tsx b/src/contexts/Settings.tsx new file mode 100644 index 0000000..2bbebcc --- /dev/null +++ b/src/contexts/Settings.tsx | |||
| @@ -0,0 +1,65 @@ | |||
| 1 | import * as React from "react"; | ||
| 2 | import { | ||
| 3 | Context, | ||
| 4 | createContext, | ||
| 5 | PropsWithChildren, | ||
| 6 | useCallback, | ||
| 7 | useEffect, | ||
| 8 | useState, | ||
| 9 | } from "react"; | ||
| 10 | import { Settings } from "~/types"; | ||
| 11 | |||
| 12 | interface SettingsApi { | ||
| 13 | darkMode: boolean; | ||
| 14 | setSetting: (key: keyof Settings, value: any) => void; | ||
| 15 | } | ||
| 16 | |||
| 17 | export const SettingsContext = createContext<Partial<SettingsApi>>( | ||
| 18 | {} | ||
| 19 | ) as Context<SettingsApi>; | ||
| 20 | |||
| 21 | export default function SettingsProvider({ children }: PropsWithChildren<any>) { | ||
| 22 | // const themeConfig = useMemo(() => new ThemeConfig(), []); | ||
| 23 | const [settings, setSettings] = useState<Settings>(() => { | ||
| 24 | const stored = localStorage.getItem("settings"); | ||
| 25 | |||
| 26 | if (stored) { | ||
| 27 | return JSON.parse(stored); | ||
| 28 | } | ||
| 29 | |||
| 30 | return { | ||
| 31 | darkMode: | ||
| 32 | window.matchMedia && | ||
| 33 | window.matchMedia("(prefers-color-scheme: dark)").matches, | ||
| 34 | }; | ||
| 35 | }); | ||
| 36 | |||
| 37 | const setSetting = useCallback( | ||
| 38 | (key: keyof Settings, value: any) => { | ||
| 39 | const newSetting = { | ||
| 40 | ...settings, | ||
| 41 | [key]: value, | ||
| 42 | }; | ||
| 43 | setSettings(newSetting); | ||
| 44 | localStorage.setItem("settings", JSON.stringify(newSetting)); | ||
| 45 | }, | ||
| 46 | [settings] | ||
| 47 | ); | ||
| 48 | |||
| 49 | useEffect(() => { | ||
| 50 | document | ||
| 51 | .querySelector("html") | ||
| 52 | ?.setAttribute("data-bs-theme", settings.darkMode ? "dark" : "light"); | ||
| 53 | }, [settings.darkMode]); | ||
| 54 | |||
| 55 | return ( | ||
| 56 | <SettingsContext.Provider | ||
| 57 | value={{ | ||
| 58 | darkMode: settings.darkMode, | ||
| 59 | setSetting, | ||
| 60 | }} | ||
| 61 | > | ||
| 62 | {children} | ||
| 63 | </SettingsContext.Provider> | ||
| 64 | ); | ||
| 65 | } | ||
