mirror of
https://github.com/Kaelio/ktx.git
synced 2026-06-13 08:15:14 +02:00
Move the live GitHub stars widget into the sidebar footer pill as a type:"icon" link, sitting opposite the Slack mark (space-between) and beside the theme switcher. Render it as inner content (GitHub mark + star + compact count) so fumadocs supplies the anchor. Replace the default fumadocs theme switcher with a custom two-icon control where each icon selects its own theme. The built-in "light-dark" mode is a single blind toggle that flips on any click, so clicking the sun while already in light mode jumps to dark. useTheme is sourced from fumadocs-ui/provider/base and the icons are inlined to avoid bare next-themes / lucide-react imports.
97 lines
2.7 KiB
TypeScript
97 lines
2.7 KiB
TypeScript
"use client";
|
|
|
|
import { useEffect, useState, type ComponentProps, type SVGProps } from "react";
|
|
import { useTheme } from "fumadocs-ui/provider/base";
|
|
|
|
/**
|
|
* Two-icon theme switcher (light / dark), each icon selecting its own theme —
|
|
* unlike fumadocs' default "light-dark" switcher, which is a single blind
|
|
* toggle that flips on any click. Dropped into the sidebar footer pill via
|
|
* `slots.themeSwitch`, so fumadocs passes the container className (left
|
|
* divider, `ms-auto`, rounded inner buttons); we merge it onto our own base.
|
|
*
|
|
* Icons are inlined (the project doesn't depend on `lucide-react` directly);
|
|
* `useTheme` is re-exported by fumadocs so we avoid a bare `next-themes` import.
|
|
*/
|
|
function SunIcon(props: SVGProps<SVGSVGElement>) {
|
|
return (
|
|
<svg
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
strokeWidth={2}
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
aria-hidden="true"
|
|
{...props}
|
|
>
|
|
<circle cx="12" cy="12" r="4" />
|
|
<path d="M12 2v2" />
|
|
<path d="M12 20v2" />
|
|
<path d="m4.93 4.93 1.41 1.41" />
|
|
<path d="m17.66 17.66 1.41 1.41" />
|
|
<path d="M2 12h2" />
|
|
<path d="M20 12h2" />
|
|
<path d="m6.34 17.66-1.41 1.41" />
|
|
<path d="m19.07 4.93-1.41 1.41" />
|
|
</svg>
|
|
);
|
|
}
|
|
|
|
function MoonIcon(props: SVGProps<SVGSVGElement>) {
|
|
return (
|
|
<svg
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
strokeWidth={2}
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
aria-hidden="true"
|
|
{...props}
|
|
>
|
|
<path d="M12 3a6 6 0 0 0 9 9 9 9 0 1 1-9-9Z" />
|
|
</svg>
|
|
);
|
|
}
|
|
|
|
const OPTIONS = [
|
|
["light", SunIcon],
|
|
["dark", MoonIcon],
|
|
] as const;
|
|
|
|
function cx(...classes: (string | false | undefined)[]): string {
|
|
return classes.filter(Boolean).join(" ");
|
|
}
|
|
|
|
export function ThemeToggle({ className, ...props }: ComponentProps<"div">) {
|
|
const { setTheme, resolvedTheme } = useTheme();
|
|
const [mounted, setMounted] = useState(false);
|
|
useEffect(() => setMounted(true), []);
|
|
const active = mounted ? resolvedTheme : null;
|
|
|
|
return (
|
|
<div
|
|
className={cx("inline-flex items-center overflow-hidden border", className)}
|
|
data-theme-toggle=""
|
|
{...props}
|
|
>
|
|
{OPTIONS.map(([key, Icon]) => (
|
|
<button
|
|
key={key}
|
|
type="button"
|
|
aria-label={key}
|
|
onClick={() => setTheme(key)}
|
|
className={cx(
|
|
"size-6.5 p-1.5 transition-colors",
|
|
active === key
|
|
? "bg-fd-accent text-fd-accent-foreground"
|
|
: "text-fd-muted-foreground hover:text-fd-accent-foreground",
|
|
)}
|
|
>
|
|
<Icon className="size-full" />
|
|
</button>
|
|
))}
|
|
</div>
|
|
);
|
|
}
|