genassist-chat-react 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -0
- package/dist/components/ChatMessage.d.ts +6 -0
- package/dist/components/ChatMessage.js +181 -54
- package/dist/components/GenAgentChat.js +178 -42
- package/dist/components/VoiceInput.js +12 -10
- package/dist/hooks/useChat.d.ts +7 -1
- package/dist/hooks/useChat.js +104 -10
- package/dist/services/chatService.d.ts +10 -3
- package/dist/services/chatService.js +168 -73
- package/dist/types/index.d.ts +27 -3
- package/dist/types/index.js +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -22,6 +22,7 @@ function App() {
|
|
|
22
22
|
<GenAgentChat
|
|
23
23
|
baseUrl="https://your-api-base-url.com"
|
|
24
24
|
apiKey="your-api-key"
|
|
25
|
+
tenant="your-tenant-id"
|
|
25
26
|
/>
|
|
26
27
|
</div>
|
|
27
28
|
);
|
|
@@ -58,6 +59,7 @@ function App() {
|
|
|
58
59
|
<GenAgentChat
|
|
59
60
|
baseUrl="https://your-api-base-url.com"
|
|
60
61
|
apiKey="your-api-key"
|
|
62
|
+
tenant="your-tenant-id"
|
|
61
63
|
userData={userData}
|
|
62
64
|
theme={theme}
|
|
63
65
|
headerTitle="Customer Support"
|
|
@@ -21,6 +21,12 @@ interface ChatMessageProps {
|
|
|
21
21
|
isFirstMessage?: boolean;
|
|
22
22
|
isNextSameSpeaker?: boolean;
|
|
23
23
|
isPrevSameSpeaker?: boolean;
|
|
24
|
+
onFeedback?: (messageId: string, value: 'good' | 'bad') => void;
|
|
25
|
+
enableTypewriter?: boolean;
|
|
26
|
+
welcomeImageUrl?: string;
|
|
27
|
+
welcomeTitle?: string;
|
|
28
|
+
possibleQueries?: string[];
|
|
29
|
+
onQuickQuery?: (query: string) => void;
|
|
24
30
|
}
|
|
25
31
|
export declare const ChatMessageComponent: React.FC<ChatMessageProps>;
|
|
26
32
|
export {};
|
|
@@ -9,9 +9,9 @@ var __assign = (this && this.__assign) || function () {
|
|
|
9
9
|
};
|
|
10
10
|
return __assign.apply(this, arguments);
|
|
11
11
|
};
|
|
12
|
-
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
12
|
+
import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
|
|
13
13
|
import React from 'react';
|
|
14
|
-
import { User, UserX, AlertCircle, FileText, FileJson, FileImage, Loader2 } from 'lucide-react';
|
|
14
|
+
import { User, UserX, AlertCircle, FileText, FileJson, FileImage, Loader2, ThumbsUp, ThumbsDown } from 'lucide-react';
|
|
15
15
|
export var AttachmentPreview = function (_a) {
|
|
16
16
|
var file = _a.file, onRemove = _a.onRemove, _b = _a.uploading, uploading = _b === void 0 ? false : _b;
|
|
17
17
|
var fileType = file.type;
|
|
@@ -92,10 +92,15 @@ export var AttachmentPreview = function (_a) {
|
|
|
92
92
|
return (_jsxs("div", { style: containerStyle, children: [isImage && imagePreview ? (_jsx("img", { src: imagePreview, alt: file.name, style: imageStyle })) : (_jsx("div", { style: { marginRight: '10px' }, children: getFileIcon(fileType) })), _jsxs("div", { style: fileInfoStyle, children: [_jsx("div", { style: fileNameStyle, title: file.name, children: file.name }), _jsxs("div", { style: fileSizeStyle, children: [(file.size / 1024).toFixed(2), " KB"] })] }), !uploading && (_jsx("button", { onClick: onRemove, style: removeButtonStyle, title: "Remove file", children: "\u2715" })), uploading && (_jsx("div", { style: uploadingOverlayStyle, children: _jsx(Loader2, { size: 24, color: "#000", style: loaderSpinStyle }) })), _jsx("style", { children: "\n @keyframes spin {\n 0% { transform: rotate(0deg); }\n 100% { transform: rotate(360deg); }\n }\n " })] }));
|
|
93
93
|
};
|
|
94
94
|
export var ChatMessageComponent = function (_a) {
|
|
95
|
-
var message = _a.message, theme = _a.theme, onPlayAudio = _a.onPlayAudio, isPlayingAudio = _a.isPlayingAudio, _b = _a.isFirstMessage, isFirstMessage = _b === void 0 ? false : _b, _c = _a.isNextSameSpeaker, isNextSameSpeaker = _c === void 0 ? false : _c, _d = _a.isPrevSameSpeaker, isPrevSameSpeaker = _d === void 0 ? false : _d;
|
|
95
|
+
var message = _a.message, theme = _a.theme, onPlayAudio = _a.onPlayAudio, isPlayingAudio = _a.isPlayingAudio, _b = _a.isFirstMessage, isFirstMessage = _b === void 0 ? false : _b, _c = _a.isNextSameSpeaker, isNextSameSpeaker = _c === void 0 ? false : _c, _d = _a.isPrevSameSpeaker, isPrevSameSpeaker = _d === void 0 ? false : _d, onFeedback = _a.onFeedback, _e = _a.enableTypewriter, enableTypewriter = _e === void 0 ? false : _e, welcomeImageUrl = _a.welcomeImageUrl, welcomeTitleProp = _a.welcomeTitle, possibleQueries = _a.possibleQueries, onQuickQuery = _a.onQuickQuery;
|
|
96
96
|
var isUser = message.speaker === 'customer';
|
|
97
97
|
var isSpecial = message.speaker === 'special';
|
|
98
98
|
var isWelcomeMessage = !isUser && !isSpecial && isFirstMessage;
|
|
99
|
+
var _f = React.useState(false), isHovered = _f[0], setIsHovered = _f[1];
|
|
100
|
+
var _g = React.useState(false), editingFeedback = _g[0], setEditingFeedback = _g[1];
|
|
101
|
+
var _h = React.useState(message.text), displayText = _h[0], setDisplayText = _h[1];
|
|
102
|
+
var _j = React.useState(false), isTypewriting = _j[0], setIsTypewriting = _j[1];
|
|
103
|
+
var lastAnimatedRef = React.useRef('');
|
|
99
104
|
var formatTimestamp = function (timestamp) {
|
|
100
105
|
try {
|
|
101
106
|
if (!timestamp || isNaN(timestamp)) {
|
|
@@ -133,9 +138,36 @@ export var ChatMessageComponent = function (_a) {
|
|
|
133
138
|
}
|
|
134
139
|
};
|
|
135
140
|
var timestamp = formatTimestamp(message.create_time);
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
141
|
+
// Fast typewriter effect for newly displayed agent messages, fix later
|
|
142
|
+
// React.useEffect(() => {
|
|
143
|
+
// if (!isUser && !isSpecial && enableTypewriter && message.text !== lastAnimatedRef.current) {
|
|
144
|
+
// lastAnimatedRef.current = message.text;
|
|
145
|
+
// setIsTypewriting(true);
|
|
146
|
+
// setDisplayText('');
|
|
147
|
+
// let i = 0;
|
|
148
|
+
// const total = message.text.length;
|
|
149
|
+
// const tickMs = 16;
|
|
150
|
+
// const targetMs = 1200;
|
|
151
|
+
// const charsPerTick = Math.max(1, Math.round(total / (targetMs / tickMs)));
|
|
152
|
+
// const interval = window.setInterval(() => {
|
|
153
|
+
// i += charsPerTick;
|
|
154
|
+
// if (i >= total) {
|
|
155
|
+
// setDisplayText(message.text);
|
|
156
|
+
// setIsTypewriting(false);
|
|
157
|
+
// window.clearInterval(interval);
|
|
158
|
+
// } else {
|
|
159
|
+
// setDisplayText(message.text.slice(0, i));
|
|
160
|
+
// }
|
|
161
|
+
// }, tickMs);
|
|
162
|
+
// return () => window.clearInterval(interval);
|
|
163
|
+
// } else {
|
|
164
|
+
// setDisplayText(message.text);
|
|
165
|
+
// setIsTypewriting(false);
|
|
166
|
+
// }
|
|
167
|
+
// }, [message.text, enableTypewriter, isUser, isSpecial]);
|
|
168
|
+
// Updated design colors
|
|
169
|
+
var userBubbleBgColor = '#E4E4E7'; // grey for user
|
|
170
|
+
var userTextColor = '#000000';
|
|
139
171
|
var agentTextColor = '#000000';
|
|
140
172
|
var fontFamily = (theme === null || theme === void 0 ? void 0 : theme.fontFamily) || 'Roboto, Arial, sans-serif';
|
|
141
173
|
var fontSize = (theme === null || theme === void 0 ? void 0 : theme.fontSize) || '15px';
|
|
@@ -153,15 +185,23 @@ export var ChatMessageComponent = function (_a) {
|
|
|
153
185
|
width: '100%',
|
|
154
186
|
justifyContent: isUser ? 'flex-end' : 'flex-start',
|
|
155
187
|
};
|
|
156
|
-
var
|
|
188
|
+
var labelRowStyle = {
|
|
157
189
|
display: isPrevSameSpeaker ? 'none' : 'flex',
|
|
158
190
|
width: '80%',
|
|
159
191
|
justifyContent: isUser ? 'flex-end' : 'flex-start',
|
|
160
|
-
|
|
192
|
+
alignItems: 'baseline',
|
|
193
|
+
gap: '8px',
|
|
194
|
+
marginBottom: '6px',
|
|
161
195
|
};
|
|
162
196
|
var messageLabelStyle = {
|
|
163
|
-
fontSize: '
|
|
164
|
-
color: '#
|
|
197
|
+
fontSize: '14px',
|
|
198
|
+
color: '#000000',
|
|
199
|
+
lineHeight: 1,
|
|
200
|
+
fontWeight: 600,
|
|
201
|
+
};
|
|
202
|
+
var topTimestampStyle = {
|
|
203
|
+
fontSize: '13px',
|
|
204
|
+
color: '#6b7280',
|
|
165
205
|
lineHeight: 1,
|
|
166
206
|
};
|
|
167
207
|
var messageBubbleContainerStyle = {
|
|
@@ -169,19 +209,35 @@ export var ChatMessageComponent = function (_a) {
|
|
|
169
209
|
flexDirection: 'column',
|
|
170
210
|
maxWidth: '80%',
|
|
171
211
|
};
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
212
|
+
// Only user's messages should render as bubbles.
|
|
213
|
+
// Agent messages are plain text with no container background.
|
|
214
|
+
var bubbleStyle = isUser
|
|
215
|
+
? {
|
|
216
|
+
backgroundColor: userBubbleBgColor,
|
|
217
|
+
color: userTextColor,
|
|
218
|
+
padding: '12px 16px',
|
|
219
|
+
borderRadius: '12px 0 12px 12px',
|
|
220
|
+
fontSize: fontSize,
|
|
221
|
+
fontFamily: fontFamily,
|
|
222
|
+
wordBreak: 'break-word',
|
|
223
|
+
lineHeight: 1.4,
|
|
224
|
+
maxWidth: '100%',
|
|
225
|
+
border: 'none',
|
|
226
|
+
boxShadow: '0 1px 2px rgba(0,0,0,0.08)'
|
|
227
|
+
}
|
|
228
|
+
: {
|
|
229
|
+
backgroundColor: 'transparent',
|
|
230
|
+
color: agentTextColor,
|
|
231
|
+
padding: 0,
|
|
232
|
+
borderRadius: 0,
|
|
233
|
+
fontSize: fontSize,
|
|
234
|
+
fontFamily: fontFamily,
|
|
235
|
+
wordBreak: 'break-word',
|
|
236
|
+
lineHeight: 1.4,
|
|
237
|
+
maxWidth: '100%',
|
|
238
|
+
border: 'none',
|
|
239
|
+
boxShadow: 'none'
|
|
240
|
+
};
|
|
185
241
|
var attachmentsContainerStyle = {
|
|
186
242
|
marginTop: '8px',
|
|
187
243
|
display: 'flex',
|
|
@@ -226,45 +282,61 @@ export var ChatMessageComponent = function (_a) {
|
|
|
226
282
|
fontSize: '12px',
|
|
227
283
|
opacity: 0.8,
|
|
228
284
|
};
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
285
|
+
// No bottom timestamp in the new design
|
|
286
|
+
var timestampStyle = { display: 'none' };
|
|
287
|
+
var speakerLabel = isUser ? 'You' : isSpecial ? 'System' : 'Agent';
|
|
288
|
+
// Feedback state and UI (thumbs up/down) for agent messages
|
|
289
|
+
var feedbackValue = (message.feedback && message.feedback.length > 0)
|
|
290
|
+
? message.feedback[message.feedback.length - 1].feedback
|
|
291
|
+
: undefined;
|
|
292
|
+
var primaryColor = (theme === null || theme === void 0 ? void 0 : theme.primaryColor) || '#2962FF';
|
|
293
|
+
var thumbsShouldShow = !isUser && !isSpecial && isHovered && !feedbackValue;
|
|
294
|
+
var showThumbsArea = !isUser && !isSpecial; // Only agents have thumbs area
|
|
295
|
+
var feedbackRowStyle = {
|
|
296
|
+
display: showThumbsArea ? 'flex' : 'none',
|
|
297
|
+
marginTop: 8,
|
|
298
|
+
alignSelf: 'flex-start',
|
|
299
|
+
alignItems: 'center',
|
|
300
|
+
height: 20,
|
|
301
|
+
gap: 8,
|
|
302
|
+
visibility: thumbsShouldShow || !!feedbackValue ? 'visible' : 'hidden',
|
|
236
303
|
};
|
|
237
|
-
var
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
304
|
+
var iconButtonStyle = {
|
|
305
|
+
background: 'transparent',
|
|
306
|
+
border: 'none',
|
|
307
|
+
padding: 0,
|
|
308
|
+
cursor: 'pointer',
|
|
309
|
+
color: '#9ca3af',
|
|
310
|
+
display: thumbsShouldShow ? 'flex' : 'none',
|
|
311
|
+
alignItems: 'center',
|
|
312
|
+
};
|
|
313
|
+
var separatorStyle = {
|
|
314
|
+
width: 1,
|
|
315
|
+
height: 14,
|
|
316
|
+
backgroundColor: '#e5e7eb',
|
|
317
|
+
display: thumbsShouldShow ? 'block' : 'none',
|
|
242
318
|
};
|
|
243
|
-
var
|
|
244
|
-
fontSize: '16px',
|
|
245
|
-
fontWeight: 'normal',
|
|
319
|
+
var singleIconStyle = {
|
|
246
320
|
color: '#000000',
|
|
321
|
+
display: feedbackValue ? 'flex' : 'none',
|
|
322
|
+
alignItems: 'center',
|
|
247
323
|
};
|
|
248
|
-
var
|
|
249
|
-
var welcomeTitle = '';
|
|
324
|
+
var welcomeTitle = welcomeTitleProp || '';
|
|
250
325
|
var welcomeContent = '';
|
|
251
326
|
if (isWelcomeMessage) {
|
|
252
327
|
var messageText = message.text;
|
|
253
|
-
if (
|
|
254
|
-
var parts = messageText.split(/[
|
|
255
|
-
if (parts.length > 0)
|
|
256
|
-
welcomeTitle = parts[0].trim();
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
welcomeContent = welcomeContent.replace(/^[,.!?\s]+/, '');
|
|
260
|
-
}
|
|
328
|
+
if (!welcomeTitleProp) {
|
|
329
|
+
var parts = messageText.split(/[\n\.!?]/);
|
|
330
|
+
if (parts.length > 0)
|
|
331
|
+
welcomeTitle = (parts[0] || '').trim();
|
|
332
|
+
welcomeContent = messageText.substring(welcomeTitle.length).trim();
|
|
333
|
+
welcomeContent = welcomeContent.replace(/^[,.!?\s]+/, '');
|
|
261
334
|
}
|
|
262
335
|
else {
|
|
263
|
-
welcomeTitle = '';
|
|
264
336
|
welcomeContent = messageText;
|
|
265
337
|
}
|
|
266
338
|
}
|
|
267
|
-
var messageLines = !isWelcomeMessage ?
|
|
339
|
+
var messageLines = !isWelcomeMessage ? displayText.split('\n').map(function (line, i) { return (_jsxs(React.Fragment, { children: [line, i < displayText.split('\n').length - 1 && _jsx("br", {})] }, i)); }) : null;
|
|
268
340
|
// Handle special messages (like takeover indicators)
|
|
269
341
|
if (isSpecial) {
|
|
270
342
|
var specialMessageStyle = {
|
|
@@ -304,9 +376,56 @@ export var ChatMessageComponent = function (_a) {
|
|
|
304
376
|
return (_jsx("div", { style: specialMessageStyle, children: _jsxs("div", { style: specialBubbleStyle, children: [icon, message.text] }) }));
|
|
305
377
|
}
|
|
306
378
|
if (isWelcomeMessage) {
|
|
307
|
-
|
|
379
|
+
var primaryColor_1 = (theme === null || theme === void 0 ? void 0 : theme.primaryColor) || '#5B3DF5';
|
|
380
|
+
var chipStyle_1 = {
|
|
381
|
+
display: 'inline-block',
|
|
382
|
+
backgroundColor: primaryColor_1,
|
|
383
|
+
color: '#fff',
|
|
384
|
+
padding: '8px 12px',
|
|
385
|
+
borderRadius: 12,
|
|
386
|
+
fontSize: '14px',
|
|
387
|
+
marginRight: 8,
|
|
388
|
+
marginBottom: 8,
|
|
389
|
+
cursor: onQuickQuery ? 'pointer' : 'default',
|
|
390
|
+
userSelect: 'none',
|
|
391
|
+
};
|
|
392
|
+
var chipContainerStyle = {
|
|
393
|
+
display: 'flex',
|
|
394
|
+
flexWrap: 'wrap',
|
|
395
|
+
marginTop: 8,
|
|
396
|
+
};
|
|
397
|
+
var imageBoxStyle = {
|
|
398
|
+
width: 120,
|
|
399
|
+
height: 120,
|
|
400
|
+
borderRadius: 12,
|
|
401
|
+
backgroundColor: '#e5e7eb',
|
|
402
|
+
display: 'flex',
|
|
403
|
+
alignItems: 'center',
|
|
404
|
+
justifyContent: 'center',
|
|
405
|
+
color: '#6b7280',
|
|
406
|
+
fontSize: 12,
|
|
407
|
+
fontWeight: 600,
|
|
408
|
+
overflow: 'hidden',
|
|
409
|
+
};
|
|
410
|
+
var welcomeTitleStyle = {
|
|
411
|
+
fontSize: '22px',
|
|
412
|
+
fontWeight: 700,
|
|
413
|
+
margin: '8px 0 4px 0',
|
|
414
|
+
color: agentTextColor,
|
|
415
|
+
fontFamily: fontFamily,
|
|
416
|
+
};
|
|
417
|
+
var welcomeSubtitleStyle = {
|
|
418
|
+
fontSize: '14px',
|
|
419
|
+
color: '#6b7280',
|
|
420
|
+
margin: 0,
|
|
421
|
+
fontFamily: fontFamily,
|
|
422
|
+
};
|
|
423
|
+
return (_jsx("div", { style: messageContainerStyle, children: _jsx("div", { style: messageBubbleContainerStyle, children: _jsxs("div", { style: { display: 'flex', flexDirection: 'column', gap: 8 }, children: [_jsx("div", { style: { display: 'flex', gap: 16 }, children: _jsx("div", { style: imageBoxStyle, children: welcomeImageUrl ? (
|
|
424
|
+
// eslint-disable-next-line jsx-a11y/alt-text
|
|
425
|
+
_jsx("img", { src: welcomeImageUrl, style: { width: '100%', height: '100%', objectFit: 'cover' } })) : (_jsx("div", { children: "IMAGE 120 \u00D7 120 px" })) }) }), welcomeTitle && _jsx("div", { style: welcomeTitleStyle, children: welcomeTitle }), welcomeContent && _jsx("div", { style: welcomeSubtitleStyle, children: welcomeContent }), possibleQueries && possibleQueries.length > 0 && (_jsx("div", { style: chipContainerStyle, children: possibleQueries.map(function (q, idx) { return (_jsx("span", { style: chipStyle_1, onClick: function () { return onQuickQuery && onQuickQuery(q); }, children: q }, idx)); }) }))] }) }) }));
|
|
308
426
|
}
|
|
309
|
-
return (_jsxs("div", { style: messageContainerStyle,
|
|
427
|
+
return (_jsxs("div", { style: messageContainerStyle, onMouseEnter: function () { return setIsHovered(true); }, onMouseLeave: function () { if (!editingFeedback)
|
|
428
|
+
setIsHovered(false); }, children: [_jsx("div", { style: labelRowStyle, children: isUser ? (_jsxs(_Fragment, { children: [_jsx("div", { style: topTimestampStyle, children: timestamp }), _jsx("div", { style: messageLabelStyle, children: "You" })] })) : (_jsxs(_Fragment, { children: [_jsx("div", { style: messageLabelStyle, children: "Agent" }), _jsx("div", { style: topTimestampStyle, children: timestamp })] })) }), message.attachments && message.attachments.length > 0 && (_jsx("div", { style: __assign(__assign({}, attachmentsContainerStyle), { alignItems: isUser ? 'flex-end' : 'flex-start' }), children: message.attachments.map(function (attachment, index) { return (_jsxs("div", { style: {
|
|
310
429
|
display: 'flex',
|
|
311
430
|
alignItems: 'center',
|
|
312
431
|
gap: '12px',
|
|
@@ -317,6 +436,14 @@ export var ChatMessageComponent = function (_a) {
|
|
|
317
436
|
minWidth: '260px',
|
|
318
437
|
maxWidth: '360px',
|
|
319
438
|
pointerEvents: 'none',
|
|
320
|
-
}, children: [attachment.type.startsWith('image/') ? (_jsx("img", { src: attachment.url, alt: attachment.name, style: { width: 56, height: 56, borderRadius: 8, objectFit: 'cover' } })) : (_jsx("div", { style: { width: 40, height: 40, display: 'flex', alignItems: 'center', justifyContent: 'center' }, children: getFileIcon(attachment.type) })), _jsxs("div", { style: { display: 'flex', flexDirection: 'column' }, children: [_jsx("div", { style: { fontWeight: 600, fontSize: 14 }, children: attachment.name }), _jsxs("div", { style: { fontSize: 12, opacity: 0.8 }, children: [attachment.type.includes('/') ? attachment.type.split('/')[1].toUpperCase() : attachment.type, attachment.size ? " \u00B7 ".concat((attachment.size / 1024).toFixed(1), " KB") : ''] })] })] }, index)); }) })), message.text && (_jsx("div", { style: bubbleStyle, children: messageLines })), _jsx("div", { style:
|
|
439
|
+
}, children: [attachment.type.startsWith('image/') ? (_jsx("img", { src: attachment.url, alt: attachment.name, style: { width: 56, height: 56, borderRadius: 8, objectFit: 'cover' } })) : (_jsx("div", { style: { width: 40, height: 40, display: 'flex', alignItems: 'center', justifyContent: 'center' }, children: getFileIcon(attachment.type) })), _jsxs("div", { style: { display: 'flex', flexDirection: 'column' }, children: [_jsx("div", { style: { fontWeight: 600, fontSize: 14 }, children: attachment.name }), _jsxs("div", { style: { fontSize: 12, opacity: 0.8 }, children: [attachment.type.includes('/') ? attachment.type.split('/')[1].toUpperCase() : attachment.type, attachment.size ? " \u00B7 ".concat((attachment.size / 1024).toFixed(1), " KB") : ''] })] })] }, index)); }) })), message.text && (_jsx("div", { style: { display: 'flex', alignItems: 'center', gap: '8px', alignSelf: isUser ? 'flex-end' : 'flex-start' }, children: _jsx("div", { style: __assign(__assign({}, bubbleStyle), { position: 'relative' }), children: messageLines }) })), showThumbsArea && (_jsx("div", { style: feedbackRowStyle, children: feedbackValue ? (_jsx("div", { style: singleIconStyle, children: feedbackValue === 'good' ? (_jsx(ThumbsUp, { size: 18 })) : (_jsx(ThumbsDown, { size: 18 })) })) : (_jsxs(_Fragment, { children: [_jsx("button", { type: "button", style: iconButtonStyle, title: "Thumbs up", onClick: function () {
|
|
440
|
+
if (message.message_id && onFeedback) {
|
|
441
|
+
onFeedback(message.message_id, 'good');
|
|
442
|
+
}
|
|
443
|
+
}, children: _jsx(ThumbsUp, { size: 18 }) }), _jsx("div", { style: separatorStyle }), _jsx("button", { type: "button", style: iconButtonStyle, title: "Thumbs down", onClick: function () {
|
|
444
|
+
if (message.message_id && onFeedback) {
|
|
445
|
+
onFeedback(message.message_id, 'bad');
|
|
446
|
+
}
|
|
447
|
+
}, children: _jsx(ThumbsDown, { size: 18 }) })] })) }))] }));
|
|
321
448
|
};
|
|
322
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ChatMessage.js","sourceRoot":"","sources":["../../src/components/ChatMessage.tsx"],"names":[],"mappings":"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","sourcesContent":["import React from 'react';\nimport { ChatMessage } from '../types';\nimport { User, UserX, AlertCircle, FileText, FileJson, FileImage, Loader2 } from 'lucide-react';\n\ninterface AttachmentPreviewProps {\n  file: File;\n  onRemove: () => void;\n  uploading?: boolean;\n}\n\nexport const AttachmentPreview: React.FC<AttachmentPreviewProps> = ({ file, onRemove, uploading = false }) => {\n  const fileType = file.type;\n  const isImage = fileType.startsWith('image/');\n  const [imagePreview, setImagePreview] = React.useState<string | null>(null);\n\n  React.useEffect(() => {\n    if (isImage) {\n      const reader = new FileReader();\n      reader.onloadend = () => {\n        setImagePreview(reader.result as string);\n      };\n      reader.readAsDataURL(file);\n    }\n  }, [file, isImage]);\n\n  const getFileIcon = (type: string) => {\n    if (type.startsWith('image/')) return <FileImage size={24} color=\"#6D28D9\" />;\n    if (type === 'application/pdf') return <FileText size={24} color=\"#B91C1C\" />;\n    if (type === 'application/json') return <FileJson size={24} color=\"#1D4ED8\" />;\n    return <FileText size={24} color=\"#4B5563\" />;\n  };\n\n  const containerStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '8px',\n    backgroundColor: '#f0f0f0',\n    borderRadius: '8px',\n    position: 'relative',\n    maxWidth: '250px',\n  };\n\n  const imageStyle: React.CSSProperties = {\n    width: '40px',\n    height: '40px',\n    borderRadius: '4px',\n    objectFit: 'cover',\n    marginRight: '10px',\n  };\n\n  const fileInfoStyle: React.CSSProperties = {\n    flex: 1,\n    overflow: 'hidden',\n    whiteSpace: 'nowrap',\n    textOverflow: 'ellipsis',\n  };\n\n  const fileNameStyle: React.CSSProperties = {\n    fontWeight: 'bold',\n    fontSize: '14px',\n  };\n\n  const fileSizeStyle: React.CSSProperties = {\n    fontSize: '12px',\n    color: '#666',\n  };\n\n  const removeButtonStyle: React.CSSProperties = {\n    background: 'none',\n    border: 'none',\n    cursor: 'pointer',\n    padding: '4px',\n    position: 'absolute',\n    top: '4px',\n    right: '4px',\n  };\n\n  const uploadingOverlayStyle: React.CSSProperties = {\n    position: 'absolute',\n    top: 0,\n    left: 0,\n    right: 0,\n    bottom: 0,\n    backgroundColor: 'rgba(255, 255, 255, 0.7)',\n    display: 'flex',\n    alignItems: 'center',\n    justifyContent: 'center',\n    borderRadius: '8px',\n  };\n\n  const loaderSpinStyle: React.CSSProperties = {\n    animation: 'spin 1s linear infinite',\n  };\n\n  return (\n    <div style={containerStyle}>\n      {isImage && imagePreview ? (\n        <img src={imagePreview} alt={file.name} style={imageStyle} />\n      ) : (\n        <div style={{ marginRight: '10px' }}>{getFileIcon(fileType)}</div>\n      )}\n      <div style={fileInfoStyle}>\n        <div style={fileNameStyle} title={file.name}>{file.name}</div>\n        <div style={fileSizeStyle}>{(file.size / 1024).toFixed(2)} KB</div>\n      </div>\n      {!uploading && (\n        <button onClick={onRemove} style={removeButtonStyle} title=\"Remove file\">\n          &#x2715;\n        </button>\n      )}\n      {uploading && (\n        <div style={uploadingOverlayStyle}>\n          <Loader2 size={24} color=\"#000\" style={loaderSpinStyle} />\n        </div>\n      )}\n      <style>{`\n        @keyframes spin {\n          0% { transform: rotate(0deg); }\n          100% { transform: rotate(360deg); }\n        }\n      `}</style>\n    </div>\n  );\n};\n\n\ninterface ChatMessageProps {\n  message: ChatMessage;\n  theme?: {\n    primaryColor?: string;\n    secondaryColor?: string;\n    fontFamily?: string;\n    fontSize?: string;\n    backgroundColor?: string;\n    textColor?: string;\n  };\n  onPlayAudio?: (text: string) => Promise<void>;\n  isPlayingAudio?: boolean;\n  isFirstMessage?: boolean;\n  isNextSameSpeaker?: boolean;\n  isPrevSameSpeaker?: boolean;\n}\n\nexport const ChatMessageComponent: React.FC<ChatMessageProps> = ({\n  message,\n  theme,\n  onPlayAudio,\n  isPlayingAudio,\n  isFirstMessage = false,\n  isNextSameSpeaker = false,\n  isPrevSameSpeaker = false\n}) => {\n  const isUser = message.speaker === 'customer';\n  const isSpecial = message.speaker === 'special';\n  const isWelcomeMessage = !isUser && !isSpecial && isFirstMessage;\n\n  const formatTimestamp = (timestamp: number) => {\n    try {\n      if (!timestamp || isNaN(timestamp)) {\n        return 'Just now';\n      }\n      \n      const timestampMs = timestamp < 1000000000000 ? timestamp * 1000 : timestamp;\n      \n      const date = new Date(timestampMs);\n      // Quick check if date is valid\n      if (isNaN(date.getTime())) {\n        return 'Just now';\n      }\n      \n      const today = new Date();\n      const yesterday = new Date(today);\n      yesterday.setDate(yesterday.getDate() - 1);\n      \n      // Format time as HH:MM AM/PM\n      const timeOptions: Intl.DateTimeFormatOptions = { hour: 'numeric', minute: '2-digit', hour12: true };\n      const timeStr = date.toLocaleTimeString(undefined, timeOptions);\n      \n      // Check if date is today, yesterday, or another day\n      if (date.toDateString() === today.toDateString()) {\n        return `Today, ${timeStr}`;\n      } else if (date.toDateString() === yesterday.toDateString()) {\n        return `Yesterday, ${timeStr}`;\n      } else {\n        // Format date as Month DD, YYYY\n        const dateOptions: Intl.DateTimeFormatOptions = { month: 'short', day: 'numeric', year: 'numeric' };\n        const dateStr = date.toLocaleDateString(undefined, dateOptions);\n        return `${dateStr}, ${timeStr}`;\n      }\n    } catch (error) {\n      console.error('Error formatting timestamp:', error);\n      return 'Just now';\n    }\n  };\n\n  const timestamp = formatTimestamp(message.create_time);\n\n  const userBubbleBgColor = theme?.primaryColor || '#2563EB';\n  const userTextColor = '#ffffff';\n  const agentBubbleBgColor = theme?.secondaryColor || '#eeeeee';\n  const agentTextColor = '#000000';\n  const fontFamily = theme?.fontFamily || 'Roboto, Arial, sans-serif';\n  const fontSize = theme?.fontSize || '15px';\n\n  const messageContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    flexDirection: 'column',\n    width: '100%',\n    marginBottom: isPrevSameSpeaker ? '8px' : '8px',\n    marginTop: isPrevSameSpeaker ? '0px' : '16px',\n    position: 'relative',\n    alignItems: isUser ? 'flex-end' : 'flex-start',\n  };\n\n  const messageRowStyle: React.CSSProperties = {\n    display: 'flex',\n    width: '100%',\n    justifyContent: isUser ? 'flex-end' : 'flex-start',\n  };\n\n  const labelContainerStyle: React.CSSProperties = {\n    display: isPrevSameSpeaker ? 'none' : 'flex',\n    width: '80%',\n    justifyContent: isUser ? 'flex-end' : 'flex-start',\n    marginBottom: '4px',\n  };\n\n  const messageLabelStyle: React.CSSProperties = {\n    fontSize: '12px',\n    color: '#757575',\n    lineHeight: 1,\n  };\n\n  const messageBubbleContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    flexDirection: 'column',\n    maxWidth: '80%',\n  };\n\n  const bubbleStyle: React.CSSProperties = {\n    backgroundColor: isUser \n      ? userBubbleBgColor\n      : agentBubbleBgColor,\n    color: isUser ? userTextColor : agentTextColor,\n    padding: '12px 16px',\n    borderRadius: '8px',\n    fontSize,\n    fontFamily,\n    wordBreak: 'break-word',\n    lineHeight: 1.4,\n    maxWidth: '100%',\n  };\n\n  const attachmentsContainerStyle: React.CSSProperties = {\n    marginTop: '8px',\n    display: 'flex',\n    flexDirection: 'column',\n    gap: '8px',\n  };\n\n  const getFileIcon = (fileType: string) => {\n    if (fileType.startsWith('image/')) return <FileImage size={24} color=\"#6D28D9\" />;\n    if (fileType === 'application/pdf') return <FileText size={24} color=\"#B91C1C\" />;\n    if (fileType === 'application/json') return <FileJson size={24} color=\"#1D4ED8\" />;\n    return <FileText size={24} color=\"#4B5563\" />;\n  };\n\n  const attachmentContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '10px',\n    backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',\n    borderRadius: '8px',\n  };\n\n  const imageAttachmentStyle: React.CSSProperties = {\n    maxWidth: '100%',\n    borderRadius: '8px',\n  };\n\n  const fileAttachmentStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '10px',\n    backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',\n    borderRadius: '8px',\n  };\n\n  const fileInfoStyle: React.CSSProperties = {\n    marginLeft: '10px',\n    flex: 1,\n  };\n  \n  const fileNameStyle: React.CSSProperties = {\n    fontWeight: 'bold',\n  };\n  \n  const fileSizeStyle: React.CSSProperties = {\n    fontSize: '12px',\n    opacity: 0.8,\n  };\n\n  const timestampStyle: React.CSSProperties = {\n    fontSize: '11px',\n    color: '#757575',\n    marginTop: '4px',\n    width: '80%',\n    textAlign: isUser ? 'right' : 'left',\n    display: isNextSameSpeaker ? 'none' : 'block', // Hide timestamp if next message is from same speaker\n  };\n\n  const welcomeTitleStyle: React.CSSProperties = {\n    fontSize: '18px',\n    fontWeight: 'bold',\n    marginBottom: '4px',\n    color: '#000000',\n  };\n\n  const welcomeContentStyle: React.CSSProperties = {\n    fontSize: '16px',\n    fontWeight: 'normal',\n    color: '#000000',\n  };\n\n  const speakerLabel = isUser ? 'You' : isSpecial ? 'System' : 'Agent';\n\n  let welcomeTitle = '';\n  let welcomeContent = '';\n  \n  if (isWelcomeMessage) {\n    const messageText = message.text;\n    if (messageText.toLowerCase().startsWith('')) {\n      const parts = messageText.split(/[,.!?]/);\n      if (parts.length > 0) {\n        welcomeTitle = parts[0].trim();\n        welcomeContent = messageText.substring(welcomeTitle.length).trim();\n        // Remove any punctuation at the start\n        welcomeContent = welcomeContent.replace(/^[,.!?\\s]+/, '');\n      }\n    } else {\n      welcomeTitle = '';\n      welcomeContent = messageText;\n    }\n  }\n\n  const messageLines = !isWelcomeMessage ? message.text.split('\\n').map((line, i) => (\n    <React.Fragment key={i}>\n      {line}\n      {i < message.text.split('\\n').length - 1 && <br />}\n    </React.Fragment>\n  )) : null;\n\n  // Handle special messages (like takeover indicators)\n  if (isSpecial) {\n    const specialMessageStyle: React.CSSProperties = {\n      display: 'flex',\n      justifyContent: 'center',\n      alignItems: 'center',\n      margin: '16px 0',\n      width: '100%',\n    };\n\n    // Determine icon and style based on message content\n    let icon;\n    let backgroundColor = '#E3F2FD';\n    let textColor = '#1976D2';\n    \n    if (message.text.toLowerCase().includes('offline') || message.text.toLowerCase().includes('inactive')) {\n      icon = <AlertCircle size={18} />;\n      backgroundColor = '#FFF3E0';\n      textColor = '#F57C00';\n    } else if (message.text.toLowerCase().includes('took over') || message.text.toLowerCase().includes('takeover')) {\n      icon = <UserX size={18} />;\n    } else {\n      icon = <User size={18} />;\n    }\n\n    const specialBubbleStyle: React.CSSProperties = {\n      backgroundColor,\n      color: textColor,\n      padding: '8px 16px',\n      borderRadius: '16px',\n      fontSize: '14px',\n      fontFamily,\n      fontWeight: '500',\n      display: 'flex',\n      alignItems: 'center',\n      gap: '8px',\n    };\n\n    return (\n      <div style={specialMessageStyle}>\n        <div style={specialBubbleStyle}>\n          {icon}\n          {message.text}\n        </div>\n      </div>\n    );\n  }\n\n  if (isWelcomeMessage) {\n    return (\n      <div style={messageContainerStyle}>\n        <div style={messageBubbleContainerStyle}>\n          <div style={labelContainerStyle}>\n            <div style={messageLabelStyle}>Agent</div>\n          </div>\n          <div style={messageRowStyle}>\n            <div style={{...bubbleStyle, backgroundColor: theme?.primaryColor || '#2563EB', color: '#ffffff'}}>\n              {message.text}\n            </div>\n          </div>\n          <div style={timestampStyle}>{timestamp}</div>\n        </div>\n      </div>\n    );\n  }\n\n  return (\n    <div style={messageContainerStyle}>\n      <div style={labelContainerStyle}>\n        <div style={messageLabelStyle}>{speakerLabel}</div>\n      </div>\n\n      {message.attachments && message.attachments.length > 0 && (\n        <div style={{ ...attachmentsContainerStyle, alignItems: isUser ? 'flex-end' : 'flex-start' }}>\n          {message.attachments.map((attachment, index) => (\n            <div\n              key={index}\n              style={{\n                display: 'flex',\n                alignItems: 'center',\n                gap: '12px',\n                padding: '12px 14px',\n                backgroundColor: isUser ? 'rgba(255,255,255,0.12)' : '#f5f5f5',\n                border: isUser ? '1px solid rgba(255,255,255,0.18)' : '1px solid #e5e5e5',\n                borderRadius: '12px',\n                minWidth: '260px',\n                maxWidth: '360px',\n                pointerEvents: 'none',\n              }}\n            >\n              {attachment.type.startsWith('image/') ? (\n                <img\n                  src={attachment.url}\n                  alt={attachment.name}\n                  style={{ width: 56, height: 56, borderRadius: 8, objectFit: 'cover' }}\n                />\n              ) : (\n                <div style={{ width: 40, height: 40, display: 'flex', alignItems: 'center', justifyContent: 'center' }}>\n                  {getFileIcon(attachment.type)}\n                </div>\n              )}\n              <div style={{ display: 'flex', flexDirection: 'column' }}>\n                <div style={{ fontWeight: 600, fontSize: 14 }}>{attachment.name}</div>\n                <div style={{ fontSize: 12, opacity: 0.8 }}>\n                  {attachment.type.includes('/') ? attachment.type.split('/')[1].toUpperCase() : attachment.type}\n                  {attachment.size ? ` · ${(attachment.size / 1024).toFixed(1)} KB` : ''}\n                </div>\n              </div>\n            </div>\n          ))}\n        </div>\n      )}\n\n      {message.text && (\n        <div style={bubbleStyle}>\n          {messageLines}\n        </div>\n      )}\n\n      <div style={timestampStyle}>{timestamp}</div>\n    </div>\n  );\n}; "]}
|
|
449
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ChatMessage.js","sourceRoot":"","sources":["../../src/components/ChatMessage.tsx"],"names":[],"mappings":";;;;;;;;;;;;AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAQtH,MAAM,CAAC,IAAM,iBAAiB,GAAqC,UAAC,EAAqC;QAAnC,IAAI,UAAA,EAAE,QAAQ,cAAA,EAAE,iBAAiB,EAAjB,SAAS,mBAAG,KAAK,KAAA;IACrG,IAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC;IAC3B,IAAM,OAAO,GAAG,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;IACxC,IAAA,KAAkC,KAAK,CAAC,QAAQ,CAAgB,IAAI,CAAC,EAApE,YAAY,QAAA,EAAE,eAAe,QAAuC,CAAC;IAE5E,KAAK,CAAC,SAAS,CAAC;QACd,IAAI,OAAO,EAAE,CAAC;YACZ,IAAM,QAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAChC,QAAM,CAAC,SAAS,GAAG;gBACjB,eAAe,CAAC,QAAM,CAAC,MAAgB,CAAC,CAAC;YAC3C,CAAC,CAAC;YACF,QAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC,EAAE,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;IAEpB,IAAM,WAAW,GAAG,UAAC,IAAY;QAC/B,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;YAAE,OAAO,KAAC,SAAS,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QAC9E,IAAI,IAAI,KAAK,iBAAiB;YAAE,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QAC9E,IAAI,IAAI,KAAK,kBAAkB;YAAE,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QAC/E,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;IAChD,CAAC,CAAC;IAEF,IAAM,cAAc,GAAwB;QAC1C,OAAO,EAAE,MAAM;QACf,UAAU,EAAE,QAAQ;QACpB,OAAO,EAAE,KAAK;QACd,eAAe,EAAE,SAAS;QAC1B,YAAY,EAAE,KAAK;QACnB,QAAQ,EAAE,UAAU;QACpB,QAAQ,EAAE,OAAO;KAClB,CAAC;IAEF,IAAM,UAAU,GAAwB;QACtC,KAAK,EAAE,MAAM;QACb,MAAM,EAAE,MAAM;QACd,YAAY,EAAE,KAAK;QACnB,SAAS,EAAE,OAAO;QAClB,WAAW,EAAE,MAAM;KACpB,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,IAAI,EAAE,CAAC;QACP,QAAQ,EAAE,QAAQ;QAClB,UAAU,EAAE,QAAQ;QACpB,YAAY,EAAE,UAAU;KACzB,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,UAAU,EAAE,MAAM;QAClB,QAAQ,EAAE,MAAM;KACjB,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,QAAQ,EAAE,MAAM;QAChB,KAAK,EAAE,MAAM;KACd,CAAC;IAEF,IAAM,iBAAiB,GAAwB;QAC7C,UAAU,EAAE,MAAM;QAClB,MAAM,EAAE,MAAM;QACd,MAAM,EAAE,SAAS;QACjB,OAAO,EAAE,KAAK;QACd,QAAQ,EAAE,UAAU;QACpB,GAAG,EAAE,KAAK;QACV,KAAK,EAAE,KAAK;KACb,CAAC;IAEF,IAAM,qBAAqB,GAAwB;QACjD,QAAQ,EAAE,UAAU;QACpB,GAAG,EAAE,CAAC;QACN,IAAI,EAAE,CAAC;QACP,KAAK,EAAE,CAAC;QACR,MAAM,EAAE,CAAC;QACT,eAAe,EAAE,0BAA0B;QAC3C,OAAO,EAAE,MAAM;QACf,UAAU,EAAE,QAAQ;QACpB,cAAc,EAAE,QAAQ;QACxB,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,IAAM,eAAe,GAAwB;QAC3C,SAAS,EAAE,yBAAyB;KACrC,CAAC;IAEF,OAAO,CACL,eAAK,KAAK,EAAE,cAAc,aACvB,OAAO,IAAI,YAAY,CAAC,CAAC,CAAC,CACzB,cAAK,GAAG,EAAE,YAAY,EAAE,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,UAAU,GAAI,CAC9D,CAAC,CAAC,CAAC,CACF,cAAK,KAAK,EAAE,EAAE,WAAW,EAAE,MAAM,EAAE,YAAG,WAAW,CAAC,QAAQ,CAAC,GAAO,CACnE,EACD,eAAK,KAAK,EAAE,aAAa,aACvB,cAAK,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,IAAI,CAAC,IAAI,YAAG,IAAI,CAAC,IAAI,GAAO,EAC9D,eAAK,KAAK,EAAE,aAAa,aAAG,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,WAAU,IAC/D,EACL,CAAC,SAAS,IAAI,CACb,iBAAQ,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,iBAAiB,EAAE,KAAK,EAAC,aAAa,uBAE/D,CACV,EACA,SAAS,IAAI,CACZ,cAAK,KAAK,EAAE,qBAAqB,YAC/B,KAAC,OAAO,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,MAAM,EAAC,KAAK,EAAE,eAAe,GAAI,GACtD,CACP,EACD,0BAAQ,0IAKP,GAAS,IACN,CACP,CAAC;AACJ,CAAC,CAAC;AA0BF,MAAM,CAAC,IAAM,oBAAoB,GAA+B,UAAC,EAchE;QAbC,OAAO,aAAA,EACP,KAAK,WAAA,EACL,WAAW,iBAAA,EACX,cAAc,oBAAA,EACd,sBAAsB,EAAtB,cAAc,mBAAG,KAAK,KAAA,EACtB,yBAAyB,EAAzB,iBAAiB,mBAAG,KAAK,KAAA,EACzB,yBAAyB,EAAzB,iBAAiB,mBAAG,KAAK,KAAA,EACzB,UAAU,gBAAA,EACV,wBAAwB,EAAxB,gBAAgB,mBAAG,KAAK,KAAA,EACxB,eAAe,qBAAA,EACD,gBAAgB,kBAAA,EAC9B,eAAe,qBAAA,EACf,YAAY,kBAAA;IAEZ,IAAM,MAAM,GAAG,OAAO,CAAC,OAAO,KAAK,UAAU,CAAC;IAC9C,IAAM,SAAS,GAAG,OAAO,CAAC,OAAO,KAAK,SAAS,CAAC;IAChD,IAAM,gBAAgB,GAAG,CAAC,MAAM,IAAI,CAAC,SAAS,IAAI,cAAc,CAAC;IAC3D,IAAA,KAA4B,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAhD,SAAS,QAAA,EAAE,YAAY,QAAyB,CAAC;IAClD,IAAA,KAAwC,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,EAA5D,eAAe,QAAA,EAAE,kBAAkB,QAAyB,CAAC;IAC9D,IAAA,KAAgC,KAAK,CAAC,QAAQ,CAAS,OAAO,CAAC,IAAI,CAAC,EAAnE,WAAW,QAAA,EAAE,cAAc,QAAwC,CAAC;IACrE,IAAA,KAAoC,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAxD,aAAa,QAAA,EAAE,gBAAgB,QAAyB,CAAC;IAChE,IAAM,eAAe,GAAG,KAAK,CAAC,MAAM,CAAS,EAAE,CAAC,CAAC;IAEjD,IAAM,eAAe,GAAG,UAAC,SAAiB;QACxC,IAAI,CAAC;YACH,IAAI,CAAC,SAAS,IAAI,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC;gBACnC,OAAO,UAAU,CAAC;YACpB,CAAC;YAED,IAAM,WAAW,GAAG,SAAS,GAAG,aAAa,CAAC,CAAC,CAAC,SAAS,GAAG,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC;YAE7E,IAAM,IAAI,GAAG,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC;YACnC,+BAA+B;YAC/B,IAAI,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC1B,OAAO,UAAU,CAAC;YACpB,CAAC;YAED,IAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;YACzB,IAAM,SAAS,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,CAAC;YAClC,SAAS,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,CAAC;YAE3C,6BAA6B;YAC7B,IAAM,WAAW,GAA+B,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;YACrG,IAAM,OAAO,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;YAEhE,oDAAoD;YACpD,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,KAAK,CAAC,YAAY,EAAE,EAAE,CAAC;gBACjD,OAAO,iBAAU,OAAO,CAAE,CAAC;YAC7B,CAAC;iBAAM,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,SAAS,CAAC,YAAY,EAAE,EAAE,CAAC;gBAC5D,OAAO,qBAAc,OAAO,CAAE,CAAC;YACjC,CAAC;iBAAM,CAAC;gBACN,gCAAgC;gBAChC,IAAM,WAAW,GAA+B,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;gBACpG,IAAM,OAAO,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;gBAChE,OAAO,UAAG,OAAO,eAAK,OAAO,CAAE,CAAC;YAClC,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;YACpD,OAAO,UAAU,CAAC;QACpB,CAAC;IACH,CAAC,CAAC;IAEF,IAAM,SAAS,GAAG,eAAe,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IAEvD,uEAAuE;IACvE,0BAA0B;IAC1B,iGAAiG;IACjG,8CAA8C;IAC9C,8BAA8B;IAC9B,0BAA0B;IAC1B,iBAAiB;IACjB,yCAAyC;IACzC,yBAAyB;IACzB,6BAA6B;IAC7B,iFAAiF;IACjF,kDAAkD;IAClD,2BAA2B;IAC3B,0BAA0B;IAC1B,wCAAwC;IACxC,mCAAmC;IACnC,0CAA0C;IAC1C,iBAAiB;IACjB,oDAAoD;IACpD,UAAU;IACV,kBAAkB;IAClB,mDAAmD;IACnD,aAAa;IACb,oCAAoC;IACpC,+BAA+B;IAC/B,MAAM;IACN,2DAA2D;IAE3D,wBAAwB;IACxB,IAAM,iBAAiB,GAAG,SAAS,CAAC,CAAC,gBAAgB;IACrD,IAAM,aAAa,GAAG,SAAS,CAAC;IAChC,IAAM,cAAc,GAAG,SAAS,CAAC;IACjC,IAAM,UAAU,GAAG,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,UAAU,KAAI,2BAA2B,CAAC;IACpE,IAAM,QAAQ,GAAG,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,QAAQ,KAAI,MAAM,CAAC;IAE3C,IAAM,qBAAqB,GAAwB;QACjD,OAAO,EAAE,MAAM;QACf,aAAa,EAAE,QAAQ;QACvB,KAAK,EAAE,MAAM;QACb,YAAY,EAAE,iBAAiB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK;QAC/C,SAAS,EAAE,iBAAiB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM;QAC7C,QAAQ,EAAE,UAAU;QACpB,UAAU,EAAE,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY;KAC/C,CAAC;IAEF,IAAM,eAAe,GAAwB;QAC3C,OAAO,EAAE,MAAM;QACf,KAAK,EAAE,MAAM;QACb,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY;KACnD,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,OAAO,EAAE,iBAAiB,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM;QAC5C,KAAK,EAAE,KAAK;QACZ,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY;QAClD,UAAU,EAAE,UAAU;QACtB,GAAG,EAAE,KAAK;QACV,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,IAAM,iBAAiB,GAAwB;QAC7C,QAAQ,EAAE,MAAM;QAChB,KAAK,EAAE,SAAS;QAChB,UAAU,EAAE,CAAC;QACb,UAAU,EAAE,GAAG;KAChB,CAAC;IAEF,IAAM,iBAAiB,GAAwB;QAC7C,QAAQ,EAAE,MAAM;QAChB,KAAK,EAAE,SAAS;QAChB,UAAU,EAAE,CAAC;KACd,CAAC;IAEF,IAAM,2BAA2B,GAAwB;QACvD,OAAO,EAAE,MAAM;QACf,aAAa,EAAE,QAAQ;QACvB,QAAQ,EAAE,KAAK;KAChB,CAAC;IAEF,iDAAiD;IACjD,8DAA8D;IAC9D,IAAM,WAAW,GAAwB,MAAM;QAC7C,CAAC,CAAC;YACE,eAAe,EAAE,iBAAiB;YAClC,KAAK,EAAE,aAAa;YACpB,OAAO,EAAE,WAAW;YACpB,YAAY,EAAE,kBAAkB;YAChC,QAAQ,UAAA;YACR,UAAU,YAAA;YACV,SAAS,EAAE,YAAY;YACvB,UAAU,EAAE,GAAG;YACf,QAAQ,EAAE,MAAM;YAChB,MAAM,EAAE,MAAM;YACd,SAAS,EAAE,4BAA4B;SACxC;QACH,CAAC,CAAC;YACE,eAAe,EAAE,aAAa;YAC9B,KAAK,EAAE,cAAc;YACrB,OAAO,EAAE,CAAC;YACV,YAAY,EAAE,CAAC;YACf,QAAQ,UAAA;YACR,UAAU,YAAA;YACV,SAAS,EAAE,YAAY;YACvB,UAAU,EAAE,GAAG;YACf,QAAQ,EAAE,MAAM;YAChB,MAAM,EAAE,MAAM;YACd,SAAS,EAAE,MAAM;SAClB,CAAC;IAEN,IAAM,yBAAyB,GAAwB;QACrD,SAAS,EAAE,KAAK;QAChB,OAAO,EAAE,MAAM;QACf,aAAa,EAAE,QAAQ;QACvB,GAAG,EAAE,KAAK;KACX,CAAC;IAEF,IAAM,WAAW,GAAG,UAAC,QAAgB;QACnC,IAAI,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC;YAAE,OAAO,KAAC,SAAS,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QAClF,IAAI,QAAQ,KAAK,iBAAiB;YAAE,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QAClF,IAAI,QAAQ,KAAK,kBAAkB;YAAE,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;QACnF,OAAO,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,EAAE,KAAK,EAAC,SAAS,GAAG,CAAC;IAChD,CAAC,CAAC;IAEF,IAAM,wBAAwB,GAAwB;QACpD,OAAO,EAAE,MAAM;QACf,UAAU,EAAE,QAAQ;QACpB,OAAO,EAAE,MAAM;QACf,eAAe,EAAE,MAAM,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC,qBAAqB;QAC5E,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,IAAM,oBAAoB,GAAwB;QAChD,QAAQ,EAAE,MAAM;QAChB,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,IAAM,mBAAmB,GAAwB;QAC/C,OAAO,EAAE,MAAM;QACf,UAAU,EAAE,QAAQ;QACpB,OAAO,EAAE,MAAM;QACf,eAAe,EAAE,MAAM,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC,qBAAqB;QAC5E,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,UAAU,EAAE,MAAM;QAClB,IAAI,EAAE,CAAC;KACR,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,UAAU,EAAE,MAAM;KACnB,CAAC;IAEF,IAAM,aAAa,GAAwB;QACzC,QAAQ,EAAE,MAAM;QAChB,OAAO,EAAE,GAAG;KACb,CAAC;IAEF,wCAAwC;IACxC,IAAM,cAAc,GAAwB,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC;IAChE,IAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC;IAErE,4DAA4D;IAC5D,IAAM,aAAa,GAAG,CAAC,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;QACrE,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,QAAQ;QACxD,CAAC,CAAC,SAAS,CAAC;IAEd,IAAM,YAAY,GAAG,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,YAAY,KAAI,SAAS,CAAC;IACtD,IAAM,gBAAgB,GAAG,CAAC,MAAM,IAAI,CAAC,SAAS,IAAI,SAAS,IAAI,CAAC,aAAa,CAAC;IAC9E,IAAM,cAAc,GAAG,CAAC,MAAM,IAAI,CAAC,SAAS,CAAC,CAAC,+BAA+B;IAE7E,IAAM,gBAAgB,GAAwB;QAC5C,OAAO,EAAE,cAAc,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM;QACzC,SAAS,EAAE,CAAC;QACZ,SAAS,EAAE,YAAY;QACvB,UAAU,EAAE,QAAQ;QACpB,MAAM,EAAE,EAAE;QACV,GAAG,EAAE,CAAC;QACN,UAAU,EAAE,gBAAgB,IAAI,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ;KACvE,CAAC;IAEF,IAAM,eAAe,GAAwB;QAC3C,UAAU,EAAE,aAAa;QACzB,MAAM,EAAE,MAAM;QACd,OAAO,EAAE,CAAC;QACV,MAAM,EAAE,SAAS;QACjB,KAAK,EAAE,SAAS;QAChB,OAAO,EAAE,gBAAgB,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM;QAC3C,UAAU,EAAE,QAAQ;KACrB,CAAC;IAEF,IAAM,cAAc,GAAwB;QAC1C,KAAK,EAAE,CAAC;QACR,MAAM,EAAE,EAAE;QACV,eAAe,EAAE,SAAS;QAC1B,OAAO,EAAE,gBAAgB,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM;KAC7C,CAAC;IAEF,IAAM,eAAe,GAAwB;QAC3C,KAAK,EAAE,SAAS;QAChB,OAAO,EAAE,aAAa,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM;QACxC,UAAU,EAAE,QAAQ;KACrB,CAAC;IAEF,IAAI,YAAY,GAAG,gBAAgB,IAAI,EAAE,CAAC;IAC1C,IAAI,cAAc,GAAG,EAAE,CAAC;IAExB,IAAI,gBAAgB,EAAE,CAAC;QACrB,IAAM,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;QACjC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,IAAM,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;YAC5C,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC;gBAAE,YAAY,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;YAC7D,cAAc,GAAG,WAAW,CAAC,SAAS,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC;YACnE,cAAc,GAAG,cAAc,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC;QAC5D,CAAC;aAAM,CAAC;YACN,cAAc,GAAG,WAAW,CAAC;QAC/B,CAAC;IACH,CAAC;IAED,IAAM,YAAY,GAAG,CAAC,gBAAgB,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,UAAC,IAAI,EAAE,CAAC,IAAK,OAAA,CAChF,MAAC,KAAK,CAAC,QAAQ,eACZ,IAAI,EACJ,CAAC,GAAG,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,IAAI,cAAM,KAF9B,CAAC,CAGL,CAClB,EALiF,CAKjF,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAEV,qDAAqD;IACrD,IAAI,SAAS,EAAE,CAAC;QACd,IAAM,mBAAmB,GAAwB;YAC/C,OAAO,EAAE,MAAM;YACf,cAAc,EAAE,QAAQ;YACxB,UAAU,EAAE,QAAQ;YACpB,MAAM,EAAE,QAAQ;YAChB,KAAK,EAAE,MAAM;SACd,CAAC;QAEF,oDAAoD;QACpD,IAAI,IAAI,SAAA,CAAC;QACT,IAAI,eAAe,GAAG,SAAS,CAAC;QAChC,IAAI,SAAS,GAAG,SAAS,CAAC;QAE1B,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YACtG,IAAI,GAAG,KAAC,WAAW,IAAC,IAAI,EAAE,EAAE,GAAI,CAAC;YACjC,eAAe,GAAG,SAAS,CAAC;YAC5B,SAAS,GAAG,SAAS,CAAC;QACxB,CAAC;aAAM,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YAC/G,IAAI,GAAG,KAAC,KAAK,IAAC,IAAI,EAAE,EAAE,GAAI,CAAC;QAC7B,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,KAAC,IAAI,IAAC,IAAI,EAAE,EAAE,GAAI,CAAC;QAC5B,CAAC;QAED,IAAM,kBAAkB,GAAwB;YAC9C,eAAe,iBAAA;YACf,KAAK,EAAE,SAAS;YAChB,OAAO,EAAE,UAAU;YACnB,YAAY,EAAE,MAAM;YACpB,QAAQ,EAAE,MAAM;YAChB,UAAU,YAAA;YACV,UAAU,EAAE,KAAK;YACjB,OAAO,EAAE,MAAM;YACf,UAAU,EAAE,QAAQ;YACpB,GAAG,EAAE,KAAK;SACX,CAAC;QAEF,OAAO,CACL,cAAK,KAAK,EAAE,mBAAmB,YAC7B,eAAK,KAAK,EAAE,kBAAkB,aAC3B,IAAI,EACJ,OAAO,CAAC,IAAI,IACT,GACF,CACP,CAAC;IACJ,CAAC;IAED,IAAI,gBAAgB,EAAE,CAAC;QACrB,IAAM,cAAY,GAAG,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,YAAY,KAAI,SAAS,CAAC;QACtD,IAAM,WAAS,GAAwB;YACrC,OAAO,EAAE,cAAc;YACvB,eAAe,EAAE,cAAY;YAC7B,KAAK,EAAE,MAAM;YACb,OAAO,EAAE,UAAU;YACnB,YAAY,EAAE,EAAE;YAChB,QAAQ,EAAE,MAAM;YAChB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;YACf,MAAM,EAAE,YAAY,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;YAC5C,UAAU,EAAE,MAAM;SACnB,CAAC;QACF,IAAM,kBAAkB,GAAwB;YAC9C,OAAO,EAAE,MAAM;YACf,QAAQ,EAAE,MAAM;YAChB,SAAS,EAAE,CAAC;SACb,CAAC;QACF,IAAM,aAAa,GAAwB;YACzC,KAAK,EAAE,GAAG;YACV,MAAM,EAAE,GAAG;YACX,YAAY,EAAE,EAAE;YAChB,eAAe,EAAE,SAAS;YAC1B,OAAO,EAAE,MAAM;YACf,UAAU,EAAE,QAAQ;YACpB,cAAc,EAAE,QAAQ;YACxB,KAAK,EAAE,SAAS;YAChB,QAAQ,EAAE,EAAE;YACZ,UAAU,EAAE,GAAG;YACf,QAAQ,EAAE,QAAQ;SACnB,CAAC;QACF,IAAM,iBAAiB,GAAwB;YAC7C,QAAQ,EAAE,MAAM;YAChB,UAAU,EAAE,GAAG;YACf,MAAM,EAAE,aAAa;YACrB,KAAK,EAAE,cAAc;YACrB,UAAU,YAAA;SACX,CAAC;QACF,IAAM,oBAAoB,GAAwB;YAChD,QAAQ,EAAE,MAAM;YAChB,KAAK,EAAE,SAAS;YAChB,MAAM,EAAE,CAAC;YACT,UAAU,YAAA;SACX,CAAC;QAEF,OAAO,CACL,cAAK,KAAK,EAAE,qBAAqB,YAC/B,cAAK,KAAK,EAAE,2BAA2B,YACrC,eAAK,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,QAAQ,EAAE,GAAG,EAAE,CAAC,EAAE,aAC9D,cAAK,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,EAAE,EAAE,YACtC,cAAK,KAAK,EAAE,aAAa,YACtB,eAAe,CAAC,CAAC,CAAC;gCACjB,6CAA6C;gCAC7C,cAAK,GAAG,EAAE,eAAe,EAAE,KAAK,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,OAAO,EAAE,GAAI,CAC5F,CAAC,CAAC,CAAC,CACF,oDAA6B,CAC9B,GACG,GACF,EACL,YAAY,IAAI,cAAK,KAAK,EAAE,iBAAiB,YAAG,YAAY,GAAO,EACnE,cAAc,IAAI,cAAK,KAAK,EAAE,oBAAoB,YAAG,cAAc,GAAO,EAC1E,eAAe,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,IAAI,CAChD,cAAK,KAAK,EAAE,kBAAkB,YAC3B,eAAe,CAAC,GAAG,CAAC,UAAC,CAAC,EAAE,GAAG,IAAK,OAAA,CAC/B,eAAgB,KAAK,EAAE,WAAS,EAAE,OAAO,EAAE,cAAM,OAAA,YAAY,IAAI,YAAY,CAAC,CAAC,CAAC,EAA/B,CAA+B,YAC7E,CAAC,IADO,GAAG,CAEP,CACR,EAJgC,CAIhC,CAAC,GACE,CACP,IACG,GACF,GACF,CACP,CAAC;IACJ,CAAC;IAED,OAAO,CACL,eACE,KAAK,EAAE,qBAAqB,EAC5B,YAAY,EAAE,cAAM,OAAA,YAAY,CAAC,IAAI,CAAC,EAAlB,CAAkB,EACtC,YAAY,EAAE,cAAQ,IAAI,CAAC,eAAe;YAAE,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,aAElE,cAAK,KAAK,EAAE,aAAa,YACtB,MAAM,CAAC,CAAC,CAAC,CACR,8BACE,cAAK,KAAK,EAAE,iBAAiB,YAAG,SAAS,GAAO,EAChD,cAAK,KAAK,EAAE,iBAAiB,oBAAW,IACvC,CACJ,CAAC,CAAC,CAAC,CACF,8BACE,cAAK,KAAK,EAAE,iBAAiB,sBAAa,EAC1C,cAAK,KAAK,EAAE,iBAAiB,YAAG,SAAS,GAAO,IAC/C,CACJ,GACG,EAEL,OAAO,CAAC,WAAW,IAAI,OAAO,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,CACxD,cAAK,KAAK,wBAAO,yBAAyB,KAAE,UAAU,EAAE,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY,eACvF,OAAO,CAAC,WAAW,CAAC,GAAG,CAAC,UAAC,UAAU,EAAE,KAAK,IAAK,OAAA,CAC9C,eAEE,KAAK,EAAE;wBACL,OAAO,EAAE,MAAM;wBACf,UAAU,EAAE,QAAQ;wBACpB,GAAG,EAAE,MAAM;wBACX,OAAO,EAAE,WAAW;wBACpB,eAAe,EAAE,MAAM,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,SAAS;wBAC9D,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,kCAAkC,CAAC,CAAC,CAAC,mBAAmB;wBACzE,YAAY,EAAE,MAAM;wBACpB,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE,OAAO;wBACjB,aAAa,EAAE,MAAM;qBACtB,aAEA,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CACtC,cACE,GAAG,EAAE,UAAU,CAAC,GAAG,EACnB,GAAG,EAAE,UAAU,CAAC,IAAI,EACpB,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,YAAY,EAAE,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,GACrE,CACH,CAAC,CAAC,CAAC,CACF,cAAK,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,cAAc,EAAE,QAAQ,EAAE,YACnG,WAAW,CAAC,UAAU,CAAC,IAAI,CAAC,GACzB,CACP,EACD,eAAK,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,QAAQ,EAAE,aACtD,cAAK,KAAK,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,QAAQ,EAAE,EAAE,EAAE,YAAG,UAAU,CAAC,IAAI,GAAO,EACtE,eAAK,KAAK,EAAE,EAAE,QAAQ,EAAE,EAAE,EAAE,OAAO,EAAE,GAAG,EAAE,aACvC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,EAC7F,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,kBAAM,CAAC,UAAU,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,QAAK,CAAC,CAAC,CAAC,EAAE,IAClE,IACF,KA/BD,KAAK,CAgCN,CACP,EAnC+C,CAmC/C,CAAC,GACE,CACP,EAEA,OAAO,CAAC,IAAI,IAAI,CACf,cAAK,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,GAAG,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY,EAAE,YAC9G,cAAK,KAAK,wBAAO,WAAW,KAAE,QAAQ,EAAE,UAAU,eAC/C,YAAY,GACT,GACF,CACP,EAGA,cAAc,IAAI,CACjB,cAAK,KAAK,EAAE,gBAAgB,YACzB,aAAa,CAAC,CAAC,CAAC,CACf,cAAK,KAAK,EAAE,eAAe,YACxB,aAAa,KAAK,MAAM,CAAC,CAAC,CAAC,CAC1B,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,GAAI,CACvB,CAAC,CAAC,CAAC,CACF,KAAC,UAAU,IAAC,IAAI,EAAE,EAAE,GAAI,CACzB,GACG,CACP,CAAC,CAAC,CAAC,CACF,8BACE,iBACE,IAAI,EAAC,QAAQ,EACb,KAAK,EAAE,eAAe,EACtB,KAAK,EAAC,WAAW,EACjB,OAAO,EAAE;gCACP,IAAI,OAAO,CAAC,UAAU,IAAI,UAAU,EAAE,CAAC;oCACrC,UAAU,CAAC,OAAO,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;gCACzC,CAAC;4BACH,CAAC,YAED,KAAC,QAAQ,IAAC,IAAI,EAAE,EAAE,GAAI,GACf,EACT,cAAK,KAAK,EAAE,cAAc,GAAI,EAC9B,iBACE,IAAI,EAAC,QAAQ,EACb,KAAK,EAAE,eAAe,EACtB,KAAK,EAAC,aAAa,EACnB,OAAO,EAAE;gCACP,IAAI,OAAO,CAAC,UAAU,IAAI,UAAU,EAAE,CAAC;oCACrC,UAAU,CAAC,OAAO,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;gCACxC,CAAC;4BACH,CAAC,YAED,KAAC,UAAU,IAAC,IAAI,EAAE,EAAE,GAAI,GACjB,IACR,CACJ,GACG,CACP,IACG,CACP,CAAC;AACJ,CAAC,CAAC","sourcesContent":["import React from 'react';\nimport { ChatMessage } from '../types';\nimport { User, UserX, AlertCircle, FileText, FileJson, FileImage, Loader2, ThumbsUp, ThumbsDown } from 'lucide-react';\n\ninterface AttachmentPreviewProps {\n  file: File;\n  onRemove: () => void;\n  uploading?: boolean;\n}\n\nexport const AttachmentPreview: React.FC<AttachmentPreviewProps> = ({ file, onRemove, uploading = false }) => {\n  const fileType = file.type;\n  const isImage = fileType.startsWith('image/');\n  const [imagePreview, setImagePreview] = React.useState<string | null>(null);\n\n  React.useEffect(() => {\n    if (isImage) {\n      const reader = new FileReader();\n      reader.onloadend = () => {\n        setImagePreview(reader.result as string);\n      };\n      reader.readAsDataURL(file);\n    }\n  }, [file, isImage]);\n\n  const getFileIcon = (type: string) => {\n    if (type.startsWith('image/')) return <FileImage size={24} color=\"#6D28D9\" />;\n    if (type === 'application/pdf') return <FileText size={24} color=\"#B91C1C\" />;\n    if (type === 'application/json') return <FileJson size={24} color=\"#1D4ED8\" />;\n    return <FileText size={24} color=\"#4B5563\" />;\n  };\n\n  const containerStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '8px',\n    backgroundColor: '#f0f0f0',\n    borderRadius: '8px',\n    position: 'relative',\n    maxWidth: '250px',\n  };\n\n  const imageStyle: React.CSSProperties = {\n    width: '40px',\n    height: '40px',\n    borderRadius: '4px',\n    objectFit: 'cover',\n    marginRight: '10px',\n  };\n\n  const fileInfoStyle: React.CSSProperties = {\n    flex: 1,\n    overflow: 'hidden',\n    whiteSpace: 'nowrap',\n    textOverflow: 'ellipsis',\n  };\n\n  const fileNameStyle: React.CSSProperties = {\n    fontWeight: 'bold',\n    fontSize: '14px',\n  };\n\n  const fileSizeStyle: React.CSSProperties = {\n    fontSize: '12px',\n    color: '#666',\n  };\n\n  const removeButtonStyle: React.CSSProperties = {\n    background: 'none',\n    border: 'none',\n    cursor: 'pointer',\n    padding: '4px',\n    position: 'absolute',\n    top: '4px',\n    right: '4px',\n  };\n\n  const uploadingOverlayStyle: React.CSSProperties = {\n    position: 'absolute',\n    top: 0,\n    left: 0,\n    right: 0,\n    bottom: 0,\n    backgroundColor: 'rgba(255, 255, 255, 0.7)',\n    display: 'flex',\n    alignItems: 'center',\n    justifyContent: 'center',\n    borderRadius: '8px',\n  };\n\n  const loaderSpinStyle: React.CSSProperties = {\n    animation: 'spin 1s linear infinite',\n  };\n\n  return (\n    <div style={containerStyle}>\n      {isImage && imagePreview ? (\n        <img src={imagePreview} alt={file.name} style={imageStyle} />\n      ) : (\n        <div style={{ marginRight: '10px' }}>{getFileIcon(fileType)}</div>\n      )}\n      <div style={fileInfoStyle}>\n        <div style={fileNameStyle} title={file.name}>{file.name}</div>\n        <div style={fileSizeStyle}>{(file.size / 1024).toFixed(2)} KB</div>\n      </div>\n      {!uploading && (\n        <button onClick={onRemove} style={removeButtonStyle} title=\"Remove file\">\n          &#x2715;\n        </button>\n      )}\n      {uploading && (\n        <div style={uploadingOverlayStyle}>\n          <Loader2 size={24} color=\"#000\" style={loaderSpinStyle} />\n        </div>\n      )}\n      <style>{`\n        @keyframes spin {\n          0% { transform: rotate(0deg); }\n          100% { transform: rotate(360deg); }\n        }\n      `}</style>\n    </div>\n  );\n};\n\n\ninterface ChatMessageProps {\n  message: ChatMessage;\n  theme?: {\n    primaryColor?: string;\n    secondaryColor?: string;\n    fontFamily?: string;\n    fontSize?: string;\n    backgroundColor?: string;\n    textColor?: string;\n  };\n  onPlayAudio?: (text: string) => Promise<void>;\n  isPlayingAudio?: boolean;\n  isFirstMessage?: boolean;\n  isNextSameSpeaker?: boolean;\n  isPrevSameSpeaker?: boolean;\n  onFeedback?: (messageId: string, value: 'good' | 'bad') => void;\n  enableTypewriter?: boolean;\n  welcomeImageUrl?: string;\n  welcomeTitle?: string;\n  possibleQueries?: string[];\n  onQuickQuery?: (query: string) => void;\n}\n\nexport const ChatMessageComponent: React.FC<ChatMessageProps> = ({\n  message,\n  theme,\n  onPlayAudio,\n  isPlayingAudio,\n  isFirstMessage = false,\n  isNextSameSpeaker = false,\n  isPrevSameSpeaker = false,\n  onFeedback,\n  enableTypewriter = false,\n  welcomeImageUrl,\n  welcomeTitle: welcomeTitleProp,\n  possibleQueries,\n  onQuickQuery,\n}) => {\n  const isUser = message.speaker === 'customer';\n  const isSpecial = message.speaker === 'special';\n  const isWelcomeMessage = !isUser && !isSpecial && isFirstMessage;\n  const [isHovered, setIsHovered] = React.useState(false);\n  const [editingFeedback, setEditingFeedback] = React.useState(false);\n  const [displayText, setDisplayText] = React.useState<string>(message.text);\n  const [isTypewriting, setIsTypewriting] = React.useState(false);\n  const lastAnimatedRef = React.useRef<string>('');\n\n  const formatTimestamp = (timestamp: number) => {\n    try {\n      if (!timestamp || isNaN(timestamp)) {\n        return 'Just now';\n      }\n      \n      const timestampMs = timestamp < 1000000000000 ? timestamp * 1000 : timestamp;\n      \n      const date = new Date(timestampMs);\n      // Quick check if date is valid\n      if (isNaN(date.getTime())) {\n        return 'Just now';\n      }\n      \n      const today = new Date();\n      const yesterday = new Date(today);\n      yesterday.setDate(yesterday.getDate() - 1);\n      \n      // Format time as HH:MM AM/PM\n      const timeOptions: Intl.DateTimeFormatOptions = { hour: 'numeric', minute: '2-digit', hour12: true };\n      const timeStr = date.toLocaleTimeString(undefined, timeOptions);\n      \n      // Check if date is today, yesterday, or another day\n      if (date.toDateString() === today.toDateString()) {\n        return `Today, ${timeStr}`;\n      } else if (date.toDateString() === yesterday.toDateString()) {\n        return `Yesterday, ${timeStr}`;\n      } else {\n        // Format date as Month DD, YYYY\n        const dateOptions: Intl.DateTimeFormatOptions = { month: 'short', day: 'numeric', year: 'numeric' };\n        const dateStr = date.toLocaleDateString(undefined, dateOptions);\n        return `${dateStr}, ${timeStr}`;\n      }\n    } catch (error) {\n      console.error('Error formatting timestamp:', error);\n      return 'Just now';\n    }\n  };\n\n  const timestamp = formatTimestamp(message.create_time);\n\n  // Fast typewriter effect for newly displayed agent messages, fix later\n  // React.useEffect(() => {\n  //   if (!isUser && !isSpecial && enableTypewriter && message.text !== lastAnimatedRef.current) {\n  //     lastAnimatedRef.current = message.text;\n  //     setIsTypewriting(true);\n  //     setDisplayText('');\n  //     let i = 0;\n  //     const total = message.text.length;\n  //     const tickMs = 16;\n  //     const targetMs = 1200;\n  //     const charsPerTick = Math.max(1, Math.round(total / (targetMs / tickMs)));\n  //     const interval = window.setInterval(() => {\n  //       i += charsPerTick;\n  //       if (i >= total) {\n  //         setDisplayText(message.text);\n  //         setIsTypewriting(false);\n  //         window.clearInterval(interval);\n  //       } else {\n  //         setDisplayText(message.text.slice(0, i));\n  //       }\n  //     }, tickMs);\n  //     return () => window.clearInterval(interval);\n  //   } else {\n  //     setDisplayText(message.text);\n  //     setIsTypewriting(false);\n  //   }\n  // }, [message.text, enableTypewriter, isUser, isSpecial]);\n\n  // Updated design colors\n  const userBubbleBgColor = '#E4E4E7'; // grey for user\n  const userTextColor = '#000000';\n  const agentTextColor = '#000000';\n  const fontFamily = theme?.fontFamily || 'Roboto, Arial, sans-serif';\n  const fontSize = theme?.fontSize || '15px';\n\n  const messageContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    flexDirection: 'column',\n    width: '100%',\n    marginBottom: isPrevSameSpeaker ? '8px' : '8px',\n    marginTop: isPrevSameSpeaker ? '0px' : '16px',\n    position: 'relative',\n    alignItems: isUser ? 'flex-end' : 'flex-start',\n  };\n\n  const messageRowStyle: React.CSSProperties = {\n    display: 'flex',\n    width: '100%',\n    justifyContent: isUser ? 'flex-end' : 'flex-start',\n  };\n\n  const labelRowStyle: React.CSSProperties = {\n    display: isPrevSameSpeaker ? 'none' : 'flex',\n    width: '80%',\n    justifyContent: isUser ? 'flex-end' : 'flex-start',\n    alignItems: 'baseline',\n    gap: '8px',\n    marginBottom: '6px',\n  };\n\n  const messageLabelStyle: React.CSSProperties = {\n    fontSize: '14px',\n    color: '#000000',\n    lineHeight: 1,\n    fontWeight: 600,\n  };\n\n  const topTimestampStyle: React.CSSProperties = {\n    fontSize: '13px',\n    color: '#6b7280',\n    lineHeight: 1,\n  };\n\n  const messageBubbleContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    flexDirection: 'column',\n    maxWidth: '80%',\n  };\n\n  // Only user's messages should render as bubbles.\n  // Agent messages are plain text with no container background.\n  const bubbleStyle: React.CSSProperties = isUser\n    ? {\n        backgroundColor: userBubbleBgColor,\n        color: userTextColor,\n        padding: '12px 16px',\n        borderRadius: '12px 0 12px 12px',\n        fontSize,\n        fontFamily,\n        wordBreak: 'break-word',\n        lineHeight: 1.4,\n        maxWidth: '100%',\n        border: 'none',\n        boxShadow: '0 1px 2px rgba(0,0,0,0.08)'\n      }\n    : {\n        backgroundColor: 'transparent',\n        color: agentTextColor,\n        padding: 0,\n        borderRadius: 0,\n        fontSize,\n        fontFamily,\n        wordBreak: 'break-word',\n        lineHeight: 1.4,\n        maxWidth: '100%',\n        border: 'none',\n        boxShadow: 'none'\n      };\n\n  const attachmentsContainerStyle: React.CSSProperties = {\n    marginTop: '8px',\n    display: 'flex',\n    flexDirection: 'column',\n    gap: '8px',\n  };\n\n  const getFileIcon = (fileType: string) => {\n    if (fileType.startsWith('image/')) return <FileImage size={24} color=\"#6D28D9\" />;\n    if (fileType === 'application/pdf') return <FileText size={24} color=\"#B91C1C\" />;\n    if (fileType === 'application/json') return <FileJson size={24} color=\"#1D4ED8\" />;\n    return <FileText size={24} color=\"#4B5563\" />;\n  };\n\n  const attachmentContainerStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '10px',\n    backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',\n    borderRadius: '8px',\n  };\n\n  const imageAttachmentStyle: React.CSSProperties = {\n    maxWidth: '100%',\n    borderRadius: '8px',\n  };\n\n  const fileAttachmentStyle: React.CSSProperties = {\n    display: 'flex',\n    alignItems: 'center',\n    padding: '10px',\n    backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',\n    borderRadius: '8px',\n  };\n\n  const fileInfoStyle: React.CSSProperties = {\n    marginLeft: '10px',\n    flex: 1,\n  };\n  \n  const fileNameStyle: React.CSSProperties = {\n    fontWeight: 'bold',\n  };\n  \n  const fileSizeStyle: React.CSSProperties = {\n    fontSize: '12px',\n    opacity: 0.8,\n  };\n\n  // No bottom timestamp in the new design\n  const timestampStyle: React.CSSProperties = { display: 'none' };\n  const speakerLabel = isUser ? 'You' : isSpecial ? 'System' : 'Agent';\n\n  // Feedback state and UI (thumbs up/down) for agent messages\n  const feedbackValue = (message.feedback && message.feedback.length > 0)\n    ? message.feedback[message.feedback.length - 1].feedback\n    : undefined;\n\n  const primaryColor = theme?.primaryColor || '#2962FF';\n  const thumbsShouldShow = !isUser && !isSpecial && isHovered && !feedbackValue;\n  const showThumbsArea = !isUser && !isSpecial; // Only agents have thumbs area\n\n  const feedbackRowStyle: React.CSSProperties = {\n    display: showThumbsArea ? 'flex' : 'none',\n    marginTop: 8,\n    alignSelf: 'flex-start',\n    alignItems: 'center',\n    height: 20,\n    gap: 8,\n    visibility: thumbsShouldShow || !!feedbackValue ? 'visible' : 'hidden',\n  };\n\n  const iconButtonStyle: React.CSSProperties = {\n    background: 'transparent',\n    border: 'none',\n    padding: 0,\n    cursor: 'pointer',\n    color: '#9ca3af',\n    display: thumbsShouldShow ? 'flex' : 'none',\n    alignItems: 'center',\n  };\n\n  const separatorStyle: React.CSSProperties = {\n    width: 1,\n    height: 14,\n    backgroundColor: '#e5e7eb',\n    display: thumbsShouldShow ? 'block' : 'none',\n  };\n\n  const singleIconStyle: React.CSSProperties = {\n    color: '#000000',\n    display: feedbackValue ? 'flex' : 'none',\n    alignItems: 'center',\n  };\n\n  let welcomeTitle = welcomeTitleProp || '';\n  let welcomeContent = '';\n  \n  if (isWelcomeMessage) {\n    const messageText = message.text;\n    if (!welcomeTitleProp) {\n      const parts = messageText.split(/[\\n\\.!?]/);\n      if (parts.length > 0) welcomeTitle = (parts[0] || '').trim();\n      welcomeContent = messageText.substring(welcomeTitle.length).trim();\n      welcomeContent = welcomeContent.replace(/^[,.!?\\s]+/, '');\n    } else {\n      welcomeContent = messageText;\n    }\n  }\n\n  const messageLines = !isWelcomeMessage ? displayText.split('\\n').map((line, i) => (\n    <React.Fragment key={i}>\n      {line}\n      {i < displayText.split('\\n').length - 1 && <br />}\n    </React.Fragment>\n  )) : null;\n\n  // Handle special messages (like takeover indicators)\n  if (isSpecial) {\n    const specialMessageStyle: React.CSSProperties = {\n      display: 'flex',\n      justifyContent: 'center',\n      alignItems: 'center',\n      margin: '16px 0',\n      width: '100%',\n    };\n\n    // Determine icon and style based on message content\n    let icon;\n    let backgroundColor = '#E3F2FD';\n    let textColor = '#1976D2';\n    \n    if (message.text.toLowerCase().includes('offline') || message.text.toLowerCase().includes('inactive')) {\n      icon = <AlertCircle size={18} />;\n      backgroundColor = '#FFF3E0';\n      textColor = '#F57C00';\n    } else if (message.text.toLowerCase().includes('took over') || message.text.toLowerCase().includes('takeover')) {\n      icon = <UserX size={18} />;\n    } else {\n      icon = <User size={18} />;\n    }\n\n    const specialBubbleStyle: React.CSSProperties = {\n      backgroundColor,\n      color: textColor,\n      padding: '8px 16px',\n      borderRadius: '16px',\n      fontSize: '14px',\n      fontFamily,\n      fontWeight: '500',\n      display: 'flex',\n      alignItems: 'center',\n      gap: '8px',\n    };\n\n    return (\n      <div style={specialMessageStyle}>\n        <div style={specialBubbleStyle}>\n          {icon}\n          {message.text}\n        </div>\n      </div>\n    );\n  }\n\n  if (isWelcomeMessage) {\n    const primaryColor = theme?.primaryColor || '#5B3DF5';\n    const chipStyle: React.CSSProperties = {\n      display: 'inline-block',\n      backgroundColor: primaryColor,\n      color: '#fff',\n      padding: '8px 12px',\n      borderRadius: 12,\n      fontSize: '14px',\n      marginRight: 8,\n      marginBottom: 8,\n      cursor: onQuickQuery ? 'pointer' : 'default',\n      userSelect: 'none',\n    };\n    const chipContainerStyle: React.CSSProperties = {\n      display: 'flex',\n      flexWrap: 'wrap',\n      marginTop: 8,\n    };\n    const imageBoxStyle: React.CSSProperties = {\n      width: 120,\n      height: 120,\n      borderRadius: 12,\n      backgroundColor: '#e5e7eb',\n      display: 'flex',\n      alignItems: 'center',\n      justifyContent: 'center',\n      color: '#6b7280',\n      fontSize: 12,\n      fontWeight: 600,\n      overflow: 'hidden',\n    };\n    const welcomeTitleStyle: React.CSSProperties = {\n      fontSize: '22px',\n      fontWeight: 700,\n      margin: '8px 0 4px 0',\n      color: agentTextColor,\n      fontFamily,\n    };\n    const welcomeSubtitleStyle: React.CSSProperties = {\n      fontSize: '14px',\n      color: '#6b7280',\n      margin: 0,\n      fontFamily,\n    };\n\n    return (\n      <div style={messageContainerStyle}>\n        <div style={messageBubbleContainerStyle}>\n          <div style={{ display: 'flex', flexDirection: 'column', gap: 8 }}>\n            <div style={{ display: 'flex', gap: 16 }}>\n              <div style={imageBoxStyle}>\n                {welcomeImageUrl ? (\n                  // eslint-disable-next-line jsx-a11y/alt-text\n                  <img src={welcomeImageUrl} style={{ width: '100%', height: '100%', objectFit: 'cover' }} />\n                ) : (\n                  <div>IMAGE 120 × 120 px</div>\n                )}\n              </div>\n            </div>\n            {welcomeTitle && <div style={welcomeTitleStyle}>{welcomeTitle}</div>}\n            {welcomeContent && <div style={welcomeSubtitleStyle}>{welcomeContent}</div>}\n            {possibleQueries && possibleQueries.length > 0 && (\n              <div style={chipContainerStyle}>\n                {possibleQueries.map((q, idx) => (\n                  <span key={idx} style={chipStyle} onClick={() => onQuickQuery && onQuickQuery(q)}>\n                    {q}\n                  </span>\n                ))}\n              </div>\n            )}\n          </div>\n        </div>\n      </div>\n    );\n  }\n\n  return (\n    <div \n      style={messageContainerStyle}\n      onMouseEnter={() => setIsHovered(true)}\n      onMouseLeave={() => { if (!editingFeedback) setIsHovered(false); }}\n    >\n      <div style={labelRowStyle}>\n        {isUser ? (\n          <>\n            <div style={topTimestampStyle}>{timestamp}</div>\n            <div style={messageLabelStyle}>You</div>\n          </>\n        ) : (\n          <>\n            <div style={messageLabelStyle}>Agent</div>\n            <div style={topTimestampStyle}>{timestamp}</div>\n          </>\n        )}\n      </div>\n\n      {message.attachments && message.attachments.length > 0 && (\n        <div style={{ ...attachmentsContainerStyle, alignItems: isUser ? 'flex-end' : 'flex-start' }}>\n          {message.attachments.map((attachment, index) => (\n            <div\n              key={index}\n              style={{\n                display: 'flex',\n                alignItems: 'center',\n                gap: '12px',\n                padding: '12px 14px',\n                backgroundColor: isUser ? 'rgba(255,255,255,0.12)' : '#f5f5f5',\n                border: isUser ? '1px solid rgba(255,255,255,0.18)' : '1px solid #e5e5e5',\n                borderRadius: '12px',\n                minWidth: '260px',\n                maxWidth: '360px',\n                pointerEvents: 'none',\n              }}\n            >\n              {attachment.type.startsWith('image/') ? (\n                <img\n                  src={attachment.url}\n                  alt={attachment.name}\n                  style={{ width: 56, height: 56, borderRadius: 8, objectFit: 'cover' }}\n                />\n              ) : (\n                <div style={{ width: 40, height: 40, display: 'flex', alignItems: 'center', justifyContent: 'center' }}>\n                  {getFileIcon(attachment.type)}\n                </div>\n              )}\n              <div style={{ display: 'flex', flexDirection: 'column' }}>\n                <div style={{ fontWeight: 600, fontSize: 14 }}>{attachment.name}</div>\n                <div style={{ fontSize: 12, opacity: 0.8 }}>\n                  {attachment.type.includes('/') ? attachment.type.split('/')[1].toUpperCase() : attachment.type}\n                  {attachment.size ? ` · ${(attachment.size / 1024).toFixed(1)} KB` : ''}\n                </div>\n              </div>\n            </div>\n          ))}\n        </div>\n      )}\n\n      {message.text && (\n        <div style={{ display: 'flex', alignItems: 'center', gap: '8px', alignSelf: isUser ? 'flex-end' : 'flex-start' }}>\n          <div style={{ ...bubbleStyle, position: 'relative' }}>\n            {messageLines}\n          </div>\n        </div>\n      )}\n\n      {/* Feedback row below agent message */}\n      {showThumbsArea && (\n        <div style={feedbackRowStyle}>\n          {feedbackValue ? (\n            <div style={singleIconStyle}>\n              {feedbackValue === 'good' ? (\n                <ThumbsUp size={18} />\n              ) : (\n                <ThumbsDown size={18} />\n              )}\n            </div>\n          ) : (\n            <>\n              <button\n                type=\"button\"\n                style={iconButtonStyle}\n                title=\"Thumbs up\"\n                onClick={() => {\n                  if (message.message_id && onFeedback) {\n                    onFeedback(message.message_id, 'good');\n                  }\n                }}\n              >\n                <ThumbsUp size={18} />\n              </button>\n              <div style={separatorStyle} />\n              <button\n                type=\"button\"\n                style={iconButtonStyle}\n                title=\"Thumbs down\"\n                onClick={() => {\n                  if (message.message_id && onFeedback) {\n                    onFeedback(message.message_id, 'bad');\n                  }\n                }}\n              >\n                <ThumbsDown size={18} />\n              </button>\n            </>\n          )}\n        </div>\n      )}\n    </div>\n  );\n};\n"]}
|