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 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
- var userBubbleBgColor = (theme === null || theme === void 0 ? void 0 : theme.primaryColor) || '#2563EB';
137
- var userTextColor = '#ffffff';
138
- var agentBubbleBgColor = (theme === null || theme === void 0 ? void 0 : theme.secondaryColor) || '#eeeeee';
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 labelContainerStyle = {
188
+ var labelRowStyle = {
157
189
  display: isPrevSameSpeaker ? 'none' : 'flex',
158
190
  width: '80%',
159
191
  justifyContent: isUser ? 'flex-end' : 'flex-start',
160
- marginBottom: '4px',
192
+ alignItems: 'baseline',
193
+ gap: '8px',
194
+ marginBottom: '6px',
161
195
  };
162
196
  var messageLabelStyle = {
163
- fontSize: '12px',
164
- color: '#757575',
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
- var bubbleStyle = {
173
- backgroundColor: isUser
174
- ? userBubbleBgColor
175
- : agentBubbleBgColor,
176
- color: isUser ? userTextColor : agentTextColor,
177
- padding: '12px 16px',
178
- borderRadius: '8px',
179
- fontSize: fontSize,
180
- fontFamily: fontFamily,
181
- wordBreak: 'break-word',
182
- lineHeight: 1.4,
183
- maxWidth: '100%',
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
- var timestampStyle = {
230
- fontSize: '11px',
231
- color: '#757575',
232
- marginTop: '4px',
233
- width: '80%',
234
- textAlign: isUser ? 'right' : 'left',
235
- display: isNextSameSpeaker ? 'none' : 'block', // Hide timestamp if next message is from same speaker
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 welcomeTitleStyle = {
238
- fontSize: '18px',
239
- fontWeight: 'bold',
240
- marginBottom: '4px',
241
- color: '#000000',
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 welcomeContentStyle = {
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 speakerLabel = isUser ? 'You' : isSpecial ? 'System' : 'Agent';
249
- var welcomeTitle = '';
324
+ var welcomeTitle = welcomeTitleProp || '';
250
325
  var welcomeContent = '';
251
326
  if (isWelcomeMessage) {
252
327
  var messageText = message.text;
253
- if (messageText.toLowerCase().startsWith('')) {
254
- var parts = messageText.split(/[,.!?]/);
255
- if (parts.length > 0) {
256
- welcomeTitle = parts[0].trim();
257
- welcomeContent = messageText.substring(welcomeTitle.length).trim();
258
- // Remove any punctuation at the start
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 ? message.text.split('\n').map(function (line, i) { return (_jsxs(React.Fragment, { children: [line, i < message.text.split('\n').length - 1 && _jsx("br", {})] }, i)); }) : null;
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
- return (_jsx("div", { style: messageContainerStyle, children: _jsxs("div", { style: messageBubbleContainerStyle, children: [_jsx("div", { style: labelContainerStyle, children: _jsx("div", { style: messageLabelStyle, children: "Agent" }) }), _jsx("div", { style: messageRowStyle, children: _jsx("div", { style: __assign(__assign({}, bubbleStyle), { backgroundColor: (theme === null || theme === void 0 ? void 0 : theme.primaryColor) || '#2563EB', color: '#ffffff' }), children: message.text }) }), _jsx("div", { style: timestampStyle, children: timestamp })] }) }));
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, children: [_jsx("div", { style: labelContainerStyle, children: _jsx("div", { style: messageLabelStyle, children: speakerLabel }) }), 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: {
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: timestampStyle, children: timestamp })] }));
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":"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","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"]}