@usepanacea/react 0.1.0 → 0.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +242 -80
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +41 -1
- package/dist/index.d.ts +41 -1
- package/dist/index.js +242 -82
- package/dist/index.js.map +1 -1
- package/dist/styles.css +1 -1
- package/package.json +1 -1
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/token.ts","../src/provider.tsx","../src/hooks/useChat.ts","../src/hooks/useLiveSession.ts","../src/hooks/useWidget.ts","../src/lib/utils.ts","../src/components/index.tsx"],"names":["createContext","useContext","useMemo","useState","useRef","jsx","useCallback","react","useEffect","twMerge","clsx","cva","jsxs","Fragment"],"mappings":"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","file":"index.cjs","sourcesContent":["import type { PanaceaConfig } from \"./types\";\n\ninterface TokenState {\n token: string;\n /** unix ms */\n expiresAt: number;\n}\n\n/**\n * Manages the short-lived widget JWT (15 min TTL).\n * Auto-refreshes 60s before expiry. Concurrent callers share one refresh promise.\n */\nexport class TokenManager {\n private state: TokenState | null = null;\n private refreshPromise: Promise<string> | null = null;\n\n constructor(\n private readonly config: PanaceaConfig,\n private readonly apiBase: string\n ) {}\n\n async getToken(): Promise<string> {\n if (!this.state || Date.now() >= this.state.expiresAt - 60_000) {\n if (!this.refreshPromise) {\n this.refreshPromise = this.refresh().finally(() => {\n this.refreshPromise = null;\n });\n }\n return this.refreshPromise;\n }\n return this.state.token;\n }\n\n private async refresh(): Promise<string> {\n const res = await fetch(`${this.apiBase}/api/v1/auth/widget-token`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ tenantId: this.config.tenantId }),\n });\n\n if (!res.ok) {\n throw new Error(`Panacea: token refresh failed (${res.status})`);\n }\n\n const data = (await res.json()) as { token: string; expiresAt: string };\n this.state = {\n token: data.token,\n expiresAt: new Date(data.expiresAt).getTime(),\n };\n return this.state.token;\n }\n}\n","\"use client\";\n\nimport React, {\n createContext,\n useContext,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport type {\n LiveMessage,\n PanaceaConfig,\n PanaceaContextValue,\n Turn,\n} from \"./types\";\nimport { TokenManager } from \"./token\";\n\n// ---------------------------------------------------------------------------\n// Context\n// ---------------------------------------------------------------------------\n\nconst PanaceaContext = createContext<PanaceaContextValue | null>(null);\n\nexport function usePanaceaContext(): PanaceaContextValue {\n const ctx = useContext(PanaceaContext);\n if (!ctx) {\n throw new Error(\n \"usePanaceaContext must be used inside <PanaceaProvider>. \" +\n \"Wrap your component tree with <PanaceaProvider tenantId=\\\"...\\\">.\"\n );\n }\n return ctx;\n}\n\n// ---------------------------------------------------------------------------\n// Provider\n// ---------------------------------------------------------------------------\n\ninterface PanaceaProviderProps {\n children: React.ReactNode;\n tenantId: string;\n apiBase?: string;\n customerToken?: string;\n escalationKeywords?: string[];\n persona?: PanaceaConfig[\"persona\"];\n}\n\nexport function PanaceaProvider({\n children,\n tenantId,\n apiBase: apiBaseProp,\n customerToken,\n escalationKeywords,\n persona,\n}: PanaceaProviderProps) {\n const apiBase =\n apiBaseProp ??\n (typeof window !== \"undefined\" ? window.location.origin : \"\");\n\n const config: PanaceaConfig = useMemo(\n () => ({ tenantId, apiBase, customerToken, escalationKeywords, persona }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [tenantId, apiBase, customerToken]\n );\n\n // Stable token manager instance across re-renders\n const tokenMgr = useMemo(\n () => new TokenManager(config, apiBase),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [tenantId, apiBase]\n );\n\n // AI conversation\n const [turns, setTurns] = useState<Turn[]>([]);\n const [loading, setLoading] = useState(false);\n const [streaming, setStreaming] = useState(\"\");\n const sessionId = useRef<string | null>(null);\n\n // Live handover\n const [liveEscalationId, setLiveEscalationId] = useState<string | null>(null);\n const [liveMessages, setLiveMessages] = useState<LiveMessage[]>([]);\n\n // Panel\n const [isOpen, setIsOpen] = useState(false);\n\n const value: PanaceaContextValue = useMemo(\n () => ({\n config,\n apiBase,\n getToken: () => tokenMgr.getToken(),\n turns,\n setTurns,\n loading,\n setLoading,\n streaming,\n setStreaming,\n sessionId,\n liveEscalationId,\n setLiveEscalationId,\n liveMessages,\n setLiveMessages,\n isOpen,\n setIsOpen,\n }),\n [\n config,\n apiBase,\n tokenMgr,\n turns,\n loading,\n streaming,\n liveEscalationId,\n liveMessages,\n isOpen,\n ]\n );\n\n return (\n <PanaceaContext.Provider value={value}>{children}</PanaceaContext.Provider>\n );\n}\n","\"use client\";\n\nimport { useCallback } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { Source, Turn } from \"../types\";\n\ninterface QueryResponse {\n answer: string;\n sources: Source[];\n confidence: number;\n flaggedForReview: boolean;\n sessionId: string;\n escalated?: boolean;\n}\n\ninterface UseChatReturn {\n /** All conversation turns so far */\n turns: Turn[];\n /** True while an AI response is in flight */\n loading: boolean;\n /** Partial token stream content during SSE streaming */\n streaming: string;\n /** Current session ID (null until first message) */\n sessionId: string | null;\n /** True once the session has been escalated to a human */\n isEscalated: boolean;\n /** Send a message. Routes to live agent if session is escalated. */\n send: (message: string) => Promise<void>;\n /** Manually trigger escalation to a human agent */\n escalate: (reason?: string) => Promise<void>;\n /** Post a thumbs-up/down reaction for a turn by index */\n react: (turnIndex: number, reaction: \"helpful\" | \"unhelpful\") => Promise<void>;\n /** Clear all turns and start fresh */\n reset: () => void;\n}\n\nexport function useChat(): UseChatReturn {\n const {\n config,\n apiBase,\n getToken,\n turns,\n setTurns,\n loading,\n setLoading,\n streaming,\n setStreaming,\n sessionId,\n liveEscalationId,\n setLiveEscalationId,\n } = usePanaceaContext();\n\n const send = useCallback(\n async (message: string) => {\n // While escalated, route to the live customer-message endpoint\n if (liveEscalationId) {\n setTurns((prev) => [...prev, { role: \"user\", content: message }]);\n try {\n const token = await getToken();\n await fetch(\n `${apiBase}/api/v1/inbox/${liveEscalationId}/customer-message`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ content: message }),\n }\n );\n } catch {\n // Message shown optimistically — non-critical\n }\n return;\n }\n\n setLoading(true);\n setStreaming(\"\");\n setTurns((prev) => [...prev, { role: \"user\", content: message }]);\n\n try {\n const token = await getToken();\n const res = await fetch(`${apiBase}/api/v1/query`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n Accept: \"text/event-stream\",\n },\n body: JSON.stringify({\n question: message,\n sessionId: sessionId.current,\n }),\n });\n\n if (!res.ok || !res.body) {\n throw new Error(`Query failed: ${res.status}`);\n }\n\n // SSE streaming\n const reader = res.body.getReader();\n const decoder = new TextDecoder();\n let buffer = \"\";\n let finalResponse: QueryResponse | null = null;\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split(\"\\n\");\n buffer = lines.pop() ?? \"\";\n\n for (const line of lines) {\n if (!line.startsWith(\"data: \")) continue;\n const raw = line.slice(6).trim();\n if (!raw || raw === \"[DONE]\") continue;\n try {\n const frame = JSON.parse(raw) as {\n type: \"token\" | \"done\" | \"error\";\n delta?: string;\n answer?: string;\n sources?: Source[];\n confidence?: number;\n sessionId?: string;\n flaggedForReview?: boolean;\n escalated?: boolean;\n message?: string;\n };\n if (frame.type === \"token\" && frame.delta) {\n setStreaming((s) => s + frame.delta);\n } else if (frame.type === \"done\") {\n finalResponse = {\n answer: frame.answer ?? \"\",\n sources: frame.sources ?? [],\n confidence: frame.confidence ?? 0,\n flaggedForReview: frame.flaggedForReview ?? false,\n sessionId: frame.sessionId ?? \"\",\n escalated: frame.escalated,\n };\n } else if (frame.type === \"error\") {\n throw new Error(frame.message ?? \"Stream error\");\n }\n } catch {\n // Ignore malformed frames\n }\n }\n }\n\n if (!finalResponse) throw new Error(\"Stream ended without done frame\");\n\n sessionId.current = finalResponse.sessionId;\n\n setTurns((prev) => [\n ...prev,\n {\n role: \"assistant\",\n content: finalResponse!.answer,\n confidence: finalResponse!.confidence,\n sources: finalResponse!.sources,\n flaggedForReview: finalResponse!.flaggedForReview,\n reaction: null,\n escalated: finalResponse!.escalated,\n },\n ]);\n } catch (err) {\n setTurns((prev) => [\n ...prev,\n {\n role: \"assistant\",\n content:\n err instanceof Error\n ? `Something went wrong: ${err.message}`\n : \"Something went wrong. Please try again.\",\n },\n ]);\n } finally {\n setLoading(false);\n setStreaming(\"\");\n }\n },\n [\n apiBase,\n getToken,\n liveEscalationId,\n sessionId,\n setLoading,\n setStreaming,\n setTurns,\n ]\n );\n\n const escalate = useCallback(\n async (reason = \"Customer requested human support\") => {\n if (!sessionId.current) return;\n try {\n const token = await getToken();\n const res = await fetch(\n `${apiBase}/api/v1/sessions/${sessionId.current}/escalate`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ reason }),\n }\n );\n if (res.ok) {\n const data = (await res.json()) as {\n data?: { escalationId?: string };\n escalationId?: string;\n };\n const id = data?.data?.escalationId ?? data?.escalationId;\n if (id) setLiveEscalationId(id);\n }\n } catch {\n // Non-critical\n }\n },\n [apiBase, getToken, sessionId, setLiveEscalationId]\n );\n\n const react = useCallback(\n async (turnIndex: number, reaction: \"helpful\" | \"unhelpful\") => {\n if (!sessionId.current) return;\n setTurns((prev) =>\n prev.map((t, i) => (i === turnIndex ? { ...t, reaction } : t))\n );\n try {\n const token = await getToken();\n await fetch(\n `${apiBase}/api/v1/sessions/${sessionId.current}/reaction`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ reaction, turnIndex }),\n }\n );\n } catch {\n // Non-critical\n }\n },\n [apiBase, getToken, sessionId, setTurns]\n );\n\n const reset = useCallback(() => {\n setTurns([]);\n setStreaming(\"\");\n sessionId.current = null;\n }, [sessionId, setStreaming, setTurns]);\n\n return {\n turns,\n loading,\n streaming,\n sessionId: sessionId.current,\n isEscalated: !!liveEscalationId,\n send,\n escalate,\n react,\n reset,\n };\n}\n","\"use client\";\n\nimport { useCallback, useEffect, useRef } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { LiveMessage } from \"../types\";\n\ninterface UseLiveSessionReturn {\n /** True once the session has been escalated */\n isLive: boolean;\n /** The active escalation ID, or null */\n escalationId: string | null;\n /** All messages exchanged since escalation (agent + customer) */\n liveMessages: LiveMessage[];\n /** Agent-only messages (for display in widget — customer messages shown optimistically) */\n agentMessages: LiveMessage[];\n /** Send a customer message to the live agent */\n sendMessage: (content: string) => Promise<void>;\n}\n\nexport function useLiveSession(): UseLiveSessionReturn {\n const {\n apiBase,\n getToken,\n liveEscalationId,\n liveMessages,\n setLiveMessages,\n } = usePanaceaContext();\n\n const seenIds = useRef<Set<string>>(new Set());\n\n // Poll for new messages every 3s while escalated\n useEffect(() => {\n if (!liveEscalationId) return;\n\n const poll = async () => {\n try {\n const token = await getToken();\n const res = await fetch(\n `${apiBase}/api/v1/inbox/${liveEscalationId}/messages`,\n { headers: { Authorization: `Bearer ${token}` } }\n );\n if (!res.ok) return;\n\n const data = (await res.json()) as {\n data?: { messages?: LiveMessage[] };\n messages?: LiveMessage[];\n };\n const all: LiveMessage[] =\n data?.data?.messages ?? data?.messages ?? [];\n\n // Only add messages we haven't seen yet\n const newMsgs = all.filter((m) => !seenIds.current.has(m.id));\n if (newMsgs.length > 0) {\n newMsgs.forEach((m) => seenIds.current.add(m.id));\n setLiveMessages((prev) => [...prev, ...newMsgs]);\n }\n } catch {\n // Non-critical\n }\n };\n\n poll();\n const id = setInterval(poll, 3_000);\n return () => clearInterval(id);\n }, [liveEscalationId, apiBase, getToken, setLiveMessages]);\n\n const sendMessage = useCallback(\n async (content: string) => {\n if (!liveEscalationId) return;\n try {\n const token = await getToken();\n await fetch(\n `${apiBase}/api/v1/inbox/${liveEscalationId}/customer-message`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ content }),\n }\n );\n } catch {\n // Non-critical — message shown optimistically by the caller\n }\n },\n [liveEscalationId, apiBase, getToken]\n );\n\n return {\n isLive: !!liveEscalationId,\n escalationId: liveEscalationId,\n liveMessages,\n agentMessages: liveMessages.filter((m) => m.senderRole === \"agent\"),\n sendMessage,\n };\n}\n","\"use client\";\n\nimport { useCallback } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\n\ninterface UseWidgetReturn {\n isOpen: boolean;\n open: () => void;\n close: () => void;\n toggle: () => void;\n}\n\nexport function useWidget(): UseWidgetReturn {\n const { isOpen, setIsOpen } = usePanaceaContext();\n\n const open = useCallback(() => setIsOpen(true), [setIsOpen]);\n const close = useCallback(() => setIsOpen(false), [setIsOpen]);\n const toggle = useCallback(() => setIsOpen((v) => !v), [setIsOpen]);\n\n return { isOpen, open, close, toggle };\n}\n","import { clsx, type ClassValue } from \"clsx\";\nimport { twMerge } from \"tailwind-merge\";\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n","\"use client\";\n\nimport React, { useEffect, useRef } from \"react\";\nimport { cva, type VariantProps } from \"class-variance-authority\";\nimport { cn } from \"../lib/utils\";\nimport { useChat } from \"../hooks/useChat\";\nimport { useLiveSession } from \"../hooks/useLiveSession\";\nimport { useWidget } from \"../hooks/useWidget\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { Turn } from \"../types\";\n\n// ---------------------------------------------------------------------------\n// Button (shadcn pattern)\n// ---------------------------------------------------------------------------\n\nconst buttonVariants = cva(\n \"inline-flex items-center justify-center gap-1.5 rounded-lg text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50\",\n {\n variants: {\n variant: {\n default: \"bg-primary text-primary-foreground hover:bg-primary/90\",\n outline:\n \"border border-input bg-background hover:bg-accent hover:text-accent-foreground\",\n ghost: \"hover:bg-accent hover:text-accent-foreground\",\n },\n size: {\n default: \"h-9 px-4 py-2\",\n sm: \"h-8 px-3 text-xs\",\n icon: \"size-9\",\n },\n },\n defaultVariants: { variant: \"default\", size: \"default\" },\n }\n);\n\ninterface ButtonProps\n extends React.ButtonHTMLAttributes<HTMLButtonElement>,\n VariantProps<typeof buttonVariants> {}\n\nfunction Button({ className, variant, size, ...props }: ButtonProps) {\n return (\n <button className={cn(buttonVariants({ variant, size }), className)} {...props} />\n );\n}\n\n// ---------------------------------------------------------------------------\n// FAB\n// ---------------------------------------------------------------------------\n\nexport function PanaceaFAB({ className }: { className?: string }) {\n const { toggle } = useWidget();\n return (\n <button\n aria-label=\"Open support chat\"\n onClick={toggle}\n className={cn(\n \"fixed bottom-6 right-6 z-[2147483647] flex size-14 items-center justify-center rounded-full bg-primary text-2xl text-primary-foreground shadow-lg transition-transform hover:scale-105\",\n className\n )}\n >\n 💬\n </button>\n );\n}\n\n// ---------------------------------------------------------------------------\n// Message bubble\n// ---------------------------------------------------------------------------\n\nfunction Bubble({ turn }: { turn: Turn }) {\n const isUser = turn.role === \"user\";\n const isAgent = turn.isLiveAgent;\n return (\n <div className={cn(\"mb-3 flex flex-col\", isUser ? \"items-end\" : \"items-start\")}>\n {isAgent && (\n <span className=\"mb-1 text-[10px] font-bold uppercase tracking-wide text-blue-600\">\n Agent\n </span>\n )}\n <div\n className={cn(\n \"max-w-[82%] rounded-2xl px-3.5 py-2.5 text-sm leading-relaxed\",\n isUser\n ? \"rounded-tr-sm bg-primary text-primary-foreground\"\n : isAgent\n ? \"rounded-tl-sm bg-blue-100 text-blue-900\"\n : \"rounded-tl-sm bg-muted text-foreground\"\n )}\n >\n {turn.content}\n </div>\n </div>\n );\n}\n\nfunction TypingDots() {\n return (\n <div className=\"mb-3 flex items-start\">\n <div className=\"flex gap-1 rounded-2xl rounded-tl-sm bg-muted px-3.5 py-3\">\n {[0, 1, 2].map((i) => (\n <span\n key={i}\n className=\"size-1.5 animate-bounce rounded-full bg-muted-foreground/50\"\n style={{ animationDelay: `${i * 0.15}s` }}\n />\n ))}\n </div>\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaMessages\n// ---------------------------------------------------------------------------\n\nexport function PanaceaMessages({ className }: { className?: string }) {\n const { turns, loading, streaming } = useChat();\n const { agentMessages } = useLiveSession();\n const bottomRef = useRef<HTMLDivElement>(null);\n\n const agentTurns: Turn[] = agentMessages.map((m) => ({\n role: \"assistant\",\n content: m.content,\n isLiveAgent: true,\n }));\n const allTurns = [...turns, ...agentTurns];\n\n useEffect(() => {\n bottomRef.current?.scrollIntoView({ behavior: \"smooth\" });\n }, [allTurns.length]);\n\n return (\n <div className={cn(\"flex flex-1 flex-col overflow-y-auto p-4\", className)}>\n {allTurns.map((turn, i) => (\n <Bubble key={i} turn={turn} />\n ))}\n {loading && !streaming && <TypingDots />}\n {streaming && <Bubble turn={{ role: \"assistant\", content: streaming }} />}\n <div ref={bottomRef} />\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaInput\n// ---------------------------------------------------------------------------\n\nexport function PanaceaInput({\n placeholder,\n className,\n}: {\n placeholder?: string;\n className?: string;\n}) {\n const { send, loading, isEscalated } = useChat();\n const inputRef = useRef<HTMLInputElement>(null);\n\n const handleSend = () => {\n const val = inputRef.current?.value.trim();\n if (!val || loading) return;\n void send(val);\n if (inputRef.current) inputRef.current.value = \"\";\n };\n\n return (\n <div className={cn(\"flex gap-2 border-t border-border bg-background p-3\", className)}>\n <input\n ref={inputRef}\n placeholder={isEscalated ? \"Reply to agent…\" : (placeholder ?? \"Ask a question…\")}\n disabled={loading}\n onKeyDown={(e) => e.key === \"Enter\" && handleSend()}\n className=\"flex-1 rounded-full border border-input bg-muted/40 px-4 py-2 text-sm outline-none placeholder:text-muted-foreground focus:border-primary focus:bg-background disabled:opacity-50\"\n />\n <Button\n onClick={handleSend}\n disabled={loading}\n size=\"sm\"\n className=\"shrink-0 rounded-full px-4\"\n >\n Send\n </Button>\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaChat — full pre-built floating panel\n// ---------------------------------------------------------------------------\n\nexport function PanaceaChat({\n title,\n placeholder,\n className,\n}: {\n title?: string;\n placeholder?: string;\n className?: string;\n}) {\n const { config } = usePanaceaContext();\n const { isOpen, close } = useWidget();\n const panelTitle = title ?? config.persona?.name ?? \"Support\";\n\n return (\n <>\n <PanaceaFAB />\n <div\n className={cn(\n \"fixed bottom-24 right-6 z-[2147483646] flex w-[360px] flex-col overflow-hidden rounded-2xl border border-border bg-background shadow-2xl transition-all duration-200\",\n isOpen\n ? \"pointer-events-auto h-[520px] opacity-100\"\n : \"pointer-events-none h-0 opacity-0\",\n className\n )}\n >\n {/* Header */}\n <div className=\"flex shrink-0 items-center justify-between bg-primary px-4 py-3.5\">\n <span className=\"text-sm font-semibold text-primary-foreground\">\n {panelTitle}\n </span>\n <button\n onClick={close}\n aria-label=\"Close\"\n className=\"text-primary-foreground/70 transition-colors hover:text-primary-foreground\"\n >\n ✕\n </button>\n </div>\n\n <PanaceaMessages />\n <PanaceaInput placeholder={placeholder} />\n </div>\n </>\n );\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/token.ts","../src/provider.tsx","../src/hooks/useChat.ts","../src/hooks/useLiveSession.ts","../src/hooks/useWidget.ts","../src/hooks/useSessionHistory.ts","../src/lib/utils.ts","../src/components/index.tsx","../src/components/PanaceaHistory.tsx"],"names":["createContext","useContext","useMemo","useState","useRef","useEffect","jsx","useCallback","react","twMerge","clsx","cva","jsxs","Fragment"],"mappings":"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","file":"index.cjs","sourcesContent":["import type { PanaceaConfig } from \"./types\";\n\ninterface TokenState {\n token: string;\n /** unix ms */\n expiresAt: number;\n}\n\n/**\n * Manages the short-lived widget JWT (15 min TTL).\n * Auto-refreshes 60s before expiry. Concurrent callers share one refresh promise.\n */\nexport class TokenManager {\n private state: TokenState | null = null;\n private refreshPromise: Promise<string> | null = null;\n\n constructor(\n private readonly config: PanaceaConfig,\n private readonly apiBase: string,\n ) {}\n\n async getToken(): Promise<string> {\n if (!this.state || Date.now() >= this.state.expiresAt - 60_000) {\n if (!this.refreshPromise) {\n this.refreshPromise = this.refresh().finally(() => {\n this.refreshPromise = null;\n });\n }\n return this.refreshPromise;\n }\n return this.state.token;\n }\n\n private async refresh(): Promise<string> {\n const res = await fetch(`${this.apiBase}/api/v1/auth/widget-token`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ tenantId: this.config.tenantId }),\n });\n\n if (!res.ok) {\n throw new Error(`Panacea: token refresh failed (${res.status})`);\n }\n\n const data = (await res.json()) as { token: string; expiresAt: string };\n this.state = {\n token: data.token,\n expiresAt: new Date(data.expiresAt).getTime(),\n };\n return this.state.token;\n }\n}\n","\"use client\";\n\nimport React, { createContext, useContext, useEffect, useMemo, useRef, useState } from \"react\";\nimport type { LiveMessage, PanaceaConfig, PanaceaContextValue, Turn } from \"./types\";\nimport { TokenManager } from \"./token\";\n\n// ---------------------------------------------------------------------------\n// Context\n// ---------------------------------------------------------------------------\n\nconst PanaceaContext = createContext<PanaceaContextValue | null>(null);\n\nexport function usePanaceaContext(): PanaceaContextValue {\n const ctx = useContext(PanaceaContext);\n if (!ctx) {\n throw new Error(\n \"usePanaceaContext must be used inside <PanaceaProvider>. \" +\n 'Wrap your component tree with <PanaceaProvider tenantId=\"...\">.',\n );\n }\n return ctx;\n}\n\n// ---------------------------------------------------------------------------\n// Provider\n// ---------------------------------------------------------------------------\n\ninterface PanaceaProviderProps {\n children: React.ReactNode;\n tenantId: string;\n apiBase?: string;\n customerToken?: string;\n escalationKeywords?: string[];\n persona?: PanaceaConfig[\"persona\"];\n}\n\nexport function PanaceaProvider({\n children,\n tenantId,\n apiBase: apiBaseProp,\n customerToken,\n escalationKeywords,\n persona,\n}: PanaceaProviderProps) {\n const apiBase = apiBaseProp ?? (typeof window !== \"undefined\" ? window.location.origin : \"\");\n\n const config: PanaceaConfig = useMemo(\n () => ({ tenantId, apiBase, customerToken, escalationKeywords, persona }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [tenantId, apiBase, customerToken],\n );\n\n // Stable token manager instance across re-renders\n const tokenMgr = useMemo(\n () => new TokenManager(config, apiBase),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [tenantId, apiBase],\n );\n\n // AI conversation\n const [turns, setTurns] = useState<Turn[]>([]);\n const [loading, setLoading] = useState(false);\n const [streaming, setStreaming] = useState(\"\");\n const sessionId = useRef<string | null>(null);\n\n // Live handover\n const [liveEscalationId, setLiveEscalationId] = useState<string | null>(null);\n const [liveMessages, setLiveMessages] = useState<LiveMessage[]>([]);\n\n // Panel\n const [isOpen, setIsOpen] = useState(false);\n\n // Session resume: on mount (or when customerToken changes), attempt to\n // restore the most recent resumable session for this identified customer.\n useEffect(() => {\n if (!customerToken) return;\n let cancelled = false;\n\n async function tryResume() {\n try {\n const token = await tokenMgr.getToken();\n const url = new URL(`${apiBase}/api/v1/sessions/resume`);\n url.searchParams.set(\"customerToken\", customerToken!);\n const res = await fetch(url.toString(), {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (!res.ok || cancelled) return;\n const data = (await res.json()) as { session: { sessionId: string } | null };\n const resumedId = data?.session?.sessionId ?? null;\n if (!resumedId || cancelled) return;\n\n sessionId.current = resumedId;\n\n // Hydrate turns from history\n const turnsRes = await fetch(`${apiBase}/api/v1/sessions/${resumedId}/turns`, {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (!turnsRes.ok || cancelled) return;\n const turnsData = (await turnsRes.json()) as {\n data?: { turns?: Array<{ query: string; response: string }> };\n turns?: Array<{ query: string; response: string }>;\n };\n const rawTurns = turnsData?.data?.turns ?? turnsData?.turns ?? [];\n if (cancelled) return;\n const hydrated: Turn[] = rawTurns.flatMap((t) => [\n { role: \"user\" as const, content: t.query },\n { role: \"assistant\" as const, content: t.response },\n ]);\n if (hydrated.length > 0) setTurns(hydrated);\n } catch {\n // Resume is best-effort — silent fail\n }\n }\n\n void tryResume();\n return () => {\n cancelled = true;\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [customerToken, apiBase]);\n\n const value: PanaceaContextValue = useMemo(\n () => ({\n config,\n apiBase,\n getToken: () => tokenMgr.getToken(),\n turns,\n setTurns,\n loading,\n setLoading,\n streaming,\n setStreaming,\n sessionId,\n liveEscalationId,\n setLiveEscalationId,\n liveMessages,\n setLiveMessages,\n isOpen,\n setIsOpen,\n }),\n [config, apiBase, tokenMgr, turns, loading, streaming, liveEscalationId, liveMessages, isOpen],\n );\n\n return <PanaceaContext.Provider value={value}>{children}</PanaceaContext.Provider>;\n}\n","\"use client\";\n\nimport { useCallback } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { Source, Turn } from \"../types\";\n\ninterface QueryResponse {\n answer: string;\n sources: Source[];\n confidence: number;\n flaggedForReview: boolean;\n sessionId: string;\n escalated?: boolean;\n}\n\ninterface UseChatReturn {\n /** All conversation turns so far */\n turns: Turn[];\n /** True while an AI response is in flight */\n loading: boolean;\n /** Partial token stream content during SSE streaming */\n streaming: string;\n /** Current session ID (null until first message) */\n sessionId: string | null;\n /** True once the session has been escalated to a human */\n isEscalated: boolean;\n /** Send a message. Routes to live agent if session is escalated. */\n send: (message: string) => Promise<void>;\n /** Manually trigger escalation to a human agent */\n escalate: (reason?: string) => Promise<void>;\n /** Post a thumbs-up/down reaction for a turn by index */\n react: (turnIndex: number, reaction: \"helpful\" | \"unhelpful\") => Promise<void>;\n /** Push page context to the active session (URL, title, content summary, etc.) */\n setPageContext: (ctx: Record<string, unknown>) => Promise<void>;\n /** Clear all turns and start fresh */\n reset: () => void;\n}\n\nexport function useChat(): UseChatReturn {\n const {\n apiBase,\n getToken,\n turns,\n setTurns,\n loading,\n setLoading,\n streaming,\n setStreaming,\n sessionId,\n liveEscalationId,\n setLiveEscalationId,\n } = usePanaceaContext();\n\n const send = useCallback(\n async (message: string) => {\n // While escalated, route to the live customer-message endpoint\n if (liveEscalationId) {\n setTurns((prev) => [...prev, { role: \"user\", content: message }]);\n try {\n const token = await getToken();\n await fetch(`${apiBase}/api/v1/inbox/${liveEscalationId}/customer-message`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ content: message }),\n });\n } catch {\n // Message shown optimistically — non-critical\n }\n return;\n }\n\n setLoading(true);\n setStreaming(\"\");\n setTurns((prev) => [...prev, { role: \"user\", content: message }]);\n\n try {\n const token = await getToken();\n const res = await fetch(`${apiBase}/api/v1/query`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n Accept: \"text/event-stream\",\n },\n body: JSON.stringify({\n question: message,\n sessionId: sessionId.current,\n }),\n });\n\n if (!res.ok || !res.body) {\n throw new Error(`Query failed: ${res.status}`);\n }\n\n // SSE streaming\n const reader = res.body.getReader();\n const decoder = new TextDecoder();\n let buffer = \"\";\n let finalResponse: QueryResponse | null = null;\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split(\"\\n\");\n buffer = lines.pop() ?? \"\";\n\n for (const line of lines) {\n if (!line.startsWith(\"data: \")) continue;\n const raw = line.slice(6).trim();\n if (!raw || raw === \"[DONE]\") continue;\n try {\n const frame = JSON.parse(raw) as {\n type: \"token\" | \"done\" | \"error\";\n delta?: string;\n answer?: string;\n sources?: Source[];\n confidence?: number;\n sessionId?: string;\n flaggedForReview?: boolean;\n escalated?: boolean;\n message?: string;\n };\n if (frame.type === \"token\" && frame.delta) {\n setStreaming((s) => s + frame.delta);\n } else if (frame.type === \"done\") {\n finalResponse = {\n answer: frame.answer ?? \"\",\n sources: frame.sources ?? [],\n confidence: frame.confidence ?? 0,\n flaggedForReview: frame.flaggedForReview ?? false,\n sessionId: frame.sessionId ?? \"\",\n escalated: frame.escalated,\n };\n } else if (frame.type === \"error\") {\n throw new Error(frame.message ?? \"Stream error\");\n }\n } catch {\n // Ignore malformed frames\n }\n }\n }\n\n if (!finalResponse) throw new Error(\"Stream ended without done frame\");\n\n sessionId.current = finalResponse.sessionId;\n\n setTurns((prev) => [\n ...prev,\n {\n role: \"assistant\",\n content: finalResponse!.answer,\n confidence: finalResponse!.confidence,\n sources: finalResponse!.sources,\n flaggedForReview: finalResponse!.flaggedForReview,\n reaction: null,\n escalated: finalResponse!.escalated,\n },\n ]);\n } catch (err) {\n setTurns((prev) => [\n ...prev,\n {\n role: \"assistant\",\n content:\n err instanceof Error\n ? `Something went wrong: ${err.message}`\n : \"Something went wrong. Please try again.\",\n },\n ]);\n } finally {\n setLoading(false);\n setStreaming(\"\");\n }\n },\n [apiBase, getToken, liveEscalationId, sessionId, setLoading, setStreaming, setTurns],\n );\n\n const escalate = useCallback(\n async (reason = \"Customer requested human support\") => {\n if (!sessionId.current) return;\n try {\n const token = await getToken();\n const res = await fetch(`${apiBase}/api/v1/sessions/${sessionId.current}/escalate`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ reason }),\n });\n if (res.ok) {\n const data = (await res.json()) as {\n data?: { escalationId?: string };\n escalationId?: string;\n };\n const id = data?.data?.escalationId ?? data?.escalationId;\n if (id) setLiveEscalationId(id);\n }\n } catch {\n // Non-critical\n }\n },\n [apiBase, getToken, sessionId, setLiveEscalationId],\n );\n\n const react = useCallback(\n async (turnIndex: number, reaction: \"helpful\" | \"unhelpful\") => {\n if (!sessionId.current) return;\n setTurns((prev) => prev.map((t, i) => (i === turnIndex ? { ...t, reaction } : t)));\n try {\n const token = await getToken();\n await fetch(`${apiBase}/api/v1/sessions/${sessionId.current}/reaction`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ reaction, turnIndex }),\n });\n } catch {\n // Non-critical\n }\n },\n [apiBase, getToken, sessionId, setTurns],\n );\n\n const reset = useCallback(() => {\n setTurns([]);\n setStreaming(\"\");\n sessionId.current = null;\n }, [sessionId, setStreaming, setTurns]);\n\n const setPageContext = useCallback(\n async (ctx: Record<string, unknown>) => {\n if (!sessionId.current) return;\n try {\n const token = await getToken();\n await fetch(`${apiBase}/api/v1/sessions/${sessionId.current}`, {\n method: \"PATCH\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ pageContext: ctx }),\n });\n } catch {\n // Non-critical\n }\n },\n [apiBase, getToken, sessionId],\n );\n\n return {\n turns,\n loading,\n streaming,\n sessionId: sessionId.current,\n isEscalated: !!liveEscalationId,\n send,\n escalate,\n react,\n setPageContext,\n reset,\n };\n}\n","\"use client\";\n\nimport { useCallback, useEffect, useRef } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { LiveMessage } from \"../types\";\n\ninterface UseLiveSessionReturn {\n /** True once the session has been escalated */\n isLive: boolean;\n /** The active escalation ID, or null */\n escalationId: string | null;\n /** All messages exchanged since escalation (agent + customer) */\n liveMessages: LiveMessage[];\n /** Agent-only messages (for display in widget — customer messages shown optimistically) */\n agentMessages: LiveMessage[];\n /** Send a customer message to the live agent */\n sendMessage: (content: string) => Promise<void>;\n}\n\nexport function useLiveSession(): UseLiveSessionReturn {\n const { apiBase, getToken, liveEscalationId, liveMessages, setLiveMessages } =\n usePanaceaContext();\n\n const seenIds = useRef<Set<string>>(new Set());\n\n // Poll for new messages every 3s while escalated\n useEffect(() => {\n if (!liveEscalationId) return;\n\n const poll = async () => {\n try {\n const token = await getToken();\n const res = await fetch(`${apiBase}/api/v1/inbox/${liveEscalationId}/messages`, {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (!res.ok) return;\n\n const data = (await res.json()) as {\n data?: { messages?: LiveMessage[] };\n messages?: LiveMessage[];\n };\n const all: LiveMessage[] = data?.data?.messages ?? data?.messages ?? [];\n\n // Only add messages we haven't seen yet\n const newMsgs = all.filter((m) => !seenIds.current.has(m.id));\n if (newMsgs.length > 0) {\n newMsgs.forEach((m) => seenIds.current.add(m.id));\n setLiveMessages((prev) => [...prev, ...newMsgs]);\n }\n } catch {\n // Non-critical\n }\n };\n\n poll();\n const id = setInterval(poll, 3_000);\n return () => clearInterval(id);\n }, [liveEscalationId, apiBase, getToken, setLiveMessages]);\n\n const sendMessage = useCallback(\n async (content: string) => {\n if (!liveEscalationId) return;\n try {\n const token = await getToken();\n await fetch(`${apiBase}/api/v1/inbox/${liveEscalationId}/customer-message`, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ content }),\n });\n } catch {\n // Non-critical — message shown optimistically by the caller\n }\n },\n [liveEscalationId, apiBase, getToken],\n );\n\n return {\n isLive: !!liveEscalationId,\n escalationId: liveEscalationId,\n liveMessages,\n agentMessages: liveMessages.filter((m) => m.senderRole === \"agent\"),\n sendMessage,\n };\n}\n","\"use client\";\n\nimport { useCallback } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\n\ninterface UseWidgetReturn {\n isOpen: boolean;\n open: () => void;\n close: () => void;\n toggle: () => void;\n}\n\nexport function useWidget(): UseWidgetReturn {\n const { isOpen, setIsOpen } = usePanaceaContext();\n\n const open = useCallback(() => setIsOpen(true), [setIsOpen]);\n const close = useCallback(() => setIsOpen(false), [setIsOpen]);\n const toggle = useCallback(() => setIsOpen((v) => !v), [setIsOpen]);\n\n return { isOpen, open, close, toggle };\n}\n","\"use client\";\n\nimport { useCallback, useEffect, useState } from \"react\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { HistorySession, HistoryTurn, Turn } from \"../types\";\n\ninterface UseSessionHistoryReturn {\n /** List of past sessions for this customer */\n sessions: HistorySession[];\n /** True while fetching the session list */\n loading: boolean;\n /** Load and display a specific past session's turns in the active chat */\n loadSession: (sessionId: string) => Promise<void>;\n /** Refetch the session list */\n refresh: () => Promise<void>;\n}\n\nexport function useSessionHistory(): UseSessionHistoryReturn {\n const { apiBase, getToken, config, setTurns, sessionId: activeSessionIdRef } = usePanaceaContext();\n\n const [sessions, setSessions] = useState<HistorySession[]>([]);\n const [loading, setLoading] = useState(false);\n\n const fetchSessions = useCallback(async () => {\n if (!config.customerToken) return;\n setLoading(true);\n try {\n const token = await getToken();\n const url = new URL(`${apiBase}/api/v1/sessions`);\n url.searchParams.set(\"customerToken\", config.customerToken);\n const res = await fetch(url.toString(), {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (!res.ok) return;\n const data = (await res.json()) as {\n data?: HistorySession[];\n sessions?: HistorySession[];\n } | HistorySession[];\n const list = Array.isArray(data)\n ? data\n : (data as { data?: HistorySession[]; sessions?: HistorySession[] })?.data ??\n (data as { sessions?: HistorySession[] })?.sessions ??\n [];\n setSessions(list);\n } catch {\n // silent\n } finally {\n setLoading(false);\n }\n }, [apiBase, getToken, config.customerToken]);\n\n useEffect(() => {\n // eslint-disable-next-line react-hooks/set-state-in-effect\n void fetchSessions();\n }, [fetchSessions]);\n\n const loadSession = useCallback(\n async (sessionId: string) => {\n try {\n const token = await getToken();\n const res = await fetch(`${apiBase}/api/v1/sessions/${sessionId}/turns`, {\n headers: { Authorization: `Bearer ${token}` },\n });\n if (!res.ok) return;\n const data = (await res.json()) as {\n data?: { turns?: HistoryTurn[] };\n turns?: HistoryTurn[];\n };\n const raw = data?.data?.turns ?? data?.turns ?? [];\n const turns: Turn[] = raw.flatMap((t) => [\n { role: \"user\" as const, content: t.query },\n { role: \"assistant\" as const, content: t.response },\n ]);\n activeSessionIdRef.current = sessionId;\n setTurns(turns);\n } catch {\n // silent\n }\n },\n [apiBase, getToken, setTurns, activeSessionIdRef],\n );\n\n return { sessions, loading, loadSession, refresh: fetchSessions };\n}\n","import { clsx, type ClassValue } from \"clsx\";\nimport { twMerge } from \"tailwind-merge\";\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n","\"use client\";\n\nimport React, { useEffect, useRef } from \"react\";\nimport { cva, type VariantProps } from \"class-variance-authority\";\nimport { cn } from \"../lib/utils\";\nimport { useChat } from \"../hooks/useChat\";\nimport { useLiveSession } from \"../hooks/useLiveSession\";\nimport { useWidget } from \"../hooks/useWidget\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { Turn } from \"../types\";\n\n// ---------------------------------------------------------------------------\n// Button (shadcn pattern)\n// ---------------------------------------------------------------------------\n\nconst buttonVariants = cva(\n \"inline-flex items-center justify-center gap-1.5 rounded-lg text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50\",\n {\n variants: {\n variant: {\n default: \"bg-primary text-primary-foreground hover:bg-primary/90\",\n outline: \"border border-input bg-background hover:bg-accent hover:text-accent-foreground\",\n ghost: \"hover:bg-accent hover:text-accent-foreground\",\n },\n size: {\n default: \"h-9 px-4 py-2\",\n sm: \"h-8 px-3 text-xs\",\n icon: \"size-9\",\n },\n },\n defaultVariants: { variant: \"default\", size: \"default\" },\n },\n);\n\ninterface ButtonProps\n extends React.ButtonHTMLAttributes<HTMLButtonElement>, VariantProps<typeof buttonVariants> {}\n\nfunction Button({ className, variant, size, ...props }: ButtonProps) {\n return <button className={cn(buttonVariants({ variant, size }), className)} {...props} />;\n}\n\n// ---------------------------------------------------------------------------\n// FAB\n// ---------------------------------------------------------------------------\n\nexport function PanaceaFAB({ className }: { className?: string }) {\n const { toggle } = useWidget();\n return (\n <button\n aria-label=\"Open support chat\"\n onClick={toggle}\n className={cn(\n \"fixed bottom-6 right-6 z-[2147483647] flex size-14 items-center justify-center rounded-full bg-primary text-2xl text-primary-foreground shadow-lg transition-transform hover:scale-105\",\n className,\n )}\n >\n 💬\n </button>\n );\n}\n\n// ---------------------------------------------------------------------------\n// Message bubble\n// ---------------------------------------------------------------------------\n\n/**\n * Parse [[Label|wiki/path]] or [[wiki/path]] citations from agent answers\n * and render them as styled inline spans (or links when sourceUrl is known).\n * Falls back to plain text for malformed tokens.\n */\nfunction renderContent(text: string): React.ReactNode[] {\n const parts = text.split(/(\\[\\[[^\\]]+\\]\\])/g);\n return parts\n .map((part, i) => {\n const m = part.match(/^\\[\\[(?:([^|\\]]+)\\|)?([^\\]]+)\\]\\]$/);\n if (m) {\n const label = (m[1] ?? m[2] ?? part).trim();\n return (\n <span\n key={i}\n className=\"inline-flex items-center gap-0.5 rounded bg-blue-50 px-1 py-0.5 text-xs font-medium text-blue-700\"\n >\n <svg className=\"inline size-3 shrink-0\" viewBox=\"0 0 16 16\" fill=\"currentColor\">\n <path d=\"M2 2h5v2H4v8h8v-3h2v5H2V2zm7 0h5v5h-2V4.414L6.707 9.707 5.293 8.293 10.586 3H9V2z\" />\n </svg>\n {label}\n </span>\n );\n }\n if (!part) return null;\n return <span key={i}>{part}</span>;\n })\n .filter(Boolean) as React.ReactNode[];\n}\n\nfunction Bubble({ turn }: { turn: Turn }) {\n const isUser = turn.role === \"user\";\n const isAgent = turn.isLiveAgent;\n return (\n <div className={cn(\"mb-3 flex flex-col\", isUser ? \"items-end\" : \"items-start\")}>\n {isAgent && (\n <span className=\"mb-1 text-[10px] font-bold uppercase tracking-wide text-blue-600\">\n Agent\n </span>\n )}\n <div\n className={cn(\n \"max-w-[82%] rounded-2xl px-3.5 py-2.5 text-sm leading-relaxed\",\n isUser\n ? \"rounded-tr-sm bg-primary text-primary-foreground\"\n : isAgent\n ? \"rounded-tl-sm bg-blue-100 text-blue-900\"\n : \"rounded-tl-sm bg-muted text-foreground\",\n )}\n >\n {isUser ? turn.content : renderContent(turn.content)}\n </div>\n </div>\n );\n}\n\nfunction TypingDots() {\n return (\n <div className=\"mb-3 flex items-start\">\n <div className=\"flex gap-1 rounded-2xl rounded-tl-sm bg-muted px-3.5 py-3\">\n {[0, 1, 2].map((i) => (\n <span\n key={i}\n className=\"size-1.5 animate-bounce rounded-full bg-muted-foreground/50\"\n style={{ animationDelay: `${i * 0.15}s` }}\n />\n ))}\n </div>\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaMessages\n// ---------------------------------------------------------------------------\n\nexport function PanaceaMessages({ className }: { className?: string }) {\n const { turns, loading, streaming } = useChat();\n const { agentMessages } = useLiveSession();\n const bottomRef = useRef<HTMLDivElement>(null);\n\n const agentTurns: Turn[] = agentMessages.map((m) => ({\n role: \"assistant\",\n content: m.content,\n isLiveAgent: true,\n }));\n const allTurns = [...turns, ...agentTurns];\n\n useEffect(() => {\n bottomRef.current?.scrollIntoView({ behavior: \"smooth\" });\n }, [allTurns.length]);\n\n return (\n <div className={cn(\"flex flex-1 flex-col overflow-y-auto p-4\", className)}>\n {allTurns.map((turn, i) => (\n <Bubble key={i} turn={turn} />\n ))}\n {loading && !streaming && <TypingDots />}\n {streaming && <Bubble turn={{ role: \"assistant\", content: streaming }} />}\n <div ref={bottomRef} />\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaInput\n// ---------------------------------------------------------------------------\n\nexport function PanaceaInput({\n placeholder,\n className,\n}: {\n placeholder?: string;\n className?: string;\n}) {\n const { send, loading, isEscalated } = useChat();\n const inputRef = useRef<HTMLInputElement>(null);\n\n const handleSend = () => {\n const val = inputRef.current?.value.trim();\n if (!val || loading) return;\n void send(val);\n if (inputRef.current) inputRef.current.value = \"\";\n };\n\n return (\n <div className={cn(\"flex gap-2 border-t border-border bg-background p-3\", className)}>\n <input\n ref={inputRef}\n placeholder={isEscalated ? \"Reply to agent…\" : (placeholder ?? \"Ask a question…\")}\n disabled={loading}\n onKeyDown={(e) => e.key === \"Enter\" && handleSend()}\n className=\"flex-1 rounded-full border border-input bg-muted/40 px-4 py-2 text-sm outline-none placeholder:text-muted-foreground focus:border-primary focus:bg-background disabled:opacity-50\"\n />\n <Button\n onClick={handleSend}\n disabled={loading}\n size=\"sm\"\n className=\"shrink-0 rounded-full px-4\"\n >\n Send\n </Button>\n </div>\n );\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaChat — full pre-built floating panel\n// ---------------------------------------------------------------------------\n\nexport function PanaceaChat({\n title,\n placeholder,\n className,\n}: {\n title?: string;\n placeholder?: string;\n className?: string;\n}) {\n const { config } = usePanaceaContext();\n const { isOpen, close } = useWidget();\n const panelTitle = title ?? config.persona?.name ?? \"Support\";\n\n return (\n <>\n <PanaceaFAB />\n <div\n className={cn(\n \"fixed bottom-24 right-6 z-[2147483646] flex w-[360px] flex-col overflow-hidden rounded-2xl border border-border bg-background shadow-2xl transition-all duration-200\",\n isOpen\n ? \"pointer-events-auto h-[520px] opacity-100\"\n : \"pointer-events-none h-0 opacity-0\",\n className,\n )}\n >\n {/* Header */}\n <div className=\"flex shrink-0 items-center justify-between bg-primary px-4 py-3.5\">\n <span className=\"text-sm font-semibold text-primary-foreground\">{panelTitle}</span>\n <button\n onClick={close}\n aria-label=\"Close\"\n className=\"text-primary-foreground/70 transition-colors hover:text-primary-foreground\"\n >\n ✕\n </button>\n </div>\n\n <PanaceaMessages />\n <PanaceaInput placeholder={placeholder} />\n </div>\n </>\n );\n}\n","\"use client\";\n\nimport React, { useState } from \"react\";\nimport { cn } from \"../lib/utils\";\nimport { useSessionHistory } from \"../hooks/useSessionHistory\";\nimport { usePanaceaContext } from \"../provider\";\nimport type { HistorySession } from \"../types\";\n\n// ---------------------------------------------------------------------------\n// Helpers\n// ---------------------------------------------------------------------------\n\nfunction formatDate(iso: string): string {\n try {\n return new Intl.DateTimeFormat(undefined, {\n month: \"short\",\n day: \"numeric\",\n hour: \"2-digit\",\n minute: \"2-digit\",\n }).format(new Date(iso));\n } catch {\n return iso;\n }\n}\n\n// ---------------------------------------------------------------------------\n// PanaceaHistory\n// ---------------------------------------------------------------------------\n\ninterface PanaceaHistoryProps {\n /** Called when the user clicks a session row and it loads into chat */\n onSessionLoaded?: (sessionId: string) => void;\n className?: string;\n}\n\n/**\n * Renders a scrollable list of past sessions for the authenticated customer.\n * Requires `customerToken` to be set on the parent `<PanaceaProvider>`.\n * Clicking a row loads the turns into the active chat.\n */\nexport function PanaceaHistory({ onSessionLoaded, className }: PanaceaHistoryProps) {\n const { config } = usePanaceaContext();\n const { sessions, loading, loadSession } = useSessionHistory();\n const [active, setActive] = useState<string | null>(null);\n const [loadingId, setLoadingId] = useState<string | null>(null);\n\n if (!config.customerToken) {\n return (\n <div className={cn(\"flex flex-col items-center justify-center gap-2 py-8 text-center text-sm text-muted-foreground\", className)}>\n <span>Sign in to see your conversation history.</span>\n </div>\n );\n }\n\n if (loading) {\n return (\n <div className={cn(\"flex flex-col gap-2 p-3\", className)}>\n {[1, 2, 3].map((i) => (\n <div\n key={i}\n className=\"h-14 animate-pulse rounded-lg bg-muted\"\n />\n ))}\n </div>\n );\n }\n\n if (sessions.length === 0) {\n return (\n <div className={cn(\"flex flex-col items-center justify-center gap-2 py-8 text-center text-sm text-muted-foreground\", className)}>\n <span>No previous conversations found.</span>\n </div>\n );\n }\n\n async function handleLoad(session: HistorySession) {\n setLoadingId(session.sessionId);\n await loadSession(session.sessionId);\n setActive(session.sessionId);\n setLoadingId(null);\n onSessionLoaded?.(session.sessionId);\n }\n\n return (\n <div className={cn(\"flex flex-col gap-1 overflow-y-auto p-2\", className)}>\n {sessions.map((session) => {\n const isActive = active === session.sessionId;\n const isLoading = loadingId === session.sessionId;\n return (\n <button\n key={session.sessionId}\n onClick={() => void handleLoad(session)}\n disabled={isLoading}\n className={cn(\n \"flex w-full flex-col gap-0.5 rounded-lg px-3 py-2.5 text-left text-sm transition-colors hover:bg-accent disabled:cursor-wait\",\n isActive && \"bg-accent\",\n )}\n >\n <span className=\"truncate font-medium leading-snug text-foreground\">\n {session.preview ?? \"Conversation\"}\n </span>\n <div className=\"flex items-center gap-2 text-xs text-muted-foreground\">\n <span>{formatDate(session.createdAt)}</span>\n {session.turnCount !== undefined && (\n <>\n <span aria-hidden>·</span>\n <span>{session.turnCount} {session.turnCount === 1 ? \"message\" : \"messages\"}</span>\n </>\n )}\n {isLoading && (\n <>\n <span aria-hidden>·</span>\n <span>Loading…</span>\n </>\n )}\n </div>\n </button>\n );\n })}\n </div>\n );\n}\n"]}
|
package/dist/index.d.cts
CHANGED
|
@@ -38,6 +38,20 @@ interface LiveMessage {
|
|
|
38
38
|
content: string;
|
|
39
39
|
createdAt: string;
|
|
40
40
|
}
|
|
41
|
+
interface HistoryTurn {
|
|
42
|
+
id: string;
|
|
43
|
+
query: string;
|
|
44
|
+
response: string;
|
|
45
|
+
createdAt: string;
|
|
46
|
+
}
|
|
47
|
+
interface HistorySession {
|
|
48
|
+
sessionId: string;
|
|
49
|
+
createdAt: string;
|
|
50
|
+
updatedAt?: string;
|
|
51
|
+
/** Preview of the first message in the session */
|
|
52
|
+
preview?: string;
|
|
53
|
+
turnCount?: number;
|
|
54
|
+
}
|
|
41
55
|
|
|
42
56
|
interface PanaceaProviderProps {
|
|
43
57
|
children: React.ReactNode;
|
|
@@ -66,6 +80,8 @@ interface UseChatReturn {
|
|
|
66
80
|
escalate: (reason?: string) => Promise<void>;
|
|
67
81
|
/** Post a thumbs-up/down reaction for a turn by index */
|
|
68
82
|
react: (turnIndex: number, reaction: "helpful" | "unhelpful") => Promise<void>;
|
|
83
|
+
/** Push page context to the active session (URL, title, content summary, etc.) */
|
|
84
|
+
setPageContext: (ctx: Record<string, unknown>) => Promise<void>;
|
|
69
85
|
/** Clear all turns and start fresh */
|
|
70
86
|
reset: () => void;
|
|
71
87
|
}
|
|
@@ -93,6 +109,18 @@ interface UseWidgetReturn {
|
|
|
93
109
|
}
|
|
94
110
|
declare function useWidget(): UseWidgetReturn;
|
|
95
111
|
|
|
112
|
+
interface UseSessionHistoryReturn {
|
|
113
|
+
/** List of past sessions for this customer */
|
|
114
|
+
sessions: HistorySession[];
|
|
115
|
+
/** True while fetching the session list */
|
|
116
|
+
loading: boolean;
|
|
117
|
+
/** Load and display a specific past session's turns in the active chat */
|
|
118
|
+
loadSession: (sessionId: string) => Promise<void>;
|
|
119
|
+
/** Refetch the session list */
|
|
120
|
+
refresh: () => Promise<void>;
|
|
121
|
+
}
|
|
122
|
+
declare function useSessionHistory(): UseSessionHistoryReturn;
|
|
123
|
+
|
|
96
124
|
declare function PanaceaFAB({ className }: {
|
|
97
125
|
className?: string;
|
|
98
126
|
}): React.JSX.Element;
|
|
@@ -109,4 +137,16 @@ declare function PanaceaChat({ title, placeholder, className, }: {
|
|
|
109
137
|
className?: string;
|
|
110
138
|
}): React.JSX.Element;
|
|
111
139
|
|
|
112
|
-
|
|
140
|
+
interface PanaceaHistoryProps {
|
|
141
|
+
/** Called when the user clicks a session row and it loads into chat */
|
|
142
|
+
onSessionLoaded?: (sessionId: string) => void;
|
|
143
|
+
className?: string;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Renders a scrollable list of past sessions for the authenticated customer.
|
|
147
|
+
* Requires `customerToken` to be set on the parent `<PanaceaProvider>`.
|
|
148
|
+
* Clicking a row loads the turns into the active chat.
|
|
149
|
+
*/
|
|
150
|
+
declare function PanaceaHistory({ onSessionLoaded, className }: PanaceaHistoryProps): React.JSX.Element;
|
|
151
|
+
|
|
152
|
+
export { type HistorySession, type HistoryTurn, type LiveMessage, PanaceaChat, type PanaceaConfig, PanaceaFAB, PanaceaHistory, PanaceaInput, PanaceaMessages, PanaceaProvider, type Source, type Turn, useChat, useLiveSession, useSessionHistory, useWidget };
|
package/dist/index.d.ts
CHANGED
|
@@ -38,6 +38,20 @@ interface LiveMessage {
|
|
|
38
38
|
content: string;
|
|
39
39
|
createdAt: string;
|
|
40
40
|
}
|
|
41
|
+
interface HistoryTurn {
|
|
42
|
+
id: string;
|
|
43
|
+
query: string;
|
|
44
|
+
response: string;
|
|
45
|
+
createdAt: string;
|
|
46
|
+
}
|
|
47
|
+
interface HistorySession {
|
|
48
|
+
sessionId: string;
|
|
49
|
+
createdAt: string;
|
|
50
|
+
updatedAt?: string;
|
|
51
|
+
/** Preview of the first message in the session */
|
|
52
|
+
preview?: string;
|
|
53
|
+
turnCount?: number;
|
|
54
|
+
}
|
|
41
55
|
|
|
42
56
|
interface PanaceaProviderProps {
|
|
43
57
|
children: React.ReactNode;
|
|
@@ -66,6 +80,8 @@ interface UseChatReturn {
|
|
|
66
80
|
escalate: (reason?: string) => Promise<void>;
|
|
67
81
|
/** Post a thumbs-up/down reaction for a turn by index */
|
|
68
82
|
react: (turnIndex: number, reaction: "helpful" | "unhelpful") => Promise<void>;
|
|
83
|
+
/** Push page context to the active session (URL, title, content summary, etc.) */
|
|
84
|
+
setPageContext: (ctx: Record<string, unknown>) => Promise<void>;
|
|
69
85
|
/** Clear all turns and start fresh */
|
|
70
86
|
reset: () => void;
|
|
71
87
|
}
|
|
@@ -93,6 +109,18 @@ interface UseWidgetReturn {
|
|
|
93
109
|
}
|
|
94
110
|
declare function useWidget(): UseWidgetReturn;
|
|
95
111
|
|
|
112
|
+
interface UseSessionHistoryReturn {
|
|
113
|
+
/** List of past sessions for this customer */
|
|
114
|
+
sessions: HistorySession[];
|
|
115
|
+
/** True while fetching the session list */
|
|
116
|
+
loading: boolean;
|
|
117
|
+
/** Load and display a specific past session's turns in the active chat */
|
|
118
|
+
loadSession: (sessionId: string) => Promise<void>;
|
|
119
|
+
/** Refetch the session list */
|
|
120
|
+
refresh: () => Promise<void>;
|
|
121
|
+
}
|
|
122
|
+
declare function useSessionHistory(): UseSessionHistoryReturn;
|
|
123
|
+
|
|
96
124
|
declare function PanaceaFAB({ className }: {
|
|
97
125
|
className?: string;
|
|
98
126
|
}): React.JSX.Element;
|
|
@@ -109,4 +137,16 @@ declare function PanaceaChat({ title, placeholder, className, }: {
|
|
|
109
137
|
className?: string;
|
|
110
138
|
}): React.JSX.Element;
|
|
111
139
|
|
|
112
|
-
|
|
140
|
+
interface PanaceaHistoryProps {
|
|
141
|
+
/** Called when the user clicks a session row and it loads into chat */
|
|
142
|
+
onSessionLoaded?: (sessionId: string) => void;
|
|
143
|
+
className?: string;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Renders a scrollable list of past sessions for the authenticated customer.
|
|
147
|
+
* Requires `customerToken` to be set on the parent `<PanaceaProvider>`.
|
|
148
|
+
* Clicking a row loads the turns into the active chat.
|
|
149
|
+
*/
|
|
150
|
+
declare function PanaceaHistory({ onSessionLoaded, className }: PanaceaHistoryProps): React.JSX.Element;
|
|
151
|
+
|
|
152
|
+
export { type HistorySession, type HistoryTurn, type LiveMessage, PanaceaChat, type PanaceaConfig, PanaceaFAB, PanaceaHistory, PanaceaInput, PanaceaMessages, PanaceaProvider, type Source, type Turn, useChat, useLiveSession, useSessionHistory, useWidget };
|