* feat(web): mobile dashboard UX polish Bottom sheets for sidebar theme/language pickers on narrow viewports with enter/exit animation and drag-to-close; inline header badges beside titles; bottom padding on the route outlet for scroll clearance; profiles loading uses a unicode braille spinner; align profile/cron card actions to the top; viewport-fit cover and supporting layout tweaks across dashboard pages. Co-authored-by: Cursor <cursoragent@cursor.com> * Fix Nix web npm hash and mobile sheet accessibility. Align fetchNpmDeps in nix/web.nix with web/package-lock.json for CI. Improve BottomPickSheet backdrop labeling, avoid aria-hidden on the dialog during exit animation, and wire theme/language sheets with listbox semantics and localized dismiss labels. Co-authored-by: Cursor <cursoragent@cursor.com> --------- Co-authored-by: Cursor <cursoragent@cursor.com>
183 lines
5.7 KiB
TypeScript
183 lines
5.7 KiB
TypeScript
import { useState, useRef, useEffect } from "react";
|
|
import { Button } from "@nous-research/ui/ui/components/button";
|
|
import { BottomPickSheet } from "@/components/BottomPickSheet";
|
|
import { Typography } from "@/components/NouiTypography";
|
|
import { useBelowBreakpoint } from "@/hooks/useBelowBreakpoint";
|
|
import { useI18n } from "@/i18n/context";
|
|
import { LOCALE_META } from "@/i18n";
|
|
import type { Locale } from "@/i18n";
|
|
import { cn } from "@/lib/utils";
|
|
|
|
/**
|
|
* Language picker — shows the current language's flag + endonym, opens a
|
|
* dropdown of all supported locales when clicked. Persists choice to
|
|
* localStorage via the I18n context.
|
|
*
|
|
* Replaces the older two-state EN↔ZH toggle now that we ship 16 locales
|
|
* (en, zh, zh-hant, ja, de, es, fr, tr, uk, af, ko, it, ga, pt, ru, hu).
|
|
*
|
|
* Locale markers use lipis/flag-icons (SVG sprites) instead of emoji so flags
|
|
* render consistently across platforms.
|
|
*
|
|
* When placed at the bottom of the sidebar (next to ThemeSwitcher), pass
|
|
* `dropUp` so the list opens above the trigger and avoids clipping below the
|
|
* viewport / overflow ancestors. Below the `sm` breakpoint, `dropUp` uses a
|
|
* bottom sheet portaled to `document.body` instead of an anchored dropdown.
|
|
*/
|
|
export function LanguageSwitcher({ dropUp = false }: LanguageSwitcherProps) {
|
|
const { locale, setLocale, t } = useI18n();
|
|
const [open, setOpen] = useState(false);
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const narrowViewport = useBelowBreakpoint(640);
|
|
const useMobileSheet = Boolean(dropUp && narrowViewport);
|
|
|
|
useEffect(() => {
|
|
if (!open) return;
|
|
function onKey(e: KeyboardEvent) {
|
|
if (e.key === "Escape") setOpen(false);
|
|
}
|
|
document.addEventListener("keydown", onKey);
|
|
return () => document.removeEventListener("keydown", onKey);
|
|
}, [open]);
|
|
|
|
// Outside-click closing only for anchored dropdown — sheet uses backdrop + portal.
|
|
useEffect(() => {
|
|
if (!open || useMobileSheet) return;
|
|
|
|
function onPointerDown(e: PointerEvent) {
|
|
if (!containerRef.current) return;
|
|
if (!containerRef.current.contains(e.target as Node)) {
|
|
setOpen(false);
|
|
}
|
|
}
|
|
|
|
document.addEventListener("pointerdown", onPointerDown);
|
|
return () => document.removeEventListener("pointerdown", onPointerDown);
|
|
}, [open, useMobileSheet]);
|
|
|
|
const current = LOCALE_META[locale];
|
|
const allLocales = Object.entries(LOCALE_META) as Array<[Locale, typeof current]>;
|
|
const sheetTitle = t.language.switchTo;
|
|
|
|
return (
|
|
<div ref={containerRef} className="relative inline-flex">
|
|
<Button
|
|
ghost
|
|
onClick={() => setOpen((v) => !v)}
|
|
title={t.language.switchTo}
|
|
aria-label={t.language.switchTo}
|
|
aria-haspopup="listbox"
|
|
aria-expanded={open}
|
|
className="px-2 py-1 normal-case tracking-normal font-normal text-xs text-muted-foreground hover:text-foreground"
|
|
>
|
|
<span className="inline-flex items-center gap-1.5">
|
|
<LocaleFlagIcon countryCode={current.flagCountryCode} />
|
|
<Typography
|
|
mondwest
|
|
className="hidden sm:inline tracking-wide uppercase text-[0.65rem]"
|
|
>
|
|
{locale === "en" ? "EN" : current.name}
|
|
</Typography>
|
|
</span>
|
|
</Button>
|
|
|
|
{useMobileSheet && (
|
|
<BottomPickSheet
|
|
backdropDismissLabel={t.common.close}
|
|
onClose={() => setOpen(false)}
|
|
open={open}
|
|
title={sheetTitle}
|
|
>
|
|
<div aria-label={sheetTitle} role="listbox">
|
|
<LanguageSwitcherOptions
|
|
allLocales={allLocales}
|
|
locale={locale}
|
|
setLocale={setLocale}
|
|
setOpen={setOpen}
|
|
/>
|
|
</div>
|
|
</BottomPickSheet>
|
|
)}
|
|
|
|
{open && !useMobileSheet && (
|
|
<div
|
|
aria-label={sheetTitle}
|
|
className={cn(
|
|
"absolute right-0 z-50 min-w-[10rem] rounded-md border border-border bg-popover shadow-md py-1 max-h-80 overflow-y-auto",
|
|
dropUp ? "bottom-full mb-1" : "top-full mt-1",
|
|
)}
|
|
role="listbox"
|
|
>
|
|
<LanguageSwitcherOptions
|
|
allLocales={allLocales}
|
|
locale={locale}
|
|
setLocale={setLocale}
|
|
setOpen={setOpen}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function LanguageSwitcherOptions({
|
|
allLocales,
|
|
locale,
|
|
setLocale,
|
|
setOpen,
|
|
}: LanguageSwitcherOptionsProps) {
|
|
return (
|
|
<>
|
|
{allLocales.map(([code, meta]) => {
|
|
const selected = code === locale;
|
|
|
|
return (
|
|
<button
|
|
aria-selected={selected}
|
|
className={
|
|
"w-full text-left px-3 py-1.5 text-xs flex items-center gap-2 hover:bg-accent hover:text-accent-foreground transition-colors " +
|
|
(selected ? "font-semibold text-foreground" : "text-muted-foreground")
|
|
}
|
|
key={code}
|
|
onClick={() => {
|
|
setLocale(code);
|
|
setOpen(false);
|
|
}}
|
|
role="option"
|
|
type="button"
|
|
>
|
|
<LocaleFlagIcon countryCode={meta.flagCountryCode} />
|
|
|
|
<span className="truncate">{meta.name}</span>
|
|
|
|
{selected && <span className="ml-auto text-xs">✓</span>}
|
|
</button>
|
|
);
|
|
})}
|
|
</>
|
|
);
|
|
}
|
|
|
|
function LocaleFlagIcon({ countryCode }: LocaleFlagIconProps) {
|
|
return (
|
|
<span
|
|
aria-hidden
|
|
className={cn("fi fis shrink-0 text-base leading-none", `fi-${countryCode}`)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
interface LanguageSwitcherOptionsProps {
|
|
allLocales: Array<[Locale, (typeof LOCALE_META)[Locale]]>;
|
|
locale: Locale;
|
|
setLocale: (code: Locale) => void;
|
|
setOpen: (open: boolean) => void;
|
|
}
|
|
|
|
interface LanguageSwitcherProps {
|
|
dropUp?: boolean;
|
|
}
|
|
|
|
interface LocaleFlagIconProps {
|
|
countryCode: string;
|
|
}
|