rowboat/apps/rowboat/app/providers/theme-provider.tsx
2025-04-28 16:34:03 +05:30

60 lines
No EOL
1.4 KiB
TypeScript

'use client'
import { createContext, useContext, useEffect, useState } from 'react'
type Theme = 'dark' | 'light'
type ThemeProviderProps = {
children: React.ReactNode
defaultTheme?: Theme
}
type ThemeProviderState = {
theme: Theme
toggleTheme: () => void
}
const ThemeProviderContext = createContext<ThemeProviderState | undefined>(undefined)
export function ThemeProvider({
children,
defaultTheme = 'light',
}: ThemeProviderProps) {
const [theme, setTheme] = useState<Theme>(defaultTheme)
useEffect(() => {
const root = document.documentElement
const storedTheme = localStorage.getItem("theme")
if (storedTheme === 'dark' || storedTheme === 'light') {
setTheme(storedTheme)
}
root.classList.remove('light', 'dark')
root.classList.add(theme)
}, [theme])
const toggleTheme = () => {
setTheme((prevTheme) => {
const newTheme = prevTheme === 'light' ? 'dark' : 'light'
if (typeof window !== 'undefined') {
localStorage.setItem("theme", newTheme)
}
return newTheme
})
}
return (
<ThemeProviderContext.Provider value={{ theme, toggleTheme }}>
{children}
</ThemeProviderContext.Provider>
)
}
export function useTheme() {
const context = useContext(ThemeProviderContext)
if (context === undefined) {
throw new Error('useTheme must be used within a ThemeProvider')
}
return context
}