import { useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from "react"; import { BarChart3, BookOpen, ChevronRight, Code2, ImageIcon, LayoutGrid, Lightbulb, MoreHorizontal, Palette, Sparkles, } from "lucide-react"; import { useTranslation } from "react-i18next"; import { ThreadComposer } from "@/components/thread/ThreadComposer"; import { ThreadHeader } from "@/components/thread/ThreadHeader"; import { StreamErrorNotice } from "@/components/thread/StreamErrorNotice"; import { ThreadViewport } from "@/components/thread/ThreadViewport"; import { useNanobotStream, type SendImage, type SendOptions } from "@/hooks/useNanobotStream"; import { useSessionHistory } from "@/hooks/useSessions"; import { fetchCliApps, listSlashCommands } from "@/lib/api"; import type { ChatSummary, CliAppInfo, SlashCommand, UIMessage } from "@/lib/types"; import { normalizeLegacyLongTaskMessages } from "@/lib/thread-display-compat"; import { scrubSubagentUiMessages } from "@/lib/subagent-channel-display"; import { useClient } from "@/providers/ClientProvider"; function projectWebuiThreadMessages(messages: UIMessage[]): UIMessage[] { return scrubSubagentUiMessages(normalizeLegacyLongTaskMessages(messages)); } interface ThreadShellProps { session: ChatSummary | null; title: string; onToggleSidebar: () => void; onGoHome?: () => void; onNewChat?: () => void; onCreateChat?: () => Promise; onTurnEnd?: () => void; theme?: "light" | "dark"; onToggleTheme?: () => void; hideSidebarToggleOnDesktop?: boolean; } function toModelBadgeLabel(modelName: string | null): string | null { if (!modelName) return null; const trimmed = modelName.trim(); if (!trimmed) return null; const leaf = trimmed.split("/").pop() ?? trimmed; return leaf || trimmed; } const QUICK_ACTION_KEYS = [ { key: "plan", icon: LayoutGrid, tone: "text-[#f25b8f]" }, { key: "analyze", icon: BarChart3, tone: "text-[#4f9de8]" }, { key: "brainstorm", icon: Lightbulb, tone: "text-[#53c59d]" }, { key: "code", icon: Code2, tone: "text-[#eba45d]" }, { key: "summarize", icon: BookOpen, tone: "text-[#a877e7]" }, { key: "more", icon: MoreHorizontal, tone: "text-muted-foreground/65" }, ] as const; const IMAGE_QUICK_ACTION_KEYS = [ { key: "icon", icon: ImageIcon, tone: "text-[#4f9de8]" }, { key: "sticker", icon: Sparkles, tone: "text-[#f25b8f]" }, { key: "poster", icon: Palette, tone: "text-[#eba45d]" }, { key: "product", icon: LayoutGrid, tone: "text-[#53c59d]" }, { key: "portrait", icon: ImageIcon, tone: "text-[#a877e7]" }, { key: "edit", icon: MoreHorizontal, tone: "text-muted-foreground/65" }, ] as const; interface PendingFirstMessage { content: string; images?: SendImage[]; options?: SendOptions; } export function ThreadShell({ session, title, onToggleSidebar, onCreateChat, onTurnEnd, theme = "light", onToggleTheme = () => {}, hideSidebarToggleOnDesktop = false, }: ThreadShellProps) { const { t } = useTranslation(); const chatId = session?.chatId ?? null; const historyKey = session?.key ?? null; const { messages: historical, loading, hasPendingToolCalls, refresh: refreshHistory, version: historyVersion, } = useSessionHistory(historyKey); const { client, modelName, token } = useClient(); const [booting, setBooting] = useState(false); const [slashCommands, setSlashCommands] = useState([]); const [cliApps, setCliApps] = useState([]); const [heroImageMode, setHeroImageMode] = useState(false); const [scrollToBottomSignal, setScrollToBottomSignal] = useState(0); const pendingFirstRef = useRef(null); const messageCacheRef = useRef>(new Map()); /** Last chatId we associated with the in-memory thread (for cache-on-switch). */ const prevChatIdForCacheRef = useRef(null); /** Skip one message-cache write right after chatId changes (messages may not match yet). */ const skipLayoutCacheRef = useRef(false); const appliedHistoryVersionRef = useRef>(new Map()); const pendingCanonicalHydrateRef = useRef>(new Set()); const sessionKeyByChatIdRef = useRef>(new Map()); const initial = useMemo(() => { if (!chatId) return historical; return messageCacheRef.current.get(chatId) ?? historical; }, [chatId, historical]); const handleTurnEnd = useCallback(() => { onTurnEnd?.(); }, [onTurnEnd]); const { messages, isStreaming, runStartedAt, goalState, send, stop, setMessages, streamError, dismissStreamError, } = useNanobotStream(chatId, initial, hasPendingToolCalls, handleTurnEnd); useEffect(() => { if (chatId && historyKey) sessionKeyByChatIdRef.current.set(chatId, historyKey); }, [chatId, historyKey]); const displayMessages = useMemo(() => projectWebuiThreadMessages(messages), [messages]); const showHeroComposer = messages.length === 0 && !loading; useEffect(() => { if (!chatId || loading) return; const cached = messageCacheRef.current.get(chatId); const appliedVersion = appliedHistoryVersionRef.current.get(chatId) ?? 0; const hasPendingCanonicalHydrate = pendingCanonicalHydrateRef.current.has(chatId); const hasNewCanonicalHistory = hasPendingCanonicalHydrate && historyVersion > appliedVersion; // When the user switches away and back, keep the local in-memory thread // state (including not-yet-persisted messages) instead of replacing it with // whatever the history endpoint currently knows about. Once a fresh // canonical replay arrives (e.g. after ``session_updated`` refresh), prefer it // so rendering converges to the same shape as a manual refresh. setMessages((prev) => { if (hasNewCanonicalHistory && historical.length > 0) { pendingCanonicalHydrateRef.current.delete(chatId); appliedHistoryVersionRef.current.set(chatId, historyVersion); const normalized = projectWebuiThreadMessages(historical); messageCacheRef.current.set(chatId, normalized); return normalized; } if (cached && cached.length > 0) return projectWebuiThreadMessages(cached); if (historical.length === 0 && prev.length > 0) return projectWebuiThreadMessages(prev); appliedHistoryVersionRef.current.set(chatId, historyVersion); const next = projectWebuiThreadMessages(historical); if (historical.length > 0) messageCacheRef.current.set(chatId, next); return next; }); // eslint-disable-next-line react-hooks/exhaustive-deps }, [loading, chatId, historical, historyVersion]); useEffect(() => { if (!chatId) return; return client.onSessionUpdate((updatedChatId, scope) => { if (updatedChatId !== chatId) return; if (scope === "metadata") return; pendingCanonicalHydrateRef.current.add(chatId); refreshHistory(); }); }, [chatId, client, refreshHistory]); useEffect(() => { if (!chatId || loading) return; setScrollToBottomSignal((value) => value + 1); }, [chatId, loading, historical]); useEffect(() => { if (chatId) return; setMessages(projectWebuiThreadMessages(historical)); }, [chatId, historical, setMessages]); useLayoutEffect(() => { if (chatId) { const prev = prevChatIdForCacheRef.current; if (prev && prev !== chatId) { messageCacheRef.current.set(prev, projectWebuiThreadMessages(messages)); skipLayoutCacheRef.current = true; } prevChatIdForCacheRef.current = chatId; } else { if (prevChatIdForCacheRef.current) { messageCacheRef.current.set( prevChatIdForCacheRef.current, projectWebuiThreadMessages(messages), ); skipLayoutCacheRef.current = true; } prevChatIdForCacheRef.current = null; } }, [chatId, messages]); // Persist thread to in-memory cache after paint so ``useNanobotStream``'s chat switch // ``useEffect`` reset has flushed; ``skipLayoutCacheRef`` drops the first run that still // sees the *previous* chat's ``messages`` (avoids stale rows leaking across sessions). useEffect(() => { if (!chatId) { return; } if (skipLayoutCacheRef.current) { skipLayoutCacheRef.current = false; return; } if (loading) { return; } messageCacheRef.current.set(chatId, projectWebuiThreadMessages(messages)); }, [chatId, loading, messages]); useEffect(() => { if (!chatId) return; const pending = pendingFirstRef.current; if (!pending) return; pendingFirstRef.current = null; setScrollToBottomSignal((value) => value + 1); send(pending.content, pending.images, pending.options); setBooting(false); }, [chatId, send]); useEffect(() => { let cancelled = false; (async () => { try { const commands = await listSlashCommands(token); if (!cancelled) setSlashCommands(commands); } catch { if (!cancelled) setSlashCommands([]); } })(); return () => { cancelled = true; }; }, [token]); const refreshCliApps = useCallback(async () => { try { const payload = await fetchCliApps(token); setCliApps(payload.apps.filter((app) => app.installed)); } catch { setCliApps([]); } }, [token]); useEffect(() => { let cancelled = false; const load = async () => { try { const payload = await fetchCliApps(token); if (!cancelled) setCliApps(payload.apps.filter((app) => app.installed)); } catch { if (!cancelled) setCliApps([]); } }; load(); const refreshOnFocus = () => { if (document.visibilityState === "hidden") return; void refreshCliApps(); }; window.addEventListener("focus", refreshOnFocus); document.addEventListener("visibilitychange", refreshOnFocus); return () => { cancelled = true; window.removeEventListener("focus", refreshOnFocus); document.removeEventListener("visibilitychange", refreshOnFocus); }; }, [refreshCliApps, token]); const handleWelcomeSend = useCallback( async (content: string, images?: SendImage[], options?: SendOptions) => { if (booting) return; setBooting(true); pendingFirstRef.current = { content, images, options }; const newId = await onCreateChat?.(); if (!newId) { pendingFirstRef.current = null; setBooting(false); } }, [booting, onCreateChat], ); const handleThreadSend = useCallback( (content: string, images?: SendImage[], options?: SendOptions) => { setScrollToBottomSignal((value) => value + 1); send(content, images, options); }, [send], ); const handleQuickAction = useCallback( (prompt: string) => { const options: SendOptions | undefined = heroImageMode ? { imageGeneration: { enabled: true, aspect_ratio: null } } : undefined; if (session) { handleThreadSend(prompt, undefined, options); return; } void handleWelcomeSend(prompt, undefined, options); }, [handleThreadSend, handleWelcomeSend, heroImageMode, session], ); const quickActionItems = heroImageMode ? IMAGE_QUICK_ACTION_KEYS : QUICK_ACTION_KEYS; const quickActionPrefix = heroImageMode ? "thread.empty.imageQuickActions" : "thread.empty.quickActions"; const quickActions = (
{quickActionItems.map(({ key, icon: Icon, tone }) => { const title = t(`${quickActionPrefix}.${key}.title`); const prompt = t(`${quickActionPrefix}.${key}.prompt`); return ( ); })}
); const composer = ( <> {streamError ? ( ) : null} {session ? ( ) : ( )} {showHeroComposer ? quickActions : null} ); const emptyState = loading ? (
{t("thread.loadingConversation")}
) : (

{t("thread.empty.greeting")}

); return (
); }