genassist-chat-react 1.0.0 → 1.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,11 @@
1
1
  import React from 'react';
2
2
  import { ChatMessage } from '../types';
3
+ interface AttachmentPreviewProps {
4
+ file: File;
5
+ onRemove: () => void;
6
+ uploading?: boolean;
7
+ }
8
+ export declare const AttachmentPreview: React.FC<AttachmentPreviewProps>;
3
9
  interface ChatMessageProps {
4
10
  message: ChatMessage;
5
11
  theme?: {
@@ -11,7 +11,86 @@ var __assign = (this && this.__assign) || function () {
11
11
  };
12
12
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
13
13
  import React from 'react';
14
- import { User, UserX, AlertCircle } from 'lucide-react';
14
+ import { User, UserX, AlertCircle, FileText, FileJson, FileImage, Loader2 } from 'lucide-react';
15
+ export var AttachmentPreview = function (_a) {
16
+ var file = _a.file, onRemove = _a.onRemove, _b = _a.uploading, uploading = _b === void 0 ? false : _b;
17
+ var fileType = file.type;
18
+ var isImage = fileType.startsWith('image/');
19
+ var _c = React.useState(null), imagePreview = _c[0], setImagePreview = _c[1];
20
+ React.useEffect(function () {
21
+ if (isImage) {
22
+ var reader_1 = new FileReader();
23
+ reader_1.onloadend = function () {
24
+ setImagePreview(reader_1.result);
25
+ };
26
+ reader_1.readAsDataURL(file);
27
+ }
28
+ }, [file, isImage]);
29
+ var getFileIcon = function (type) {
30
+ if (type.startsWith('image/'))
31
+ return _jsx(FileImage, { size: 24, color: "#6D28D9" });
32
+ if (type === 'application/pdf')
33
+ return _jsx(FileText, { size: 24, color: "#B91C1C" });
34
+ if (type === 'application/json')
35
+ return _jsx(FileJson, { size: 24, color: "#1D4ED8" });
36
+ return _jsx(FileText, { size: 24, color: "#4B5563" });
37
+ };
38
+ var containerStyle = {
39
+ display: 'flex',
40
+ alignItems: 'center',
41
+ padding: '8px',
42
+ backgroundColor: '#f0f0f0',
43
+ borderRadius: '8px',
44
+ position: 'relative',
45
+ maxWidth: '250px',
46
+ };
47
+ var imageStyle = {
48
+ width: '40px',
49
+ height: '40px',
50
+ borderRadius: '4px',
51
+ objectFit: 'cover',
52
+ marginRight: '10px',
53
+ };
54
+ var fileInfoStyle = {
55
+ flex: 1,
56
+ overflow: 'hidden',
57
+ whiteSpace: 'nowrap',
58
+ textOverflow: 'ellipsis',
59
+ };
60
+ var fileNameStyle = {
61
+ fontWeight: 'bold',
62
+ fontSize: '14px',
63
+ };
64
+ var fileSizeStyle = {
65
+ fontSize: '12px',
66
+ color: '#666',
67
+ };
68
+ var removeButtonStyle = {
69
+ background: 'none',
70
+ border: 'none',
71
+ cursor: 'pointer',
72
+ padding: '4px',
73
+ position: 'absolute',
74
+ top: '4px',
75
+ right: '4px',
76
+ };
77
+ var uploadingOverlayStyle = {
78
+ position: 'absolute',
79
+ top: 0,
80
+ left: 0,
81
+ right: 0,
82
+ bottom: 0,
83
+ backgroundColor: 'rgba(255, 255, 255, 0.7)',
84
+ display: 'flex',
85
+ alignItems: 'center',
86
+ justifyContent: 'center',
87
+ borderRadius: '8px',
88
+ };
89
+ var loaderSpinStyle = {
90
+ animation: 'spin 1s linear infinite',
91
+ };
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
+ };
15
94
  export var ChatMessageComponent = function (_a) {
16
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;
17
96
  var isUser = message.speaker === 'customer';
@@ -103,6 +182,50 @@ export var ChatMessageComponent = function (_a) {
103
182
  lineHeight: 1.4,
104
183
  maxWidth: '100%',
105
184
  };
185
+ var attachmentsContainerStyle = {
186
+ marginTop: '8px',
187
+ display: 'flex',
188
+ flexDirection: 'column',
189
+ gap: '8px',
190
+ };
191
+ var getFileIcon = function (fileType) {
192
+ if (fileType.startsWith('image/'))
193
+ return _jsx(FileImage, { size: 24, color: "#6D28D9" });
194
+ if (fileType === 'application/pdf')
195
+ return _jsx(FileText, { size: 24, color: "#B91C1C" });
196
+ if (fileType === 'application/json')
197
+ return _jsx(FileJson, { size: 24, color: "#1D4ED8" });
198
+ return _jsx(FileText, { size: 24, color: "#4B5563" });
199
+ };
200
+ var attachmentContainerStyle = {
201
+ display: 'flex',
202
+ alignItems: 'center',
203
+ padding: '10px',
204
+ backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',
205
+ borderRadius: '8px',
206
+ };
207
+ var imageAttachmentStyle = {
208
+ maxWidth: '100%',
209
+ borderRadius: '8px',
210
+ };
211
+ var fileAttachmentStyle = {
212
+ display: 'flex',
213
+ alignItems: 'center',
214
+ padding: '10px',
215
+ backgroundColor: isUser ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.05)',
216
+ borderRadius: '8px',
217
+ };
218
+ var fileInfoStyle = {
219
+ marginLeft: '10px',
220
+ flex: 1,
221
+ };
222
+ var fileNameStyle = {
223
+ fontWeight: 'bold',
224
+ };
225
+ var fileSizeStyle = {
226
+ fontSize: '12px',
227
+ opacity: 0.8,
228
+ };
106
229
  var timestampStyle = {
107
230
  fontSize: '11px',
108
231
  color: '#757575',
@@ -183,6 +306,17 @@ export var ChatMessageComponent = function (_a) {
183
306
  if (isWelcomeMessage) {
184
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 })] }) }));
185
308
  }
186
- return (_jsxs("div", { style: messageContainerStyle, children: [_jsx("div", { style: labelContainerStyle, children: _jsx("div", { style: messageLabelStyle, children: speakerLabel }) }), _jsx("div", { style: bubbleStyle, children: messageLines }), _jsx("div", { style: timestampStyle, children: timestamp })] }));
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: {
310
+ display: 'flex',
311
+ alignItems: 'center',
312
+ gap: '12px',
313
+ padding: '12px 14px',
314
+ backgroundColor: isUser ? 'rgba(255,255,255,0.12)' : '#f5f5f5',
315
+ border: isUser ? '1px solid rgba(255,255,255,0.18)' : '1px solid #e5e5e5',
316
+ borderRadius: '12px',
317
+ minWidth: '260px',
318
+ maxWidth: '360px',
319
+ 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 })] }));
187
321
  };
188
- //# 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 } from 'lucide-react';\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 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      <div style={bubbleStyle}>\n        {messageLines}\n      </div>\n      <div style={timestampStyle}>{timestamp}</div>\n    </div>\n  );\n}; "]}
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}; "]}