open-ask-ai 0.2.3 → 0.2.4

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.esm.js CHANGED
@@ -1,11 +1,13 @@
1
1
  import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
2
2
  import * as React from 'react';
3
- import { useState, useCallback, useEffect } from 'react';
3
+ import { useState, useCallback, useEffect, useRef } from 'react';
4
4
  import * as DialogPrimitive from '@radix-ui/react-dialog';
5
- import { Sparkles, X } from 'lucide-react';
5
+ import { Sparkles, RotateCcw, X, ArrowUp } from 'lucide-react';
6
6
  import { Slot } from '@radix-ui/react-slot';
7
7
  import ReactMarkdown from 'react-markdown';
8
8
  import remarkGfm from 'remark-gfm';
9
+ import rehypeHighlight from 'rehype-highlight';
10
+ import { common } from 'lowlight';
9
11
 
10
12
  /**
11
13
  * API client for Ask AI widget
@@ -50,7 +52,7 @@ class APIClient {
50
52
  /**
51
53
  * Ask a question and return the SSE stream response
52
54
  */
53
- async askQuestion(sessionId, question, system) {
55
+ async askQuestion(sessionId, question, system, signal) {
54
56
  const response = await fetch(`${this.baseUrl}/api/ask`, {
55
57
  method: 'POST',
56
58
  headers: {
@@ -61,6 +63,7 @@ class APIClient {
61
63
  question,
62
64
  system,
63
65
  }),
66
+ signal,
64
67
  });
65
68
  if (!response.ok) {
66
69
  throw new Error(`Failed to ask question: ${response.statusText}`);
@@ -91,17 +94,10 @@ function useSession({ apiClient }) {
91
94
  setIsCreating(false);
92
95
  }
93
96
  }, [apiClient]);
94
- const recreateSession = useCallback(async () => {
95
- // Clean up old session if it exists
96
- if (sessionId) {
97
- try {
98
- await apiClient.deleteSession(sessionId);
99
- }
100
- catch (err) {
101
- console.error('Failed to delete old session:', err);
102
- }
103
- }
104
- await initializeSession();
97
+ const clearSession = useCallback(async () => {
98
+ setSessionId(null);
99
+ setIsCreating(false);
100
+ setError(null);
105
101
  }, [sessionId, apiClient, initializeSession]);
106
102
  // Cleanup on unmount
107
103
  useEffect(() => {
@@ -118,7 +114,7 @@ function useSession({ apiClient }) {
118
114
  isCreating,
119
115
  error,
120
116
  initializeSession,
121
- recreateSession,
117
+ clearSession,
122
118
  };
123
119
  }
124
120
 
@@ -201,6 +197,10 @@ function useSSE(options = {}) {
201
197
  }
202
198
  }
203
199
  catch (err) {
200
+ // Don't report AbortError as it's an intentional cancellation
201
+ if (err instanceof Error && err.name === 'AbortError') {
202
+ return;
203
+ }
204
204
  const error = err instanceof Error ? err : new Error('SSE stream error');
205
205
  onError?.(error);
206
206
  }
@@ -217,7 +217,8 @@ function useChat({ apiClient, systemPrompt }) {
217
217
  const [messages, setMessages] = useState([]);
218
218
  const [isStreaming, setIsStreaming] = useState(false);
219
219
  const [error, setError] = useState(null);
220
- const { sessionId, isCreating: isCreatingSession, error: sessionError, initializeSession } = useSession({ apiClient });
220
+ const abortControllerRef = useRef(null);
221
+ const { sessionId, isCreating: isCreatingSession, error: sessionError, initializeSession, clearSession } = useSession({ apiClient });
221
222
  const { handleSSEStream } = useSSE({
222
223
  onConnected: () => {
223
224
  console.log('SSE connected');
@@ -269,12 +270,36 @@ function useChat({ apiClient, systemPrompt }) {
269
270
  if (!text.trim()) {
270
271
  return;
271
272
  }
273
+ // Abort any ongoing request
274
+ if (abortControllerRef.current) {
275
+ abortControllerRef.current.abort();
276
+ }
277
+ // Create new AbortController for this request
278
+ const abortController = new AbortController();
279
+ abortControllerRef.current = abortController;
272
280
  setError(null);
281
+ setIsStreaming(true);
282
+ // Add user message immediately (optimistic UI)
283
+ const userMessage = {
284
+ id: crypto.randomUUID(),
285
+ role: 'user',
286
+ content: text,
287
+ timestamp: Date.now(),
288
+ };
289
+ setMessages((prev) => [...prev, userMessage]);
290
+ // Add placeholder for assistant response
291
+ const assistantMessage = {
292
+ id: crypto.randomUUID(),
293
+ role: 'assistant',
294
+ content: '',
295
+ timestamp: Date.now(),
296
+ isStreaming: true,
297
+ };
298
+ setMessages((prev) => [...prev, assistantMessage]);
273
299
  // Create session if it doesn't exist yet
274
300
  let currentSessionId = sessionId;
275
301
  if (!currentSessionId && !isCreatingSession) {
276
302
  try {
277
- setIsStreaming(true);
278
303
  currentSessionId = await initializeSession();
279
304
  }
280
305
  catch (err) {
@@ -292,40 +317,39 @@ function useChat({ apiClient, systemPrompt }) {
292
317
  setError(new Error('No active session'));
293
318
  return;
294
319
  }
295
- setIsStreaming(true);
296
- // Add user message immediately (optimistic UI)
297
- const userMessage = {
298
- id: crypto.randomUUID(),
299
- role: 'user',
300
- content: text,
301
- timestamp: Date.now(),
302
- };
303
- setMessages((prev) => [...prev, userMessage]);
304
- // Add placeholder for assistant response
305
- const assistantMessage = {
306
- id: crypto.randomUUID(),
307
- role: 'assistant',
308
- content: '',
309
- timestamp: Date.now(),
310
- isStreaming: true,
311
- };
312
- setMessages((prev) => [...prev, assistantMessage]);
313
320
  try {
314
- const response = await apiClient.askQuestion(currentSessionId, text, systemPrompt);
321
+ const response = await apiClient.askQuestion(currentSessionId, text, systemPrompt, abortController.signal);
315
322
  await handleSSEStream(response);
316
323
  }
317
324
  catch (err) {
325
+ // Don't treat aborted requests as errors
326
+ if (err instanceof Error && err.name === 'AbortError') {
327
+ return;
328
+ }
318
329
  const error = err instanceof Error ? err : new Error('Failed to send message');
319
330
  setError(error);
320
331
  setIsStreaming(false);
321
332
  // Remove the failed messages
322
333
  setMessages((prev) => prev.slice(0, -2));
323
334
  }
335
+ finally {
336
+ // Clear the abort controller reference if this was the current one
337
+ if (abortControllerRef.current === abortController) {
338
+ abortControllerRef.current = null;
339
+ }
340
+ }
324
341
  }, [sessionId, isCreatingSession, initializeSession, apiClient, handleSSEStream, systemPrompt]);
325
- const clearMessages = useCallback(() => {
342
+ const resetChat = useCallback(async () => {
343
+ // Abort any ongoing request
344
+ if (abortControllerRef.current) {
345
+ abortControllerRef.current.abort();
346
+ abortControllerRef.current = null;
347
+ }
326
348
  setMessages([]);
327
349
  setError(null);
328
- }, []);
350
+ setIsStreaming(false);
351
+ await clearSession();
352
+ }, [clearSession]);
329
353
  return {
330
354
  messages,
331
355
  isStreaming,
@@ -333,7 +357,7 @@ function useChat({ apiClient, systemPrompt }) {
333
357
  sessionError,
334
358
  isCreatingSession,
335
359
  sendMessage,
336
- clearMessages,
360
+ resetChat,
337
361
  };
338
362
  }
339
363
 
@@ -382,11 +406,11 @@ const Button = React.forwardRef(({ className, variant = 'default', size = 'defau
382
406
  });
383
407
  Button.displayName = 'Button';
384
408
 
385
- var css_248z$4 = "/* Drawer 组件样式 */\n\n.overlay_M4Ctc {\n position: fixed;\n inset: 0;\n z-index: 50;\n background-color: rgba(0, 0, 0, 0.5);\n}\n\n.overlay_M4Ctc[data-state=\"open\"] {\n animation: fadeIn_ofwCi 300ms ease-in-out;\n}\n\n.overlay_M4Ctc[data-state=\"closed\"] {\n animation: fadeOut_gGY8D 300ms ease-in-out;\n}\n\n.content_CFbqn {\n position: fixed;\n z-index: 50;\n display: flex;\n flex-direction: column;\n background-color: var(--ask-ai-background);\n box-shadow: var(--ask-ai-shadow);\n top: 16px;\n bottom: 16px;\n height: calc(100% - 32px);\n border-radius: var(--ask-ai-radius-lg);\n transition: transform 300ms cubic-bezier(0.4, 0, 0.2, 1);\n width: 100%;\n max-width: calc(100% - 32px);\n overflow: hidden;\n}\n\n@media (min-width: 632px) {\n .content_CFbqn {\n max-width: 600px;\n }\n}\n\n.content_CFbqn[data-state=\"open\"].position-right_jfDyQ {\n animation: slideInFromRight_1XcEG 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"closed\"].position-right_jfDyQ {\n animation: slideOutToRight_R3Q0I 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"open\"].position-left_GhNmP {\n animation: slideInFromLeft_g-jJq 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"closed\"].position-left_GhNmP {\n animation: slideOutToLeft_K488H 300ms ease-in-out;\n}\n\n.position-right_jfDyQ {\n right: 16px;\n}\n\n.position-left_GhNmP {\n left: 16px;\n}\n\n.header_m6xZ6 {\n display: flex;\n align-items: center;\n justify-content: space-between;\n border-bottom: 1px solid var(--ask-ai-border);\n padding: 8px 16px;\n}\n\n.title_jiPst {\n font-size: 18px;\n font-weight: 600;\n color: var(--ask-ai-foreground);\n margin: 0;\n display: inline-flex;\n gap: 8px;\n align-items: center;\n}\n\n.icon_VOWAV {\n width: 20px;\n height: 20px;\n color: var(--ask-ai-primary);\n}\n\n.closeButton_HTzsf {\n border-radius: 4px;\n opacity: 0.7;\n}\n\n.closeButton_HTzsf:hover {\n opacity: 1;\n}\n\n.closeIcon_QvQtm {\n height: 20px;\n width: 20px;\n}\n\n.body_EPX-M {\n flex: 1;\n overflow: hidden;\n}\n\n/* Animations */\n@keyframes fadeIn_ofwCi {\n from {\n opacity: 0;\n }\n to {\n opacity: 1;\n }\n}\n\n@keyframes fadeOut_gGY8D {\n from {\n opacity: 1;\n }\n to {\n opacity: 0;\n }\n}\n\n@keyframes slideInFromRight_1XcEG {\n from {\n transform: translateX(100%);\n }\n to {\n transform: translateX(0);\n }\n}\n\n@keyframes slideOutToRight_R3Q0I {\n from {\n transform: translateX(0);\n }\n to {\n transform: translateX(100%);\n }\n}\n\n@keyframes slideInFromLeft_g-jJq {\n from {\n transform: translateX(-100%);\n }\n to {\n transform: translateX(0);\n }\n}\n\n@keyframes slideOutToLeft_K488H {\n from {\n transform: translateX(0);\n }\n to {\n transform: translateX(-100%);\n }\n}\n";
386
- var styles$2 = {"overlay":"overlay_M4Ctc","fadeIn":"fadeIn_ofwCi","fadeOut":"fadeOut_gGY8D","content":"content_CFbqn","position-right":"position-right_jfDyQ","slideInFromRight":"slideInFromRight_1XcEG","slideOutToRight":"slideOutToRight_R3Q0I","position-left":"position-left_GhNmP","slideInFromLeft":"slideInFromLeft_g-jJq","slideOutToLeft":"slideOutToLeft_K488H","header":"header_m6xZ6","title":"title_jiPst","icon":"icon_VOWAV","closeButton":"closeButton_HTzsf","closeIcon":"closeIcon_QvQtm","body":"body_EPX-M"};
409
+ var css_248z$4 = "/* Drawer 组件样式 */\n\n.overlay_M4Ctc {\n position: fixed;\n inset: 0;\n z-index: 50;\n background-color: rgba(0, 0, 0, 0.5);\n}\n\n.overlay_M4Ctc[data-state=\"open\"] {\n animation: fadeIn_ofwCi 300ms ease-in-out;\n}\n\n.overlay_M4Ctc[data-state=\"closed\"] {\n animation: fadeOut_gGY8D 300ms ease-in-out;\n}\n\n.content_CFbqn {\n position: fixed;\n z-index: 50;\n display: flex;\n flex-direction: column;\n background-color: var(--ask-ai-background);\n box-shadow: var(--ask-ai-shadow);\n top: 16px;\n bottom: 16px;\n height: calc(100% - 32px);\n border-radius: var(--ask-ai-radius-lg);\n transition: transform 300ms cubic-bezier(0.4, 0, 0.2, 1);\n width: 100%;\n max-width: calc(100% - 32px);\n overflow: hidden;\n}\n\n@media (min-width: 632px) {\n .content_CFbqn {\n max-width: 600px;\n }\n}\n\n.content_CFbqn[data-state=\"open\"].position-right_jfDyQ {\n animation: slideInFromRight_1XcEG 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"closed\"].position-right_jfDyQ {\n animation: slideOutToRight_R3Q0I 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"open\"].position-left_GhNmP {\n animation: slideInFromLeft_g-jJq 300ms ease-in-out;\n}\n\n.content_CFbqn[data-state=\"closed\"].position-left_GhNmP {\n animation: slideOutToLeft_K488H 300ms ease-in-out;\n}\n\n.position-right_jfDyQ {\n right: 16px;\n}\n\n.position-left_GhNmP {\n left: 16px;\n}\n\n.header_m6xZ6 {\n display: flex;\n align-items: center;\n justify-content: space-between;\n border-bottom: 1px solid var(--ask-ai-border);\n padding: 8px 16px;\n}\n\n.headerActions_p1WQU {\n display: flex;\n align-items: center;\n gap: 4px;\n}\n\n.title_jiPst {\n font-size: 18px;\n font-weight: 600;\n color: var(--ask-ai-foreground);\n margin: 0;\n display: inline-flex;\n gap: 8px;\n align-items: center;\n}\n\n.icon_VOWAV {\n width: 20px;\n height: 20px;\n color: var(--ask-ai-primary);\n}\n\n.closeButton_HTzsf {\n border-radius: 4px;\n opacity: 0.7;\n}\n\n.closeButton_HTzsf:hover {\n opacity: 1;\n}\n\n.newSessionButton_-WoCc {\n border-radius: 4px;\n opacity: 0.7;\n}\n\n.newSessionButton_-WoCc:hover {\n opacity: 1;\n}\n\n.closeIcon_QvQtm {\n height: 20px;\n width: 20px;\n}\n\n.newSessionIcon_edUZ8 {\n height: 18px;\n width: 18px;\n}\n\n.body_EPX-M {\n flex: 1;\n overflow: hidden;\n}\n\n/* Animations */\n@keyframes fadeIn_ofwCi {\n from {\n opacity: 0;\n }\n to {\n opacity: 1;\n }\n}\n\n@keyframes fadeOut_gGY8D {\n from {\n opacity: 1;\n }\n to {\n opacity: 0;\n }\n}\n\n@keyframes slideInFromRight_1XcEG {\n from {\n transform: translateX(100%);\n }\n to {\n transform: translateX(0);\n }\n}\n\n@keyframes slideOutToRight_R3Q0I {\n from {\n transform: translateX(0);\n }\n to {\n transform: translateX(100%);\n }\n}\n\n@keyframes slideInFromLeft_g-jJq {\n from {\n transform: translateX(-100%);\n }\n to {\n transform: translateX(0);\n }\n}\n\n@keyframes slideOutToLeft_K488H {\n from {\n transform: translateX(0);\n }\n to {\n transform: translateX(-100%);\n }\n}\n";
410
+ var styles$2 = {"overlay":"overlay_M4Ctc","fadeIn":"fadeIn_ofwCi","fadeOut":"fadeOut_gGY8D","content":"content_CFbqn","position-right":"position-right_jfDyQ","slideInFromRight":"slideInFromRight_1XcEG","slideOutToRight":"slideOutToRight_R3Q0I","position-left":"position-left_GhNmP","slideInFromLeft":"slideInFromLeft_g-jJq","slideOutToLeft":"slideOutToLeft_K488H","header":"header_m6xZ6","headerActions":"headerActions_p1WQU","title":"title_jiPst","icon":"icon_VOWAV","closeButton":"closeButton_HTzsf","newSessionButton":"newSessionButton_-WoCc","closeIcon":"closeIcon_QvQtm","newSessionIcon":"newSessionIcon_edUZ8","body":"body_EPX-M"};
387
411
  styleInject(css_248z$4);
388
412
 
389
- function Drawer({ isOpen, onClose, position = 'right', width = 600, title = 'Ask AI', closeAriaLabel = 'Close', children, theme = 'light', }) {
413
+ function Drawer({ isOpen, onClose, onNewSession, hasMessages, position = 'right', width = 600, title = 'Ask AI', closeAriaLabel = 'Close', newSessionAriaLabel = 'New session', children, theme = 'light', }) {
390
414
  const widthStyle = typeof width === 'number' ? `${width}px` : width;
391
415
  const contentClasses = [
392
416
  styles$2.content,
@@ -406,7 +430,7 @@ function Drawer({ isOpen, onClose, position = 'right', width = 600, title = 'Ask
406
430
  document.body.removeChild(container);
407
431
  };
408
432
  }, [theme]);
409
- return (jsx(DialogPrimitive.Root, { open: isOpen, onOpenChange: (open) => !open && onClose(), children: jsxs(DialogPrimitive.Portal, { container: portalContainer, children: [jsx(DialogPrimitive.Overlay, { className: styles$2.overlay }), jsxs(DialogPrimitive.Content, { className: contentClasses, style: { maxWidth: widthStyle }, children: [jsxs("div", { className: styles$2.header, children: [jsxs(DialogPrimitive.Title, { className: styles$2.title, children: [jsx(Sparkles, { className: styles$2.icon }), title] }), jsx(DialogPrimitive.Close, { asChild: true, children: jsx(Button, { variant: "ghost", size: "icon", "aria-label": closeAriaLabel, className: styles$2.closeButton, children: jsx(X, { className: styles$2.closeIcon }) }) })] }), jsx("div", { className: styles$2.body, children: children })] })] }) }));
433
+ return (jsx(DialogPrimitive.Root, { open: isOpen, onOpenChange: (open) => !open && onClose(), children: jsxs(DialogPrimitive.Portal, { container: portalContainer, children: [jsx(DialogPrimitive.Overlay, { className: styles$2.overlay }), jsxs(DialogPrimitive.Content, { className: contentClasses, style: { maxWidth: widthStyle }, children: [jsxs("div", { className: styles$2.header, children: [jsxs(DialogPrimitive.Title, { className: styles$2.title, children: [jsx(Sparkles, { className: styles$2.icon }), title] }), jsxs("div", { className: styles$2.headerActions, children: [onNewSession && hasMessages && (jsx(Button, { variant: "ghost", size: "icon", "aria-label": newSessionAriaLabel, className: styles$2.newSessionButton, onClick: onNewSession, children: jsx(RotateCcw, { className: styles$2.newSessionIcon }) })), jsx(DialogPrimitive.Close, { asChild: true, children: jsx(Button, { variant: "ghost", size: "icon", "aria-label": closeAriaLabel, className: styles$2.closeButton, children: jsx(X, { className: styles$2.closeIcon }) }) })] })] }), jsx("div", { className: styles$2.body, children: children })] })] }) }));
410
434
  }
411
435
 
412
436
  var css_248z$3 = "/* Trigger 组件样式 */\n\n.trigger_whnWp {\n display: flex;\n flex-direction: row;\n align-items: center;\n gap: 8px;\n box-shadow: var(--ask-ai-shadow);\n}\n\n.icon_KBJUj {\n height: 16px;\n width: 16px;\n}\n";
@@ -418,8 +442,8 @@ function Trigger({ onClick, text = 'Ask AI', ariaLabel = 'Open AI assistant', cl
418
442
  return (jsxs(Button, { onClick: onClick, variant: "outline", "aria-label": ariaLabel, className: classNames, children: [jsx(Sparkles, { className: styles$1.icon }), text && jsx("span", { children: text })] }));
419
443
  }
420
444
 
421
- var css_248z$2 = "/* ChatContainer 组件样式 */\n\n.container_FRFCj {\n display: flex;\n flex-direction: column;\n height: 100%;\n}\n\n.messagesArea_Hi-49 {\n flex: 1;\n overflow-y: auto;\n padding: 16px;\n display: flex;\n flex-direction: column;\n gap: 16px;\n}\n\n/* Welcome Screen */\n.welcomeScreen_fJ8-E {\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n height: 100%;\n text-align: center;\n}\n\n.welcomeMessage_unMIK {\n font-size: 18px;\n color: var(--ask-ai-foreground);\n margin-bottom: 16px;\n}\n\n.exampleQuestionsContainer_RBWq6 {\n margin-top: 24px;\n display: flex;\n flex-direction: column;\n gap: 8px;\n}\n\n.exampleQuestionsTitle_Vurj8 {\n font-size: 14px;\n color: var(--ask-ai-muted);\n margin-bottom: 8px;\n}\n\n.exampleButton_6sOzR {\n display: block;\n width: 100%;\n text-align: left;\n padding: 8px 16px;\n font-size: 14px;\n color: var(--ask-ai-foreground);\n background-color: var(--ask-ai-button-bg);\n border: none;\n border-radius: var(--ask-ai-radius-md);\n cursor: pointer;\n transition: background-color var(--ask-ai-transition);\n}\n\n.exampleButton_6sOzR:hover {\n background-color: var(--ask-ai-border);\n}\n\n/* Messages */\n.messageWrapper_nsQr8 {\n display: flex;\n}\n\n.messageWrapper_nsQr8.user_s1Gzu {\n justify-content: flex-end;\n}\n\n.messageWrapper_nsQr8.assistant_xU0-H {\n justify-content: flex-start;\n}\n\n.message_jxpo4 {\n max-width: 85%;\n border-radius: var(--ask-ai-radius-lg);\n padding: 8px 16px;\n}\n\n.message_jxpo4.user_s1Gzu {\n background-color: var(--ask-ai-user-message-bg);\n color: var(--ask-ai-user-message-text);\n}\n\n.message_jxpo4.assistant_xU0-H {\n background-color: var(--ask-ai-ai-message-bg);\n color: var(--ask-ai-ai-message-text);\n}\n\n.messageText_neW3m {\n font-size: 14px;\n white-space: pre-wrap;\n}\n\n/* Markdown Styles */\n.markdown_mNC3q {\n font-size: 14px;\n line-height: 1.6;\n}\n\n.markdown_mNC3q > *:first-child {\n margin-top: 0;\n}\n\n.markdown_mNC3q > *:last-child {\n margin-bottom: 0;\n}\n\n/* 段落 */\n.markdown_mNC3q p {\n display: block;\n margin: 0.5em 0;\n}\n\n/* 标题 */\n.markdown_mNC3q h1,\n.markdown_mNC3q h2,\n.markdown_mNC3q h3,\n.markdown_mNC3q h4,\n.markdown_mNC3q h5,\n.markdown_mNC3q h6 {\n display: block;\n margin: 1em 0 0.5em;\n font-weight: 600;\n line-height: 1.3;\n}\n\n.markdown_mNC3q h1 {\n font-size: 1.5em;\n}\n\n.markdown_mNC3q h2 {\n font-size: 1.3em;\n}\n\n.markdown_mNC3q h3 {\n font-size: 1.1em;\n}\n\n/* 无序列表 */\n.markdown_mNC3q ul {\n display: block;\n margin: 0.5em 0;\n padding-left: 1.5em;\n list-style-type: disc;\n list-style-position: outside;\n}\n\n/* 有序列表 */\n.markdown_mNC3q ol {\n display: block;\n margin: 0.5em 0;\n padding-left: 1.5em;\n list-style-type: decimal;\n list-style-position: outside;\n}\n\n/* 嵌套列表 */\n.markdown_mNC3q ul ul {\n list-style-type: circle;\n}\n\n.markdown_mNC3q ul ul ul {\n list-style-type: square;\n}\n\n.markdown_mNC3q ol ol {\n list-style-type: lower-alpha;\n}\n\n.markdown_mNC3q ol ol ol {\n list-style-type: lower-roman;\n}\n\n/* 列表项 */\n.markdown_mNC3q li {\n display: list-item;\n margin: 0.25em 0;\n}\n\n/* 文本样式 */\n.markdown_mNC3q strong,\n.markdown_mNC3q b {\n font-weight: bold;\n}\n\n.markdown_mNC3q em,\n.markdown_mNC3q i {\n font-style: italic;\n}\n\n.markdown_mNC3q u {\n text-decoration: underline;\n}\n\n.markdown_mNC3q s,\n.markdown_mNC3q del {\n text-decoration: line-through;\n}\n\n.markdown_mNC3q code {\n font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;\n font-size: 0.9em;\n padding: 0.15em 0.4em;\n background-color: var(--ask-ai-border);\n border-radius: 4px;\n}\n\n.markdown_mNC3q pre {\n margin: 0.75em 0;\n padding: 0.75em 1em;\n background-color: var(--ask-ai-border);\n border-radius: var(--ask-ai-radius-md);\n overflow-x: auto;\n}\n\n.markdown_mNC3q pre code {\n padding: 0;\n background-color: transparent;\n border-radius: 0;\n font-size: 0.85em;\n}\n\n.markdown_mNC3q blockquote {\n margin: 0.5em 0;\n padding: 0.5em 1em;\n border-left: 3px solid var(--ask-ai-primary);\n background-color: var(--ask-ai-border);\n border-radius: 0 var(--ask-ai-radius-sm) var(--ask-ai-radius-sm) 0;\n}\n\n.markdown_mNC3q blockquote p {\n margin: 0;\n}\n\n.markdown_mNC3q a {\n color: var(--ask-ai-primary);\n text-decoration: none;\n}\n\n.markdown_mNC3q a:hover {\n text-decoration: underline;\n}\n\n.markdown_mNC3q table {\n width: 100%;\n margin: 0.75em 0;\n border-collapse: collapse;\n font-size: 0.9em;\n}\n\n.markdown_mNC3q th,\n.markdown_mNC3q td {\n padding: 0.5em 0.75em;\n border: 1px solid var(--ask-ai-border);\n text-align: left;\n}\n\n.markdown_mNC3q th {\n background-color: var(--ask-ai-border);\n font-weight: 600;\n}\n\n.markdown_mNC3q hr {\n margin: 1em 0;\n border: none;\n border-top: 1px solid var(--ask-ai-border);\n}\n\n.markdown_mNC3q img {\n max-width: 100%;\n height: auto;\n border-radius: var(--ask-ai-radius-md);\n}\n\n.cursor_9Dhwg {\n display: inline-block;\n width: 2px;\n height: 16px;\n margin-left: 4px;\n background-color: currentColor;\n animation: pulse_STM-e 1s cubic-bezier(0.4, 0, 0.6, 1) infinite;\n}\n\n@keyframes pulse_STM-e {\n 0%, 100% {\n opacity: 1;\n }\n 50% {\n opacity: 0.5;\n }\n}\n\n/* Error */\n.error_uFX9a {\n padding: 16px;\n background-color: rgba(220, 38, 38, 0.1);\n color: var(--ask-ai-error);\n border-radius: var(--ask-ai-radius-md);\n font-size: 14px;\n}\n\n/* Input Area */\n.inputForm_nC0l- {\n border-top: 1px solid var(--ask-ai-border);\n padding: 16px;\n margin: 0;\n}\n\n.inputWrapper_zGeKy {\n display: flex;\n gap: 8px;\n align-items: flex-end;\n}\n\n.input_7lMOc {\n flex: 1;\n padding: 8px 16px;\n font-size: 14px;\n background-color: var(--ask-ai-background);\n border: 1px solid var(--ask-ai-border);\n border-radius: var(--ask-ai-radius-md);\n color: var(--ask-ai-foreground);\n font-family: inherit;\n resize: none;\n min-height: 40px;\n max-height: 200px;\n line-height: 1.5;\n overflow-y: auto;\n}\n\n.input_7lMOc:focus {\n outline: none;\n box-shadow: 0 0 0 2px var(--ask-ai-primary);\n}\n\n.input_7lMOc:disabled {\n opacity: 0.5;\n}\n\n.submitButton_2XrPY {\n height: 40px;\n padding: 8px 16px;\n font-size: 14px;\n background-color: var(--ask-ai-primary);\n color: #ffffff;\n border: none;\n border-radius: var(--ask-ai-radius-md);\n cursor: pointer;\n transition: all var(--ask-ai-transition);\n}\n\n.submitButton_2XrPY:hover:not(:disabled) {\n background-color: var(--ask-ai-primary-hover);\n}\n\n.submitButton_2XrPY:disabled {\n opacity: 0.5;\n cursor: not-allowed;\n}\n";
422
- var styles = {"container":"container_FRFCj","messagesArea":"messagesArea_Hi-49","welcomeScreen":"welcomeScreen_fJ8-E","welcomeMessage":"welcomeMessage_unMIK","exampleQuestionsContainer":"exampleQuestionsContainer_RBWq6","exampleQuestionsTitle":"exampleQuestionsTitle_Vurj8","exampleButton":"exampleButton_6sOzR","messageWrapper":"messageWrapper_nsQr8","user":"user_s1Gzu","assistant":"assistant_xU0-H","message":"message_jxpo4","messageText":"messageText_neW3m","markdown":"markdown_mNC3q","cursor":"cursor_9Dhwg","pulse":"pulse_STM-e","error":"error_uFX9a","inputForm":"inputForm_nC0l-","inputWrapper":"inputWrapper_zGeKy","input":"input_7lMOc","submitButton":"submitButton_2XrPY"};
445
+ var css_248z$2 = "/* ChatContainer 组件样式 */\n\n.container_FRFCj {\n display: flex;\n flex-direction: column;\n height: 100%;\n}\n\n.messagesArea_Hi-49 {\n flex: 1;\n overflow-y: auto;\n padding: 16px;\n display: flex;\n flex-direction: column;\n gap: 16px;\n}\n\n/* Welcome Screen */\n.welcomeScreen_fJ8-E {\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n height: 100%;\n text-align: center;\n}\n\n.welcomeMessage_unMIK {\n font-size: 18px;\n color: var(--ask-ai-foreground);\n margin-bottom: 16px;\n}\n\n.exampleQuestionsContainer_RBWq6 {\n margin-top: 24px;\n display: flex;\n flex-direction: column;\n gap: 8px;\n}\n\n.exampleQuestionsTitle_Vurj8 {\n font-size: 14px;\n color: var(--ask-ai-muted);\n margin-bottom: 8px;\n}\n\n.exampleButton_6sOzR {\n display: block;\n width: 100%;\n text-align: left;\n padding: 8px 16px;\n font-size: 14px;\n color: var(--ask-ai-foreground);\n background-color: var(--ask-ai-button-bg);\n border: none;\n border-radius: var(--ask-ai-radius-md);\n cursor: pointer;\n transition: background-color var(--ask-ai-transition);\n}\n\n.exampleButton_6sOzR:hover {\n background-color: var(--ask-ai-border);\n}\n\n/* Messages */\n.messageWrapper_nsQr8 {\n display: flex;\n}\n\n.messageWrapper_nsQr8.user_s1Gzu {\n justify-content: flex-end;\n}\n\n.messageWrapper_nsQr8.assistant_xU0-H {\n justify-content: flex-start;\n}\n\n.message_jxpo4 {\n max-width: 85%;\n border-radius: var(--ask-ai-radius-lg);\n padding: 8px 16px;\n}\n\n.message_jxpo4.user_s1Gzu {\n background-color: var(--ask-ai-user-message-bg);\n color: var(--ask-ai-user-message-text);\n}\n\n.message_jxpo4.assistant_xU0-H {\n background-color: var(--ask-ai-ai-message-bg);\n color: var(--ask-ai-ai-message-text);\n}\n\n.messageText_neW3m {\n font-size: 14px;\n white-space: pre-wrap;\n}\n\n/* Markdown Styles */\n.markdown_mNC3q {\n font-size: 14px;\n line-height: 1.6;\n}\n\n.markdown_mNC3q > *:first-child {\n margin-top: 0;\n}\n\n.markdown_mNC3q > *:last-child {\n margin-bottom: 0;\n}\n\n/* 段落 */\n.markdown_mNC3q p {\n display: block;\n margin: 0.5em 0;\n}\n\n/* 标题 */\n.markdown_mNC3q h1,\n.markdown_mNC3q h2,\n.markdown_mNC3q h3,\n.markdown_mNC3q h4,\n.markdown_mNC3q h5,\n.markdown_mNC3q h6 {\n display: block;\n margin: 1em 0 0.5em;\n font-weight: 600;\n line-height: 1.3;\n}\n\n.markdown_mNC3q h1 {\n font-size: 1.5em;\n}\n\n.markdown_mNC3q h2 {\n font-size: 1.3em;\n}\n\n.markdown_mNC3q h3 {\n font-size: 1.1em;\n}\n\n/* 无序列表 */\n.markdown_mNC3q ul {\n display: block;\n margin: 0.5em 0;\n padding-left: 1.5em;\n list-style-type: disc;\n list-style-position: outside;\n}\n\n/* 有序列表 */\n.markdown_mNC3q ol {\n display: block;\n margin: 0.5em 0;\n padding-left: 1.5em;\n list-style-type: decimal;\n list-style-position: outside;\n}\n\n/* 嵌套列表 */\n.markdown_mNC3q ul ul {\n list-style-type: circle;\n}\n\n.markdown_mNC3q ul ul ul {\n list-style-type: square;\n}\n\n.markdown_mNC3q ol ol {\n list-style-type: lower-alpha;\n}\n\n.markdown_mNC3q ol ol ol {\n list-style-type: lower-roman;\n}\n\n/* 列表项 */\n.markdown_mNC3q li {\n display: list-item;\n margin: 0.25em 0;\n}\n\n/* 文本样式 */\n.markdown_mNC3q strong,\n.markdown_mNC3q b {\n font-weight: bold;\n}\n\n.markdown_mNC3q em,\n.markdown_mNC3q i {\n font-style: italic;\n}\n\n.markdown_mNC3q u {\n text-decoration: underline;\n}\n\n.markdown_mNC3q s,\n.markdown_mNC3q del {\n text-decoration: line-through;\n}\n\n.markdown_mNC3q code {\n font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;\n font-size: 0.9em;\n padding: 0.15em 0.4em;\n background-color: var(--ask-ai-border);\n border-radius: 4px;\n}\n\n/* 代码块容器 */\n.markdown_mNC3q pre {\n margin: 0.75em 0;\n padding: 0.75em 1em;\n background-color: var(--ask-ai-code-bg);\n border-radius: var(--ask-ai-radius-md);\n overflow-x: auto;\n line-height: 1.5;\n}\n\n.markdown_mNC3q pre code {\n padding: 0;\n background-color: transparent;\n border-radius: 0;\n font-size: 0.85em;\n color: var(--ask-ai-code-text);\n font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;\n}\n\n/* Highlight.js 语法高亮样式 */\n.markdown_mNC3q .hljs-keyword,\n.markdown_mNC3q .hljs-selector-tag,\n.markdown_mNC3q .hljs-literal,\n.markdown_mNC3q .hljs-section,\n.markdown_mNC3q .hljs-link {\n color: var(--ask-ai-code-keyword);\n font-weight: bold;\n}\n\n.markdown_mNC3q .hljs-string,\n.markdown_mNC3q .hljs-regexp,\n.markdown_mNC3q .hljs-addition,\n.markdown_mNC3q .hljs-attribute,\n.markdown_mNC3q .hljs-meta-string {\n color: var(--ask-ai-code-string);\n}\n\n.markdown_mNC3q .hljs-number,\n.markdown_mNC3q .hljs-meta .hljs-keyword {\n color: var(--ask-ai-code-number);\n}\n\n.markdown_mNC3q .hljs-comment,\n.markdown_mNC3q .hljs-quote {\n color: var(--ask-ai-code-comment);\n font-style: italic;\n}\n\n.markdown_mNC3q .hljs-function,\n.markdown_mNC3q .hljs-title {\n color: var(--ask-ai-code-function);\n}\n\n.markdown_mNC3q .hljs-operator,\n.markdown_mNC3q .hljs-punctuation {\n color: var(--ask-ai-code-operator);\n}\n\n.markdown_mNC3q .hljs-variable,\n.markdown_mNC3q .hljs-property,\n.markdown_mNC3q .hljs-params {\n color: var(--ask-ai-code-variable);\n}\n\n.markdown_mNC3q .hljs-type,\n.markdown_mNC3q .hljs-class,\n.markdown_mNC3q .hljs-built_in {\n color: var(--ask-ai-code-type);\n}\n\n.markdown_mNC3q .hljs-builtin,\n.markdown_mNC3q .hljs-name {\n color: var(--ask-ai-code-builtin);\n}\n\n.markdown_mNC3q .hljs-tag {\n color: var(--ask-ai-code-tag);\n}\n\n.markdown_mNC3q .hljs-attr {\n color: var(--ask-ai-code-attr);\n}\n\n.markdown_mNC3q .hljs-deletion {\n color: var(--ask-ai-error);\n}\n\n.markdown_mNC3q .hljs-emphasis {\n font-style: italic;\n}\n\n.markdown_mNC3q .hljs-strong {\n font-weight: bold;\n}\n\n.markdown_mNC3q blockquote {\n margin: 0.5em 0;\n padding: 0.5em 1em;\n border-left: 3px solid var(--ask-ai-primary);\n background-color: var(--ask-ai-border);\n border-radius: 0 var(--ask-ai-radius-sm) var(--ask-ai-radius-sm) 0;\n}\n\n.markdown_mNC3q blockquote p {\n margin: 0;\n}\n\n.markdown_mNC3q a {\n color: var(--ask-ai-primary);\n text-decoration: none;\n}\n\n.markdown_mNC3q a:hover {\n text-decoration: underline;\n}\n\n.markdown_mNC3q table {\n width: 100%;\n margin: 0.75em 0;\n border-collapse: collapse;\n font-size: 0.9em;\n}\n\n.markdown_mNC3q th,\n.markdown_mNC3q td {\n padding: 0.5em 0.75em;\n border: 1px solid var(--ask-ai-border);\n text-align: left;\n}\n\n.markdown_mNC3q th {\n background-color: var(--ask-ai-border);\n font-weight: 600;\n}\n\n.markdown_mNC3q hr {\n margin: 1em 0;\n border: none;\n border-top: 1px solid var(--ask-ai-border);\n}\n\n.markdown_mNC3q img {\n max-width: 100%;\n height: auto;\n border-radius: var(--ask-ai-radius-md);\n}\n\n.cursor_9Dhwg {\n display: inline-block;\n width: 6px;\n height: 16px;\n background-color: currentColor;\n animation: blink_-4y-x 1s step-end infinite;\n}\n\n@keyframes blink_-4y-x {\n 0%, 50% {\n opacity: 0.75;\n }\n 50.01%, 100% {\n opacity: 0;\n }\n}\n\n/* Error */\n.error_uFX9a {\n padding: 16px;\n background-color: rgba(220, 38, 38, 0.1);\n color: var(--ask-ai-error);\n border-radius: var(--ask-ai-radius-md);\n font-size: 14px;\n}\n\n/* Input Area */\n.inputForm_nC0l- {\n border-top: 1px solid var(--ask-ai-border);\n padding: 16px;\n margin: 0;\n}\n\n.inputWrapper_zGeKy {\n display: flex;\n gap: 8px;\n align-items: flex-end;\n}\n\n.input_7lMOc {\n flex: 1;\n padding: 8px 16px;\n font-size: 14px;\n background-color: var(--ask-ai-background);\n border: 1px solid var(--ask-ai-border);\n border-radius: var(--ask-ai-radius-md);\n color: var(--ask-ai-foreground);\n font-family: inherit;\n resize: none;\n min-height: 40px;\n max-height: 200px;\n line-height: 1.5;\n overflow-y: auto;\n}\n\n.input_7lMOc:focus {\n outline: none;\n box-shadow: 0 0 0 2px var(--ask-ai-primary);\n}\n\n.input_7lMOc:disabled {\n opacity: 0.5;\n}\n\n.submitButton_2XrPY {\n height: 40px;\n padding: 8px;\n font-size: 14px;\n background-color: var(--ask-ai-primary);\n color: #ffffff;\n border: none;\n border-radius: var(--ask-ai-radius-md);\n cursor: pointer;\n transition: all var(--ask-ai-transition);\n}\n\n.submitButton_2XrPY:hover:not(:disabled) {\n background-color: var(--ask-ai-primary-hover);\n}\n\n.submitButton_2XrPY:disabled {\n opacity: 0.5;\n cursor: not-allowed;\n}\n";
446
+ var styles = {"container":"container_FRFCj","messagesArea":"messagesArea_Hi-49","welcomeScreen":"welcomeScreen_fJ8-E","welcomeMessage":"welcomeMessage_unMIK","exampleQuestionsContainer":"exampleQuestionsContainer_RBWq6","exampleQuestionsTitle":"exampleQuestionsTitle_Vurj8","exampleButton":"exampleButton_6sOzR","messageWrapper":"messageWrapper_nsQr8","user":"user_s1Gzu","assistant":"assistant_xU0-H","message":"message_jxpo4","messageText":"messageText_neW3m","markdown":"markdown_mNC3q","cursor":"cursor_9Dhwg","blink":"blink_-4y-x","error":"error_uFX9a","inputForm":"inputForm_nC0l-","inputWrapper":"inputWrapper_zGeKy","input":"input_7lMOc","submitButton":"submitButton_2XrPY"};
423
447
  styleInject(css_248z$2);
424
448
 
425
449
  function ChatContainer({ texts, exampleQuestions, onMessage, onError, messages, isStreaming, error, sendMessage, input, setInput, }) {
@@ -475,12 +499,13 @@ function ChatContainer({ texts, exampleQuestions, onMessage, onError, messages,
475
499
  }, [messages, onMessage]);
476
500
  const handleSubmit = async (e) => {
477
501
  e.preventDefault();
478
- if (!input.trim() || isStreaming)
502
+ const inputValue = input.trim();
503
+ if (!inputValue || isStreaming)
479
504
  return;
480
505
  // Enable auto-scroll for new user message
481
506
  shouldAutoScrollRef.current = true;
482
- await sendMessage(input);
483
507
  setInput('');
508
+ await sendMessage(inputValue);
484
509
  // Reset textarea height after submission
485
510
  setTimeout(() => {
486
511
  if (textareaRef.current) {
@@ -516,7 +541,10 @@ function ChatContainer({ texts, exampleQuestions, onMessage, onError, messages,
516
541
  // Welcome Screen
517
542
  jsxs("div", { className: styles.welcomeScreen, children: [jsx("p", { className: styles.welcomeMessage, children: welcomeMessage }), exampleQuestions && exampleQuestions.length > 0 && (jsxs("div", { className: styles.exampleQuestionsContainer, children: [jsx("p", { className: styles.exampleQuestionsTitle, children: exampleQuestionsTitle }), exampleQuestions.map((question, index) => (jsx("button", { onClick: () => handleExampleClick(question), className: styles.exampleButton, children: question }, index)))] }))] })) : (
518
543
  // Messages
519
- jsx(Fragment, { children: messages.map((message) => (jsx("div", { className: `${styles.messageWrapper} ${styles[message.role]}`, children: jsx("div", { className: `${styles.message} ${styles[message.role]}`, children: message.role === 'assistant' ? (jsxs("div", { className: styles.markdown, children: [jsx(ReactMarkdown, { remarkPlugins: [remarkGfm], children: message.content }), message.isStreaming && (jsx("span", { className: styles.cursor }))] })) : (jsx("div", { className: styles.messageText, children: message.content })) }) }, message.id))) })), error && (jsx("div", { className: styles.error, children: error.message }))] }), jsx("form", { onSubmit: handleSubmit, className: styles.inputForm, children: jsxs("div", { className: styles.inputWrapper, children: [jsx("textarea", { ref: textareaRef, value: input, onChange: handleInputChange, onKeyDown: handleKeyDown, placeholder: inputPlaceholder, disabled: isStreaming, className: styles.input, rows: 1 }), jsx("button", { type: "submit", disabled: !input.trim() || isStreaming, className: styles.submitButton, children: isStreaming ? 'Sending...' : 'Send' })] }) })] }));
544
+ jsx(Fragment, { children: messages.map((message) => (jsx("div", { className: `${styles.messageWrapper} ${styles[message.role]}`, children: jsx("div", { className: `${styles.message} ${styles[message.role]}`, children: message.role === 'assistant' ? (jsxs("div", { className: styles.markdown, children: [jsx(ReactMarkdown, { remarkPlugins: [remarkGfm], rehypePlugins: [[rehypeHighlight, {
545
+ languages: common,
546
+ prefix: 'hljs-'
547
+ }]], children: message.content }), message.isStreaming && (jsx("span", { className: styles.cursor }))] })) : (jsx("div", { className: styles.messageText, children: message.content })) }) }, message.id))) })), error && (jsx("div", { className: styles.error, children: error.message }))] }), jsx("form", { onSubmit: handleSubmit, className: styles.inputForm, children: jsxs("div", { className: styles.inputWrapper, children: [jsx("textarea", { ref: textareaRef, value: input, onChange: handleInputChange, onKeyDown: handleKeyDown, placeholder: inputPlaceholder, className: styles.input, rows: 1 }), jsx("button", { type: "submit", disabled: !input.trim() || isStreaming, className: styles.submitButton, children: jsx(ArrowUp, {}) })] }) })] }));
520
548
  }
521
549
 
522
550
  function Widget(props) {
@@ -524,7 +552,7 @@ function Widget(props) {
524
552
  const [isOpen, setIsOpen] = React.useState(false);
525
553
  const [apiClient] = React.useState(() => new APIClient(apiUrl));
526
554
  // Lift chat state to Widget level to persist across drawer open/close
527
- const { messages, isStreaming, error, sendMessage } = useChat({ apiClient, systemPrompt });
555
+ const { messages, isStreaming, error, sendMessage, resetChat } = useChat({ apiClient, systemPrompt });
528
556
  // Input state also needs to persist
529
557
  const [input, setInput] = React.useState('');
530
558
  // Handle drawer open
@@ -537,6 +565,11 @@ function Widget(props) {
537
565
  setIsOpen(false);
538
566
  onClose?.();
539
567
  }, [onClose]);
568
+ // Handle new session
569
+ const handleNewSession = React.useCallback(async () => {
570
+ setInput('');
571
+ await resetChat();
572
+ }, [resetChat]);
540
573
  // Handle keyboard shortcut
541
574
  React.useEffect(() => {
542
575
  if (!enableHotkey || !hotkey)
@@ -571,10 +604,10 @@ function Widget(props) {
571
604
  // Then call our handleOpen
572
605
  handleOpen();
573
606
  },
574
- })) : (jsx(Trigger, { onClick: handleOpen, text: texts?.triggerButtonText, ariaLabel: texts?.triggerButtonAriaLabel })), jsx(Drawer, { isOpen: isOpen, onClose: handleClose, position: drawerPosition, width: drawerWidth, title: texts?.drawerTitle, closeAriaLabel: texts?.drawerCloseAriaLabel, theme: theme, children: jsx(ChatContainer, { texts: texts, exampleQuestions: exampleQuestions, onMessage: onMessage, onError: onError, messages: messages, isStreaming: isStreaming, error: error, sendMessage: sendMessage, input: input, setInput: setInput }) })] }));
607
+ })) : (jsx(Trigger, { onClick: handleOpen, text: texts?.triggerButtonText, ariaLabel: texts?.triggerButtonAriaLabel })), jsx(Drawer, { isOpen: isOpen, onClose: handleClose, onNewSession: handleNewSession, hasMessages: messages.length > 0, position: drawerPosition, width: drawerWidth, title: texts?.drawerTitle, closeAriaLabel: texts?.drawerCloseAriaLabel, newSessionAriaLabel: texts?.drawerNewSessionAriaLabel, theme: theme, children: jsx(ChatContainer, { texts: texts, exampleQuestions: exampleQuestions, onMessage: onMessage, onError: onError, messages: messages, isStreaming: isStreaming, error: error, sendMessage: sendMessage, input: input, setInput: setInput }) })] }));
575
608
  }
