patrickulrich.com/src/components/ThemeProvider.tsx

70 lines
1.6 KiB
TypeScript
Raw Normal View History

2025-05-30 22:37:13 +02:00
import { useEffect, useState } from "react"
2025-06-01 09:37:49 -05:00
import { type Theme, ThemeContext } from "@/lib/ThemeContext"
2025-05-30 22:37:13 +02:00
type ThemeProviderProps = {
children: React.ReactNode
defaultTheme?: Theme
storageKey?: string
}
export function ThemeProvider({
children,
defaultTheme = "system",
storageKey = "theme",
...props
}: ThemeProviderProps) {
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: dark)")
.matches
? "dark"
: "light"
root.classList.add(systemTheme)
return
}
root.classList.add(theme)
}, [theme])
useEffect(() => {
const mediaQuery = window.matchMedia("(prefers-color-scheme: dark)")
const handleChange = () => {
if (theme === "system") {
const root = window.document.documentElement
root.classList.remove("light", "dark")
const systemTheme = mediaQuery.matches ? "dark" : "light"
root.classList.add(systemTheme)
}
}
mediaQuery.addEventListener("change", handleChange)
return () => mediaQuery.removeEventListener("change", handleChange)
}, [theme])
const value = {
theme,
setTheme: (theme: Theme) => {
localStorage.setItem(storageKey, theme)
setTheme(theme)
},
}
return (
2025-06-01 09:37:49 -05:00
<ThemeContext.Provider {...props} value={value}>
2025-05-30 22:37:13 +02:00
{children}
2025-06-01 09:37:49 -05:00
</ThemeContext.Provider>
2025-05-30 22:37:13 +02:00
)
}