feat: setup theme context config
This commit is contained in:
parent
79cfd0a722
commit
f9664d48e7
4 changed files with 92 additions and 11 deletions
63
ui/components/theme/Switcher.tsx
Normal file
63
ui/components/theme/Switcher.tsx
Normal file
|
@ -0,0 +1,63 @@
|
|||
'use client';
|
||||
import { useTheme } from 'next-themes';
|
||||
import { SunIcon, MoonIcon, MonitorIcon } from 'lucide-react';
|
||||
import { useCallback, useEffect, useState } from 'react';
|
||||
|
||||
type Theme = 'dark' | 'light' | 'system';
|
||||
|
||||
export function ThemeSwitcher() {
|
||||
const [mounted, setMounted] = useState(false);
|
||||
|
||||
const { theme, setTheme } = useTheme();
|
||||
|
||||
const isTheme = useCallback((t: Theme) => t === theme, [theme]);
|
||||
|
||||
const handleThemeSwitch = (theme: Theme) => {
|
||||
setTheme(theme);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
setMounted(true);
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
if (isTheme('system')) {
|
||||
const preferDarkScheme = window.matchMedia(
|
||||
'(prefers-color-scheme: dark)',
|
||||
);
|
||||
|
||||
const detectThemeChange = (event: MediaQueryListEvent) => {
|
||||
const theme: Theme = event.matches ? 'dark' : 'light';
|
||||
setTheme(theme);
|
||||
};
|
||||
|
||||
preferDarkScheme.addEventListener('change', detectThemeChange);
|
||||
|
||||
return () => {
|
||||
preferDarkScheme.removeEventListener('change', detectThemeChange);
|
||||
};
|
||||
}
|
||||
}, [isTheme, setTheme, theme]);
|
||||
|
||||
// Avoid Hydration Mismatch
|
||||
if (!mounted) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return isTheme('dark') ? (
|
||||
<SunIcon
|
||||
className="cursor-pointer"
|
||||
onClick={() => handleThemeSwitch('light')}
|
||||
/>
|
||||
) : isTheme('light') ? (
|
||||
<MoonIcon
|
||||
className="cursor-pointer"
|
||||
onClick={() => handleThemeSwitch('dark')}
|
||||
/>
|
||||
) : (
|
||||
<MonitorIcon
|
||||
className="cursor-pointer"
|
||||
onClick={() => handleThemeSwitch('system')}
|
||||
/>
|
||||
);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue