ai-site-pilot 0.2.2 → 0.2.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,6 @@
1
1
  import { LanguageModel } from 'ai';
2
2
  import { T as ToolDefinition } from '../types--7jDyUM6.mjs';
3
- import { S as StreamEvent } from '../types-2ZpUTVRN.mjs';
3
+ import { S as StreamEvent } from '../types-K00dDlBC.mjs';
4
4
 
5
5
  /**
6
6
  * Factory for creating Next.js API route handlers
@@ -1,6 +1,6 @@
1
1
  import { LanguageModel } from 'ai';
2
2
  import { T as ToolDefinition } from '../types--7jDyUM6.js';
3
- import { S as StreamEvent } from '../types-2ZpUTVRN.js';
3
+ import { S as StreamEvent } from '../types-K00dDlBC.js';
4
4
 
5
5
  /**
6
6
  * Factory for creating Next.js API route handlers
@@ -1,4 +1,4 @@
1
- import { C as ChatMessage } from '../types-2ZpUTVRN.mjs';
1
+ import { C as ChatMessage, T as ToolExecution } from '../types-K00dDlBC.mjs';
2
2
 
3
3
  interface UseChatOptions {
4
4
  /** API endpoint for chat */
@@ -11,6 +11,9 @@ interface UseChatOptions {
11
11
  onStreamStart?: () => void;
12
12
  /** Callback when streaming ends */
13
13
  onStreamEnd?: () => void;
14
+ /** Generate a fallback message when AI uses tools but provides no text.
15
+ * If not provided, uses a generic fallback message. */
16
+ generateFallbackMessage?: (toolCalls: ToolExecution[]) => string;
14
17
  }
15
18
  interface UseChatReturn {
16
19
  /** All messages in the conversation */
@@ -1,4 +1,4 @@
1
- import { C as ChatMessage } from '../types-2ZpUTVRN.js';
1
+ import { C as ChatMessage, T as ToolExecution } from '../types-K00dDlBC.js';
2
2
 
3
3
  interface UseChatOptions {
4
4
  /** API endpoint for chat */
@@ -11,6 +11,9 @@ interface UseChatOptions {
11
11
  onStreamStart?: () => void;
12
12
  /** Callback when streaming ends */
13
13
  onStreamEnd?: () => void;
14
+ /** Generate a fallback message when AI uses tools but provides no text.
15
+ * If not provided, uses a generic fallback message. */
16
+ generateFallbackMessage?: (toolCalls: ToolExecution[]) => string;
14
17
  }
15
18
  interface UseChatReturn {
16
19
  /** All messages in the conversation */
@@ -4,7 +4,7 @@ var react = require('react');
4
4
 
5
5
  // src/hooks/useChat.ts
6
6
  function useChat(options) {
7
- const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd } = options;
7
+ const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd, generateFallbackMessage } = options;
8
8
  const [messages, setMessages] = react.useState(initialMessages);
9
9
  const [input, setInput] = react.useState("");
10
10
  const [isLoading, setIsLoading] = react.useState(false);
@@ -103,7 +103,14 @@ function useChat(options) {
103
103
  )
104
104
  );
105
105
  }
106
- if (!fullText) {
106
+ if (!fullText && toolCalls.length > 0 && generateFallbackMessage) {
107
+ const fallbackMessage = generateFallbackMessage(toolCalls);
108
+ setMessages(
109
+ (prev) => prev.map(
110
+ (m) => m.id === assistantMessageId ? { ...m, content: fallbackMessage } : m
111
+ )
112
+ );
113
+ } else if (!fullText) {
107
114
  setMessages(
108
115
  (prev) => prev.map(
109
116
  (m) => m.id === assistantMessageId ? { ...m, content: "I've made some changes. Take a look!" } : m
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/hooks/useChat.ts","../../src/hooks/useSpeech.ts"],"names":["useState","useRef","useCallback","useEffect"],"mappings":"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","file":"index.js","sourcesContent":["'use client';\n\nimport { useState, useCallback, useRef } from 'react';\nimport type { ChatMessage, ToolExecution, StreamEvent } from '../types';\n\nexport interface UseChatOptions {\n /** API endpoint for chat */\n apiEndpoint: string;\n /** Initial messages */\n initialMessages?: ChatMessage[];\n /** Callback when a tool is called */\n onToolCall?: (toolName: string, args: Record<string, unknown>) => void | Promise<void>;\n /** Callback when streaming starts */\n onStreamStart?: () => void;\n /** Callback when streaming ends */\n onStreamEnd?: () => void;\n}\n\nexport interface UseChatReturn {\n /** All messages in the conversation */\n messages: ChatMessage[];\n /** Current input value */\n input: string;\n /** Set the input value */\n setInput: (value: string) => void;\n /** Whether the assistant is currently responding */\n isLoading: boolean;\n /** ID of the currently streaming message */\n streamingMessageId: string | null;\n /** Send a message */\n sendMessage: (content?: string) => Promise<void>;\n /** Clear all messages */\n clearMessages: () => void;\n /** Add a message manually */\n addMessage: (message: Omit<ChatMessage, 'id' | 'timestamp'>) => void;\n}\n\n/**\n * Hook for managing chat state and streaming\n */\nexport function useChat(options: UseChatOptions): UseChatReturn {\n const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd } = options;\n\n const [messages, setMessages] = useState<ChatMessage[]>(initialMessages);\n const [input, setInput] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [streamingMessageId, setStreamingMessageId] = useState<string | null>(null);\n\n const abortControllerRef = useRef<AbortController | null>(null);\n\n const addMessage = useCallback((message: Omit<ChatMessage, 'id' | 'timestamp'>) => {\n const newMessage: ChatMessage = {\n ...message,\n id: Date.now().toString(),\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, newMessage]);\n return newMessage;\n }, []);\n\n const clearMessages = useCallback(() => {\n setMessages(initialMessages);\n }, [initialMessages]);\n\n const sendMessage = useCallback(\n async (content?: string) => {\n const messageContent = content || input;\n if (!messageContent.trim() || isLoading) return;\n\n // Cancel any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n }\n abortControllerRef.current = new AbortController();\n\n // Add user message\n const userMessage: ChatMessage = {\n id: Date.now().toString(),\n role: 'user',\n content: messageContent,\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, userMessage]);\n setInput('');\n setIsLoading(true);\n onStreamStart?.();\n\n // Add placeholder for assistant message\n const assistantMessageId = (Date.now() + 1).toString();\n const assistantMessage: ChatMessage = {\n id: assistantMessageId,\n role: 'assistant',\n content: '',\n timestamp: new Date(),\n toolCalls: [],\n };\n setMessages((prev) => [...prev, assistantMessage]);\n setStreamingMessageId(assistantMessageId);\n\n let fullText = '';\n const toolCalls: ToolExecution[] = [];\n\n try {\n // Prepare messages for API (convert to format expected by API)\n const apiMessages = messages.concat(userMessage).map((m) => ({\n role: m.role,\n content: m.content,\n }));\n\n const response = await fetch(apiEndpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ messages: apiMessages }),\n signal: abortControllerRef.current.signal,\n });\n\n if (!response.ok) {\n throw new Error('Failed to get response');\n }\n\n const reader = response.body?.getReader();\n if (!reader) throw new Error('No reader available');\n\n const decoder = new TextDecoder();\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n const text = decoder.decode(value);\n const lines = text.split('\\n');\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n try {\n const data = JSON.parse(line.slice(6)) as StreamEvent;\n\n if (data.type === 'text') {\n fullText += data.content;\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, content: fullText } : m\n )\n );\n } else if (data.type === 'tool') {\n const toolCall = { name: data.name, args: data.args };\n toolCalls.push(toolCall);\n\n // Execute tool call callback\n if (onToolCall) {\n try {\n await onToolCall(data.name, data.args);\n } catch (e) {\n console.error('Tool execution error:', e);\n }\n }\n } else if (data.type === 'done') {\n // Update message with tool calls\n if (toolCalls.length > 0) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, toolCalls } : m\n )\n );\n }\n\n // If no text, show default message\n if (!fullText) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: \"I've made some changes. Take a look!\" }\n : m\n )\n );\n }\n } else if (data.type === 'error') {\n throw new Error(data.message);\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n }\n }\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n // Request was cancelled, ignore\n return;\n }\n\n console.error('Chat error:', error);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: 'Sorry, I encountered an error. Please try again.' }\n : m\n )\n );\n } finally {\n setIsLoading(false);\n setStreamingMessageId(null);\n onStreamEnd?.();\n }\n },\n [apiEndpoint, input, isLoading, messages, onToolCall, onStreamStart, onStreamEnd]\n );\n\n return {\n messages,\n input,\n setInput,\n isLoading,\n streamingMessageId,\n sendMessage,\n clearMessages,\n addMessage,\n };\n}\n","'use client';\n\nimport { useState, useRef, useEffect, useCallback } from 'react';\n\n// Web Speech API types\ninterface SpeechRecognitionEvent extends Event {\n results: SpeechRecognitionResultList;\n}\n\ninterface SpeechRecognitionInstance {\n continuous: boolean;\n interimResults: boolean;\n lang: string;\n start(): void;\n stop(): void;\n onresult: ((event: SpeechRecognitionEvent) => void) | null;\n onend: (() => void) | null;\n onerror: (() => void) | null;\n}\n\nexport interface UseSpeechOptions {\n /** Language for speech recognition */\n lang?: string;\n /** Callback when speech is recognized */\n onResult?: (transcript: string, isFinal: boolean) => void;\n}\n\nexport interface UseSpeechReturn {\n /** Whether speech recognition is supported */\n isSupported: boolean;\n /** Whether currently listening */\n isListening: boolean;\n /** Toggle listening on/off */\n toggleListening: () => void;\n /** Start listening */\n startListening: () => void;\n /** Stop listening */\n stopListening: () => void;\n /** Speak text using TTS */\n speak: (text: string) => void;\n /** Whether TTS is enabled */\n ttsEnabled: boolean;\n /** Toggle TTS on/off */\n setTtsEnabled: (enabled: boolean) => void;\n /** Cancel current speech */\n cancelSpeech: () => void;\n}\n\n/**\n * Hook for speech recognition and text-to-speech\n */\nexport function useSpeech(options: UseSpeechOptions = {}): UseSpeechReturn {\n const { lang = 'en-US', onResult } = options;\n\n const [isSupported, setIsSupported] = useState(false);\n const [isListening, setIsListening] = useState(false);\n const [ttsEnabled, setTtsEnabled] = useState(false);\n\n const recognitionRef = useRef<SpeechRecognitionInstance | null>(null);\n\n // Initialize speech recognition\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const windowWithSpeech = window as any;\n const SpeechRecognitionAPI =\n windowWithSpeech.SpeechRecognition || windowWithSpeech.webkitSpeechRecognition;\n\n if (SpeechRecognitionAPI) {\n setIsSupported(true);\n recognitionRef.current = new SpeechRecognitionAPI() as SpeechRecognitionInstance;\n recognitionRef.current.continuous = false;\n recognitionRef.current.interimResults = true;\n recognitionRef.current.lang = lang;\n\n recognitionRef.current.onresult = (event) => {\n const result = event.results[0];\n const transcript = result[0].transcript;\n onResult?.(transcript, result.isFinal);\n\n if (result.isFinal) {\n setIsListening(false);\n }\n };\n\n recognitionRef.current.onend = () => setIsListening(false);\n recognitionRef.current.onerror = () => setIsListening(false);\n }\n }, [lang, onResult]);\n\n const startListening = useCallback(() => {\n if (!recognitionRef.current) return;\n try {\n recognitionRef.current.start();\n setIsListening(true);\n } catch (e) {\n console.error('Speech recognition error:', e);\n }\n }, []);\n\n const stopListening = useCallback(() => {\n if (!recognitionRef.current) return;\n recognitionRef.current.stop();\n setIsListening(false);\n }, []);\n\n const toggleListening = useCallback(() => {\n if (isListening) {\n stopListening();\n } else {\n startListening();\n }\n }, [isListening, startListening, stopListening]);\n\n const speak = useCallback(\n (text: string) => {\n if (!ttsEnabled || typeof window === 'undefined' || !('speechSynthesis' in window)) {\n return;\n }\n\n speechSynthesis.cancel();\n const utterance = new SpeechSynthesisUtterance(text);\n utterance.rate = 1.0;\n utterance.pitch = 1.0;\n\n // Try to find a good voice\n const voices = speechSynthesis.getVoices();\n const preferredVoice = voices.find(\n (v) => v.name.includes('Google') || v.name.includes('Samantha') || v.name.includes('Alex')\n );\n if (preferredVoice) {\n utterance.voice = preferredVoice;\n }\n\n speechSynthesis.speak(utterance);\n },\n [ttsEnabled]\n );\n\n const cancelSpeech = useCallback(() => {\n if (typeof window !== 'undefined' && 'speechSynthesis' in window) {\n speechSynthesis.cancel();\n }\n }, []);\n\n return {\n isSupported,\n isListening,\n toggleListening,\n startListening,\n stopListening,\n speak,\n ttsEnabled,\n setTtsEnabled,\n cancelSpeech,\n };\n}\n"]}
1
+ {"version":3,"sources":["../../src/hooks/useChat.ts","../../src/hooks/useSpeech.ts"],"names":["useState","useRef","useCallback","useEffect"],"mappings":"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","file":"index.js","sourcesContent":["'use client';\n\nimport { useState, useCallback, useRef } from 'react';\nimport type { ChatMessage, ToolExecution, StreamEvent } from '../types';\n\nexport interface UseChatOptions {\n /** API endpoint for chat */\n apiEndpoint: string;\n /** Initial messages */\n initialMessages?: ChatMessage[];\n /** Callback when a tool is called */\n onToolCall?: (toolName: string, args: Record<string, unknown>) => void | Promise<void>;\n /** Callback when streaming starts */\n onStreamStart?: () => void;\n /** Callback when streaming ends */\n onStreamEnd?: () => void;\n /** Generate a fallback message when AI uses tools but provides no text.\n * If not provided, uses a generic fallback message. */\n generateFallbackMessage?: (toolCalls: ToolExecution[]) => string;\n}\n\nexport interface UseChatReturn {\n /** All messages in the conversation */\n messages: ChatMessage[];\n /** Current input value */\n input: string;\n /** Set the input value */\n setInput: (value: string) => void;\n /** Whether the assistant is currently responding */\n isLoading: boolean;\n /** ID of the currently streaming message */\n streamingMessageId: string | null;\n /** Send a message */\n sendMessage: (content?: string) => Promise<void>;\n /** Clear all messages */\n clearMessages: () => void;\n /** Add a message manually */\n addMessage: (message: Omit<ChatMessage, 'id' | 'timestamp'>) => void;\n}\n\n/**\n * Hook for managing chat state and streaming\n */\nexport function useChat(options: UseChatOptions): UseChatReturn {\n const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd, generateFallbackMessage } = options;\n\n const [messages, setMessages] = useState<ChatMessage[]>(initialMessages);\n const [input, setInput] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [streamingMessageId, setStreamingMessageId] = useState<string | null>(null);\n\n const abortControllerRef = useRef<AbortController | null>(null);\n\n const addMessage = useCallback((message: Omit<ChatMessage, 'id' | 'timestamp'>) => {\n const newMessage: ChatMessage = {\n ...message,\n id: Date.now().toString(),\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, newMessage]);\n return newMessage;\n }, []);\n\n const clearMessages = useCallback(() => {\n setMessages(initialMessages);\n }, [initialMessages]);\n\n const sendMessage = useCallback(\n async (content?: string) => {\n const messageContent = content || input;\n if (!messageContent.trim() || isLoading) return;\n\n // Cancel any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n }\n abortControllerRef.current = new AbortController();\n\n // Add user message\n const userMessage: ChatMessage = {\n id: Date.now().toString(),\n role: 'user',\n content: messageContent,\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, userMessage]);\n setInput('');\n setIsLoading(true);\n onStreamStart?.();\n\n // Add placeholder for assistant message\n const assistantMessageId = (Date.now() + 1).toString();\n const assistantMessage: ChatMessage = {\n id: assistantMessageId,\n role: 'assistant',\n content: '',\n timestamp: new Date(),\n toolCalls: [],\n };\n setMessages((prev) => [...prev, assistantMessage]);\n setStreamingMessageId(assistantMessageId);\n\n let fullText = '';\n const toolCalls: ToolExecution[] = [];\n\n try {\n // Prepare messages for API (convert to format expected by API)\n const apiMessages = messages.concat(userMessage).map((m) => ({\n role: m.role,\n content: m.content,\n }));\n\n const response = await fetch(apiEndpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ messages: apiMessages }),\n signal: abortControllerRef.current.signal,\n });\n\n if (!response.ok) {\n throw new Error('Failed to get response');\n }\n\n const reader = response.body?.getReader();\n if (!reader) throw new Error('No reader available');\n\n const decoder = new TextDecoder();\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n const text = decoder.decode(value);\n const lines = text.split('\\n');\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n try {\n const data = JSON.parse(line.slice(6)) as StreamEvent;\n\n if (data.type === 'text') {\n fullText += data.content;\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, content: fullText } : m\n )\n );\n } else if (data.type === 'tool') {\n const toolCall = { name: data.name, args: data.args };\n toolCalls.push(toolCall);\n\n // Execute tool call callback\n if (onToolCall) {\n try {\n await onToolCall(data.name, data.args);\n } catch (e) {\n console.error('Tool execution error:', e);\n }\n }\n } else if (data.type === 'done') {\n // Update message with tool calls\n if (toolCalls.length > 0) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, toolCalls } : m\n )\n );\n }\n\n // If no text, generate contextual fallback based on tools used\n if (!fullText && toolCalls.length > 0 && generateFallbackMessage) {\n const fallbackMessage = generateFallbackMessage(toolCalls);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: fallbackMessage }\n : m\n )\n );\n } else if (!fullText) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: \"I've made some changes. Take a look!\" }\n : m\n )\n );\n }\n } else if (data.type === 'error') {\n throw new Error(data.message);\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n }\n }\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n // Request was cancelled, ignore\n return;\n }\n\n console.error('Chat error:', error);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: 'Sorry, I encountered an error. Please try again.' }\n : m\n )\n );\n } finally {\n setIsLoading(false);\n setStreamingMessageId(null);\n onStreamEnd?.();\n }\n },\n [apiEndpoint, input, isLoading, messages, onToolCall, onStreamStart, onStreamEnd]\n );\n\n return {\n messages,\n input,\n setInput,\n isLoading,\n streamingMessageId,\n sendMessage,\n clearMessages,\n addMessage,\n };\n}\n","'use client';\n\nimport { useState, useRef, useEffect, useCallback } from 'react';\n\n// Web Speech API types\ninterface SpeechRecognitionEvent extends Event {\n results: SpeechRecognitionResultList;\n}\n\ninterface SpeechRecognitionInstance {\n continuous: boolean;\n interimResults: boolean;\n lang: string;\n start(): void;\n stop(): void;\n onresult: ((event: SpeechRecognitionEvent) => void) | null;\n onend: (() => void) | null;\n onerror: (() => void) | null;\n}\n\nexport interface UseSpeechOptions {\n /** Language for speech recognition */\n lang?: string;\n /** Callback when speech is recognized */\n onResult?: (transcript: string, isFinal: boolean) => void;\n}\n\nexport interface UseSpeechReturn {\n /** Whether speech recognition is supported */\n isSupported: boolean;\n /** Whether currently listening */\n isListening: boolean;\n /** Toggle listening on/off */\n toggleListening: () => void;\n /** Start listening */\n startListening: () => void;\n /** Stop listening */\n stopListening: () => void;\n /** Speak text using TTS */\n speak: (text: string) => void;\n /** Whether TTS is enabled */\n ttsEnabled: boolean;\n /** Toggle TTS on/off */\n setTtsEnabled: (enabled: boolean) => void;\n /** Cancel current speech */\n cancelSpeech: () => void;\n}\n\n/**\n * Hook for speech recognition and text-to-speech\n */\nexport function useSpeech(options: UseSpeechOptions = {}): UseSpeechReturn {\n const { lang = 'en-US', onResult } = options;\n\n const [isSupported, setIsSupported] = useState(false);\n const [isListening, setIsListening] = useState(false);\n const [ttsEnabled, setTtsEnabled] = useState(false);\n\n const recognitionRef = useRef<SpeechRecognitionInstance | null>(null);\n\n // Initialize speech recognition\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const windowWithSpeech = window as any;\n const SpeechRecognitionAPI =\n windowWithSpeech.SpeechRecognition || windowWithSpeech.webkitSpeechRecognition;\n\n if (SpeechRecognitionAPI) {\n setIsSupported(true);\n recognitionRef.current = new SpeechRecognitionAPI() as SpeechRecognitionInstance;\n recognitionRef.current.continuous = false;\n recognitionRef.current.interimResults = true;\n recognitionRef.current.lang = lang;\n\n recognitionRef.current.onresult = (event) => {\n const result = event.results[0];\n const transcript = result[0].transcript;\n onResult?.(transcript, result.isFinal);\n\n if (result.isFinal) {\n setIsListening(false);\n }\n };\n\n recognitionRef.current.onend = () => setIsListening(false);\n recognitionRef.current.onerror = () => setIsListening(false);\n }\n }, [lang, onResult]);\n\n const startListening = useCallback(() => {\n if (!recognitionRef.current) return;\n try {\n recognitionRef.current.start();\n setIsListening(true);\n } catch (e) {\n console.error('Speech recognition error:', e);\n }\n }, []);\n\n const stopListening = useCallback(() => {\n if (!recognitionRef.current) return;\n recognitionRef.current.stop();\n setIsListening(false);\n }, []);\n\n const toggleListening = useCallback(() => {\n if (isListening) {\n stopListening();\n } else {\n startListening();\n }\n }, [isListening, startListening, stopListening]);\n\n const speak = useCallback(\n (text: string) => {\n if (!ttsEnabled || typeof window === 'undefined' || !('speechSynthesis' in window)) {\n return;\n }\n\n speechSynthesis.cancel();\n const utterance = new SpeechSynthesisUtterance(text);\n utterance.rate = 1.0;\n utterance.pitch = 1.0;\n\n // Try to find a good voice\n const voices = speechSynthesis.getVoices();\n const preferredVoice = voices.find(\n (v) => v.name.includes('Google') || v.name.includes('Samantha') || v.name.includes('Alex')\n );\n if (preferredVoice) {\n utterance.voice = preferredVoice;\n }\n\n speechSynthesis.speak(utterance);\n },\n [ttsEnabled]\n );\n\n const cancelSpeech = useCallback(() => {\n if (typeof window !== 'undefined' && 'speechSynthesis' in window) {\n speechSynthesis.cancel();\n }\n }, []);\n\n return {\n isSupported,\n isListening,\n toggleListening,\n startListening,\n stopListening,\n speak,\n ttsEnabled,\n setTtsEnabled,\n cancelSpeech,\n };\n}\n"]}
@@ -2,7 +2,7 @@ import { useState, useRef, useCallback, useEffect } from 'react';
2
2
 
3
3
  // src/hooks/useChat.ts
4
4
  function useChat(options) {
5
- const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd } = options;
5
+ const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd, generateFallbackMessage } = options;
6
6
  const [messages, setMessages] = useState(initialMessages);
7
7
  const [input, setInput] = useState("");
8
8
  const [isLoading, setIsLoading] = useState(false);
@@ -101,7 +101,14 @@ function useChat(options) {
101
101
  )
102
102
  );
103
103
  }
104
- if (!fullText) {
104
+ if (!fullText && toolCalls.length > 0 && generateFallbackMessage) {
105
+ const fallbackMessage = generateFallbackMessage(toolCalls);
106
+ setMessages(
107
+ (prev) => prev.map(
108
+ (m) => m.id === assistantMessageId ? { ...m, content: fallbackMessage } : m
109
+ )
110
+ );
111
+ } else if (!fullText) {
105
112
  setMessages(
106
113
  (prev) => prev.map(
107
114
  (m) => m.id === assistantMessageId ? { ...m, content: "I've made some changes. Take a look!" } : m
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/hooks/useChat.ts","../../src/hooks/useSpeech.ts"],"names":["useState","useRef","useCallback"],"mappings":";;;AAwCO,SAAS,QAAQ,OAAA,EAAwC;AAC9D,EAAA,MAAM,EAAE,aAAa,eAAA,GAAkB,IAAI,UAAA,EAAY,aAAA,EAAe,aAAY,GAAI,OAAA;AAEtF,EAAA,MAAM,CAAC,QAAA,EAAU,WAAW,CAAA,GAAI,SAAwB,eAAe,CAAA;AACvE,EAAA,MAAM,CAAC,KAAA,EAAO,QAAQ,CAAA,GAAI,SAAS,EAAE,CAAA;AACrC,EAAA,MAAM,CAAC,SAAA,EAAW,YAAY,CAAA,GAAI,SAAS,KAAK,CAAA;AAChD,EAAA,MAAM,CAAC,kBAAA,EAAoB,qBAAqB,CAAA,GAAI,SAAwB,IAAI,CAAA;AAEhF,EAAA,MAAM,kBAAA,GAAqB,OAA+B,IAAI,CAAA;AAE9D,EAAA,MAAM,UAAA,GAAa,WAAA,CAAY,CAAC,OAAA,KAAmD;AACjF,IAAA,MAAM,UAAA,GAA0B;AAAA,MAC9B,GAAG,OAAA;AAAA,MACH,EAAA,EAAI,IAAA,CAAK,GAAA,EAAI,CAAE,QAAA,EAAS;AAAA,MACxB,SAAA,sBAAe,IAAA;AAAK,KACtB;AACA,IAAA,WAAA,CAAY,CAAC,IAAA,KAAS,CAAC,GAAG,IAAA,EAAM,UAAU,CAAC,CAAA;AAC3C,IAAA,OAAO,UAAA;AAAA,EACT,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,MAAM,aAAA,GAAgB,YAAY,MAAM;AACtC,IAAA,WAAA,CAAY,eAAe,CAAA;AAAA,EAC7B,CAAA,EAAG,CAAC,eAAe,CAAC,CAAA;AAEpB,EAAA,MAAM,WAAA,GAAc,WAAA;AAAA,IAClB,OAAO,OAAA,KAAqB;AAC1B,MAAA,MAAM,iBAAiB,OAAA,IAAW,KAAA;AAClC,MAAA,IAAI,CAAC,cAAA,CAAe,IAAA,EAAK,IAAK,SAAA,EAAW;AAGzC,MAAA,IAAI,mBAAmB,OAAA,EAAS;AAC9B,QAAA,kBAAA,CAAmB,QAAQ,KAAA,EAAM;AAAA,MACnC;AACA,MAAA,kBAAA,CAAmB,OAAA,GAAU,IAAI,eAAA,EAAgB;AAGjD,MAAA,MAAM,WAAA,GAA2B;AAAA,QAC/B,EAAA,EAAI,IAAA,CAAK,GAAA,EAAI,CAAE,QAAA,EAAS;AAAA,QACxB,IAAA,EAAM,MAAA;AAAA,QACN,OAAA,EAAS,cAAA;AAAA,QACT,SAAA,sBAAe,IAAA;AAAK,OACtB;AACA,MAAA,WAAA,CAAY,CAAC,IAAA,KAAS,CAAC,GAAG,IAAA,EAAM,WAAW,CAAC,CAAA;AAC5C,MAAA,QAAA,CAAS,EAAE,CAAA;AACX,MAAA,YAAA,CAAa,IAAI,CAAA;AACjB,MAAA,aAAA,IAAgB;AAGhB,MAAA,MAAM,kBAAA,GAAA,CAAsB,IAAA,CAAK,GAAA,EAAI,GAAI,GAAG,QAAA,EAAS;AACrD,MAAA,MAAM,gBAAA,GAAgC;AAAA,QACpC,EAAA,EAAI,kBAAA;AAAA,QACJ,IAAA,EAAM,WAAA;AAAA,QACN,OAAA,EAAS,EAAA;AAAA,QACT,SAAA,sBAAe,IAAA,EAAK;AAAA,QACpB,WAAW;AAAC,OACd;AACA,MAAA,WAAA,CAAY,CAAC,IAAA,KAAS,CAAC,GAAG,IAAA,EAAM,gBAAgB,CAAC,CAAA;AACjD,MAAA,qBAAA,CAAsB,kBAAkB,CAAA;AAExC,MAAA,IAAI,QAAA,GAAW,EAAA;AACf,MAAA,MAAM,YAA6B,EAAC;AAEpC,MAAA,IAAI;AAEF,QAAA,MAAM,cAAc,QAAA,CAAS,MAAA,CAAO,WAAW,CAAA,CAAE,GAAA,CAAI,CAAC,CAAA,MAAO;AAAA,UAC3D,MAAM,CAAA,CAAE,IAAA;AAAA,UACR,SAAS,CAAA,CAAE;AAAA,SACb,CAAE,CAAA;AAEF,QAAA,MAAM,QAAA,GAAW,MAAM,KAAA,CAAM,WAAA,EAAa;AAAA,UACxC,MAAA,EAAQ,MAAA;AAAA,UACR,OAAA,EAAS,EAAE,cAAA,EAAgB,kBAAA,EAAmB;AAAA,UAC9C,MAAM,IAAA,CAAK,SAAA,CAAU,EAAE,QAAA,EAAU,aAAa,CAAA;AAAA,UAC9C,MAAA,EAAQ,mBAAmB,OAAA,CAAQ;AAAA,SACpC,CAAA;AAED,QAAA,IAAI,CAAC,SAAS,EAAA,EAAI;AAChB,UAAA,MAAM,IAAI,MAAM,wBAAwB,CAAA;AAAA,QAC1C;AAEA,QAAA,MAAM,MAAA,GAAS,QAAA,CAAS,IAAA,EAAM,SAAA,EAAU;AACxC,QAAA,IAAI,CAAC,MAAA,EAAQ,MAAM,IAAI,MAAM,qBAAqB,CAAA;AAElD,QAAA,MAAM,OAAA,GAAU,IAAI,WAAA,EAAY;AAEhC,QAAA,OAAO,IAAA,EAAM;AACX,UAAA,MAAM,EAAE,IAAA,EAAM,KAAA,EAAM,GAAI,MAAM,OAAO,IAAA,EAAK;AAC1C,UAAA,IAAI,IAAA,EAAM;AAEV,UAAA,MAAM,IAAA,GAAO,OAAA,CAAQ,MAAA,CAAO,KAAK,CAAA;AACjC,UAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,KAAA,CAAM,IAAI,CAAA;AAE7B,UAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,YAAA,IAAI,IAAA,CAAK,UAAA,CAAW,QAAQ,CAAA,EAAG;AAC7B,cAAA,IAAI;AACF,gBAAA,MAAM,OAAO,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,CAAC,CAAC,CAAA;AAErC,gBAAA,IAAI,IAAA,CAAK,SAAS,MAAA,EAAQ;AACxB,kBAAA,QAAA,IAAY,IAAA,CAAK,OAAA;AACjB,kBAAA,WAAA;AAAA,oBAAY,CAAC,SACX,IAAA,CAAK,GAAA;AAAA,sBAAI,CAAC,CAAA,KACR,CAAA,CAAE,EAAA,KAAO,kBAAA,GAAqB,EAAE,GAAG,CAAA,EAAG,OAAA,EAAS,QAAA,EAAS,GAAI;AAAA;AAC9D,mBACF;AAAA,gBACF,CAAA,MAAA,IAAW,IAAA,CAAK,IAAA,KAAS,MAAA,EAAQ;AAC/B,kBAAA,MAAM,WAAW,EAAE,IAAA,EAAM,KAAK,IAAA,EAAM,IAAA,EAAM,KAAK,IAAA,EAAK;AACpD,kBAAA,SAAA,CAAU,KAAK,QAAQ,CAAA;AAGvB,kBAAA,IAAI,UAAA,EAAY;AACd,oBAAA,IAAI;AACF,sBAAA,MAAM,UAAA,CAAW,IAAA,CAAK,IAAA,EAAM,IAAA,CAAK,IAAI,CAAA;AAAA,oBACvC,SAAS,CAAA,EAAG;AACV,sBAAA,OAAA,CAAQ,KAAA,CAAM,yBAAyB,CAAC,CAAA;AAAA,oBAC1C;AAAA,kBACF;AAAA,gBACF,CAAA,MAAA,IAAW,IAAA,CAAK,IAAA,KAAS,MAAA,EAAQ;AAE/B,kBAAA,IAAI,SAAA,CAAU,SAAS,CAAA,EAAG;AACxB,oBAAA,WAAA;AAAA,sBAAY,CAAC,SACX,IAAA,CAAK,GAAA;AAAA,wBAAI,CAAC,MACR,CAAA,CAAE,EAAA,KAAO,qBAAqB,EAAE,GAAG,CAAA,EAAG,SAAA,EAAU,GAAI;AAAA;AACtD,qBACF;AAAA,kBACF;AAGA,kBAAA,IAAI,CAAC,QAAA,EAAU;AACb,oBAAA,WAAA;AAAA,sBAAY,CAAC,SACX,IAAA,CAAK,GAAA;AAAA,wBAAI,CAAC,CAAA,KACR,CAAA,CAAE,EAAA,KAAO,kBAAA,GACL,EAAE,GAAG,CAAA,EAAG,OAAA,EAAS,sCAAA,EAAuC,GACxD;AAAA;AACN,qBACF;AAAA,kBACF;AAAA,gBACF,CAAA,MAAA,IAAW,IAAA,CAAK,IAAA,KAAS,OAAA,EAAS;AAChC,kBAAA,MAAM,IAAI,KAAA,CAAM,IAAA,CAAK,OAAO,CAAA;AAAA,gBAC9B;AAAA,cACF,CAAA,CAAA,MAAQ;AAAA,cAER;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,MACF,SAAS,KAAA,EAAO;AACd,QAAA,IAAI,KAAA,YAAiB,KAAA,IAAS,KAAA,CAAM,IAAA,KAAS,YAAA,EAAc;AAEzD,UAAA;AAAA,QACF;AAEA,QAAA,OAAA,CAAQ,KAAA,CAAM,eAAe,KAAK,CAAA;AAClC,QAAA,WAAA;AAAA,UAAY,CAAC,SACX,IAAA,CAAK,GAAA;AAAA,YAAI,CAAC,CAAA,KACR,CAAA,CAAE,EAAA,KAAO,kBAAA,GACL,EAAE,GAAG,CAAA,EAAG,OAAA,EAAS,kDAAA,EAAmD,GACpE;AAAA;AACN,SACF;AAAA,MACF,CAAA,SAAE;AACA,QAAA,YAAA,CAAa,KAAK,CAAA;AAClB,QAAA,qBAAA,CAAsB,IAAI,CAAA;AAC1B,QAAA,WAAA,IAAc;AAAA,MAChB;AAAA,IACF,CAAA;AAAA,IACA,CAAC,WAAA,EAAa,KAAA,EAAO,WAAW,QAAA,EAAU,UAAA,EAAY,eAAe,WAAW;AAAA,GAClF;AAEA,EAAA,OAAO;AAAA,IACL,QAAA;AAAA,IACA,KAAA;AAAA,IACA,QAAA;AAAA,IACA,SAAA;AAAA,IACA,kBAAA;AAAA,IACA,WAAA;AAAA,IACA,aAAA;AAAA,IACA;AAAA,GACF;AACF;ACvKO,SAAS,SAAA,CAAU,OAAA,GAA4B,EAAC,EAAoB;AACzE,EAAA,MAAM,EAAE,IAAA,GAAO,OAAA,EAAS,QAAA,EAAS,GAAI,OAAA;AAErC,EAAA,MAAM,CAAC,WAAA,EAAa,cAAc,CAAA,GAAIA,SAAS,KAAK,CAAA;AACpD,EAAA,MAAM,CAAC,WAAA,EAAa,cAAc,CAAA,GAAIA,SAAS,KAAK,CAAA;AACpD,EAAA,MAAM,CAAC,UAAA,EAAY,aAAa,CAAA,GAAIA,SAAS,KAAK,CAAA;AAElD,EAAA,MAAM,cAAA,GAAiBC,OAAyC,IAAI,CAAA;AAGpE,EAAA,SAAA,CAAU,MAAM;AACd,IAAA,IAAI,OAAO,WAAW,WAAA,EAAa;AAGnC,IAAA,MAAM,gBAAA,GAAmB,MAAA;AACzB,IAAA,MAAM,oBAAA,GACJ,gBAAA,CAAiB,iBAAA,IAAqB,gBAAA,CAAiB,uBAAA;AAEzD,IAAA,IAAI,oBAAA,EAAsB;AACxB,MAAA,cAAA,CAAe,IAAI,CAAA;AACnB,MAAA,cAAA,CAAe,OAAA,GAAU,IAAI,oBAAA,EAAqB;AAClD,MAAA,cAAA,CAAe,QAAQ,UAAA,GAAa,KAAA;AACpC,MAAA,cAAA,CAAe,QAAQ,cAAA,GAAiB,IAAA;AACxC,MAAA,cAAA,CAAe,QAAQ,IAAA,GAAO,IAAA;AAE9B,MAAA,cAAA,CAAe,OAAA,CAAQ,QAAA,GAAW,CAAC,KAAA,KAAU;AAC3C,QAAA,MAAM,MAAA,GAAS,KAAA,CAAM,OAAA,CAAQ,CAAC,CAAA;AAC9B,QAAA,MAAM,UAAA,GAAa,MAAA,CAAO,CAAC,CAAA,CAAE,UAAA;AAC7B,QAAA,QAAA,GAAW,UAAA,EAAY,OAAO,OAAO,CAAA;AAErC,QAAA,IAAI,OAAO,OAAA,EAAS;AAClB,UAAA,cAAA,CAAe,KAAK,CAAA;AAAA,QACtB;AAAA,MACF,CAAA;AAEA,MAAA,cAAA,CAAe,OAAA,CAAQ,KAAA,GAAQ,MAAM,cAAA,CAAe,KAAK,CAAA;AACzD,MAAA,cAAA,CAAe,OAAA,CAAQ,OAAA,GAAU,MAAM,cAAA,CAAe,KAAK,CAAA;AAAA,IAC7D;AAAA,EACF,CAAA,EAAG,CAAC,IAAA,EAAM,QAAQ,CAAC,CAAA;AAEnB,EAAA,MAAM,cAAA,GAAiBC,YAAY,MAAM;AACvC,IAAA,IAAI,CAAC,eAAe,OAAA,EAAS;AAC7B,IAAA,IAAI;AACF,MAAA,cAAA,CAAe,QAAQ,KAAA,EAAM;AAC7B,MAAA,cAAA,CAAe,IAAI,CAAA;AAAA,IACrB,SAAS,CAAA,EAAG;AACV,MAAA,OAAA,CAAQ,KAAA,CAAM,6BAA6B,CAAC,CAAA;AAAA,IAC9C;AAAA,EACF,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,MAAM,aAAA,GAAgBA,YAAY,MAAM;AACtC,IAAA,IAAI,CAAC,eAAe,OAAA,EAAS;AAC7B,IAAA,cAAA,CAAe,QAAQ,IAAA,EAAK;AAC5B,IAAA,cAAA,CAAe,KAAK,CAAA;AAAA,EACtB,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,MAAM,eAAA,GAAkBA,YAAY,MAAM;AACxC,IAAA,IAAI,WAAA,EAAa;AACf,MAAA,aAAA,EAAc;AAAA,IAChB,CAAA,MAAO;AACL,MAAA,cAAA,EAAe;AAAA,IACjB;AAAA,EACF,CAAA,EAAG,CAAC,WAAA,EAAa,cAAA,EAAgB,aAAa,CAAC,CAAA;AAE/C,EAAA,MAAM,KAAA,GAAQA,WAAAA;AAAA,IACZ,CAAC,IAAA,KAAiB;AAChB,MAAA,IAAI,CAAC,UAAA,IAAc,OAAO,WAAW,WAAA,IAAe,EAAE,qBAAqB,MAAA,CAAA,EAAS;AAClF,QAAA;AAAA,MACF;AAEA,MAAA,eAAA,CAAgB,MAAA,EAAO;AACvB,MAAA,MAAM,SAAA,GAAY,IAAI,wBAAA,CAAyB,IAAI,CAAA;AACnD,MAAA,SAAA,CAAU,IAAA,GAAO,CAAA;AACjB,MAAA,SAAA,CAAU,KAAA,GAAQ,CAAA;AAGlB,MAAA,MAAM,MAAA,GAAS,gBAAgB,SAAA,EAAU;AACzC,MAAA,MAAM,iBAAiB,MAAA,CAAO,IAAA;AAAA,QAC5B,CAAC,CAAA,KAAM,CAAA,CAAE,IAAA,CAAK,SAAS,QAAQ,CAAA,IAAK,CAAA,CAAE,IAAA,CAAK,SAAS,UAAU,CAAA,IAAK,CAAA,CAAE,IAAA,CAAK,SAAS,MAAM;AAAA,OAC3F;AACA,MAAA,IAAI,cAAA,EAAgB;AAClB,QAAA,SAAA,CAAU,KAAA,GAAQ,cAAA;AAAA,MACpB;AAEA,MAAA,eAAA,CAAgB,MAAM,SAAS,CAAA;AAAA,IACjC,CAAA;AAAA,IACA,CAAC,UAAU;AAAA,GACb;AAEA,EAAA,MAAM,YAAA,GAAeA,YAAY,MAAM;AACrC,IAAA,IAAI,OAAO,MAAA,KAAW,WAAA,IAAe,iBAAA,IAAqB,MAAA,EAAQ;AAChE,MAAA,eAAA,CAAgB,MAAA,EAAO;AAAA,IACzB;AAAA,EACF,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,OAAO;AAAA,IACL,WAAA;AAAA,IACA,WAAA;AAAA,IACA,eAAA;AAAA,IACA,cAAA;AAAA,IACA,aAAA;AAAA,IACA,KAAA;AAAA,IACA,UAAA;AAAA,IACA,aAAA;AAAA,IACA;AAAA,GACF;AACF","file":"index.mjs","sourcesContent":["'use client';\n\nimport { useState, useCallback, useRef } from 'react';\nimport type { ChatMessage, ToolExecution, StreamEvent } from '../types';\n\nexport interface UseChatOptions {\n /** API endpoint for chat */\n apiEndpoint: string;\n /** Initial messages */\n initialMessages?: ChatMessage[];\n /** Callback when a tool is called */\n onToolCall?: (toolName: string, args: Record<string, unknown>) => void | Promise<void>;\n /** Callback when streaming starts */\n onStreamStart?: () => void;\n /** Callback when streaming ends */\n onStreamEnd?: () => void;\n}\n\nexport interface UseChatReturn {\n /** All messages in the conversation */\n messages: ChatMessage[];\n /** Current input value */\n input: string;\n /** Set the input value */\n setInput: (value: string) => void;\n /** Whether the assistant is currently responding */\n isLoading: boolean;\n /** ID of the currently streaming message */\n streamingMessageId: string | null;\n /** Send a message */\n sendMessage: (content?: string) => Promise<void>;\n /** Clear all messages */\n clearMessages: () => void;\n /** Add a message manually */\n addMessage: (message: Omit<ChatMessage, 'id' | 'timestamp'>) => void;\n}\n\n/**\n * Hook for managing chat state and streaming\n */\nexport function useChat(options: UseChatOptions): UseChatReturn {\n const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd } = options;\n\n const [messages, setMessages] = useState<ChatMessage[]>(initialMessages);\n const [input, setInput] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [streamingMessageId, setStreamingMessageId] = useState<string | null>(null);\n\n const abortControllerRef = useRef<AbortController | null>(null);\n\n const addMessage = useCallback((message: Omit<ChatMessage, 'id' | 'timestamp'>) => {\n const newMessage: ChatMessage = {\n ...message,\n id: Date.now().toString(),\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, newMessage]);\n return newMessage;\n }, []);\n\n const clearMessages = useCallback(() => {\n setMessages(initialMessages);\n }, [initialMessages]);\n\n const sendMessage = useCallback(\n async (content?: string) => {\n const messageContent = content || input;\n if (!messageContent.trim() || isLoading) return;\n\n // Cancel any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n }\n abortControllerRef.current = new AbortController();\n\n // Add user message\n const userMessage: ChatMessage = {\n id: Date.now().toString(),\n role: 'user',\n content: messageContent,\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, userMessage]);\n setInput('');\n setIsLoading(true);\n onStreamStart?.();\n\n // Add placeholder for assistant message\n const assistantMessageId = (Date.now() + 1).toString();\n const assistantMessage: ChatMessage = {\n id: assistantMessageId,\n role: 'assistant',\n content: '',\n timestamp: new Date(),\n toolCalls: [],\n };\n setMessages((prev) => [...prev, assistantMessage]);\n setStreamingMessageId(assistantMessageId);\n\n let fullText = '';\n const toolCalls: ToolExecution[] = [];\n\n try {\n // Prepare messages for API (convert to format expected by API)\n const apiMessages = messages.concat(userMessage).map((m) => ({\n role: m.role,\n content: m.content,\n }));\n\n const response = await fetch(apiEndpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ messages: apiMessages }),\n signal: abortControllerRef.current.signal,\n });\n\n if (!response.ok) {\n throw new Error('Failed to get response');\n }\n\n const reader = response.body?.getReader();\n if (!reader) throw new Error('No reader available');\n\n const decoder = new TextDecoder();\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n const text = decoder.decode(value);\n const lines = text.split('\\n');\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n try {\n const data = JSON.parse(line.slice(6)) as StreamEvent;\n\n if (data.type === 'text') {\n fullText += data.content;\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, content: fullText } : m\n )\n );\n } else if (data.type === 'tool') {\n const toolCall = { name: data.name, args: data.args };\n toolCalls.push(toolCall);\n\n // Execute tool call callback\n if (onToolCall) {\n try {\n await onToolCall(data.name, data.args);\n } catch (e) {\n console.error('Tool execution error:', e);\n }\n }\n } else if (data.type === 'done') {\n // Update message with tool calls\n if (toolCalls.length > 0) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, toolCalls } : m\n )\n );\n }\n\n // If no text, show default message\n if (!fullText) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: \"I've made some changes. Take a look!\" }\n : m\n )\n );\n }\n } else if (data.type === 'error') {\n throw new Error(data.message);\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n }\n }\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n // Request was cancelled, ignore\n return;\n }\n\n console.error('Chat error:', error);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: 'Sorry, I encountered an error. Please try again.' }\n : m\n )\n );\n } finally {\n setIsLoading(false);\n setStreamingMessageId(null);\n onStreamEnd?.();\n }\n },\n [apiEndpoint, input, isLoading, messages, onToolCall, onStreamStart, onStreamEnd]\n );\n\n return {\n messages,\n input,\n setInput,\n isLoading,\n streamingMessageId,\n sendMessage,\n clearMessages,\n addMessage,\n };\n}\n","'use client';\n\nimport { useState, useRef, useEffect, useCallback } from 'react';\n\n// Web Speech API types\ninterface SpeechRecognitionEvent extends Event {\n results: SpeechRecognitionResultList;\n}\n\ninterface SpeechRecognitionInstance {\n continuous: boolean;\n interimResults: boolean;\n lang: string;\n start(): void;\n stop(): void;\n onresult: ((event: SpeechRecognitionEvent) => void) | null;\n onend: (() => void) | null;\n onerror: (() => void) | null;\n}\n\nexport interface UseSpeechOptions {\n /** Language for speech recognition */\n lang?: string;\n /** Callback when speech is recognized */\n onResult?: (transcript: string, isFinal: boolean) => void;\n}\n\nexport interface UseSpeechReturn {\n /** Whether speech recognition is supported */\n isSupported: boolean;\n /** Whether currently listening */\n isListening: boolean;\n /** Toggle listening on/off */\n toggleListening: () => void;\n /** Start listening */\n startListening: () => void;\n /** Stop listening */\n stopListening: () => void;\n /** Speak text using TTS */\n speak: (text: string) => void;\n /** Whether TTS is enabled */\n ttsEnabled: boolean;\n /** Toggle TTS on/off */\n setTtsEnabled: (enabled: boolean) => void;\n /** Cancel current speech */\n cancelSpeech: () => void;\n}\n\n/**\n * Hook for speech recognition and text-to-speech\n */\nexport function useSpeech(options: UseSpeechOptions = {}): UseSpeechReturn {\n const { lang = 'en-US', onResult } = options;\n\n const [isSupported, setIsSupported] = useState(false);\n const [isListening, setIsListening] = useState(false);\n const [ttsEnabled, setTtsEnabled] = useState(false);\n\n const recognitionRef = useRef<SpeechRecognitionInstance | null>(null);\n\n // Initialize speech recognition\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const windowWithSpeech = window as any;\n const SpeechRecognitionAPI =\n windowWithSpeech.SpeechRecognition || windowWithSpeech.webkitSpeechRecognition;\n\n if (SpeechRecognitionAPI) {\n setIsSupported(true);\n recognitionRef.current = new SpeechRecognitionAPI() as SpeechRecognitionInstance;\n recognitionRef.current.continuous = false;\n recognitionRef.current.interimResults = true;\n recognitionRef.current.lang = lang;\n\n recognitionRef.current.onresult = (event) => {\n const result = event.results[0];\n const transcript = result[0].transcript;\n onResult?.(transcript, result.isFinal);\n\n if (result.isFinal) {\n setIsListening(false);\n }\n };\n\n recognitionRef.current.onend = () => setIsListening(false);\n recognitionRef.current.onerror = () => setIsListening(false);\n }\n }, [lang, onResult]);\n\n const startListening = useCallback(() => {\n if (!recognitionRef.current) return;\n try {\n recognitionRef.current.start();\n setIsListening(true);\n } catch (e) {\n console.error('Speech recognition error:', e);\n }\n }, []);\n\n const stopListening = useCallback(() => {\n if (!recognitionRef.current) return;\n recognitionRef.current.stop();\n setIsListening(false);\n }, []);\n\n const toggleListening = useCallback(() => {\n if (isListening) {\n stopListening();\n } else {\n startListening();\n }\n }, [isListening, startListening, stopListening]);\n\n const speak = useCallback(\n (text: string) => {\n if (!ttsEnabled || typeof window === 'undefined' || !('speechSynthesis' in window)) {\n return;\n }\n\n speechSynthesis.cancel();\n const utterance = new SpeechSynthesisUtterance(text);\n utterance.rate = 1.0;\n utterance.pitch = 1.0;\n\n // Try to find a good voice\n const voices = speechSynthesis.getVoices();\n const preferredVoice = voices.find(\n (v) => v.name.includes('Google') || v.name.includes('Samantha') || v.name.includes('Alex')\n );\n if (preferredVoice) {\n utterance.voice = preferredVoice;\n }\n\n speechSynthesis.speak(utterance);\n },\n [ttsEnabled]\n );\n\n const cancelSpeech = useCallback(() => {\n if (typeof window !== 'undefined' && 'speechSynthesis' in window) {\n speechSynthesis.cancel();\n }\n }, []);\n\n return {\n isSupported,\n isListening,\n toggleListening,\n startListening,\n stopListening,\n speak,\n ttsEnabled,\n setTtsEnabled,\n cancelSpeech,\n };\n}\n"]}
1
+ {"version":3,"sources":["../../src/hooks/useChat.ts","../../src/hooks/useSpeech.ts"],"names":["useState","useRef","useCallback"],"mappings":"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","file":"index.mjs","sourcesContent":["'use client';\n\nimport { useState, useCallback, useRef } from 'react';\nimport type { ChatMessage, ToolExecution, StreamEvent } from '../types';\n\nexport interface UseChatOptions {\n /** API endpoint for chat */\n apiEndpoint: string;\n /** Initial messages */\n initialMessages?: ChatMessage[];\n /** Callback when a tool is called */\n onToolCall?: (toolName: string, args: Record<string, unknown>) => void | Promise<void>;\n /** Callback when streaming starts */\n onStreamStart?: () => void;\n /** Callback when streaming ends */\n onStreamEnd?: () => void;\n /** Generate a fallback message when AI uses tools but provides no text.\n * If not provided, uses a generic fallback message. */\n generateFallbackMessage?: (toolCalls: ToolExecution[]) => string;\n}\n\nexport interface UseChatReturn {\n /** All messages in the conversation */\n messages: ChatMessage[];\n /** Current input value */\n input: string;\n /** Set the input value */\n setInput: (value: string) => void;\n /** Whether the assistant is currently responding */\n isLoading: boolean;\n /** ID of the currently streaming message */\n streamingMessageId: string | null;\n /** Send a message */\n sendMessage: (content?: string) => Promise<void>;\n /** Clear all messages */\n clearMessages: () => void;\n /** Add a message manually */\n addMessage: (message: Omit<ChatMessage, 'id' | 'timestamp'>) => void;\n}\n\n/**\n * Hook for managing chat state and streaming\n */\nexport function useChat(options: UseChatOptions): UseChatReturn {\n const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd, generateFallbackMessage } = options;\n\n const [messages, setMessages] = useState<ChatMessage[]>(initialMessages);\n const [input, setInput] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [streamingMessageId, setStreamingMessageId] = useState<string | null>(null);\n\n const abortControllerRef = useRef<AbortController | null>(null);\n\n const addMessage = useCallback((message: Omit<ChatMessage, 'id' | 'timestamp'>) => {\n const newMessage: ChatMessage = {\n ...message,\n id: Date.now().toString(),\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, newMessage]);\n return newMessage;\n }, []);\n\n const clearMessages = useCallback(() => {\n setMessages(initialMessages);\n }, [initialMessages]);\n\n const sendMessage = useCallback(\n async (content?: string) => {\n const messageContent = content || input;\n if (!messageContent.trim() || isLoading) return;\n\n // Cancel any ongoing request\n if (abortControllerRef.current) {\n abortControllerRef.current.abort();\n }\n abortControllerRef.current = new AbortController();\n\n // Add user message\n const userMessage: ChatMessage = {\n id: Date.now().toString(),\n role: 'user',\n content: messageContent,\n timestamp: new Date(),\n };\n setMessages((prev) => [...prev, userMessage]);\n setInput('');\n setIsLoading(true);\n onStreamStart?.();\n\n // Add placeholder for assistant message\n const assistantMessageId = (Date.now() + 1).toString();\n const assistantMessage: ChatMessage = {\n id: assistantMessageId,\n role: 'assistant',\n content: '',\n timestamp: new Date(),\n toolCalls: [],\n };\n setMessages((prev) => [...prev, assistantMessage]);\n setStreamingMessageId(assistantMessageId);\n\n let fullText = '';\n const toolCalls: ToolExecution[] = [];\n\n try {\n // Prepare messages for API (convert to format expected by API)\n const apiMessages = messages.concat(userMessage).map((m) => ({\n role: m.role,\n content: m.content,\n }));\n\n const response = await fetch(apiEndpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ messages: apiMessages }),\n signal: abortControllerRef.current.signal,\n });\n\n if (!response.ok) {\n throw new Error('Failed to get response');\n }\n\n const reader = response.body?.getReader();\n if (!reader) throw new Error('No reader available');\n\n const decoder = new TextDecoder();\n\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n const text = decoder.decode(value);\n const lines = text.split('\\n');\n\n for (const line of lines) {\n if (line.startsWith('data: ')) {\n try {\n const data = JSON.parse(line.slice(6)) as StreamEvent;\n\n if (data.type === 'text') {\n fullText += data.content;\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, content: fullText } : m\n )\n );\n } else if (data.type === 'tool') {\n const toolCall = { name: data.name, args: data.args };\n toolCalls.push(toolCall);\n\n // Execute tool call callback\n if (onToolCall) {\n try {\n await onToolCall(data.name, data.args);\n } catch (e) {\n console.error('Tool execution error:', e);\n }\n }\n } else if (data.type === 'done') {\n // Update message with tool calls\n if (toolCalls.length > 0) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId ? { ...m, toolCalls } : m\n )\n );\n }\n\n // If no text, generate contextual fallback based on tools used\n if (!fullText && toolCalls.length > 0 && generateFallbackMessage) {\n const fallbackMessage = generateFallbackMessage(toolCalls);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: fallbackMessage }\n : m\n )\n );\n } else if (!fullText) {\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: \"I've made some changes. Take a look!\" }\n : m\n )\n );\n }\n } else if (data.type === 'error') {\n throw new Error(data.message);\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n }\n }\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n // Request was cancelled, ignore\n return;\n }\n\n console.error('Chat error:', error);\n setMessages((prev) =>\n prev.map((m) =>\n m.id === assistantMessageId\n ? { ...m, content: 'Sorry, I encountered an error. Please try again.' }\n : m\n )\n );\n } finally {\n setIsLoading(false);\n setStreamingMessageId(null);\n onStreamEnd?.();\n }\n },\n [apiEndpoint, input, isLoading, messages, onToolCall, onStreamStart, onStreamEnd]\n );\n\n return {\n messages,\n input,\n setInput,\n isLoading,\n streamingMessageId,\n sendMessage,\n clearMessages,\n addMessage,\n };\n}\n","'use client';\n\nimport { useState, useRef, useEffect, useCallback } from 'react';\n\n// Web Speech API types\ninterface SpeechRecognitionEvent extends Event {\n results: SpeechRecognitionResultList;\n}\n\ninterface SpeechRecognitionInstance {\n continuous: boolean;\n interimResults: boolean;\n lang: string;\n start(): void;\n stop(): void;\n onresult: ((event: SpeechRecognitionEvent) => void) | null;\n onend: (() => void) | null;\n onerror: (() => void) | null;\n}\n\nexport interface UseSpeechOptions {\n /** Language for speech recognition */\n lang?: string;\n /** Callback when speech is recognized */\n onResult?: (transcript: string, isFinal: boolean) => void;\n}\n\nexport interface UseSpeechReturn {\n /** Whether speech recognition is supported */\n isSupported: boolean;\n /** Whether currently listening */\n isListening: boolean;\n /** Toggle listening on/off */\n toggleListening: () => void;\n /** Start listening */\n startListening: () => void;\n /** Stop listening */\n stopListening: () => void;\n /** Speak text using TTS */\n speak: (text: string) => void;\n /** Whether TTS is enabled */\n ttsEnabled: boolean;\n /** Toggle TTS on/off */\n setTtsEnabled: (enabled: boolean) => void;\n /** Cancel current speech */\n cancelSpeech: () => void;\n}\n\n/**\n * Hook for speech recognition and text-to-speech\n */\nexport function useSpeech(options: UseSpeechOptions = {}): UseSpeechReturn {\n const { lang = 'en-US', onResult } = options;\n\n const [isSupported, setIsSupported] = useState(false);\n const [isListening, setIsListening] = useState(false);\n const [ttsEnabled, setTtsEnabled] = useState(false);\n\n const recognitionRef = useRef<SpeechRecognitionInstance | null>(null);\n\n // Initialize speech recognition\n useEffect(() => {\n if (typeof window === 'undefined') return;\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const windowWithSpeech = window as any;\n const SpeechRecognitionAPI =\n windowWithSpeech.SpeechRecognition || windowWithSpeech.webkitSpeechRecognition;\n\n if (SpeechRecognitionAPI) {\n setIsSupported(true);\n recognitionRef.current = new SpeechRecognitionAPI() as SpeechRecognitionInstance;\n recognitionRef.current.continuous = false;\n recognitionRef.current.interimResults = true;\n recognitionRef.current.lang = lang;\n\n recognitionRef.current.onresult = (event) => {\n const result = event.results[0];\n const transcript = result[0].transcript;\n onResult?.(transcript, result.isFinal);\n\n if (result.isFinal) {\n setIsListening(false);\n }\n };\n\n recognitionRef.current.onend = () => setIsListening(false);\n recognitionRef.current.onerror = () => setIsListening(false);\n }\n }, [lang, onResult]);\n\n const startListening = useCallback(() => {\n if (!recognitionRef.current) return;\n try {\n recognitionRef.current.start();\n setIsListening(true);\n } catch (e) {\n console.error('Speech recognition error:', e);\n }\n }, []);\n\n const stopListening = useCallback(() => {\n if (!recognitionRef.current) return;\n recognitionRef.current.stop();\n setIsListening(false);\n }, []);\n\n const toggleListening = useCallback(() => {\n if (isListening) {\n stopListening();\n } else {\n startListening();\n }\n }, [isListening, startListening, stopListening]);\n\n const speak = useCallback(\n (text: string) => {\n if (!ttsEnabled || typeof window === 'undefined' || !('speechSynthesis' in window)) {\n return;\n }\n\n speechSynthesis.cancel();\n const utterance = new SpeechSynthesisUtterance(text);\n utterance.rate = 1.0;\n utterance.pitch = 1.0;\n\n // Try to find a good voice\n const voices = speechSynthesis.getVoices();\n const preferredVoice = voices.find(\n (v) => v.name.includes('Google') || v.name.includes('Samantha') || v.name.includes('Alex')\n );\n if (preferredVoice) {\n utterance.voice = preferredVoice;\n }\n\n speechSynthesis.speak(utterance);\n },\n [ttsEnabled]\n );\n\n const cancelSpeech = useCallback(() => {\n if (typeof window !== 'undefined' && 'speechSynthesis' in window) {\n speechSynthesis.cancel();\n }\n }, []);\n\n return {\n isSupported,\n isListening,\n toggleListening,\n startListening,\n stopListening,\n speak,\n ttsEnabled,\n setTtsEnabled,\n cancelSpeech,\n };\n}\n"]}
package/dist/index.d.mts CHANGED
@@ -1,7 +1,9 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
- import { a as SitePilotProps, C as ChatMessage$1, b as Suggestion } from './types-2ZpUTVRN.mjs';
3
- export { d as SitePilotFeatures, c as SitePilotTheme, S as StreamEvent, T as ToolExecution } from './types-2ZpUTVRN.mjs';
2
+ import { a as SitePilotProps, C as ChatMessage$1, b as Suggestion } from './types-K00dDlBC.mjs';
3
+ export { d as SitePilotFeatures, c as SitePilotTheme, S as StreamEvent, T as ToolExecution } from './types-K00dDlBC.mjs';
4
4
  export { useChat, useSpeech } from './hooks/index.mjs';
5
+ export { ToolRegistry, createFallbackMessageGenerator, createToolRegistry, defineSimpleTool, defineTool } from './tools/index.mjs';
6
+ import './types--7jDyUM6.mjs';
5
7
 
6
8
  /**
7
9
  * SitePilot - AI chat widget that can control and navigate your website
@@ -22,7 +24,7 @@ export { useChat, useSpeech } from './hooks/index.mjs';
22
24
  * />
23
25
  * ```
24
26
  */
25
- declare function SitePilot({ apiEndpoint, theme, suggestions, features, onToolCall, defaultOpen, placeholder, welcomeMessage, className, }: SitePilotProps): react_jsx_runtime.JSX.Element;
27
+ declare function SitePilot({ apiEndpoint, theme, suggestions, features, onToolCall, generateFallbackMessage, defaultOpen, placeholder, welcomeMessage, className, }: SitePilotProps): react_jsx_runtime.JSX.Element;
26
28
 
27
29
  interface ChatMessageProps {
28
30
  message: ChatMessage$1;
package/dist/index.d.ts CHANGED
@@ -1,7 +1,9 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
- import { a as SitePilotProps, C as ChatMessage$1, b as Suggestion } from './types-2ZpUTVRN.js';
3
- export { d as SitePilotFeatures, c as SitePilotTheme, S as StreamEvent, T as ToolExecution } from './types-2ZpUTVRN.js';
2
+ import { a as SitePilotProps, C as ChatMessage$1, b as Suggestion } from './types-K00dDlBC.js';
3
+ export { d as SitePilotFeatures, c as SitePilotTheme, S as StreamEvent, T as ToolExecution } from './types-K00dDlBC.js';
4
4
  export { useChat, useSpeech } from './hooks/index.js';
5
+ export { ToolRegistry, createFallbackMessageGenerator, createToolRegistry, defineSimpleTool, defineTool } from './tools/index.js';
6
+ import './types--7jDyUM6.js';
5
7
 
6
8
  /**
7
9
  * SitePilot - AI chat widget that can control and navigate your website
@@ -22,7 +24,7 @@ export { useChat, useSpeech } from './hooks/index.js';
22
24
  * />
23
25
  * ```
24
26
  */
25
- declare function SitePilot({ apiEndpoint, theme, suggestions, features, onToolCall, defaultOpen, placeholder, welcomeMessage, className, }: SitePilotProps): react_jsx_runtime.JSX.Element;
27
+ declare function SitePilot({ apiEndpoint, theme, suggestions, features, onToolCall, generateFallbackMessage, defaultOpen, placeholder, welcomeMessage, className, }: SitePilotProps): react_jsx_runtime.JSX.Element;
26
28
 
27
29
  interface ChatMessageProps {
28
30
  message: ChatMessage$1;
package/dist/index.js CHANGED
@@ -14,7 +14,7 @@ var ReactMarkdown__default = /*#__PURE__*/_interopDefault(ReactMarkdown);
14
14
 
15
15
  // src/components/SitePilot.tsx
16
16
  function useChat(options) {
17
- const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd } = options;
17
+ const { apiEndpoint, initialMessages = [], onToolCall, onStreamStart, onStreamEnd, generateFallbackMessage } = options;
18
18
  const [messages, setMessages] = react.useState(initialMessages);
19
19
  const [input, setInput] = react.useState("");
20
20
  const [isLoading, setIsLoading] = react.useState(false);
@@ -113,7 +113,14 @@ function useChat(options) {
113
113
  )
114
114
  );
115
115
  }
116
- if (!fullText) {
116
+ if (!fullText && toolCalls.length > 0 && generateFallbackMessage) {
117
+ const fallbackMessage = generateFallbackMessage(toolCalls);
118
+ setMessages(
119
+ (prev) => prev.map(
120
+ (m) => m.id === assistantMessageId ? { ...m, content: fallbackMessage } : m
121
+ )
122
+ );
123
+ } else if (!fullText) {
117
124
  setMessages(
118
125
  (prev) => prev.map(
119
126
  (m) => m.id === assistantMessageId ? { ...m, content: "I've made some changes. Take a look!" } : m
@@ -394,6 +401,7 @@ function SitePilot({
394
401
  suggestions = [],
395
402
  features = {},
396
403
  onToolCall,
404
+ generateFallbackMessage,
397
405
  defaultOpen = false,
398
406
  placeholder = "Type a message...",
399
407
  welcomeMessage = "Hi! I'm here to help you navigate and explore. What would you like to know?",
@@ -440,6 +448,7 @@ function SitePilot({
440
448
  apiEndpoint,
441
449
  initialMessages,
442
450
  onToolCall,
451
+ generateFallbackMessage,
443
452
  onStreamStart: () => setShowSuggestions(false)
444
453
  });
445
454
  const {
@@ -614,11 +623,147 @@ function SitePilot({
614
623
  }
615
624
  var SitePilot_default = SitePilot;
616
625
 
626
+ // src/tools/registry.ts
627
+ function createFallbackMessageGenerator(toolMessages, defaultMessage = "I've made some changes. Take a look!") {
628
+ return (toolCalls) => {
629
+ if (toolCalls.length === 0) return defaultMessage;
630
+ const messages = [];
631
+ for (const tool of toolCalls) {
632
+ const generator = toolMessages[tool.name];
633
+ if (generator) {
634
+ messages.push(generator(tool.args));
635
+ } else {
636
+ messages.push(defaultMessage);
637
+ }
638
+ }
639
+ return messages.join(" ");
640
+ };
641
+ }
642
+ function defineTool(config) {
643
+ return config;
644
+ }
645
+ function defineSimpleTool(name, description, handler) {
646
+ return {
647
+ name,
648
+ description,
649
+ parameters: {
650
+ type: "object",
651
+ properties: {},
652
+ required: []
653
+ },
654
+ handler
655
+ };
656
+ }
657
+ var ToolRegistry = class {
658
+ constructor(tools = []) {
659
+ this.tools = /* @__PURE__ */ new Map();
660
+ for (const tool of tools) {
661
+ this.register(tool);
662
+ }
663
+ }
664
+ /**
665
+ * Register a tool
666
+ */
667
+ register(tool) {
668
+ this.tools.set(tool.name, tool);
669
+ }
670
+ /**
671
+ * Unregister a tool
672
+ */
673
+ unregister(name) {
674
+ this.tools.delete(name);
675
+ }
676
+ /**
677
+ * Get a tool by name
678
+ */
679
+ get(name) {
680
+ return this.tools.get(name);
681
+ }
682
+ /**
683
+ * Get all tools
684
+ */
685
+ getAll() {
686
+ return Array.from(this.tools.values());
687
+ }
688
+ /**
689
+ * Execute a tool by name with given arguments
690
+ */
691
+ async execute(execution) {
692
+ const tool = this.tools.get(execution.name);
693
+ if (!tool) {
694
+ console.warn(`Tool not found: ${execution.name}`);
695
+ return;
696
+ }
697
+ if (tool.handler) {
698
+ await tool.handler(execution.args);
699
+ }
700
+ }
701
+ /**
702
+ * Execute multiple tools in sequence
703
+ */
704
+ async executeAll(executions, delayMs = 300) {
705
+ for (const execution of executions) {
706
+ await this.execute(execution);
707
+ if (delayMs > 0 && executions.indexOf(execution) < executions.length - 1) {
708
+ await new Promise((resolve) => setTimeout(resolve, delayMs));
709
+ }
710
+ }
711
+ }
712
+ /**
713
+ * Convert to Vercel AI SDK tool format
714
+ */
715
+ toVercelAITools() {
716
+ const result = {};
717
+ for (const tool of this.tools.values()) {
718
+ result[tool.name] = {
719
+ description: tool.description,
720
+ parameters: {
721
+ type: "object",
722
+ properties: tool.parameters.properties,
723
+ required: tool.parameters.required
724
+ }
725
+ };
726
+ }
727
+ return result;
728
+ }
729
+ /**
730
+ * Convert to Gemini function declarations format
731
+ */
732
+ toGeminiFunctionDeclarations() {
733
+ return this.getAll().map((tool) => ({
734
+ name: tool.name,
735
+ description: tool.description,
736
+ parameters: {
737
+ type: "object",
738
+ properties: Object.fromEntries(
739
+ Object.entries(tool.parameters.properties).map(([key, value]) => [
740
+ key,
741
+ {
742
+ type: value.type,
743
+ description: value.description,
744
+ enum: value.enum
745
+ }
746
+ ])
747
+ ),
748
+ required: tool.parameters.required
749
+ }
750
+ }));
751
+ }
752
+ };
753
+ function createToolRegistry(tools) {
754
+ return new ToolRegistry(tools);
755
+ }
756
+
617
757
  exports.ChatInput = ChatInput;
618
758
  exports.ChatMessage = ChatMessage;
619
759
  exports.SitePilot = SitePilot;
620
760
  exports.Suggestions = Suggestions;
761
+ exports.ToolRegistry = ToolRegistry;
762
+ exports.createFallbackMessageGenerator = createFallbackMessageGenerator;
763
+ exports.createToolRegistry = createToolRegistry;
621
764
  exports.default = SitePilot_default;
765
+ exports.defineSimpleTool = defineSimpleTool;
766
+ exports.defineTool = defineTool;
622
767
  exports.useChat = useChat;
623
768
  exports.useSpeech = useSpeech;
624
769
  //# sourceMappingURL=index.js.map