576
609
 
577
- var css_248z$1 = "/* CSS 变量定义 - 亮色主题(默认) */\n.ask-ai {\n /* 颜色 */\n --ask-ai-primary: #2563eb;\n --ask-ai-primary-hover: #1d4ed8;\n --ask-ai-background: #ffffff;\n --ask-ai-foreground: #0f172a;\n --ask-ai-muted: #64748b;\n --ask-ai-border: #e2e8f0;\n --ask-ai-error: #dc2626;\n --ask-ai-button-bg: #f1f5f9;\n\n /* 用户消息 */\n --ask-ai-user-message-bg: var(--ask-ai-button-bg);\n --ask-ai-user-message-text: var(--ask-ai-foreground);\n\n /* AI 消息 */\n --ask-ai-ai-message-bg: none;\n --ask-ai-ai-message-text: var(--ask-ai-foreground);\n\n /* 字体 */\n --ask-ai-font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n --ask-ai-font-size-base: 14px;\n --ask-ai-line-height: 1.5;\n\n /* 间距 */\n --ask-ai-spacing-xs: 4px;\n --ask-ai-spacing-sm: 8px;\n --ask-ai-spacing-md: 16px;\n --ask-ai-spacing-lg: 24px;\n\n /* 圆角 */\n --ask-ai-radius-sm: 4px;\n --ask-ai-radius-md: 8px;\n --ask-ai-radius-lg: 12px;\n\n /* 阴影 */\n --ask-ai-shadow: none;\n\n /* 动画 */\n --ask-ai-transition: 150ms cubic-bezier(0.4, 0, 0.2, 1);\n}\n\n/* 明确指定暗色主题 */\n.ask-ai.dark {\n --ask-ai-primary: #3b82f6;\n --ask-ai-primary-hover: #2563eb;\n --ask-ai-background: #0f172a;\n --ask-ai-foreground: #f1f5f9;\n --ask-ai-muted: #94a3b8;\n --ask-ai-border: #334155;\n --ask-ai-error: #ef4444;\n --ask-ai-button-bg: #1e293b;\n}\n";
610
+ var css_248z$1 = "/* CSS 变量定义 - 亮色主题(默认) */\n.ask-ai {\n /* 颜色 */\n --ask-ai-primary: #2563eb;\n --ask-ai-primary-hover: #1d4ed8;\n --ask-ai-background: #ffffff;\n --ask-ai-foreground: #0f172a;\n --ask-ai-muted: #64748b;\n --ask-ai-border: #e2e8f0;\n --ask-ai-error: #dc2626;\n --ask-ai-button-bg: #f1f5f9;\n\n /* 用户消息 */\n --ask-ai-user-message-bg: var(--ask-ai-button-bg);\n --ask-ai-user-message-text: var(--ask-ai-foreground);\n\n /* AI 消息 */\n --ask-ai-ai-message-bg: none;\n --ask-ai-ai-message-text: var(--ask-ai-foreground);\n\n /* 字体 */\n --ask-ai-font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n --ask-ai-font-size-base: 14px;\n --ask-ai-line-height: 1.5;\n\n /* 间距 */\n --ask-ai-spacing-xs: 4px;\n --ask-ai-spacing-sm: 8px;\n --ask-ai-spacing-md: 16px;\n --ask-ai-spacing-lg: 24px;\n\n /* 圆角 */\n --ask-ai-radius-sm: 4px;\n --ask-ai-radius-md: 8px;\n --ask-ai-radius-lg: 12px;\n\n /* 阴影 */\n --ask-ai-shadow: none;\n\n /* 动画 */\n --ask-ai-transition: 150ms cubic-bezier(0.4, 0, 0.2, 1);\n\n /* 语法高亮 - 亮色主题 (VS Code Light+) */\n --ask-ai-code-bg: #f8f8f8;\n --ask-ai-code-text: #000000;\n --ask-ai-code-keyword: #0000ff; /* 关键字:蓝色 */\n --ask-ai-code-string: #a31515; /* 字符串:红色 */\n --ask-ai-code-number: #098658; /* 数字:绿色 */\n --ask-ai-code-comment: #008000; /* 注释:绿色 */\n --ask-ai-code-function: #795e26; /* 函数:棕色 */\n --ask-ai-code-operator: #000000; /* 操作符:黑色 */\n --ask-ai-code-variable: #001080; /* 变量:深蓝 */\n --ask-ai-code-type: #267f99; /* 类型:青蓝 */\n --ask-ai-code-builtin: #0000ff; /* 内置:蓝色 */\n --ask-ai-code-tag: #800000; /* 标签:深红 */\n --ask-ai-code-attr: #ff0000; /* 属性:红色 */\n}\n\n/* 明确指定暗色主题 */\n.ask-ai.dark {\n --ask-ai-primary: #3b82f6;\n --ask-ai-primary-hover: #2563eb;\n --ask-ai-background: #0f172a;\n --ask-ai-foreground: #f1f5f9;\n --ask-ai-muted: #94a3b8;\n --ask-ai-border: #334155;\n --ask-ai-error: #ef4444;\n --ask-ai-button-bg: #1e293b;\n\n /* 语法高亮 - 暗色主题 (VS Code Dark+) */\n --ask-ai-code-bg: #1e1e1e;\n --ask-ai-code-text: #d4d4d4;\n --ask-ai-code-keyword: #569cd6; /* 关键字:蓝色 */\n --ask-ai-code-string: #ce9178; /* 字符串:米色 */\n --ask-ai-code-number: #b5cea8; /* 数字:浅绿 */\n --ask-ai-code-comment: #6a9955; /* 注释:绿色 */\n --ask-ai-code-function: #dcdcaa; /* 函数:黄色 */\n --ask-ai-code-operator: #d4d4d4; /* 操作符:浅灰 */\n --ask-ai-code-variable: #9cdcfe; /* 变量:青色 */\n --ask-ai-code-type: #4ec9b0; /* 类型:青绿 */\n --ask-ai-code-builtin: #4fc1ff; /* 内置:亮蓝 */\n --ask-ai-code-tag: #569cd6; /* 标签:蓝色 */\n --ask-ai-code-attr: #9cdcfe; /* 属性:青色 */\n}\n";
578
611
  styleInject(css_248z$1);
