genassist-chat-react 1.0.0

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.
@@ -0,0 +1,51 @@
1
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { Mic, Loader2 } from 'lucide-react';
3
+ import { useVoiceInput } from '../hooks/useVoiceInput';
4
+ export var VoiceInput = function (_a) {
5
+ var onTranscription = _a.onTranscription, onError = _a.onError, baseUrl = _a.baseUrl, apiKey = _a.apiKey, theme = _a.theme;
6
+ var _b = useVoiceInput({
7
+ baseUrl: baseUrl,
8
+ apiKey: apiKey,
9
+ onTranscription: onTranscription,
10
+ onError: onError,
11
+ }), isRecording = _b.isRecording, isLoading = _b.isLoading, toggleRecording = _b.toggleRecording;
12
+ var handleClick = function (e) {
13
+ e.preventDefault();
14
+ e.stopPropagation();
15
+ toggleRecording();
16
+ };
17
+ var getButtonContent = function () {
18
+ if (isLoading) {
19
+ return (_jsx(Loader2, { size: 20, color: "#757575", style: {
20
+ animation: 'spin 1s linear infinite',
21
+ } }));
22
+ }
23
+ return _jsx(Mic, { size: 20, color: isRecording ? '#ff0000' : '#757575' });
24
+ };
25
+ var getTitle = function () {
26
+ if (isLoading)
27
+ return 'Connecting...';
28
+ return isRecording ? 'Stop Recording' : 'Start Recording';
29
+ };
30
+ var buttonStyle = {
31
+ backgroundColor: '#ffffff',
32
+ color: isRecording ? '#ff0000' : '#757575',
33
+ border: 'none',
34
+ borderRadius: '8px',
35
+ width: '48px',
36
+ height: '48px',
37
+ display: 'flex',
38
+ alignItems: 'center',
39
+ justifyContent: 'center',
40
+ cursor: isLoading ? 'not-allowed' : 'pointer',
41
+ outline: 'none',
42
+ transition: 'all 0.2s ease',
43
+ flexShrink: 0,
44
+ boxShadow: '0 1px 3px rgba(0, 0, 0, 0.05)',
45
+ opacity: isLoading ? 0.7 : 1,
46
+ };
47
+ // Keyframe animation for the spinner
48
+ var spinKeyframes = "\n @keyframes spin {\n from {\n transform: rotate(0deg);\n }\n to {\n transform: rotate(360deg);\n }\n }\n ";
49
+ return (_jsxs(_Fragment, { children: [_jsx("style", { children: spinKeyframes }), _jsx("button", { type: "button", style: buttonStyle, onClick: handleClick, title: getTitle(), disabled: isLoading, children: getButtonContent() })] }));
50
+ };
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVm9pY2VJbnB1dC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL1ZvaWNlSW5wdXQudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFDQSxPQUFPLEVBQUUsR0FBRyxFQUFFLE9BQU8sRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUM1QyxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFjdkQsTUFBTSxDQUFDLElBQU0sVUFBVSxHQUE4QixVQUFDLEVBTXJEO1FBTEMsZUFBZSxxQkFBQSxFQUNmLE9BQU8sYUFBQSxFQUNQLE9BQU8sYUFBQSxFQUNQLE1BQU0sWUFBQSxFQUNOLEtBQUssV0FBQTtJQUVDLElBQUEsS0FBOEMsYUFBYSxDQUFDO1FBQ2hFLE9BQU8sU0FBQTtRQUNQLE1BQU0sUUFBQTtRQUNOLGVBQWUsaUJBQUE7UUFDZixPQUFPLFNBQUE7S0FDUixDQUFDLEVBTE0sV0FBVyxpQkFBQSxFQUFFLFNBQVMsZUFBQSxFQUFFLGVBQWUscUJBSzdDLENBQUM7SUFFSCxJQUFNLFdBQVcsR0FBRyxVQUFDLENBQW1CO1FBQ3RDLENBQUMsQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUNuQixDQUFDLENBQUMsZUFBZSxFQUFFLENBQUM7UUFDcEIsZUFBZSxFQUFFLENBQUM7SUFDcEIsQ0FBQyxDQUFDO0lBRUYsSUFBTSxnQkFBZ0IsR0FBRztRQUN2QixJQUFJLFNBQVMsRUFBRSxDQUFDO1lBQ2QsT0FBTyxDQUNMLEtBQUMsT0FBTyxJQUNOLElBQUksRUFBRSxFQUFFLEVBQ1IsS0FBSyxFQUFDLFNBQVMsRUFDZixLQUFLLEVBQUU7b0JBQ0wsU0FBUyxFQUFFLHlCQUF5QjtpQkFDckMsR0FDRCxDQUNILENBQUM7UUFDSixDQUFDO1FBQ0QsT0FBTyxLQUFDLEdBQUcsSUFBQyxJQUFJLEVBQUUsRUFBRSxFQUFFLEtBQUssRUFBRSxXQUFXLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsU0FBUyxHQUFJLENBQUM7SUFDdkUsQ0FBQyxDQUFDO0lBRUYsSUFBTSxRQUFRLEdBQUc7UUFDZixJQUFJLFNBQVM7WUFBRSxPQUFPLGVBQWUsQ0FBQztRQUN0QyxPQUFPLFdBQVcsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixDQUFDO0lBQzVELENBQUMsQ0FBQztJQUVGLElBQU0sV0FBVyxHQUF3QjtRQUN2QyxlQUFlLEVBQUUsU0FBUztRQUMxQixLQUFLLEVBQUUsV0FBVyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLFNBQVM7UUFDMUMsTUFBTSxFQUFFLE1BQU07UUFDZCxZQUFZLEVBQUUsS0FBSztRQUNuQixLQUFLLEVBQUUsTUFBTTtRQUNiLE1BQU0sRUFBRSxNQUFNO1FBQ2QsT0FBTyxFQUFFLE1BQU07UUFDZixVQUFVLEVBQUUsUUFBUTtRQUNwQixjQUFjLEVBQUUsUUFBUTtRQUN4QixNQUFNLEVBQUUsU0FBUyxDQUFDLENBQUMsQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLFNBQVM7UUFDN0MsT0FBTyxFQUFFLE1BQU07UUFDZixVQUFVLEVBQUUsZUFBZTtRQUMzQixVQUFVLEVBQUUsQ0FBQztRQUNiLFNBQVMsRUFBRSwrQkFBK0I7UUFDMUMsT0FBTyxFQUFFLFNBQVMsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO0tBQzdCLENBQUM7SUFFRixxQ0FBcUM7SUFDckMsSUFBTSxhQUFhLEdBQUcsc0pBU3JCLENBQUM7SUFFRixPQUFPLENBQ0wsOEJBQ0UsMEJBQVEsYUFBYSxHQUFTLEVBQzlCLGlCQUNFLElBQUksRUFBQyxRQUFRLEVBQ2IsS0FBSyxFQUFFLFdBQVcsRUFDbEIsT0FBTyxFQUFFLFdBQVcsRUFDcEIsS0FBSyxFQUFFLFFBQVEsRUFBRSxFQUNqQixRQUFRLEVBQUUsU0FBUyxZQUVsQixnQkFBZ0IsRUFBRSxHQUNaLElBQ1IsQ0FDSixDQUFDO0FBQ0osQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcbmltcG9ydCB7IE1pYywgTG9hZGVyMiB9IGZyb20gJ2x1Y2lkZS1yZWFjdCc7XG5pbXBvcnQgeyB1c2VWb2ljZUlucHV0IH0gZnJvbSAnLi4vaG9va3MvdXNlVm9pY2VJbnB1dCc7XG5cbmludGVyZmFjZSBWb2ljZUlucHV0UHJvcHMge1xuICBvblRyYW5zY3JpcHRpb246ICh0ZXh0OiBzdHJpbmcpID0+IHZvaWQ7XG4gIG9uRXJyb3I6IChlcnJvcjogRXJyb3IpID0+IHZvaWQ7XG4gIGJhc2VVcmw6IHN0cmluZztcbiAgYXBpS2V5OiBzdHJpbmc7XG4gIHRoZW1lPzoge1xuICAgIHByaW1hcnlDb2xvcj86IHN0cmluZztcbiAgICBiYWNrZ3JvdW5kQ29sb3I/OiBzdHJpbmc7XG4gICAgZm9udEZhbWlseT86IHN0cmluZztcbiAgfTtcbn1cblxuZXhwb3J0IGNvbnN0IFZvaWNlSW5wdXQ6IFJlYWN0LkZDPFZvaWNlSW5wdXRQcm9wcz4gPSAoe1xuICBvblRyYW5zY3JpcHRpb24sXG4gIG9uRXJyb3IsXG4gIGJhc2VVcmwsXG4gIGFwaUtleSxcbiAgdGhlbWVcbn0pID0+IHtcbiAgY29uc3QgeyBpc1JlY29yZGluZywgaXNMb2FkaW5nLCB0b2dnbGVSZWNvcmRpbmcgfSA9IHVzZVZvaWNlSW5wdXQoe1xuICAgIGJhc2VVcmwsXG4gICAgYXBpS2V5LFxuICAgIG9uVHJhbnNjcmlwdGlvbixcbiAgICBvbkVycm9yLFxuICB9KTtcblxuICBjb25zdCBoYW5kbGVDbGljayA9IChlOiBSZWFjdC5Nb3VzZUV2ZW50KSA9PiB7XG4gICAgZS5wcmV2ZW50RGVmYXVsdCgpO1xuICAgIGUuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgdG9nZ2xlUmVjb3JkaW5nKCk7XG4gIH07XG5cbiAgY29uc3QgZ2V0QnV0dG9uQ29udGVudCA9ICgpID0+IHtcbiAgICBpZiAoaXNMb2FkaW5nKSB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICA8TG9hZGVyMiBcbiAgICAgICAgICBzaXplPXsyMH0gXG4gICAgICAgICAgY29sb3I9XCIjNzU3NTc1XCIgXG4gICAgICAgICAgc3R5bGU9e3tcbiAgICAgICAgICAgIGFuaW1hdGlvbjogJ3NwaW4gMXMgbGluZWFyIGluZmluaXRlJyxcbiAgICAgICAgICB9fVxuICAgICAgICAvPlxuICAgICAgKTtcbiAgICB9XG4gICAgcmV0dXJuIDxNaWMgc2l6ZT17MjB9IGNvbG9yPXtpc1JlY29yZGluZyA/ICcjZmYwMDAwJyA6ICcjNzU3NTc1J30gLz47XG4gIH07XG5cbiAgY29uc3QgZ2V0VGl0bGUgPSAoKSA9PiB7XG4gICAgaWYgKGlzTG9hZGluZykgcmV0dXJuICdDb25uZWN0aW5nLi4uJztcbiAgICByZXR1cm4gaXNSZWNvcmRpbmcgPyAnU3RvcCBSZWNvcmRpbmcnIDogJ1N0YXJ0IFJlY29yZGluZyc7XG4gIH07XG5cbiAgY29uc3QgYnV0dG9uU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgYmFja2dyb3VuZENvbG9yOiAnI2ZmZmZmZicsXG4gICAgY29sb3I6IGlzUmVjb3JkaW5nID8gJyNmZjAwMDAnIDogJyM3NTc1NzUnLFxuICAgIGJvcmRlcjogJ25vbmUnLFxuICAgIGJvcmRlclJhZGl1czogJzhweCcsXG4gICAgd2lkdGg6ICc0OHB4JyxcbiAgICBoZWlnaHQ6ICc0OHB4JyxcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAganVzdGlmeUNvbnRlbnQ6ICdjZW50ZXInLFxuICAgIGN1cnNvcjogaXNMb2FkaW5nID8gJ25vdC1hbGxvd2VkJyA6ICdwb2ludGVyJyxcbiAgICBvdXRsaW5lOiAnbm9uZScsXG4gICAgdHJhbnNpdGlvbjogJ2FsbCAwLjJzIGVhc2UnLFxuICAgIGZsZXhTaHJpbms6IDAsXG4gICAgYm94U2hhZG93OiAnMCAxcHggM3B4IHJnYmEoMCwgMCwgMCwgMC4wNSknLFxuICAgIG9wYWNpdHk6IGlzTG9hZGluZyA/IDAuNyA6IDEsXG4gIH07XG5cbiAgLy8gS2V5ZnJhbWUgYW5pbWF0aW9uIGZvciB0aGUgc3Bpbm5lclxuICBjb25zdCBzcGluS2V5ZnJhbWVzID0gYFxuICAgIEBrZXlmcmFtZXMgc3BpbiB7XG4gICAgICBmcm9tIHtcbiAgICAgICAgdHJhbnNmb3JtOiByb3RhdGUoMGRlZyk7XG4gICAgICB9XG4gICAgICB0byB7XG4gICAgICAgIHRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7XG4gICAgICB9XG4gICAgfVxuICBgO1xuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIDxzdHlsZT57c3BpbktleWZyYW1lc308L3N0eWxlPlxuICAgICAgPGJ1dHRvblxuICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgc3R5bGU9e2J1dHRvblN0eWxlfVxuICAgICAgICBvbkNsaWNrPXtoYW5kbGVDbGlja31cbiAgICAgICAgdGl0bGU9e2dldFRpdGxlKCl9XG4gICAgICAgIGRpc2FibGVkPXtpc0xvYWRpbmd9XG4gICAgICA+XG4gICAgICAgIHtnZXRCdXR0b25Db250ZW50KCl9XG4gICAgICA8L2J1dHRvbj5cbiAgICA8Lz5cbiAgKTtcbn07ICJdfQ==
@@ -0,0 +1,21 @@
1
+ import { ChatMessage } from '../types';
2
+ export interface UseChatProps {
3
+ baseUrl: string;
4
+ apiKey: string;
5
+ metadata?: Record<string, any>;
6
+ onError?: (error: Error) => void;
7
+ onTakeover?: () => void;
8
+ onFinalize?: () => void;
9
+ }
10
+ export declare const useChat: ({ baseUrl, apiKey, metadata, onError, onTakeover, onFinalize }: UseChatProps) => {
11
+ messages: ChatMessage[];
12
+ isLoading: boolean;
13
+ sendMessage: (text: string) => Promise<void>;
14
+ resetConversation: () => Promise<void>;
15
+ startConversation: () => Promise<void>;
16
+ connectionState: "connecting" | "connected" | "disconnected";
17
+ conversationId: string | null;
18
+ possibleQueries: string[];
19
+ isTakenOver: boolean;
20
+ isFinalized: boolean;
21
+ };
@@ -0,0 +1,256 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ var __generator = (this && this.__generator) || function (thisArg, body) {
11
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
12
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
13
+ function verb(n) { return function (v) { return step([n, v]); }; }
14
+ function step(op) {
15
+ if (f) throw new TypeError("Generator is already executing.");
16
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
17
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
18
+ if (y = 0, t) op = [op[0] & 2, t.value];
19
+ switch (op[0]) {
20
+ case 0: case 1: t = op; break;
21
+ case 4: _.label++; return { value: op[1], done: false };
22
+ case 5: _.label++; y = op[1]; op = [0]; continue;
23
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
24
+ default:
25
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
26
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
27
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
28
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
29
+ if (t[2]) _.ops.pop();
30
+ _.trys.pop(); continue;
31
+ }
32
+ op = body.call(thisArg, _);
33
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
34
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
+ }
36
+ };
37
+ var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
38
+ if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
39
+ if (ar || !(i in from)) {
40
+ if (!ar) ar = Array.prototype.slice.call(from, 0, i);
41
+ ar[i] = from[i];
42
+ }
43
+ }
44
+ return to.concat(ar || Array.prototype.slice.call(from));
45
+ };
46
+ import { useState, useEffect, useCallback, useRef } from 'react';
47
+ import { ChatService } from '../services/chatService';
48
+ export var useChat = function (_a) {
49
+ var baseUrl = _a.baseUrl, apiKey = _a.apiKey, metadata = _a.metadata, onError = _a.onError, onTakeover = _a.onTakeover, onFinalize = _a.onFinalize;
50
+ var _b = useState(function () {
51
+ try {
52
+ var storedMessages = localStorage.getItem('chatMessages');
53
+ return storedMessages ? JSON.parse(storedMessages) : [];
54
+ }
55
+ catch (error) {
56
+ console.error('Error loading messages from localStorage:', error);
57
+ return [];
58
+ }
59
+ }), messages = _b[0], setMessages = _b[1];
60
+ var _c = useState(false), isLoading = _c[0], setIsLoading = _c[1];
61
+ var _d = useState('disconnected'), connectionState = _d[0], setConnectionState = _d[1];
62
+ var chatServiceRef = useRef(null);
63
+ var _e = useState(null), conversationId = _e[0], setConversationId = _e[1];
64
+ var _f = useState([]), possibleQueries = _f[0], setPossibleQueries = _f[1];
65
+ var _g = useState(false), isTakenOver = _g[0], setIsTakenOver = _g[1];
66
+ var _h = useState(false), isFinalized = _h[0], setIsFinalized = _h[1];
67
+ // Initialize chat service
68
+ useEffect(function () {
69
+ chatServiceRef.current = new ChatService(baseUrl, apiKey, metadata);
70
+ chatServiceRef.current.setMessageHandler(function (message) {
71
+ setMessages(function (prevMessages) { return __spreadArray(__spreadArray([], prevMessages, true), [message], false); });
72
+ });
73
+ chatServiceRef.current.setTakeoverHandler(function () {
74
+ setIsTakenOver(true);
75
+ if (onTakeover) {
76
+ onTakeover();
77
+ }
78
+ });
79
+ chatServiceRef.current.setFinalizedHandler(function () {
80
+ setIsFinalized(true);
81
+ if (onFinalize) {
82
+ onFinalize();
83
+ }
84
+ });
85
+ chatServiceRef.current.setConnectionStateHandler(setConnectionState);
86
+ // Check for a saved conversation and connect to it
87
+ var convId = chatServiceRef.current.getConversationId();
88
+ if (convId) {
89
+ setConversationId(convId);
90
+ if (chatServiceRef.current.isConversationFinalized()) {
91
+ setIsFinalized(true);
92
+ }
93
+ else {
94
+ chatServiceRef.current.connectWebSocket();
95
+ }
96
+ }
97
+ // Cleanup
98
+ return function () {
99
+ if (chatServiceRef.current) {
100
+ chatServiceRef.current.disconnect();
101
+ }
102
+ };
103
+ }, [baseUrl, apiKey, metadata, onError, onTakeover, onFinalize]);
104
+ useEffect(function () {
105
+ try {
106
+ localStorage.setItem('chatMessages', JSON.stringify(messages));
107
+ }
108
+ catch (error) {
109
+ console.error('Error saving messages to localStorage:', error);
110
+ }
111
+ }, [messages]);
112
+ // Reset conversation
113
+ var resetConversation = useCallback(function () { return __awaiter(void 0, void 0, void 0, function () {
114
+ var convId, queries, error_1;
115
+ return __generator(this, function (_a) {
116
+ switch (_a.label) {
117
+ case 0:
118
+ if (!chatServiceRef.current) {
119
+ return [2 /*return*/];
120
+ }
121
+ setConnectionState('connecting');
122
+ setIsLoading(true);
123
+ setMessages([]);
124
+ setPossibleQueries([]);
125
+ localStorage.removeItem('chatMessages');
126
+ setIsFinalized(false);
127
+ _a.label = 1;
128
+ case 1:
129
+ _a.trys.push([1, 3, 4, 5]);
130
+ // Reset the conversation in the chat service
131
+ chatServiceRef.current.resetConversation();
132
+ return [4 /*yield*/, chatServiceRef.current.startConversation()];
133
+ case 2:
134
+ convId = _a.sent();
135
+ setConversationId(convId);
136
+ setConnectionState('connected');
137
+ // Get possible queries from API response
138
+ if (chatServiceRef.current.getPossibleQueries) {
139
+ queries = chatServiceRef.current.getPossibleQueries();
140
+ if (queries && queries.length > 0) {
141
+ setPossibleQueries(queries);
142
+ }
143
+ }
144
+ return [3 /*break*/, 5];
145
+ case 3:
146
+ error_1 = _a.sent();
147
+ setConnectionState('disconnected');
148
+ if (onError && error_1 instanceof Error) {
149
+ onError(error_1);
150
+ }
151
+ else {
152
+ console.error('Failed to reset conversation:', error_1);
153
+ }
154
+ return [3 /*break*/, 5];
155
+ case 4:
156
+ setIsLoading(false);
157
+ return [7 /*endfinally*/];
158
+ case 5: return [2 /*return*/];
159
+ }
160
+ });
161
+ }); }, [onError]);
162
+ // Send message
163
+ var sendMessage = useCallback(function (text) { return __awaiter(void 0, void 0, void 0, function () {
164
+ var error_2;
165
+ return __generator(this, function (_a) {
166
+ switch (_a.label) {
167
+ case 0:
168
+ if (!chatServiceRef.current) {
169
+ throw new Error('Chat service not initialized');
170
+ }
171
+ _a.label = 1;
172
+ case 1:
173
+ _a.trys.push([1, 3, 4, 5]);
174
+ setIsLoading(true);
175
+ return [4 /*yield*/, chatServiceRef.current.sendMessage(text)];
176
+ case 2:
177
+ _a.sent();
178
+ return [3 /*break*/, 5];
179
+ case 3:
180
+ error_2 = _a.sent();
181
+ if (onError && error_2 instanceof Error) {
182
+ onError(error_2);
183
+ }
184
+ else {
185
+ console.error('Failed to send message:', error_2);
186
+ }
187
+ return [3 /*break*/, 5];
188
+ case 4:
189
+ setIsLoading(false);
190
+ return [7 /*endfinally*/];
191
+ case 5: return [2 /*return*/];
192
+ }
193
+ });
194
+ }); }, [onError]);
195
+ var startConversation = useCallback(function () { return __awaiter(void 0, void 0, void 0, function () {
196
+ var convId, queries, error_3;
197
+ return __generator(this, function (_a) {
198
+ switch (_a.label) {
199
+ case 0:
200
+ if (!chatServiceRef.current) {
201
+ return [2 /*return*/];
202
+ }
203
+ _a.label = 1;
204
+ case 1:
205
+ _a.trys.push([1, 3, 4, 5]);
206
+ setConnectionState('connecting');
207
+ setIsLoading(true);
208
+ // Reset state for new conversation
209
+ setMessages([]);
210
+ setPossibleQueries([]);
211
+ localStorage.removeItem('chatMessages');
212
+ setIsFinalized(false);
213
+ chatServiceRef.current.resetConversation();
214
+ return [4 /*yield*/, chatServiceRef.current.startConversation()];
215
+ case 2:
216
+ convId = _a.sent();
217
+ setConversationId(convId);
218
+ setConnectionState('connected');
219
+ if (chatServiceRef.current.getPossibleQueries) {
220
+ queries = chatServiceRef.current.getPossibleQueries();
221
+ if (queries && queries.length > 0) {
222
+ setPossibleQueries(queries);
223
+ }
224
+ }
225
+ return [3 /*break*/, 5];
226
+ case 3:
227
+ error_3 = _a.sent();
228
+ setConnectionState('disconnected');
229
+ if (onError && error_3 instanceof Error) {
230
+ onError(error_3);
231
+ }
232
+ else {
233
+ console.error('Failed to start conversation:', error_3);
234
+ }
235
+ return [3 /*break*/, 5];
236
+ case 4:
237
+ setIsLoading(false);
238
+ return [7 /*endfinally*/];
239
+ case 5: return [2 /*return*/];
240
+ }
241
+ });
242
+ }); }, [onError]);
243
+ return {
244
+ messages: messages,
245
+ isLoading: isLoading,
246
+ sendMessage: sendMessage,
247
+ resetConversation: resetConversation,
248
+ startConversation: startConversation,
249
+ connectionState: connectionState,
250
+ conversationId: conversationId,
251
+ possibleQueries: possibleQueries,
252
+ isTakenOver: isTakenOver,
253
+ isFinalized: isFinalized
254
+ };
255
+ };
256
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"useChat.js","sourceRoot":"","sources":["../../src/hooks/useChat.ts"],"names":[],"mappings":"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","sourcesContent":["import { useState, useEffect, useCallback, useRef } from 'react';\nimport { ChatService } from '../services/chatService';\nimport { ChatMessage } from '../types';\n\nexport interface UseChatProps {\n  baseUrl: string;\n  apiKey: string;\n  metadata?: Record<string, any>;\n  onError?: (error: Error) => void;\n  onTakeover?: () => void;\n  onFinalize?: () => void;\n}\n\nexport const useChat = ({ baseUrl, apiKey, metadata, onError, onTakeover, onFinalize }: UseChatProps) => {\n  const [messages, setMessages] = useState<ChatMessage[]>(() => {\n    try {\n      const storedMessages = localStorage.getItem('chatMessages');\n      return storedMessages ? JSON.parse(storedMessages) : [];\n    } catch (error) {\n      console.error('Error loading messages from localStorage:', error);\n      return [];\n    }\n  });\n  const [isLoading, setIsLoading] = useState<boolean>(false);\n  const [connectionState, setConnectionState] = useState<'connecting' | 'connected' | 'disconnected'>('disconnected');\n  const chatServiceRef = useRef<ChatService | null>(null);\n  const [conversationId, setConversationId] = useState<string | null>(null);\n  const [possibleQueries, setPossibleQueries] = useState<string[]>([]);\n  const [isTakenOver, setIsTakenOver] = useState<boolean>(false);\n  const [isFinalized, setIsFinalized] = useState<boolean>(false);\n\n  // Initialize chat service\n  useEffect(() => {\n    chatServiceRef.current = new ChatService(baseUrl, apiKey, metadata);\n    \n    chatServiceRef.current.setMessageHandler((message: ChatMessage) => {\n      setMessages(prevMessages => [...prevMessages, message]);\n    });\n\n    chatServiceRef.current.setTakeoverHandler(() => {\n      setIsTakenOver(true);\n      if (onTakeover) {\n        onTakeover();\n      }\n    });\n\n    chatServiceRef.current.setFinalizedHandler(() => {\n      setIsFinalized(true);\n      if (onFinalize) {\n        onFinalize();\n      }\n    });\n\n    chatServiceRef.current.setConnectionStateHandler(setConnectionState);\n\n    // Check for a saved conversation and connect to it\n    const convId = chatServiceRef.current.getConversationId();\n    if (convId) {\n        setConversationId(convId);\n        if (chatServiceRef.current.isConversationFinalized()) {\n          setIsFinalized(true);\n        } else {\n          chatServiceRef.current.connectWebSocket();\n        }\n    }\n\n    // Cleanup\n    return () => {\n      if (chatServiceRef.current) {\n        chatServiceRef.current.disconnect();\n      }\n    };\n  }, [baseUrl, apiKey, metadata, onError, onTakeover, onFinalize]);\n\n  useEffect(() => {\n    try {\n      localStorage.setItem('chatMessages', JSON.stringify(messages));\n    } catch (error) {\n      console.error('Error saving messages to localStorage:', error);\n    }\n  }, [messages]);\n\n  // Reset conversation\n  const resetConversation = useCallback(async () => {\n    if (!chatServiceRef.current) {\n      return;\n    }\n    \n    setConnectionState('connecting');\n    setIsLoading(true);\n    setMessages([]);\n    setPossibleQueries([]);\n    localStorage.removeItem('chatMessages');\n    setIsFinalized(false);\n    \n    try {\n      // Reset the conversation in the chat service\n      chatServiceRef.current.resetConversation();\n      \n      // Start a new conversation\n      const convId = await chatServiceRef.current.startConversation();\n      setConversationId(convId);\n      setConnectionState('connected');\n\n      // Get possible queries from API response\n      if (chatServiceRef.current.getPossibleQueries) {\n        const queries = chatServiceRef.current.getPossibleQueries();\n        if (queries && queries.length > 0) {\n          setPossibleQueries(queries);\n        }\n      }\n    } catch (error) {\n      setConnectionState('disconnected');\n      if (onError && error instanceof Error) {\n        onError(error);\n      } else {\n        console.error('Failed to reset conversation:', error);\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, [onError]);\n\n  // Send message\n  const sendMessage = useCallback(async (text: string) => {\n    if (!chatServiceRef.current) {\n      throw new Error('Chat service not initialized');\n    }\n\n    try {\n      setIsLoading(true);\n      await chatServiceRef.current.sendMessage(text);\n    } catch (error) {\n      if (onError && error instanceof Error) {\n        onError(error);\n      } else {\n        console.error('Failed to send message:', error);\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, [onError]);\n\n  const startConversation = useCallback(async () => {\n    if (!chatServiceRef.current) {\n      return;\n    }\n    try {\n      setConnectionState('connecting');\n      setIsLoading(true);\n      \n      // Reset state for new conversation\n      setMessages([]);\n      setPossibleQueries([]);\n      localStorage.removeItem('chatMessages');\n      setIsFinalized(false);\n      chatServiceRef.current.resetConversation();\n\n      const convId = await chatServiceRef.current.startConversation();\n      setConversationId(convId);\n      setConnectionState('connected');\n\n      if (chatServiceRef.current.getPossibleQueries) {\n        const queries = chatServiceRef.current.getPossibleQueries();\n        if (queries && queries.length > 0) {\n          setPossibleQueries(queries);\n        }\n      }\n    } catch (error) {\n      setConnectionState('disconnected');\n      if (onError && error instanceof Error) {\n        onError(error);\n      } else {\n        console.error('Failed to start conversation:', error);\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, [onError]);\n\n  return {\n    messages,\n    isLoading,\n    sendMessage,\n    resetConversation,\n    startConversation,\n    connectionState,\n    conversationId,\n    possibleQueries,\n    isTakenOver,\n    isFinalized\n  };\n}; "]}
@@ -0,0 +1,16 @@
1
+ interface UseVoiceInputProps {
2
+ baseUrl: string;
3
+ apiKey: string;
4
+ onTranscription: (text: string) => void;
5
+ onError: (error: Error) => void;
6
+ }
7
+ interface UseVoiceInputReturn {
8
+ isRecording: boolean;
9
+ isLoading: boolean;
10
+ isSessionActive: boolean;
11
+ startSession: () => Promise<void>;
12
+ stopSession: () => void;
13
+ toggleRecording: () => void;
14
+ }
15
+ export declare const useVoiceInput: ({ baseUrl, apiKey, onTranscription, onError, }: UseVoiceInputProps) => UseVoiceInputReturn;
16
+ export {};
@@ -0,0 +1,210 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ var __generator = (this && this.__generator) || function (thisArg, body) {
11
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
12
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
13
+ function verb(n) { return function (v) { return step([n, v]); }; }
14
+ function step(op) {
15
+ if (f) throw new TypeError("Generator is already executing.");
16
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
17
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
18
+ if (y = 0, t) op = [op[0] & 2, t.value];
19
+ switch (op[0]) {
20
+ case 0: case 1: t = op; break;
21
+ case 4: _.label++; return { value: op[1], done: false };
22
+ case 5: _.label++; y = op[1]; op = [0]; continue;
23
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
24
+ default:
25
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
26
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
27
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
28
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
29
+ if (t[2]) _.ops.pop();
30
+ _.trys.pop(); continue;
31
+ }
32
+ op = body.call(thisArg, _);
33
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
34
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
+ }
36
+ };
37
+ import { useRef, useState, useEffect, useCallback } from 'react';
38
+ export var useVoiceInput = function (_a) {
39
+ var baseUrl = _a.baseUrl, apiKey = _a.apiKey, onTranscription = _a.onTranscription, onError = _a.onError;
40
+ var _b = useState(false), isRecording = _b[0], setIsRecording = _b[1];
41
+ var _c = useState(false), isLoading = _c[0], setIsLoading = _c[1];
42
+ var _d = useState(false), isSessionActive = _d[0], setIsSessionActive = _d[1];
43
+ var peerConnection = useRef(null);
44
+ var dataChannel = useRef(null);
45
+ var audioElement = useRef(null);
46
+ var cleanup = useCallback(function () {
47
+ if (dataChannel.current) {
48
+ dataChannel.current.close();
49
+ dataChannel.current = null;
50
+ }
51
+ if (peerConnection.current) {
52
+ peerConnection.current.getSenders().forEach(function (sender) {
53
+ if (sender.track) {
54
+ sender.track.stop();
55
+ }
56
+ });
57
+ peerConnection.current.close();
58
+ peerConnection.current = null;
59
+ }
60
+ if (audioElement.current) {
61
+ audioElement.current.srcObject = null;
62
+ audioElement.current = null;
63
+ }
64
+ setIsRecording(false);
65
+ setIsSessionActive(false);
66
+ setIsLoading(false);
67
+ }, []);
68
+ var handleDataChannelMessage = useCallback(function (event) {
69
+ var data = JSON.parse(event.data);
70
+ // Handle conversation item created events which may contain transcripts
71
+ if (data.type === 'conversation.item.created' && data.item) {
72
+ // Check if the item has content with transcript
73
+ if (data.item.content && Array.isArray(data.item.content)) {
74
+ for (var _i = 0, _a = data.item.content; _i < _a.length; _i++) {
75
+ var contentItem = _a[_i];
76
+ if (contentItem.type === 'input_audio' && contentItem.transcript) {
77
+ onTranscription(contentItem.transcript);
78
+ return;
79
+ }
80
+ }
81
+ }
82
+ // Check if the item itself has a transcript field
83
+ if (data.item.transcript) {
84
+ onTranscription(data.item.transcript);
85
+ return;
86
+ }
87
+ }
88
+ // Handle input audio transcription completed events
89
+ if (data.type === 'conversation.item.input_audio_transcription.completed' && data.transcript) {
90
+ onTranscription(data.transcript);
91
+ }
92
+ // Handle response audio transcript events
93
+ if (data.type === 'response.audio_transcript.done' && data.transcript) {
94
+ onTranscription(data.transcript);
95
+ }
96
+ // Check for any other events with transcript field
97
+ if (data.transcript && !data.type.includes('error')) {
98
+ onTranscription(data.transcript);
99
+ }
100
+ }, [onTranscription]);
101
+ var startSession = useCallback(function () { return __awaiter(void 0, void 0, void 0, function () {
102
+ var tokenResponse, ephemeralKey, pc, mediaStream, dc, offer, response, answer, error_1;
103
+ var _a;
104
+ return __generator(this, function (_b) {
105
+ switch (_b.label) {
106
+ case 0:
107
+ setIsLoading(true);
108
+ _b.label = 1;
109
+ case 1:
110
+ _b.trys.push([1, 10, , 11]);
111
+ return [4 /*yield*/, fetch("".concat(baseUrl, "api/voice/openai/session"), {
112
+ headers: {
113
+ 'x-api-key': apiKey
114
+ }
115
+ })];
116
+ case 2:
117
+ tokenResponse = _b.sent();
118
+ if (!tokenResponse.ok) {
119
+ throw new Error('Failed to get ephemeral API key');
120
+ }
121
+ return [4 /*yield*/, tokenResponse.json()];
122
+ case 3:
123
+ ephemeralKey = _b.sent();
124
+ pc = new RTCPeerConnection();
125
+ // Set up to play remote audio from the model
126
+ audioElement.current = document.createElement("audio");
127
+ audioElement.current.autoplay = true;
128
+ pc.ontrack = function (e) {
129
+ if (audioElement.current) {
130
+ audioElement.current.srcObject = e.streams[0];
131
+ }
132
+ };
133
+ return [4 /*yield*/, navigator.mediaDevices.getUserMedia({
134
+ audio: true,
135
+ })];
136
+ case 4:
137
+ mediaStream = _b.sent();
138
+ pc.addTrack(mediaStream.getTracks()[0]);
139
+ dc = pc.createDataChannel("oai-events");
140
+ dataChannel.current = dc;
141
+ dc.onmessage = handleDataChannelMessage;
142
+ dc.onopen = function () {
143
+ setIsSessionActive(true);
144
+ setIsLoading(false);
145
+ };
146
+ return [4 /*yield*/, pc.createOffer()];
147
+ case 5:
148
+ offer = _b.sent();
149
+ return [4 /*yield*/, pc.setLocalDescription(offer)];
150
+ case 6:
151
+ _b.sent();
152
+ return [4 /*yield*/, fetch('https://api.openai.com/v1/realtime', {
153
+ method: 'POST',
154
+ headers: {
155
+ 'Authorization': "Bearer ".concat(ephemeralKey),
156
+ 'Content-Type': 'application/sdp',
157
+ },
158
+ body: offer.sdp,
159
+ })];
160
+ case 7:
161
+ response = _b.sent();
162
+ _a = {
163
+ type: 'answer'
164
+ };
165
+ return [4 /*yield*/, response.text()];
166
+ case 8:
167
+ answer = (_a.sdp = _b.sent(),
168
+ _a);
169
+ return [4 /*yield*/, pc.setRemoteDescription(new RTCSessionDescription(answer))];
170
+ case 9:
171
+ _b.sent();
172
+ peerConnection.current = pc;
173
+ setIsRecording(true);
174
+ return [3 /*break*/, 11];
175
+ case 10:
176
+ error_1 = _b.sent();
177
+ onError(error_1);
178
+ cleanup();
179
+ return [3 /*break*/, 11];
180
+ case 11: return [2 /*return*/];
181
+ }
182
+ });
183
+ }); }, [baseUrl, apiKey, handleDataChannelMessage, onError, cleanup]);
184
+ var stopSession = useCallback(function () {
185
+ cleanup();
186
+ }, [cleanup]);
187
+ var toggleRecording = useCallback(function () {
188
+ if (isLoading)
189
+ return;
190
+ if (isRecording) {
191
+ stopSession();
192
+ }
193
+ else {
194
+ startSession();
195
+ }
196
+ }, [isLoading, isRecording, startSession, stopSession]);
197
+ // Cleanup on component unmount
198
+ useEffect(function () {
199
+ return cleanup;
200
+ }, [cleanup]);
201
+ return {
202
+ isRecording: isRecording,
203
+ isLoading: isLoading,
204
+ isSessionActive: isSessionActive,
205
+ startSession: startSession,
206
+ stopSession: stopSession,
207
+ toggleRecording: toggleRecording,
208
+ };
209
+ };
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"useVoiceInput.js","sourceRoot":"","sources":["../../src/hooks/useVoiceInput.ts"],"names":[],"mappings":"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","sourcesContent":["import { useRef, useState, useEffect, useCallback } from 'react';\n\ninterface UseVoiceInputProps {\n  baseUrl: string;\n  apiKey: string;\n  onTranscription: (text: string) => void;\n  onError: (error: Error) => void;\n}\n\ninterface UseVoiceInputReturn {\n  isRecording: boolean;\n  isLoading: boolean;\n  isSessionActive: boolean;\n  startSession: () => Promise<void>;\n  stopSession: () => void;\n  toggleRecording: () => void;\n}\n\nexport const useVoiceInput = ({\n  baseUrl,\n  apiKey,\n  onTranscription,\n  onError,\n}: UseVoiceInputProps): UseVoiceInputReturn => {\n  const [isRecording, setIsRecording] = useState(false);\n  const [isLoading, setIsLoading] = useState(false);\n  const [isSessionActive, setIsSessionActive] = useState(false);\n  \n  const peerConnection = useRef<RTCPeerConnection | null>(null);\n  const dataChannel = useRef<RTCDataChannel | null>(null);\n  const audioElement = useRef<HTMLAudioElement | null>(null);\n\n  const cleanup = useCallback(() => {\n    if (dataChannel.current) {\n      dataChannel.current.close();\n      dataChannel.current = null;\n    }\n\n    if (peerConnection.current) {\n      peerConnection.current.getSenders().forEach((sender) => {\n        if (sender.track) {\n          sender.track.stop();\n        }\n      });\n      peerConnection.current.close();\n      peerConnection.current = null;\n    }\n\n    if (audioElement.current) {\n      audioElement.current.srcObject = null;\n      audioElement.current = null;\n    }\n\n    setIsRecording(false);\n    setIsSessionActive(false);\n    setIsLoading(false);\n  }, []);\n\n  const handleDataChannelMessage = useCallback((event: MessageEvent) => {\n    const data = JSON.parse(event.data);\n    \n    // Handle conversation item created events which may contain transcripts\n    if (data.type === 'conversation.item.created' && data.item) {\n      // Check if the item has content with transcript\n      if (data.item.content && Array.isArray(data.item.content)) {\n        for (const contentItem of data.item.content) {\n          if (contentItem.type === 'input_audio' && contentItem.transcript) {\n            onTranscription(contentItem.transcript);\n            return;\n          }\n        }\n      }\n      \n      // Check if the item itself has a transcript field\n      if (data.item.transcript) {\n        onTranscription(data.item.transcript);\n        return;\n      }\n    }\n    \n    // Handle input audio transcription completed events\n    if (data.type === 'conversation.item.input_audio_transcription.completed' && data.transcript) {\n      onTranscription(data.transcript);\n    }\n    \n    // Handle response audio transcript events\n    if (data.type === 'response.audio_transcript.done' && data.transcript) {\n      onTranscription(data.transcript);\n    }\n    \n    // Check for any other events with transcript field\n    if (data.transcript && !data.type.includes('error')) {\n      onTranscription(data.transcript);\n    }\n  }, [onTranscription]);\n\n  const startSession = useCallback(async () => {\n    setIsLoading(true);\n    try {\n      // Get ephemeral API key from backend\n      const tokenResponse = await fetch(`${baseUrl}api/voice/openai/session`, {\n        headers: {\n          'x-api-key': apiKey\n        }\n      });\n      \n      if (!tokenResponse.ok) {\n        throw new Error('Failed to get ephemeral API key');\n      }\n      \n      const ephemeralKey = await tokenResponse.json();\n\n      // Create a peer connection\n      const pc = new RTCPeerConnection();\n\n      // Set up to play remote audio from the model\n      audioElement.current = document.createElement(\"audio\");\n      audioElement.current.autoplay = true;\n      pc.ontrack = (e) => {\n        if (audioElement.current) {\n          audioElement.current.srcObject = e.streams[0];\n        }\n      };\n\n      // Add local audio track for microphone input\n      const mediaStream = await navigator.mediaDevices.getUserMedia({\n        audio: true,\n      });\n      pc.addTrack(mediaStream.getTracks()[0]);\n\n      // Set up data channel for sending and receiving events\n      const dc = pc.createDataChannel(\"oai-events\");\n      dataChannel.current = dc;\n\n      dc.onmessage = handleDataChannelMessage;\n      \n      dc.onopen = () => {\n        setIsSessionActive(true);\n        setIsLoading(false);\n      };\n\n      // Start the session using SDP\n      const offer = await pc.createOffer();\n      await pc.setLocalDescription(offer);\n\n      const response = await fetch('https://api.openai.com/v1/realtime', {\n        method: 'POST',\n        headers: {\n          'Authorization': `Bearer ${ephemeralKey}`,\n          'Content-Type': 'application/sdp',\n        },\n        body: offer.sdp,\n      });\n\n      const answer: RTCSessionDescriptionInit = {\n        type: 'answer',\n        sdp: await response.text(),\n      };\n\n      await pc.setRemoteDescription(new RTCSessionDescription(answer));\n\n      peerConnection.current = pc;\n      setIsRecording(true);\n    } catch (error) {\n      onError(error as Error);\n      cleanup();\n    }\n  }, [baseUrl, apiKey, handleDataChannelMessage, onError, cleanup]);\n\n  const stopSession = useCallback(() => {\n    cleanup();\n  }, [cleanup]);\n\n  const toggleRecording = useCallback(() => {\n    if (isLoading) return;\n    \n    if (isRecording) {\n      stopSession();\n    } else {\n      startSession();\n    }\n  }, [isLoading, isRecording, startSession, stopSession]);\n\n  // Cleanup on component unmount\n  useEffect(() => {\n    return cleanup;\n  }, [cleanup]);\n\n  return {\n    isRecording,\n    isLoading,\n    isSessionActive,\n    startSession,\n    stopSession,\n    toggleRecording,\n  };\n}; "]}
@@ -0,0 +1,2 @@
1
+ export { GenAgentChat } from './components/GenAgentChat';
2
+ export type { GenAgentChatProps, ChatMessage } from './types';
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ export { GenAgentChat } from './components/GenAgentChat';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgR2VuQWdlbnRDaGF0IH0gZnJvbSAnLi9jb21wb25lbnRzL0dlbkFnZW50Q2hhdCc7XG5leHBvcnQgdHlwZSB7IEdlbkFnZW50Q2hhdFByb3BzLCBDaGF0TWVzc2FnZSB9IGZyb20gJy4vdHlwZXMnOyAiXX0=
@@ -0,0 +1,17 @@
1
+ interface AudioServiceConfig {
2
+ baseUrl: string;
3
+ apiKey: string;
4
+ }
5
+ export declare class AudioService {
6
+ private baseUrl;
7
+ private apiKey;
8
+ private ws;
9
+ private audioChunks;
10
+ private resolvePromise;
11
+ private rejectPromise;
12
+ constructor(config: AudioServiceConfig);
13
+ textToSpeech(text: string, voice?: string): Promise<Blob>;
14
+ private cleanup;
15
+ playAudio(audioBlob: Blob): Promise<void>;
16
+ }
17
+ export {};