import { useState, useEffect } from "react";
import { Button } from "@cloudflare/kumo";
import { SunIcon, MoonIcon } from "@phosphor-icons/react";
export function ThemeToggle() {
const [theme, setTheme] = useState<"light" | "dark">("light");
const [mounted, setMounted] = useState(false);
useEffect(() => {
setMounted(true);
const stored = localStorage.getItem("theme");
if (stored === "dark" || stored === "light") {
setTheme(stored);
} else if (window.matchMedia("(prefers-color-scheme: dark)").matches) {
setTheme("dark");
}
}, []);
const toggleTheme = () => {
const newTheme = theme === "light" ? "dark" : "light";
setTheme(newTheme);
localStorage.setItem("theme", newTheme);
document.documentElement.setAttribute("data-mode", newTheme);
};
// Prevent hydration mismatch
if (!mounted) {
return (
<Button variant="ghost" shape="square" aria-label="Toggle theme">
<SunIcon size={20} />
</Button>
);
}
return (
<Button
variant="ghost"
shape="square"
aria-label={`Switch to ${theme === "light" ? "dark" : "light"} mode`}
onClick={toggleTheme}
>
{theme === "light" ? <MoonIcon size={20} /> : <SunIcon size={20} />}
</Button>
);
}cloudflare/kumo
Publicmirrored fromhttps://github.com/cloudflare/kumoAvailable
packages/kumo-docs-astro/src/components/ThemeToggle.tsx
45lines · modepreview