579
612
 
580
613
  var css_248z = "/* 全局样式 - 所有样式都在 .ask-ai 命名空间下 */\n\n/* 基础样式重置 */\n.ask-ai *,\n.ask-ai *::before,\n.ask-ai *::after {\n box-sizing: border-box;\n}\n\n.ask-ai {\n font-family: var(--ask-ai-font-family);\n font-size: var(--ask-ai-font-size-base);\n line-height: var(--ask-ai-line-height);\n color: var(--ask-ai-foreground);\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n}\n\n/* 确保组件不受外部样式影响 */\n.ask-ai button {\n font-family: inherit;\n}\n\n.ask-ai input,\n.ask-ai textarea {\n font-family: inherit;\n font-size: inherit;\n}\n\n/* 滚动条样式 */\n.ask-ai ::-webkit-scrollbar {\n width: 8px;\n height: 8px;\n}\n\n.ask-ai ::-webkit-scrollbar-track {\n background: transparent;\n}\n\n.ask-ai ::-webkit-scrollbar-thumb {\n background: var(--ask-ai-border);\n border-radius: var(--ask-ai-radius-sm);\n}\n\n.ask-ai ::-webkit-scrollbar-thumb:hover {\n background: var(--ask-ai-muted);\n}\n";
@@ -1 +1 @@
1
- {"version":3,"file":"index.esm.js","sources":["../src/core/api/client.ts","../src/core/hooks/useSession.ts","../src/core/hooks/useSSE.ts","../src/core/hooks/useChat.ts","../node_modules/style-inject/dist/style-inject.es.js","../src/components/ui/Button.tsx","../src/components/widget/Drawer.tsx","../src/components/widget/Trigger.tsx","../src/components/chat/ChatContainer.tsx","../src/components/widget/Widget.tsx"],"sourcesContent":["import type { SessionResponse } from '../types/index.js';\n\n/**\n * API client for Ask AI widget\n * Supports dynamic API URL configuration\n */\nexport class APIClient {\n private baseUrl: string;\n\n constructor(baseUrl: string) {\n this.baseUrl = baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n }\n\n /**\n * Create a new session\n */\n async createSession(): Promise<SessionResponse> {\n const response = await fetch(`${this.baseUrl}/api/session`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n throw new Error(`Failed to create session: ${response.statusText}`);\n }\n\n return response.json();\n }\n\n /**\n * Delete a session\n */\n async deleteSession(sessionId: string): Promise<void> {\n const response = await fetch(`${this.baseUrl}/api/session/${sessionId}`, {\n method: 'DELETE',\n });\n\n if (!response.ok) {\n throw new Error(`Failed to delete session: ${response.statusText}`);\n }\n }\n\n /**\n * Ask a question and return the SSE stream response\n */\n async askQuestion(\n sessionId: string,\n question: string,\n system?: string\n ): Promise<Response> {\n const response = await fetch(`${this.baseUrl}/api/ask`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n sessionId,\n question,\n system,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`Failed to ask question: ${response.statusText}`);\n }\n\n return response;\n }\n}\n","import { useState, useEffect, useCallback } from 'react';\nimport type { APIClient } from '../api/client.js';\n\ninterface UseSessionOptions {\n apiClient: APIClient;\n}\n\ninterface UseSessionReturn {\n sessionId: string | null;\n isCreating: boolean;\n error: Error | null;\n initializeSession: () => Promise<string>;\n recreateSession: () => Promise<void>;\n}\n\nexport function useSession({ apiClient }: UseSessionOptions): UseSessionReturn {\n const [sessionId, setSessionId] = useState<string | null>(null);\n const [isCreating, setIsCreating] = useState(false);\n const [error, setError] = useState<Error | null>(null);\n\n const initializeSession = useCallback(async () => {\n setIsCreating(true);\n setError(null);\n\n try {\n const { sessionId: newSessionId } = await apiClient.createSession();\n setSessionId(newSessionId);\n return newSessionId;\n } catch (err) {\n const error = err instanceof Error ? err : new Error('Failed to create session');\n setError(error);\n console.error('Failed to create session:', error);\n throw error;\n } finally {\n setIsCreating(false);\n }\n }, [apiClient]);\n\n const recreateSession = useCallback(async () => {\n // Clean up old session if it exists\n if (sessionId) {\n try {\n await apiClient.deleteSession(sessionId);\n } catch (err) {\n console.error('Failed to delete old session:', err);\n }\n }\n\n await initializeSession();\n }, [sessionId, apiClient, initializeSession]);\n\n // Cleanup on unmount\n useEffect(() => {\n return () => {\n if (sessionId) {\n apiClient.deleteSession(sessionId).catch((err) => {\n console.error('Failed to cleanup session on unmount:', err);\n });\n }\n };\n }, [sessionId, apiClient]);\n\n return {\n sessionId,\n isCreating,\n error,\n initializeSession,\n recreateSession,\n };\n}\n","import { useCallback } from 'react';\nimport type { SSEData } from '../types/index.js';\n\ninterface UseSSEOptions {\n onConnected?: () => void;\n onMessage?: (data: SSEData) => void;\n onDone?: () => void;\n onError?: (error: Error) => void;\n}\n\ninterface UseSSEReturn {\n handleSSEStream: (response: Response) => Promise<void>;\n}\n\n/**\n * Parse SSE chunk format: \"event: type\\ndata: json\\n\\n\"\n */\nfunction parseSSEChunk(chunk: string): Array<{ event: string; data: any }> {\n const events: Array<{ event: string; data: any }> = [];\n const lines = chunk.split('\\n');\n let currentEvent = '';\n let currentData = '';\n\n for (const line of lines) {\n if (line.startsWith('event: ')) {\n currentEvent = line.slice(7).trim();\n } else if (line.startsWith('data: ')) {\n currentData = line.slice(6).trim();\n } else if (line === '') {\n // Empty line marks end of event\n if (currentEvent && currentData) {\n try {\n const parsedData = JSON.parse(currentData);\n events.push({\n event: currentEvent,\n data: parsedData,\n });\n } catch (err) {\n console.error('Failed to parse SSE data:', currentData, err);\n }\n currentEvent = '';\n currentData = '';\n }\n }\n }\n\n return events;\n}\n\nexport function useSSE(options: UseSSEOptions = {}): UseSSEReturn {\n const { onConnected, onMessage, onDone, onError } = options;\n\n const handleSSEStream = useCallback(\n async (response: Response) => {\n const reader = response.body?.getReader();\n const decoder = new TextDecoder();\n\n if (!reader) {\n onError?.(new Error('No response body'));\n return;\n }\n\n let buffer = '';\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n\n if (done) {\n break;\n }\n\n buffer += decoder.decode(value, { stream: true });\n\n // Process complete events\n const events = parseSSEChunk(buffer);\n\n for (const { event, data } of events) {\n if (event === 'connected') {\n onConnected?.();\n onMessage?.(data);\n } else if (event === 'answer') {\n onMessage?.(data);\n } else if (event === 'done') {\n onDone?.();\n onMessage?.(data);\n } else if (event === 'error') {\n onError?.(new Error(data.error || 'Unknown error'));\n onMessage?.(data);\n }\n }\n\n // Clear processed events from buffer (keep incomplete data)\n const lastEventEnd = buffer.lastIndexOf('\\n\\n');\n if (lastEventEnd !== -1) {\n buffer = buffer.slice(lastEventEnd + 2);\n }\n }\n } catch (err) {\n const error = err instanceof Error ? err : new Error('SSE stream error');\n onError?.(error);\n } finally {\n reader.releaseLock();\n }\n },\n [onConnected, onMessage, onDone, onError]\n );\n\n return {\n handleSSEStream,\n };\n}\n","import { useState, useCallback } from 'react';\nimport { useSession } from './useSession.js';\nimport { useSSE } from './useSSE.js';\nimport type { Message } from '../types/index.js';\nimport type { APIClient } from '../api/client.js';\n\ninterface UseChatOptions {\n apiClient: APIClient;\n systemPrompt?: string;\n}\n\ninterface UseChatReturn {\n messages: Message[];\n isStreaming: boolean;\n error: Error | null;\n sessionError: Error | null;\n isCreatingSession: boolean;\n sendMessage: (text: string) => Promise<void>;\n clearMessages: () => void;\n}\n\nexport function useChat({ apiClient, systemPrompt }: UseChatOptions): UseChatReturn {\n const [messages, setMessages] = useState<Message[]>([]);\n const [isStreaming, setIsStreaming] = useState(false);\n const [error, setError] = useState<Error | null>(null);\n\n const { sessionId, isCreating: isCreatingSession, error: sessionError, initializeSession } = useSession({ apiClient });\n\n const { handleSSEStream } = useSSE({\n onConnected: () => {\n console.log('SSE connected');\n },\n onMessage: (data) => {\n if (data.type === 'answer' && data.text) {\n // Accumulate the answer text\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant' && lastMessage.isStreaming) {\n // Update the last assistant message\n return prev.slice(0, -1).concat({\n ...lastMessage,\n content: lastMessage.content + data.text,\n });\n }\n return prev;\n });\n }\n },\n onDone: () => {\n // Mark streaming complete\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant') {\n return prev.slice(0, -1).concat({\n ...lastMessage,\n isStreaming: false,\n });\n }\n return prev;\n });\n setIsStreaming(false);\n },\n onError: (err) => {\n setError(err);\n setIsStreaming(false);\n // Remove the failed assistant message\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant' && lastMessage.isStreaming) {\n return prev.slice(0, -1);\n }\n return prev;\n });\n },\n });\n\n const sendMessage = useCallback(\n async (text: string) => {\n if (!text.trim()) {\n return;\n }\n\n setError(null);\n\n // Create session if it doesn't exist yet\n let currentSessionId = sessionId;\n if (!currentSessionId && !isCreatingSession) {\n try {\n setIsStreaming(true);\n currentSessionId = await initializeSession();\n } catch (err) {\n setError(new Error('Failed to create session'));\n setIsStreaming(false);\n return;\n }\n }\n\n // Wait for session to be created if it's currently being created\n if (isCreatingSession) {\n setError(new Error('Session is being created, please try again'));\n return;\n }\n\n if (!currentSessionId) {\n setError(new Error('No active session'));\n return;\n }\n\n setIsStreaming(true);\n\n // Add user message immediately (optimistic UI)\n const userMessage: Message = {\n id: crypto.randomUUID(),\n role: 'user',\n content: text,\n timestamp: Date.now(),\n };\n setMessages((prev) => [...prev, userMessage]);\n\n // Add placeholder for assistant response\n const assistantMessage: Message = {\n id: crypto.randomUUID(),\n role: 'assistant',\n content: '',\n timestamp: Date.now(),\n isStreaming: true,\n };\n setMessages((prev) => [...prev, assistantMessage]);\n\n try {\n const response = await apiClient.askQuestion(\n currentSessionId,\n text,\n systemPrompt\n );\n await handleSSEStream(response);\n } catch (err) {\n const error = err instanceof Error ? err : new Error('Failed to send message');\n setError(error);\n setIsStreaming(false);\n\n // Remove the failed messages\n setMessages((prev) => prev.slice(0, -2));\n }\n },\n [sessionId, isCreatingSession, initializeSession, apiClient, handleSSEStream, systemPrompt]\n );\n\n const clearMessages = useCallback(() => {\n setMessages([]);\n setError(null);\n }, []);\n\n return {\n messages,\n isStreaming,\n error,\n sessionError,\n isCreatingSession,\n sendMessage,\n clearMessages,\n };\n}\n","function styleInject(css, ref) {\n if ( ref === void 0 ) ref = {};\n var insertAt = ref.insertAt;\n\n if (!css || typeof document === 'undefined') { return; }\n\n var head = document.head || document.getElementsByTagName('head')[0];\n var style = document.createElement('style');\n style.type = 'text/css';\n\n if (insertAt === 'top') {\n if (head.firstChild) {\n head.insertBefore(style, head.firstChild);\n } else {\n head.appendChild(style);\n }\n } else {\n head.appendChild(style);\n }\n\n if (style.styleSheet) {\n style.styleSheet.cssText = css;\n } else {\n style.appendChild(document.createTextNode(css));\n }\n}\n\nexport default styleInject;\n","import * as React from 'react';\nimport { Slot } from '@radix-ui/react-slot';\nimport styles from './Button.module.css';\n\ntype ButtonVariant = 'default' | 'outline' | 'ghost' | 'icon';\ntype ButtonSize = 'default' | 'sm' | 'lg' | 'icon';\n\nexport interface ButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {\n asChild?: boolean;\n variant?: ButtonVariant;\n size?: ButtonSize;\n}\n\nconst Button = React.forwardRef<HTMLButtonElement, ButtonProps>(\n ({ className, variant = 'default', size = 'default', asChild = false, ...props }, ref) => {\n const Comp = asChild ? Slot : 'button';\n\n const classNames = [\n styles.button,\n styles[`variant-${variant}`],\n styles[`size-${size}`],\n className,\n ]\n .filter(Boolean)\n .join(' ');\n\n return (\n <Comp\n className={classNames}\n ref={ref}\n {...props}\n />\n );\n }\n);\n\nButton.displayName = 'Button';\n\nexport { Button };\n","import * as React from 'react';\nimport * as DialogPrimitive from '@radix-ui/react-dialog';\nimport { Sparkles, X } from 'lucide-react';\nimport { Button } from '../ui/Button.js';\nimport styles from './Drawer.module.css';\n\ninterface DrawerProps {\n isOpen: boolean;\n onClose: () => void;\n position?: 'right' | 'left';\n width?: number | string;\n title?: string;\n closeAriaLabel?: string;\n children?: React.ReactNode;\n theme?: 'light' | 'dark';\n}\n\nexport function Drawer({\n isOpen,\n onClose,\n position = 'right',\n width = 600,\n title = 'Ask AI',\n closeAriaLabel = 'Close',\n children,\n theme = 'light',\n}: DrawerProps) {\n const widthStyle = typeof width === 'number' ? `${width}px` : width;\n\n const contentClasses = [\n styles.content,\n styles[`position-${position}`],\n ].join(' ');\n\n // Create a portal container that inherits theme from the nearest .ask-ai ancestor\n const [portalContainer, setPortalContainer] = React.useState<HTMLElement | null>(null);\n\n React.useEffect(() => {\n if (typeof document === 'undefined') return;\n\n // Create a portal container with ask-ai class\n const container = document.createElement('div');\n container.className = `ask-ai${theme === 'dark' ? ' dark' : ''}`;\n document.body.appendChild(container);\n setPortalContainer(container);\n\n return () => {\n document.body.removeChild(container);\n };\n }, [theme]);\n\n return (\n <DialogPrimitive.Root open={isOpen} onOpenChange={(open) => !open && onClose()}>\n <DialogPrimitive.Portal container={portalContainer}>\n {/* Overlay */}\n <DialogPrimitive.Overlay className={styles.overlay} />\n\n {/* Drawer Content */}\n <DialogPrimitive.Content\n className={contentClasses}\n style={{ maxWidth: widthStyle }}\n >\n {/* Header */}\n <div className={styles.header}>\n <DialogPrimitive.Title className={styles.title}>\n <Sparkles className={styles.icon} />\n {title}\n </DialogPrimitive.Title>\n <DialogPrimitive.Close asChild>\n <Button\n variant=\"ghost\"\n size=\"icon\"\n aria-label={closeAriaLabel}\n className={styles.closeButton}\n >\n <X className={styles.closeIcon} />\n </Button>\n </DialogPrimitive.Close>\n </div>\n\n {/* Content */}\n <div className={styles.body}>\n {children}\n </div>\n </DialogPrimitive.Content>\n </DialogPrimitive.Portal>\n </DialogPrimitive.Root>\n );\n}\n","import { Sparkles } from 'lucide-react';\nimport { Button } from '../ui/Button.js';\nimport styles from './Trigger.module.css';\n\ninterface TriggerProps {\n onClick: () => void;\n text?: string;\n ariaLabel?: string;\n className?: string;\n}\n\nexport function Trigger({\n onClick,\n text = 'Ask AI',\n ariaLabel = 'Open AI assistant',\n className,\n}: TriggerProps) {\n const classNames = [styles.trigger, className].filter(Boolean).join(' ');\n\n return (\n <Button\n onClick={onClick}\n variant=\"outline\"\n aria-label={ariaLabel}\n className={classNames}\n >\n <Sparkles className={styles.icon} />\n {text && <span>{text}</span>}\n </Button>\n );\n}\n","import * as React from 'react';\nimport ReactMarkdown from 'react-markdown';\nimport remarkGfm from 'remark-gfm';\nimport type { WidgetTexts, Message } from '../../core/types/index.js';\nimport styles from './ChatContainer.module.css';\n\ninterface ChatContainerProps {\n texts?: WidgetTexts;\n exampleQuestions?: string[];\n onMessage?: (message: any) => void;\n onError?: (error: Error) => void;\n // Lifted state from parent\n messages: Message[];\n isStreaming: boolean;\n error: Error | null;\n sendMessage: (text: string) => Promise<void>;\n input: string;\n setInput: React.Dispatch<React.SetStateAction<string>>;\n}\n\nexport function ChatContainer({\n texts,\n exampleQuestions,\n onMessage,\n onError,\n messages,\n isStreaming,\n error,\n sendMessage,\n input,\n setInput,\n}: ChatContainerProps) {\n const textareaRef = React.useRef<HTMLTextAreaElement>(null);\n const messagesAreaRef = React.useRef<HTMLDivElement>(null);\n const shouldAutoScrollRef = React.useRef(true); // Track if auto-scroll is enabled\n\n // Check if user is at the bottom of the messages area\n const isAtBottom = React.useCallback(() => {\n const messagesArea = messagesAreaRef.current;\n if (!messagesArea) return true;\n\n const threshold = 50; // pixels from bottom to consider \"at bottom\"\n const scrollBottom = messagesArea.scrollHeight - messagesArea.scrollTop - messagesArea.clientHeight;\n return scrollBottom < threshold;\n }, []);\n\n // Scroll to bottom of messages area\n const scrollToBottom = React.useCallback(() => {\n const messagesArea = messagesAreaRef.current;\n if (messagesArea) {\n messagesArea.scrollTop = messagesArea.scrollHeight;\n }\n }, []);\n\n // Handle user scroll events\n const handleScroll = React.useCallback(() => {\n const atBottom = isAtBottom();\n shouldAutoScrollRef.current = atBottom;\n }, [isAtBottom]);\n\n // Auto-scroll when messages change (if enabled)\n React.useEffect(() => {\n if (shouldAutoScrollRef.current) {\n scrollToBottom();\n }\n }, [messages, scrollToBottom]);\n\n // Auto-resize textarea based on content\n const adjustTextareaHeight = React.useCallback(() => {\n const textarea = textareaRef.current;\n if (textarea) {\n textarea.style.height = 'auto';\n textarea.style.height = `${textarea.scrollHeight}px`;\n }\n }, []);\n\n // Call onError callback when error occurs\n React.useEffect(() => {\n if (error && onError) {\n onError(error);\n }\n }, [error, onError]);\n\n // Call onMessage callback when new message arrives\n React.useEffect(() => {\n if (messages.length > 0 && onMessage) {\n onMessage(messages[messages.length - 1]);\n }\n }, [messages, onMessage]);\n\n const handleSubmit = async (e: React.FormEvent) => {\n e.preventDefault();\n if (!input.trim() || isStreaming) return;\n\n // Enable auto-scroll for new user message\n shouldAutoScrollRef.current = true;\n\n await sendMessage(input);\n setInput('');\n // Reset textarea height after submission\n setTimeout(() => {\n if (textareaRef.current) {\n textareaRef.current.style.height = 'auto';\n }\n }, 0);\n };\n\n const handleInputChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {\n setInput(e.target.value);\n adjustTextareaHeight();\n };\n\n const handleKeyDown = (e: React.KeyboardEvent<HTMLTextAreaElement>) => {\n // Submit on Enter (without Shift)\n if (e.key === 'Enter' && !e.shiftKey) {\n e.preventDefault();\n if (input.trim() && !isStreaming) {\n handleSubmit(e as any);\n }\n }\n // Allow Shift+Enter for new line (default textarea behavior)\n };\n\n const handleExampleClick = async (question: string) => {\n if (isStreaming) return;\n // Enable auto-scroll for new user message\n shouldAutoScrollRef.current = true;\n await sendMessage(question);\n };\n\n const inputPlaceholder = texts?.inputPlaceholder || 'Type your question...';\n const welcomeMessage = texts?.welcomeMessage || 'Hi! How can I help you today?';\n const exampleQuestionsTitle = texts?.exampleQuestionsTitle || 'Example questions:';\n\n return (\n <div className={styles.container}>\n {/* Messages Area */}\n <div\n ref={messagesAreaRef}\n onScroll={handleScroll}\n className={styles.messagesArea}\n >\n {messages.length === 0 ? (\n // Welcome Screen\n <div className={styles.welcomeScreen}>\n <p className={styles.welcomeMessage}>\n {welcomeMessage}\n </p>\n\n {exampleQuestions && exampleQuestions.length > 0 && (\n <div className={styles.exampleQuestionsContainer}>\n <p className={styles.exampleQuestionsTitle}>\n {exampleQuestionsTitle}\n </p>\n {exampleQuestions.map((question, index) => (\n <button\n key={index}\n onClick={() => handleExampleClick(question)}\n className={styles.exampleButton}\n >\n {question}\n </button>\n ))}\n </div>\n )}\n </div>\n ) : (\n // Messages\n <>\n {messages.map((message) => (\n <div\n key={message.id}\n className={`${styles.messageWrapper} ${styles[message.role]}`}\n >\n <div className={`${styles.message} ${styles[message.role]}`}>\n {message.role === 'assistant' ? (\n <div className={styles.markdown}>\n <ReactMarkdown remarkPlugins={[remarkGfm]}>\n {message.content}\n </ReactMarkdown>\n {message.isStreaming && (\n <span className={styles.cursor} />\n )}\n </div>\n ) : (\n <div className={styles.messageText}>\n {message.content}\n </div>\n )}\n </div>\n </div>\n ))}\n </>\n )}\n\n {error && (\n <div className={styles.error}>\n {error.message}\n </div>\n )}\n </div>\n\n {/* Input Area */}\n <form onSubmit={handleSubmit} className={styles.inputForm}>\n <div className={styles.inputWrapper}>\n <textarea\n ref={textareaRef}\n value={input}\n onChange={handleInputChange}\n onKeyDown={handleKeyDown}\n placeholder={inputPlaceholder}\n disabled={isStreaming}\n className={styles.input}\n rows={1}\n />\n <button\n type=\"submit\"\n disabled={!input.trim() || isStreaming}\n className={styles.submitButton}\n >\n {isStreaming ? 'Sending...' : 'Send'}\n </button>\n </div>\n </form>\n </div>\n );\n}\n","import * as React from 'react';\nimport { APIClient } from '../../core/api/client.js';\nimport { useChat } from '../../core/hooks/useChat.js';\nimport type { WidgetProps } from '../../core/types/index.js';\nimport { Drawer } from './Drawer.js';\nimport { Trigger } from './Trigger.js';\nimport { ChatContainer } from '../chat/ChatContainer.js';\n\nexport function Widget(props: WidgetProps) {\n const {\n apiUrl,\n drawerPosition = 'right',\n drawerWidth = 600,\n theme = 'light',\n texts,\n exampleQuestions,\n systemPrompt,\n hotkey,\n enableHotkey = true,\n onOpen,\n onClose,\n onMessage,\n onError,\n className,\n style,\n children,\n } = props;\n\n const [isOpen, setIsOpen] = React.useState(false);\n const [apiClient] = React.useState(() => new APIClient(apiUrl));\n\n // Lift chat state to Widget level to persist across drawer open/close\n const { messages, isStreaming, error, sendMessage } = useChat({ apiClient, systemPrompt });\n\n // Input state also needs to persist\n const [input, setInput] = React.useState('');\n\n // Handle drawer open\n const handleOpen = React.useCallback(() => {\n setIsOpen(true);\n onOpen?.();\n }, [onOpen]);\n\n // Handle drawer close\n const handleClose = React.useCallback(() => {\n setIsOpen(false);\n onClose?.();\n }, [onClose]);\n\n // Handle keyboard shortcut\n React.useEffect(() => {\n if (!enableHotkey || !hotkey) return;\n\n const handleKeyDown = (e: KeyboardEvent) => {\n const keys = hotkey.toLowerCase().split('+');\n const ctrl = keys.includes('ctrl') || keys.includes('control');\n const cmd = keys.includes('cmd') || keys.includes('command') || keys.includes('meta');\n const shift = keys.includes('shift');\n const alt = keys.includes('alt');\n const key = keys[keys.length - 1];\n\n const ctrlPressed = ctrl && (e.ctrlKey || e.metaKey);\n const cmdPressed = cmd && (e.metaKey || e.ctrlKey);\n const shiftPressed = !shift || e.shiftKey;\n const altPressed = !alt || e.altKey;\n\n if ((ctrlPressed || cmdPressed) && shiftPressed && altPressed && e.key.toLowerCase() === key) {\n e.preventDefault();\n setIsOpen((prev) => !prev);\n }\n };\n\n document.addEventListener('keydown', handleKeyDown);\n return () => document.removeEventListener('keydown', handleKeyDown);\n }, [enableHotkey, hotkey]);\n\n return (\n <div className={`ask-ai ${className || ''} ${theme === 'dark' ? 'dark' : ''}`} style={style}>\n {/* Trigger Button - Custom or Default */}\n {children && React.isValidElement(children) ? (\n React.cloneElement(children, {\n ...children.props,\n onClick: (e: React.MouseEvent) => {\n // Call existing onClick if present\n const existingOnClick = children.props?.onClick;\n if (existingOnClick) {\n existingOnClick(e);\n }\n // Then call our handleOpen\n handleOpen();\n },\n })\n ) : (\n <Trigger\n onClick={handleOpen}\n text={texts?.triggerButtonText}\n ariaLabel={texts?.triggerButtonAriaLabel}\n />\n )}\n\n {/* Drawer with Chat */}\n <Drawer\n isOpen={isOpen}\n onClose={handleClose}\n position={drawerPosition}\n width={drawerWidth}\n title={texts?.drawerTitle}\n closeAriaLabel={texts?.drawerCloseAriaLabel}\n theme={theme}\n >\n <ChatContainer\n texts={texts}\n exampleQuestions={exampleQuestions}\n onMessage={onMessage}\n onError={onError}\n messages={messages}\n isStreaming={isStreaming}\n error={error}\n sendMessage={sendMessage}\n input={input}\n setInput={setInput}\n />\n </Drawer>\n </div>\n );\n}\n"],"names":["styles","_jsx","_jsxs","_Fragment"],"mappings":"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","x_google_ignoreList":[4]}
1
+ {"version":3,"file":"index.esm.js","sources":["../src/core/api/client.ts","../src/core/hooks/useSession.ts","../src/core/hooks/useSSE.ts","../src/core/hooks/useChat.ts","../node_modules/style-inject/dist/style-inject.es.js","../src/components/ui/Button.tsx","../src/components/widget/Drawer.tsx","../src/components/widget/Trigger.tsx","../src/components/chat/ChatContainer.tsx","../src/components/widget/Widget.tsx"],"sourcesContent":["import type { SessionResponse } from '../types/index.js';\n\n/**\n * API client for Ask AI widget\n * Supports dynamic API URL configuration\n */\nexport class APIClient {\n private baseUrl: string;\n\n constructor(baseUrl: string) {\n this.baseUrl = baseUrl.replace(/\\/$/, ''); // Remove trailing slash\n }\n\n /**\n * Create a new session\n */\n async createSession(): Promise<SessionResponse> {\n const response = await fetch(`${this.baseUrl}/api/session`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n throw new Error(`Failed to create session: ${response.statusText}`);\n }\n\n return response.json();\n }\n\n /**\n * Delete a session\n */\n async deleteSession(sessionId: string): Promise<void> {\n const response = await fetch(`${this.baseUrl}/api/session/${sessionId}`, {\n method: 'DELETE',\n });\n\n if (!response.ok) {\n throw new Error(`Failed to delete session: ${response.statusText}`);\n }\n }\n\n /**\n * Ask a question and return the SSE stream response\n */\n async askQuestion(\n sessionId: string,\n question: string,\n system?: string,\n signal?: AbortSignal\n ): Promise<Response> {\n const response = await fetch(`${this.baseUrl}/api/ask`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n sessionId,\n question,\n system,\n }),\n signal,\n });\n\n if (!response.ok) {\n throw new Error(`Failed to ask question: ${response.statusText}`);\n }\n\n return response;\n }\n}\n","import { useState, useEffect, useCallback } from 'react';\nimport type { APIClient } from '../api/client.js';\n\ninterface UseSessionOptions {\n apiClient: APIClient;\n}\n\ninterface UseSessionReturn {\n sessionId: string | null;\n isCreating: boolean;\n error: Error | null;\n initializeSession: () => Promise<string>;\n clearSession: () => Promise<void>;\n}\n\nexport function useSession({ apiClient }: UseSessionOptions): UseSessionReturn {\n const [sessionId, setSessionId] = useState<string | null>(null);\n const [isCreating, setIsCreating] = useState(false);\n const [error, setError] = useState<Error | null>(null);\n\n const initializeSession = useCallback(async () => {\n setIsCreating(true);\n setError(null);\n\n try {\n const { sessionId: newSessionId } = await apiClient.createSession();\n setSessionId(newSessionId);\n return newSessionId;\n } catch (err) {\n const error = err instanceof Error ? err : new Error('Failed to create session');\n setError(error);\n console.error('Failed to create session:', error);\n throw error;\n } finally {\n setIsCreating(false);\n }\n }, [apiClient]);\n\n const clearSession = useCallback(async () => {\n setSessionId(null);\n setIsCreating(false);\n setError(null);\n }, [sessionId, apiClient, initializeSession]);\n\n // Cleanup on unmount\n useEffect(() => {\n return () => {\n if (sessionId) {\n apiClient.deleteSession(sessionId).catch((err) => {\n console.error('Failed to cleanup session on unmount:', err);\n });\n }\n };\n }, [sessionId, apiClient]);\n\n return {\n sessionId,\n isCreating,\n error,\n initializeSession,\n clearSession,\n };\n}\n","import { useCallback } from 'react';\nimport type { SSEData } from '../types/index.js';\n\ninterface UseSSEOptions {\n onConnected?: () => void;\n onMessage?: (data: SSEData) => void;\n onDone?: () => void;\n onError?: (error: Error) => void;\n}\n\ninterface UseSSEReturn {\n handleSSEStream: (response: Response) => Promise<void>;\n}\n\n/**\n * Parse SSE chunk format: \"event: type\\ndata: json\\n\\n\"\n */\nfunction parseSSEChunk(chunk: string): Array<{ event: string; data: any }> {\n const events: Array<{ event: string; data: any }> = [];\n const lines = chunk.split('\\n');\n let currentEvent = '';\n let currentData = '';\n\n for (const line of lines) {\n if (line.startsWith('event: ')) {\n currentEvent = line.slice(7).trim();\n } else if (line.startsWith('data: ')) {\n currentData = line.slice(6).trim();\n } else if (line === '') {\n // Empty line marks end of event\n if (currentEvent && currentData) {\n try {\n const parsedData = JSON.parse(currentData);\n events.push({\n event: currentEvent,\n data: parsedData,\n });\n } catch (err) {\n console.error('Failed to parse SSE data:', currentData, err);\n }\n currentEvent = '';\n currentData = '';\n }\n }\n }\n\n return events;\n}\n\nexport function useSSE(options: UseSSEOptions = {}): UseSSEReturn {\n const { onConnected, onMessage, onDone, onError } = options;\n\n const handleSSEStream = useCallback(\n async (response: Response) => {\n const reader = response.body?.getReader();\n const decoder = new TextDecoder();\n\n if (!reader) {\n onError?.(new Error('No response body'));\n return;\n }\n\n let buffer = '';\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n\n if (done) {\n break;\n }\n\n buffer += decoder.decode(value, { stream: true });\n\n // Process complete events\n const events = parseSSEChunk(buffer);\n\n for (const { event, data } of events) {\n if (event === 'connected') {\n onConnected?.();\n onMessage?.(data);\n } else if (event === 'answer') {\n onMessage?.(data);\n } else if (event === 'done') {\n onDone?.();\n onMessage?.(data);\n } else if (event === 'error') {\n onError?.(new Error(data.error || 'Unknown error'));\n onMessage?.(data);\n }\n }\n\n // Clear processed events from buffer (keep incomplete data)\n const lastEventEnd = buffer.lastIndexOf('\\n\\n');\n if (lastEventEnd !== -1) {\n buffer = buffer.slice(lastEventEnd + 2);\n }\n }\n } catch (err) {\n // Don't report AbortError as it's an intentional cancellation\n if (err instanceof Error && err.name === 'AbortError') {\n return;\n }\n const error = err instanceof Error ? err : new Error('SSE stream error');\n onError?.(error);\n } finally {\n reader.releaseLock();\n }\n },\n [onConnected, onMessage, onDone, onError]\n );\n\n return {\n handleSSEStream,\n };\n}\n","import { useState, useCallback, useRef } from 'react';\nimport { useSession } from './useSession.js';\nimport { useSSE } from './useSSE.js';\nimport type { Message } from '../types/index.js';\nimport type { APIClient } from '../api/client.js';\n\ninterface UseChatOptions {\n apiClient: APIClient;\n systemPrompt?: string;\n}\n\ninterface UseChatReturn {\n messages: Message[];\n isStreaming: boolean;\n error: Error | null;\n sessionError: Error | null;\n isCreatingSession: boolean;\n sendMessage: (text: string) => Promise<void>;\n resetChat: () => Promise<void>;\n}\n\nexport function useChat({ apiClient, systemPrompt }: UseChatOptions): UseChatReturn {\n const [messages, setMessages] = useState<Message[]>([]);\n const [isStreaming, setIsStreaming] = useState(false);\n const [error, setError] = useState<Error | null>(null);\n const abortControllerRef = useRef<AbortController | null>(null);\n\n const { sessionId, isCreating: isCreatingSession, error: sessionError, initializeSession, clearSession } = useSession({ apiClient });\n\n const { handleSSEStream } = useSSE({\n onConnected: () => {\n console.log('SSE connected');\n },\n onMessage: (data) => {\n if (data.type === 'answer' && data.text) {\n // Accumulate the answer text\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant' && lastMessage.isStreaming) {\n // Update the last assistant message\n return prev.slice(0, -1).concat({\n ...lastMessage,\n content: lastMessage.content + data.text,\n });\n }\n return prev;\n });\n }\n },\n onDone: () => {\n // Mark streaming complete\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant') {\n return prev.slice(0, -1).concat({\n ...lastMessage,\n isStreaming: false,\n });\n }\n return prev;\n });\n setIsStreaming(false);\n },\n onError: (err) => {\n setError(err);\n setIsStreaming(false);\n // Remove the failed assistant message\n setMessages((prev) => {\n const lastMessage = prev[prev.length - 1];\n if (lastMessage && lastMessage.role === 'assistant' && lastMessage.isStreaming) {\n return prev.slice(0, -1);\n }\n return prev;\n });\n },\n });\n\n const sendMessage = useCallback(\n async (text: string) => {\n if (!text.trim()) {\n return;\n }\n\n // Abort any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n }\n\n // Create new AbortController for this request\n const abortController = new AbortController();\n abortControllerRef.current = abortController;\n\n setError(null);\n\n setIsStreaming(true);\n\n // Add user message immediately (optimistic UI)\n const userMessage: Message = {\n id: crypto.randomUUID(),\n role: 'user',\n content: text,\n timestamp: Date.now(),\n };\n setMessages((prev) => [...prev, userMessage]);\n\n // Add placeholder for assistant response\n const assistantMessage: Message = {\n id: crypto.randomUUID(),\n role: 'assistant',\n content: '',\n timestamp: Date.now(),\n isStreaming: true,\n };\n setMessages((prev) => [...prev, assistantMessage]);\n\n // Create session if it doesn't exist yet\n let currentSessionId = sessionId;\n if (!currentSessionId && !isCreatingSession) {\n try {\n currentSessionId = await initializeSession();\n } catch (err) {\n setError(new Error('Failed to create session'));\n setIsStreaming(false);\n return;\n }\n }\n\n // Wait for session to be created if it's currently being created\n if (isCreatingSession) {\n setError(new Error('Session is being created, please try again'));\n return;\n }\n\n if (!currentSessionId) {\n setError(new Error('No active session'));\n return;\n }\n\n try {\n const response = await apiClient.askQuestion(\n currentSessionId,\n text,\n systemPrompt,\n abortController.signal\n );\n await handleSSEStream(response);\n } catch (err) {\n // Don't treat aborted requests as errors\n if (err instanceof Error && err.name === 'AbortError') {\n return;\n }\n\n const error = err instanceof Error ? err : new Error('Failed to send message');\n setError(error);\n setIsStreaming(false);\n\n // Remove the failed messages\n setMessages((prev) => prev.slice(0, -2));\n } finally {\n // Clear the abort controller reference if this was the current one\n if (abortControllerRef.current === abortController) {\n abortControllerRef.current = null;\n }\n }\n },\n [sessionId, isCreatingSession, initializeSession, apiClient, handleSSEStream, systemPrompt]\n );\n\n const resetChat = useCallback(async () => {\n // Abort any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n abortControllerRef.current = null;\n }\n\n setMessages([]);\n setError(null);\n setIsStreaming(false);\n await clearSession();\n }, [clearSession]);\n\n return {\n messages,\n isStreaming,\n error,\n sessionError,\n isCreatingSession,\n sendMessage,\n resetChat,\n };\n}\n","function styleInject(css, ref) {\n if ( ref === void 0 ) ref = {};\n var insertAt = ref.insertAt;\n\n if (!css || typeof document === 'undefined') { return; }\n\n var head = document.head || document.getElementsByTagName('head')[0];\n var style = document.createElement('style');\n style.type = 'text/css';\n\n if (insertAt === 'top') {\n if (head.firstChild) {\n head.insertBefore(style, head.firstChild);\n } else {\n head.appendChild(style);\n }\n } else {\n head.appendChild(style);\n }\n\n if (style.styleSheet) {\n style.styleSheet.cssText = css;\n } else {\n style.appendChild(document.createTextNode(css));\n }\n}\n\nexport default styleInject;\n","import * as React from 'react';\nimport { Slot } from '@radix-ui/react-slot';\nimport styles from './Button.module.css';\n\ntype ButtonVariant = 'default' | 'outline' | 'ghost' | 'icon';\ntype ButtonSize = 'default' | 'sm' | 'lg' | 'icon';\n\nexport interface ButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {\n asChild?: boolean;\n variant?: ButtonVariant;\n size?: ButtonSize;\n}\n\nconst Button = React.forwardRef<HTMLButtonElement, ButtonProps>(\n ({ className, variant = 'default', size = 'default', asChild = false, ...props }, ref) => {\n const Comp = asChild ? Slot : 'button';\n\n const classNames = [\n styles.button,\n styles[`variant-${variant}`],\n styles[`size-${size}`],\n className,\n ]\n .filter(Boolean)\n .join(' ');\n\n return (\n <Comp\n className={classNames}\n ref={ref}\n {...props}\n />\n );\n }\n);\n\nButton.displayName = 'Button';\n\nexport { Button };\n","import * as React from 'react';\nimport * as DialogPrimitive from '@radix-ui/react-dialog';\nimport { Sparkles, X, RotateCcw } from 'lucide-react';\nimport { Button } from '../ui/Button.js';\nimport styles from './Drawer.module.css';\n\ninterface DrawerProps {\n isOpen: boolean;\n onClose: () => void;\n onNewSession?: () => void;\n hasMessages?: boolean;\n position?: 'right' | 'left';\n width?: number | string;\n title?: string;\n closeAriaLabel?: string;\n newSessionAriaLabel?: string;\n children?: React.ReactNode;\n theme?: 'light' | 'dark';\n}\n\nexport function Drawer({\n isOpen,\n onClose,\n onNewSession,\n hasMessages,\n position = 'right',\n width = 600,\n title = 'Ask AI',\n closeAriaLabel = 'Close',\n newSessionAriaLabel = 'New session',\n children,\n theme = 'light',\n}: DrawerProps) {\n const widthStyle = typeof width === 'number' ? `${width}px` : width;\n\n const contentClasses = [\n styles.content,\n styles[`position-${position}`],\n ].join(' ');\n\n // Create a portal container that inherits theme from the nearest .ask-ai ancestor\n const [portalContainer, setPortalContainer] = React.useState<HTMLElement | null>(null);\n\n React.useEffect(() => {\n if (typeof document === 'undefined') return;\n\n // Create a portal container with ask-ai class\n const container = document.createElement('div');\n container.className = `ask-ai${theme === 'dark' ? ' dark' : ''}`;\n document.body.appendChild(container);\n setPortalContainer(container);\n\n return () => {\n document.body.removeChild(container);\n };\n }, [theme]);\n\n return (\n <DialogPrimitive.Root open={isOpen} onOpenChange={(open) => !open && onClose()}>\n <DialogPrimitive.Portal container={portalContainer}>\n {/* Overlay */}\n <DialogPrimitive.Overlay className={styles.overlay} />\n\n {/* Drawer Content */}\n <DialogPrimitive.Content\n className={contentClasses}\n style={{ maxWidth: widthStyle }}\n >\n {/* Header */}\n <div className={styles.header}>\n <DialogPrimitive.Title className={styles.title}>\n <Sparkles className={styles.icon} />\n {title}\n </DialogPrimitive.Title>\n <div className={styles.headerActions}>\n {onNewSession && hasMessages && (\n <Button\n variant=\"ghost\"\n size=\"icon\"\n aria-label={newSessionAriaLabel}\n className={styles.newSessionButton}\n onClick={onNewSession}\n >\n <RotateCcw className={styles.newSessionIcon} />\n </Button>\n )}\n <DialogPrimitive.Close asChild>\n <Button\n variant=\"ghost\"\n size=\"icon\"\n aria-label={closeAriaLabel}\n className={styles.closeButton}\n >\n <X className={styles.closeIcon} />\n </Button>\n </DialogPrimitive.Close>\n </div>\n </div>\n\n {/* Content */}\n <div className={styles.body}>\n {children}\n </div>\n </DialogPrimitive.Content>\n </DialogPrimitive.Portal>\n </DialogPrimitive.Root>\n );\n}\n","import { Sparkles } from 'lucide-react';\nimport { Button } from '../ui/Button.js';\nimport styles from './Trigger.module.css';\n\ninterface TriggerProps {\n onClick: () => void;\n text?: string;\n ariaLabel?: string;\n className?: string;\n}\n\nexport function Trigger({\n onClick,\n text = 'Ask AI',\n ariaLabel = 'Open AI assistant',\n className,\n}: TriggerProps) {\n const classNames = [styles.trigger, className].filter(Boolean).join(' ');\n\n return (\n <Button\n onClick={onClick}\n variant=\"outline\"\n aria-label={ariaLabel}\n className={classNames}\n >\n <Sparkles className={styles.icon} />\n {text && <span>{text}</span>}\n </Button>\n );\n}\n","import * as React from 'react';\nimport ReactMarkdown from 'react-markdown';\nimport remarkGfm from 'remark-gfm';\nimport rehypeHighlight from 'rehype-highlight';\nimport { common } from 'lowlight';\nimport type { WidgetTexts, Message } from '../../core/types/index.js';\nimport styles from './ChatContainer.module.css';\nimport { ArrowUp } from 'lucide-react';\n\ninterface ChatContainerProps {\n texts?: WidgetTexts;\n exampleQuestions?: string[];\n onMessage?: (message: any) => void;\n onError?: (error: Error) => void;\n // Lifted state from parent\n messages: Message[];\n isStreaming: boolean;\n error: Error | null;\n sendMessage: (text: string) => Promise<void>;\n input: string;\n setInput: React.Dispatch<React.SetStateAction<string>>;\n}\n\nexport function ChatContainer({\n texts,\n exampleQuestions,\n onMessage,\n onError,\n messages,\n isStreaming,\n error,\n sendMessage,\n input,\n setInput,\n}: ChatContainerProps) {\n const textareaRef = React.useRef<HTMLTextAreaElement>(null);\n const messagesAreaRef = React.useRef<HTMLDivElement>(null);\n const shouldAutoScrollRef = React.useRef(true); // Track if auto-scroll is enabled\n\n // Check if user is at the bottom of the messages area\n const isAtBottom = React.useCallback(() => {\n const messagesArea = messagesAreaRef.current;\n if (!messagesArea) return true;\n\n const threshold = 50; // pixels from bottom to consider \"at bottom\"\n const scrollBottom = messagesArea.scrollHeight - messagesArea.scrollTop - messagesArea.clientHeight;\n return scrollBottom < threshold;\n }, []);\n\n // Scroll to bottom of messages area\n const scrollToBottom = React.useCallback(() => {\n const messagesArea = messagesAreaRef.current;\n if (messagesArea) {\n messagesArea.scrollTop = messagesArea.scrollHeight;\n }\n }, []);\n\n // Handle user scroll events\n const handleScroll = React.useCallback(() => {\n const atBottom = isAtBottom();\n shouldAutoScrollRef.current = atBottom;\n }, [isAtBottom]);\n\n // Auto-scroll when messages change (if enabled)\n React.useEffect(() => {\n if (shouldAutoScrollRef.current) {\n scrollToBottom();\n }\n }, [messages, scrollToBottom]);\n\n // Auto-resize textarea based on content\n const adjustTextareaHeight = React.useCallback(() => {\n const textarea = textareaRef.current;\n if (textarea) {\n textarea.style.height = 'auto';\n textarea.style.height = `${textarea.scrollHeight}px`;\n }\n }, []);\n\n // Call onError callback when error occurs\n React.useEffect(() => {\n if (error && onError) {\n onError(error);\n }\n }, [error, onError]);\n\n // Call onMessage callback when new message arrives\n React.useEffect(() => {\n if (messages.length > 0 && onMessage) {\n onMessage(messages[messages.length - 1]);\n }\n }, [messages, onMessage]);\n\n const handleSubmit = async (e: React.FormEvent) => {\n e.preventDefault();\n const inputValue = input.trim();\n if (!inputValue || isStreaming) return;\n\n // Enable auto-scroll for new user message\n shouldAutoScrollRef.current = true;\n\n setInput('');\n await sendMessage(inputValue);\n // Reset textarea height after submission\n setTimeout(() => {\n if (textareaRef.current) {\n textareaRef.current.style.height = 'auto';\n }\n }, 0);\n };\n\n const handleInputChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {\n setInput(e.target.value);\n adjustTextareaHeight();\n };\n\n const handleKeyDown = (e: React.KeyboardEvent<HTMLTextAreaElement>) => {\n // Submit on Enter (without Shift)\n if (e.key === 'Enter' && !e.shiftKey) {\n e.preventDefault();\n if (input.trim() && !isStreaming) {\n handleSubmit(e as any);\n }\n }\n // Allow Shift+Enter for new line (default textarea behavior)\n };\n\n const handleExampleClick = async (question: string) => {\n if (isStreaming) return;\n // Enable auto-scroll for new user message\n shouldAutoScrollRef.current = true;\n await sendMessage(question);\n };\n\n const inputPlaceholder = texts?.inputPlaceholder || 'Type your question...';\n const welcomeMessage = texts?.welcomeMessage || 'Hi! How can I help you today?';\n const exampleQuestionsTitle = texts?.exampleQuestionsTitle || 'Example questions:';\n\n return (\n <div className={styles.container}>\n {/* Messages Area */}\n <div\n ref={messagesAreaRef}\n onScroll={handleScroll}\n className={styles.messagesArea}\n >\n {messages.length === 0 ? (\n // Welcome Screen\n <div className={styles.welcomeScreen}>\n <p className={styles.welcomeMessage}>\n {welcomeMessage}\n </p>\n\n {exampleQuestions && exampleQuestions.length > 0 && (\n <div className={styles.exampleQuestionsContainer}>\n <p className={styles.exampleQuestionsTitle}>\n {exampleQuestionsTitle}\n </p>\n {exampleQuestions.map((question, index) => (\n <button\n key={index}\n onClick={() => handleExampleClick(question)}\n className={styles.exampleButton}\n >\n {question}\n </button>\n ))}\n </div>\n )}\n </div>\n ) : (\n // Messages\n <>\n {messages.map((message) => (\n <div\n key={message.id}\n className={`${styles.messageWrapper} ${styles[message.role]}`}\n >\n <div className={`${styles.message} ${styles[message.role]}`}>\n {message.role === 'assistant' ? (\n <div className={styles.markdown}>\n <ReactMarkdown\n remarkPlugins={[remarkGfm]}\n rehypePlugins={[[rehypeHighlight, {\n languages: common,\n prefix: 'hljs-'\n }]]}\n >\n {message.content}\n </ReactMarkdown>\n {message.isStreaming && (\n <span className={styles.cursor} />\n )}\n </div>\n ) : (\n <div className={styles.messageText}>\n {message.content}\n </div>\n )}\n </div>\n </div>\n ))}\n </>\n )}\n\n {error && (\n <div className={styles.error}>\n {error.message}\n </div>\n )}\n </div>\n\n {/* Input Area */}\n <form onSubmit={handleSubmit} className={styles.inputForm}>\n <div className={styles.inputWrapper}>\n <textarea\n ref={textareaRef}\n value={input}\n onChange={handleInputChange}\n onKeyDown={handleKeyDown}\n placeholder={inputPlaceholder}\n className={styles.input}\n rows={1}\n />\n <button\n type=\"submit\"\n disabled={!input.trim() || isStreaming}\n className={styles.submitButton}\n >\n <ArrowUp />\n </button>\n </div>\n </form>\n </div>\n );\n}\n","import * as React from 'react';\nimport { APIClient } from '../../core/api/client.js';\nimport { useChat } from '../../core/hooks/useChat.js';\nimport type { WidgetProps } from '../../core/types/index.js';\nimport { Drawer } from './Drawer.js';\nimport { Trigger } from './Trigger.js';\nimport { ChatContainer } from '../chat/ChatContainer.js';\n\nexport function Widget(props: WidgetProps) {\n const {\n apiUrl,\n drawerPosition = 'right',\n drawerWidth = 600,\n theme = 'light',\n texts,\n exampleQuestions,\n systemPrompt,\n hotkey,\n enableHotkey = true,\n onOpen,\n onClose,\n onMessage,\n onError,\n className,\n style,\n children,\n } = props;\n\n const [isOpen, setIsOpen] = React.useState(false);\n const [apiClient] = React.useState(() => new APIClient(apiUrl));\n\n // Lift chat state to Widget level to persist across drawer open/close\n const { messages, isStreaming, error, sendMessage, resetChat } = useChat({ apiClient, systemPrompt });\n\n // Input state also needs to persist\n const [input, setInput] = React.useState('');\n\n // Handle drawer open\n const handleOpen = React.useCallback(() => {\n setIsOpen(true);\n onOpen?.();\n }, [onOpen]);\n\n // Handle drawer close\n const handleClose = React.useCallback(() => {\n setIsOpen(false);\n onClose?.();\n }, [onClose]);\n\n // Handle new session\n const handleNewSession = React.useCallback(async () => {\n setInput('');\n await resetChat();\n }, [resetChat]);\n\n // Handle keyboard shortcut\n React.useEffect(() => {\n if (!enableHotkey || !hotkey) return;\n\n const handleKeyDown = (e: KeyboardEvent) => {\n const keys = hotkey.toLowerCase().split('+');\n const ctrl = keys.includes('ctrl') || keys.includes('control');\n const cmd = keys.includes('cmd') || keys.includes('command') || keys.includes('meta');\n const shift = keys.includes('shift');\n const alt = keys.includes('alt');\n const key = keys[keys.length - 1];\n\n const ctrlPressed = ctrl && (e.ctrlKey || e.metaKey);\n const cmdPressed = cmd && (e.metaKey || e.ctrlKey);\n const shiftPressed = !shift || e.shiftKey;\n const altPressed = !alt || e.altKey;\n\n if ((ctrlPressed || cmdPressed) && shiftPressed && altPressed && e.key.toLowerCase() === key) {\n e.preventDefault();\n setIsOpen((prev) => !prev);\n }\n };\n\n document.addEventListener('keydown', handleKeyDown);\n return () => document.removeEventListener('keydown', handleKeyDown);\n }, [enableHotkey, hotkey]);\n\n return (\n <div className={`ask-ai ${className || ''} ${theme === 'dark' ? 'dark' : ''}`} style={style}>\n {/* Trigger Button - Custom or Default */}\n {children && React.isValidElement(children) ? (\n React.cloneElement(children, {\n ...children.props,\n onClick: (e: React.MouseEvent) => {\n // Call existing onClick if present\n const existingOnClick = children.props?.onClick;\n if (existingOnClick) {\n existingOnClick(e);\n }\n // Then call our handleOpen\n handleOpen();\n },\n })\n ) : (\n <Trigger\n onClick={handleOpen}\n text={texts?.triggerButtonText}\n ariaLabel={texts?.triggerButtonAriaLabel}\n />\n )}\n\n {/* Drawer with Chat */}\n <Drawer\n isOpen={isOpen}\n onClose={handleClose}\n onNewSession={handleNewSession}\n hasMessages={messages.length > 0}\n position={drawerPosition}\n width={drawerWidth}\n title={texts?.drawerTitle}\n closeAriaLabel={texts?.drawerCloseAriaLabel}\n newSessionAriaLabel={texts?.drawerNewSessionAriaLabel}\n theme={theme}\n >\n <ChatContainer\n texts={texts}\n exampleQuestions={exampleQuestions}\n onMessage={onMessage}\n onError={onError}\n messages={messages}\n isStreaming={isStreaming}\n error={error}\n sendMessage={sendMessage}\n input={input}\n setInput={setInput}\n />\n </Drawer>\n </div>\n );\n}\n"],"names":["styles","_jsx","_jsxs","_Fragment"],"mappings":"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","x_google_ignoreList":[4]}