mirror of
https://github.com/HKUDS/nanobot.git
synced 2026-05-20 08:32:25 +00:00
122 lines
4.2 KiB
TypeScript
122 lines
4.2 KiB
TypeScript
import { useMemo, useState } from "react";
|
|
import {
|
|
Menu,
|
|
Search,
|
|
SquarePen,
|
|
} from "lucide-react";
|
|
import { useTranslation } from "react-i18next";
|
|
|
|
import { ChatList } from "@/components/ChatList";
|
|
import { ConnectionBadge } from "@/components/ConnectionBadge";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Separator } from "@/components/ui/separator";
|
|
import { cn } from "@/lib/utils";
|
|
import type { ChatSummary } from "@/lib/types";
|
|
|
|
interface SidebarProps {
|
|
sessions: ChatSummary[];
|
|
activeKey: string | null;
|
|
loading: boolean;
|
|
onNewChat: () => void;
|
|
onSelect: (key: string) => void;
|
|
onRequestDelete: (key: string, label: string) => void;
|
|
onCollapse: () => void;
|
|
}
|
|
|
|
export function Sidebar(props: SidebarProps) {
|
|
const { t } = useTranslation();
|
|
const [query, setQuery] = useState("");
|
|
const normalizedQuery = query.trim().toLowerCase();
|
|
const filteredSessions = useMemo(() => {
|
|
if (!normalizedQuery) return props.sessions;
|
|
return props.sessions.filter((session) => {
|
|
const haystack = [
|
|
session.preview,
|
|
session.chatId,
|
|
session.channel,
|
|
session.key,
|
|
]
|
|
.filter(Boolean)
|
|
.join(" ")
|
|
.toLowerCase();
|
|
return haystack.includes(normalizedQuery);
|
|
});
|
|
}, [normalizedQuery, props.sessions]);
|
|
|
|
return (
|
|
<nav
|
|
aria-label={t("sidebar.navigation")}
|
|
className="flex h-full w-full flex-col border-r border-sidebar-border/60 bg-sidebar text-sidebar-foreground"
|
|
>
|
|
<div className="flex items-center justify-between px-3 pb-2.5 pt-3">
|
|
<picture className="block min-w-0">
|
|
<source srcSet="/brand/nanobot_logo.webp" type="image/webp" />
|
|
<img
|
|
src="/brand/nanobot_logo.png"
|
|
alt="nanobot"
|
|
className="h-6 w-auto select-none object-contain opacity-95"
|
|
draggable={false}
|
|
/>
|
|
</picture>
|
|
<Button
|
|
variant="ghost"
|
|
size="icon"
|
|
aria-label={t("sidebar.collapse")}
|
|
onClick={props.onCollapse}
|
|
className="h-7 w-7 rounded-lg text-muted-foreground/85 hover:bg-sidebar-accent/75 hover:text-sidebar-foreground"
|
|
>
|
|
<Menu className="h-3.5 w-3.5" />
|
|
</Button>
|
|
</div>
|
|
|
|
<div className="space-y-1.5 px-2 pb-2">
|
|
<label className="relative block">
|
|
<span className="sr-only">{t("sidebar.searchAria")}</span>
|
|
<Search
|
|
className="pointer-events-none absolute left-3 top-1/2 h-3.5 w-3.5 -translate-y-1/2 text-muted-foreground/70"
|
|
aria-hidden
|
|
/>
|
|
<input
|
|
value={query}
|
|
onChange={(event) => setQuery(event.target.value)}
|
|
placeholder={t("sidebar.searchPlaceholder")}
|
|
aria-label={t("sidebar.searchAria")}
|
|
className={cn(
|
|
"h-8 w-full rounded-full border border-transparent bg-sidebar-accent/45",
|
|
"pl-8 pr-3 text-[12.5px] text-sidebar-foreground outline-none",
|
|
"placeholder:text-muted-foreground/75",
|
|
"transition-colors hover:bg-sidebar-accent/65",
|
|
"focus:border-sidebar-border/80 focus:bg-sidebar-accent/70",
|
|
"focus:ring-1 focus:ring-sidebar-border/70",
|
|
)}
|
|
/>
|
|
</label>
|
|
<Button
|
|
onClick={props.onNewChat}
|
|
className="h-8 w-full justify-start gap-2 rounded-full px-3 text-[12.5px] font-medium text-sidebar-foreground/92 hover:bg-sidebar-accent/75 hover:text-sidebar-foreground"
|
|
variant="ghost"
|
|
>
|
|
<SquarePen className="h-3.5 w-3.5" />
|
|
{t("sidebar.newChat")}
|
|
</Button>
|
|
</div>
|
|
<div className="flex-1 overflow-hidden">
|
|
<ChatList
|
|
sessions={filteredSessions}
|
|
activeKey={props.activeKey}
|
|
loading={props.loading}
|
|
emptyLabel={
|
|
normalizedQuery ? t("sidebar.noSearchResults") : t("chat.noSessions")
|
|
}
|
|
onSelect={props.onSelect}
|
|
onRequestDelete={props.onRequestDelete}
|
|
/>
|
|
</div>
|
|
<Separator className="bg-sidebar-border/50" />
|
|
<div className="flex items-center px-2.5 py-2.5 text-xs">
|
|
<ConnectionBadge />
|
|
</div>
|
|
</nav>
|
|
);
|
|
}
|