genassist-chat-react 1.0.23 → 1.0.24
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 +5 -0
- package/dist/components/ChatMessage.js +18 -4
- package/dist/components/GenAgentChat.js +80 -33
- package/dist/hooks/useChat.d.ts +4 -1
- package/dist/hooks/useChat.js +45 -15
- package/dist/services/chatService.d.ts +5 -0
- package/dist/services/chatService.js +20 -14
- package/dist/types/index.d.ts +5 -0
- package/dist/types/index.js +1 -1
- package/dist/utils/en.json +3 -0
- package/dist/utils/i18n.d.ts +3 -0
- package/dist/utils/i18n.js +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -92,10 +92,15 @@ export default App;
|
|
|
92
92
|
| theme | object | No | - | Custom theme options |
|
|
93
93
|
| headerTitle | string | No | 'Chat' | Title displayed in the chat header |
|
|
94
94
|
| placeholder | string | No | 'Type a message...' | Placeholder text for the input |
|
|
95
|
+
| description | string | No | 'Support' | A short agent description text below header title |
|
|
96
|
+
| serverUnavailableMessage | string | No | 'The agent is currently offline, please check back later. Thank you!' | Custom message for when agent is offline (Agent is off or server is down) |
|
|
97
|
+
| serverUnavailableContactUrl | string | No | - | Url to redirect user for contact support |
|
|
98
|
+
| serverUnavailableContactLabel | string | No | Contact support | Label for contact support |
|
|
95
99
|
| useWS | boolean | false | true | 'Enable or disable websocket...' |
|
|
96
100
|
| useAudio | boolean | false | false | 'Enable or disable audio on chat input...' |
|
|
97
101
|
| useFile | boolean | false | false | 'Enable or disable file attachments on chat input...' |
|
|
98
102
|
| reCaptchaKey | string | false | undefined | 'Use google reCaptchaV3 site-key...' |
|
|
103
|
+
| mode | string | true | flotaing | 'Chat mode, floating or fullscreen' |
|
|
99
104
|
| allowedExtensions | string[] | false | undefined | 'Look for type AllowedExtension and see the supported list of extensions' |
|
|
100
105
|
|
|
101
106
|
|
|
@@ -205,10 +205,15 @@ export var ChatMessageComponent = function (_a) {
|
|
|
205
205
|
width: '100%',
|
|
206
206
|
};
|
|
207
207
|
// Determine icon and style based on message content
|
|
208
|
+
// Agent off & server down
|
|
209
|
+
var isAgentOffOrServerDown = message.text.toLowerCase().includes('offline') ||
|
|
210
|
+
message.text.toLowerCase().includes('inactive') ||
|
|
211
|
+
message.text.toLowerCase().includes('unavailable') ||
|
|
212
|
+
(message.linkUrl && message.linkLabel);
|
|
208
213
|
var icon = void 0;
|
|
209
214
|
var backgroundColor = '#E3F2FD';
|
|
210
215
|
var textColor = '#1976D2';
|
|
211
|
-
if (
|
|
216
|
+
if (isAgentOffOrServerDown) {
|
|
212
217
|
icon = _jsx(AlertCircle, { size: 18 });
|
|
213
218
|
backgroundColor = '#FFF3E0';
|
|
214
219
|
textColor = '#F57C00';
|
|
@@ -228,10 +233,19 @@ export var ChatMessageComponent = function (_a) {
|
|
|
228
233
|
fontFamily: fontFamily,
|
|
229
234
|
fontWeight: '500',
|
|
230
235
|
display: 'flex',
|
|
231
|
-
|
|
236
|
+
flexDirection: 'column',
|
|
237
|
+
alignItems: 'flex-start',
|
|
232
238
|
gap: '8px',
|
|
233
239
|
};
|
|
234
|
-
|
|
240
|
+
var linkStyle = {
|
|
241
|
+
color: textColor,
|
|
242
|
+
textDecoration: 'underline',
|
|
243
|
+
cursor: 'pointer',
|
|
244
|
+
fontSize: '14px',
|
|
245
|
+
fontFamily: fontFamily,
|
|
246
|
+
fontWeight: '500',
|
|
247
|
+
};
|
|
248
|
+
return (_jsx("div", { style: specialMessageStyle, children: _jsxs("div", { style: __assign(__assign({}, specialBubbleStyle), { flexDirection: 'row', alignItems: 'center' }), children: [icon, _jsxs("div", { style: { display: 'flex', flexDirection: 'column', gap: '4px' }, children: [_jsx("span", { children: message.text }), message.linkUrl && message.linkLabel && (_jsx("a", { href: message.linkUrl, target: "_blank", rel: "noopener noreferrer", style: linkStyle, onClick: function (e) { return e.stopPropagation(); }, children: message.linkLabel }))] })] }) }));
|
|
235
249
|
}
|
|
236
250
|
if (isWelcomeMessage) {
|
|
237
251
|
return (_jsx("div", { style: messageContainerStyle, children: _jsx("div", { style: __assign({}, messageBubbleContainerStyle), children: _jsx(WelcomeCard, { theme: theme, imageUrl: welcomeImageUrl, title: welcomeTitle, content: welcomeContent, possibleQueries: possibleQueries, onQuickQuery: onQuickQuery, isAgentTyping: isAgentTyping }) }) }));
|
|
@@ -257,4 +271,4 @@ export var ChatMessageComponent = function (_a) {
|
|
|
257
271
|
}
|
|
258
272
|
}, children: _jsx(ThumbsDown, { size: 18 }) })] })) }))] }));
|
|
259
273
|
};
|
|
260
|
-
//# 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 { WelcomeCard } from './WelcomeCard';\nimport { ChatMessage, ScheduleItem, Translations } from '../types';\nimport { User, UserX, AlertCircle, ThumbsUp, ThumbsDown } from 'lucide-react';\nimport { formatTimestamp } from '../utils/time';\nimport { InteractiveContent } from './InteractiveContent';\nimport { parseInteractiveContentBlocks } from '../utils/interactiveContent';\nimport { defaultTranslations, getTranslationString, mergeTranslations } from '../utils/i18n';\nimport { UploadFilePreview } from './common/UploadFilePreview';\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  isLastMessage?: boolean;\n  onQuickAction?: (text: string) => void;\n  onScheduleConfirm?: (schedule: ScheduleItem) => void;\n  translations?: Translations;\n  language?: string;\n  agentName?: string; // Custom agent name to display instead of translation\n  isAgentTyping?: boolean; // Whether agent is currently typing/thinking\n}\n\nexport const ChatMessageComponent: React.FC<ChatMessageProps> = ({\n  message,\n  theme,\n  onPlayAudio,\n  isFirstMessage = false,\n  isPrevSameSpeaker = false,\n  onFeedback,\n  welcomeImageUrl,\n  welcomeTitle: welcomeTitleProp,\n  possibleQueries,\n  onQuickQuery,\n  isLastMessage = false,\n  onQuickAction,\n  onScheduleConfirm,\n  translations: customTranslations,\n  language,\n  agentName,\n  isAgentTyping = false,\n}) => {\n  // Merge translations with defaults\n  const translations = React.useMemo(\n    () => mergeTranslations(customTranslations, defaultTranslations),\n    [customTranslations]\n  );\n\n  // Translation helper\n  const t = (key: string, fallback?: string): string => {\n    return getTranslationString(key, translations, fallback);\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] = React.useState(false);\n  const [displayText] = React.useState<string>(message.text);\n  const contentBlocks = React.useMemo(\n    () => parseInteractiveContentBlocks(displayText, message?.type as 'file' | 'message' | undefined),\n    [displayText]\n  );\n\n  // Format timestamp with translations\n  const timestamp = React.useMemo(() => {\n    const timeTranslations = translations?.time ? {\n      justNow: translations.time.justNow,\n      today: translations.time.today,\n      yesterday: translations.time.yesterday,\n    } : undefined;\n    return formatTimestamp(message.create_time, language, timeTranslations);\n  }, [message.create_time, language, translations]);\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 = theme?.textColor || '#000000';\n  const primaryColor = theme?.primaryColor || '#4f46e5';\n  const fontFamily = theme?.fontFamily || 'Roboto, Arial, sans-serif';\n  const fontSize = theme?.fontSize || '15px';\n  const bubbleTextColor = isUser ? userTextColor : agentTextColor;\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 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  // 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 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  // 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          <WelcomeCard\n            theme={theme}\n            imageUrl={welcomeImageUrl}\n            title={welcomeTitle}\n            content={welcomeContent}\n            possibleQueries={possibleQueries}\n            onQuickQuery={onQuickQuery}\n            isAgentTyping={isAgentTyping}\n          />\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}>{t('labels.you')}</div>\n          </>\n        ) : (\n          <>\n            <div style={messageLabelStyle}>{agentName || t('labels.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            <UploadFilePreview key={index} file={attachment} />\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            <div style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>\n              <InteractiveContent\n                blocks={contentBlocks}\n                primaryColor={primaryColor}\n                textColor={bubbleTextColor}\n                isActionable={!isUser && isLastMessage && !isAgentTyping}\n                onQuickAction={onQuickAction}\n                onScheduleConfirm={onScheduleConfirm}\n              />\n            </div>\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={t('feedback.thumbsUp')}\n                onClick={() => {\n                  // Use message_id from socket, fallback to id if message_id doesn't exist\n                  const msgId = message.message_id || (message as any).id;\n                  if (msgId && onFeedback) {\n                    onFeedback(msgId, 'good');\n                  } else {\n                    // ignore\n                  }\n                }}\n              >\n                <ThumbsUp size={18} />\n              </button>\n              <div style={separatorStyle} />\n              <button\n                type=\"button\"\n                style={iconButtonStyle}\n                title={t('feedback.thumbsDown')}\n                onClick={() => {\n                  // Use message_id from socket, fallback to id if message_id doesn't exist\n                  const msgId = message.message_id || (message as any).id;\n                  if (msgId && onFeedback) {\n                    onFeedback(msgId, 'bad');\n                  } else {\n                    // ignore\n                  }\n                }}\n              >\n                <ThumbsDown size={18} />\n              </button>\n            </>\n          )}\n        </div>\n      )}\n    </div>\n  );\n};\n"]}
|
|
274
|
+
//# 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 { WelcomeCard } from './WelcomeCard';\nimport { ChatMessage, ScheduleItem, Translations } from '../types';\nimport { User, UserX, AlertCircle, ThumbsUp, ThumbsDown } from 'lucide-react';\nimport { formatTimestamp } from '../utils/time';\nimport { InteractiveContent } from './InteractiveContent';\nimport { parseInteractiveContentBlocks } from '../utils/interactiveContent';\nimport { defaultTranslations, getTranslationString, mergeTranslations } from '../utils/i18n';\nimport { UploadFilePreview } from './common/UploadFilePreview';\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  isLastMessage?: boolean;\n  onQuickAction?: (text: string) => void;\n  onScheduleConfirm?: (schedule: ScheduleItem) => void;\n  translations?: Translations;\n  language?: string;\n  agentName?: string; // Custom agent name to display instead of translation\n  isAgentTyping?: boolean; // Whether agent is currently typing/thinking\n}\n\nexport const ChatMessageComponent: React.FC<ChatMessageProps> = ({\n  message,\n  theme,\n  onPlayAudio,\n  isFirstMessage = false,\n  isPrevSameSpeaker = false,\n  onFeedback,\n  welcomeImageUrl,\n  welcomeTitle: welcomeTitleProp,\n  possibleQueries,\n  onQuickQuery,\n  isLastMessage = false,\n  onQuickAction,\n  onScheduleConfirm,\n  translations: customTranslations,\n  language,\n  agentName,\n  isAgentTyping = false,\n}) => {\n  // Merge translations with defaults\n  const translations = React.useMemo(\n    () => mergeTranslations(customTranslations, defaultTranslations),\n    [customTranslations]\n  );\n\n  // Translation helper\n  const t = (key: string, fallback?: string): string => {\n    return getTranslationString(key, translations, fallback);\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] = React.useState(false);\n  const [displayText] = React.useState<string>(message.text);\n  const contentBlocks = React.useMemo(\n    () => parseInteractiveContentBlocks(displayText, message?.type as 'file' | 'message' | undefined),\n    [displayText]\n  );\n\n  // Format timestamp with translations\n  const timestamp = React.useMemo(() => {\n    const timeTranslations = translations?.time ? {\n      justNow: translations.time.justNow,\n      today: translations.time.today,\n      yesterday: translations.time.yesterday,\n    } : undefined;\n    return formatTimestamp(message.create_time, language, timeTranslations);\n  }, [message.create_time, language, translations]);\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 = theme?.textColor || '#000000';\n  const primaryColor = theme?.primaryColor || '#4f46e5';\n  const fontFamily = theme?.fontFamily || 'Roboto, Arial, sans-serif';\n  const fontSize = theme?.fontSize || '15px';\n  const bubbleTextColor = isUser ? userTextColor : agentTextColor;\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 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  // 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 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  // 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    // Agent off & server down\n    const isAgentOffOrServerDown =\n      message.text.toLowerCase().includes('offline') ||\n      message.text.toLowerCase().includes('inactive') ||\n      message.text.toLowerCase().includes('unavailable') ||\n      (message.linkUrl && message.linkLabel);\n    let icon;\n    let backgroundColor = '#E3F2FD';\n    let textColor = '#1976D2';\n    if (isAgentOffOrServerDown) {\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      flexDirection: 'column',\n      alignItems: 'flex-start',\n      gap: '8px',\n    };\n\n    const linkStyle: React.CSSProperties = {\n      color: textColor,\n      textDecoration: 'underline',\n      cursor: 'pointer',\n      fontSize: '14px',\n      fontFamily,\n      fontWeight: '500',\n    };\n\n    return (\n      <div style={specialMessageStyle}>\n        <div style={{ ...specialBubbleStyle, flexDirection: 'row', alignItems: 'center' }}>\n          {icon}\n          <div style={{ display: 'flex', flexDirection: 'column', gap: '4px' }}>\n            <span>{message.text}</span>\n            {message.linkUrl && message.linkLabel && (\n              <a\n                href={message.linkUrl}\n                target=\"_blank\"\n                rel=\"noopener noreferrer\"\n                style={linkStyle}\n                onClick={(e) => e.stopPropagation()}\n              >\n                {message.linkLabel}\n              </a>\n            )}\n          </div>\n        </div>\n      </div>\n    );\n  }\n\n  if (isWelcomeMessage) {\n    return (\n      <div style={messageContainerStyle}>\n        <div style={{ ...messageBubbleContainerStyle }}>\n          <WelcomeCard\n            theme={theme}\n            imageUrl={welcomeImageUrl}\n            title={welcomeTitle}\n            content={welcomeContent}\n            possibleQueries={possibleQueries}\n            onQuickQuery={onQuickQuery}\n            isAgentTyping={isAgentTyping}\n          />\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}>{t('labels.you')}</div>\n          </>\n        ) : (\n          <>\n            <div style={messageLabelStyle}>{agentName || t('labels.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            <UploadFilePreview key={index} file={attachment} />\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            <div style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>\n              <InteractiveContent\n                blocks={contentBlocks}\n                primaryColor={primaryColor}\n                textColor={bubbleTextColor}\n                isActionable={!isUser && isLastMessage && !isAgentTyping}\n                onQuickAction={onQuickAction}\n                onScheduleConfirm={onScheduleConfirm}\n              />\n            </div>\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={t('feedback.thumbsUp')}\n                onClick={() => {\n                  // Use message_id from socket, fallback to id if message_id doesn't exist\n                  const msgId = message.message_id || (message as any).id;\n                  if (msgId && onFeedback) {\n                    onFeedback(msgId, 'good');\n                  } else {\n                    // ignore\n                  }\n                }}\n              >\n                <ThumbsUp size={18} />\n              </button>\n              <div style={separatorStyle} />\n              <button\n                type=\"button\"\n                style={iconButtonStyle}\n                title={t('feedback.thumbsDown')}\n                onClick={() => {\n                  // Use message_id from socket, fallback to id if message_id doesn't exist\n                  const msgId = message.message_id || (message as any).id;\n                  if (msgId && onFeedback) {\n                    onFeedback(msgId, 'bad');\n                  } else {\n                    // ignore\n                  }\n                }}\n              >\n                <ThumbsDown size={18} />\n              </button>\n            </>\n          )}\n        </div>\n      )}\n    </div>\n  );\n};\n"]}
|