Perplexica/ui/components/theme/Switcher.tsx

60 lines
1.5 KiB
TypeScript
Raw Normal View History

2024-07-05 14:36:50 +08:00
"use client";
import { useTheme } from "next-themes";
import { SunIcon, MoonIcon, MonitorIcon } from "lucide-react";
import { useCallback, useEffect, useState } from "react";
import { Select } from "../SettingsDialog";
2024-05-24 18:20:15 +08:00
2024-07-05 14:36:50 +08:00
type Theme = "dark" | "light" | "system";
2024-05-24 18:20:15 +08:00
const ThemeSwitcher = ({ className }: { className?: string }) => {
2024-05-24 18:20:15 +08:00
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(() => {
2024-07-05 14:36:50 +08:00
if (isTheme("system")) {
const preferDarkScheme = window.matchMedia("(prefers-color-scheme: dark)");
2024-05-24 18:20:15 +08:00
const detectThemeChange = (event: MediaQueryListEvent) => {
2024-07-05 14:36:50 +08:00
const theme: Theme = event.matches ? "dark" : "light";
2024-05-24 18:20:15 +08:00
setTheme(theme);
};
2024-07-05 14:36:50 +08:00
preferDarkScheme.addEventListener("change", detectThemeChange);
2024-05-24 18:20:15 +08:00
return () => {
2024-07-05 14:36:50 +08:00
preferDarkScheme.removeEventListener("change", detectThemeChange);
2024-05-24 18:20:15 +08:00
};
}
}, [isTheme, setTheme, theme]);
// Avoid Hydration Mismatch
if (!mounted) {
return null;
}
return (
<Select
className={className}
value={theme}
2024-07-05 14:36:50 +08:00
onChange={e => handleThemeSwitch(e.target.value as Theme)}
options={[
2024-07-05 14:36:50 +08:00
{ value: "light", label: "Light" },
{ value: "dark", label: "Dark" },
]}
2024-05-24 18:20:15 +08:00
/>
);
2024-05-30 21:38:37 +05:30
};
export default ThemeSwitcher;