mirror of
https://bitbucket.org/Mattrixwv/mattrixwvreactcomponents.git
synced 2025-12-07 14:13:58 -05:00
62 lines
1.4 KiB
TypeScript
62 lines
1.4 KiB
TypeScript
import type { Theme, ThemeProviderProps, ThemeProviderState } from "$/types/Theme";
|
|
import { createContext, useContext, useEffect, useMemo, useState } from "react";
|
|
|
|
|
|
const themeInitialState: ThemeProviderState = {
|
|
theme: "system",
|
|
setTheme: () => null
|
|
}
|
|
|
|
export const ThemeProviderContext = createContext<ThemeProviderState>(themeInitialState);
|
|
|
|
|
|
export default function ThemeProvider(props: ThemeProviderProps){
|
|
const {
|
|
children,
|
|
defaultTheme = "system",
|
|
storageKey = "mattrixwv-ui-theme"
|
|
} = props;
|
|
|
|
const [ theme, setTheme ] = useState<Theme>((localStorage.getItem(storageKey) as Theme) || defaultTheme);
|
|
|
|
useEffect(() => {
|
|
const root = window.document.documentElement;
|
|
|
|
root.classList.remove("light", "dark");
|
|
|
|
if(theme === "system"){
|
|
const systemTheme = window.matchMedia("(prefers-color-scheme: light)").matches ? "light" : "dark";
|
|
|
|
root.classList.add(systemTheme);
|
|
}
|
|
else{
|
|
root.classList.add(theme);
|
|
}
|
|
}, [ theme ]);
|
|
|
|
const value: ThemeProviderState = useMemo(() => ({
|
|
theme,
|
|
setTheme: (theme: Theme) => {
|
|
localStorage.setItem(storageKey, theme);
|
|
setTheme(theme);
|
|
}
|
|
}), [ theme ]);
|
|
|
|
return (
|
|
<ThemeProviderContext.Provider value={value}>
|
|
{children}
|
|
</ThemeProviderContext.Provider>
|
|
);
|
|
}
|
|
|
|
|
|
export function useTheme(){
|
|
const context = useContext(ThemeProviderContext);
|
|
|
|
if(!context){
|
|
throw new Error("useTheme must be used within a ThemeProvider");
|
|
}
|
|
|
|
return context;
|
|
}
|