genassist-chat-react 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +113 -0
- package/assets/chat-logo.png +0 -0
- package/dist/components/ChatMessage.d.ts +20 -0
- package/dist/components/ChatMessage.js +188 -0
- package/dist/components/GenAgentChat.d.ts +3 -0
- package/dist/components/GenAgentChat.js +520 -0
- package/dist/components/VoiceInput.d.ts +14 -0
- package/dist/components/VoiceInput.js +51 -0
- package/dist/hooks/useChat.d.ts +21 -0
- package/dist/hooks/useChat.js +256 -0
- package/dist/hooks/useVoiceInput.d.ts +16 -0
- package/dist/hooks/useVoiceInput.js +210 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +2 -0
- package/dist/services/audioService.d.ts +17 -0
- package/dist/services/audioService.js +118 -0
- package/dist/services/chatService.d.ts +48 -0
- package/dist/services/chatService.js +376 -0
- package/dist/types/index.d.ts +42 -0
- package/dist/types/index.js +2 -0
- package/package.json +48 -0
package/README.md
ADDED
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
# GenAssist Chat React
|
|
2
|
+
|
|
3
|
+
A reusable React chat component for integrating with GenAssist API.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install genassist-chat-react
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Usage
|
|
12
|
+
|
|
13
|
+
### Basic Usage
|
|
14
|
+
|
|
15
|
+
```jsx
|
|
16
|
+
import React from 'react';
|
|
17
|
+
import { GenAgentChat } from 'genassist-chat-react';
|
|
18
|
+
|
|
19
|
+
function App() {
|
|
20
|
+
return (
|
|
21
|
+
<div style={{ height: '600px', width: '400px' }}>
|
|
22
|
+
<GenAgentChat
|
|
23
|
+
baseUrl="https://your-api-base-url.com"
|
|
24
|
+
apiKey="your-api-key"
|
|
25
|
+
/>
|
|
26
|
+
</div>
|
|
27
|
+
);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export default App;
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
### With Custom Theme and User Data
|
|
34
|
+
|
|
35
|
+
```jsx
|
|
36
|
+
import React from 'react';
|
|
37
|
+
import { GenAgentChat } from 'genassist-chat-react';
|
|
38
|
+
|
|
39
|
+
function App() {
|
|
40
|
+
const userData = {
|
|
41
|
+
userId: '12345',
|
|
42
|
+
username: 'johndoe',
|
|
43
|
+
email: 'john@example.com',
|
|
44
|
+
// Any additional metadata
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
const theme = {
|
|
48
|
+
primaryColor: '#4a90e2',
|
|
49
|
+
secondaryColor: '#f5f5f5',
|
|
50
|
+
backgroundColor: '#ffffff',
|
|
51
|
+
textColor: '#333333',
|
|
52
|
+
fontFamily: 'Roboto, sans-serif',
|
|
53
|
+
fontSize: '15px',
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
return (
|
|
57
|
+
<div style={{ height: '600px', width: '400px' }}>
|
|
58
|
+
<GenAgentChat
|
|
59
|
+
baseUrl="https://your-api-base-url.com"
|
|
60
|
+
apiKey="your-api-key"
|
|
61
|
+
userData={userData}
|
|
62
|
+
theme={theme}
|
|
63
|
+
headerTitle="Customer Support"
|
|
64
|
+
placeholder="Ask a question..."
|
|
65
|
+
onError={(error) => console.error('Chat error:', error)}
|
|
66
|
+
onTakeover={handleTakeover}
|
|
67
|
+
/>
|
|
68
|
+
</div>
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export default App;
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
## Props
|
|
76
|
+
|
|
77
|
+
| Prop | Type | Required | Default | Description |
|
|
78
|
+
|------|------|----------|---------|-------------|
|
|
79
|
+
| baseUrl | string | Yes | - | Base URL for the API endpoints |
|
|
80
|
+
| apiKey | string | Yes | - | API key for authentication |
|
|
81
|
+
| userData | object | No | - | Any user metadata to include |
|
|
82
|
+
| onError | function | No | - | Error handler callback |
|
|
83
|
+
| onTakeover | function | No | - | Callback triggered when a takeover event occurs |
|
|
84
|
+
| theme | object | No | - | Custom theme options |
|
|
85
|
+
| headerTitle | string | No | 'Chat' | Title displayed in the chat header |
|
|
86
|
+
| placeholder | string | No | 'Type a message...' | Placeholder text for the input |
|
|
87
|
+
|
|
88
|
+
## Theme Options
|
|
89
|
+
|
|
90
|
+
| Option | Type | Description |
|
|
91
|
+
|--------|------|-------------|
|
|
92
|
+
| primaryColor | string | Primary color for buttons and user messages |
|
|
93
|
+
| secondaryColor | string | Background color for agent messages |
|
|
94
|
+
| backgroundColor | string | Background color for the chat container |
|
|
95
|
+
| textColor | string | Text color for agent messages |
|
|
96
|
+
| fontFamily | string | Font family for all text |
|
|
97
|
+
| fontSize | string | Font size for messages |
|
|
98
|
+
|
|
99
|
+
## API Endpoints
|
|
100
|
+
|
|
101
|
+
The component interacts with the following endpoints:
|
|
102
|
+
|
|
103
|
+
1. Start Conversation: `POST /api/conversations/in-progress/start`
|
|
104
|
+
2. Update Conversation: `POST /api/conversations/in-progress/update/{conversation_id}`
|
|
105
|
+
3. WebSocket: `/api/conversations/ws/{conversation_id}?access_token={token}&lang=en&topics=message&topics=takeover`
|
|
106
|
+
|
|
107
|
+
## Run
|
|
108
|
+
./
|
|
109
|
+
yarn run build
|
|
110
|
+
|
|
111
|
+
cd example-app
|
|
112
|
+
|
|
113
|
+
yarn run dev
|
|
Binary file
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import { ChatMessage } from '../types';
|
|
3
|
+
interface ChatMessageProps {
|
|
4
|
+
message: ChatMessage;
|
|
5
|
+
theme?: {
|
|
6
|
+
primaryColor?: string;
|
|
7
|
+
secondaryColor?: string;
|
|
8
|
+
fontFamily?: string;
|
|
9
|
+
fontSize?: string;
|
|
10
|
+
backgroundColor?: string;
|
|
11
|
+
textColor?: string;
|
|
12
|
+
};
|
|
13
|
+
onPlayAudio?: (text: string) => Promise<void>;
|
|
14
|
+
isPlayingAudio?: boolean;
|
|
15
|
+
isFirstMessage?: boolean;
|
|
16
|
+
isNextSameSpeaker?: boolean;
|
|
17
|
+
isPrevSameSpeaker?: boolean;
|
|
18
|
+
}
|
|
19
|
+
export declare const ChatMessageComponent: React.FC<ChatMessageProps>;
|
|
20
|
+
export {};
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
var __assign = (this && this.__assign) || function () {
|
|
2
|
+
__assign = Object.assign || function(t) {
|
|
3
|
+
for (var s, i = 1, n = arguments.length; i < n; i++) {
|
|
4
|
+
s = arguments[i];
|
|
5
|
+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
|
|
6
|
+
t[p] = s[p];
|
|
7
|
+
}
|
|
8
|
+
return t;
|
|
9
|
+
};
|
|
10
|
+
return __assign.apply(this, arguments);
|
|
11
|
+
};
|
|
12
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
13
|
+
import React from 'react';
|
|
14
|
+
import { User, UserX, AlertCircle } from 'lucide-react';
|
|
15
|
+
export var ChatMessageComponent = function (_a) {
|
|
16
|
+
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
|
+
var isUser = message.speaker === 'customer';
|
|
18
|
+
var isSpecial = message.speaker === 'special';
|
|
19
|
+
var isWelcomeMessage = !isUser && !isSpecial && isFirstMessage;
|
|
20
|
+
var formatTimestamp = function (timestamp) {
|
|
21
|
+
try {
|
|
22
|
+
if (!timestamp || isNaN(timestamp)) {
|
|
23
|
+
return 'Just now';
|
|
24
|
+
}
|
|
25
|
+
var timestampMs = timestamp < 1000000000000 ? timestamp * 1000 : timestamp;
|
|
26
|
+
var date = new Date(timestampMs);
|
|
27
|
+
// Quick check if date is valid
|
|
28
|
+
if (isNaN(date.getTime())) {
|
|
29
|
+
return 'Just now';
|
|
30
|
+
}
|
|
31
|
+
var today = new Date();
|
|
32
|
+
var yesterday = new Date(today);
|
|
33
|
+
yesterday.setDate(yesterday.getDate() - 1);
|
|
34
|
+
// Format time as HH:MM AM/PM
|
|
35
|
+
var timeOptions = { hour: 'numeric', minute: '2-digit', hour12: true };
|
|
36
|
+
var timeStr = date.toLocaleTimeString(undefined, timeOptions);
|
|
37
|
+
// Check if date is today, yesterday, or another day
|
|
38
|
+
if (date.toDateString() === today.toDateString()) {
|
|
39
|
+
return "Today, ".concat(timeStr);
|
|
40
|
+
}
|
|
41
|
+
else if (date.toDateString() === yesterday.toDateString()) {
|
|
42
|
+
return "Yesterday, ".concat(timeStr);
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
// Format date as Month DD, YYYY
|
|
46
|
+
var dateOptions = { month: 'short', day: 'numeric', year: 'numeric' };
|
|
47
|
+
var dateStr = date.toLocaleDateString(undefined, dateOptions);
|
|
48
|
+
return "".concat(dateStr, ", ").concat(timeStr);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
catch (error) {
|
|
52
|
+
console.error('Error formatting timestamp:', error);
|
|
53
|
+
return 'Just now';
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
var timestamp = formatTimestamp(message.create_time);
|
|
57
|
+
var userBubbleBgColor = (theme === null || theme === void 0 ? void 0 : theme.primaryColor) || '#2563EB';
|
|
58
|
+
var userTextColor = '#ffffff';
|
|
59
|
+
var agentBubbleBgColor = (theme === null || theme === void 0 ? void 0 : theme.secondaryColor) || '#eeeeee';
|
|
60
|
+
var agentTextColor = '#000000';
|
|
61
|
+
var fontFamily = (theme === null || theme === void 0 ? void 0 : theme.fontFamily) || 'Roboto, Arial, sans-serif';
|
|
62
|
+
var fontSize = (theme === null || theme === void 0 ? void 0 : theme.fontSize) || '15px';
|
|
63
|
+
var messageContainerStyle = {
|
|
64
|
+
display: 'flex',
|
|
65
|
+
flexDirection: 'column',
|
|
66
|
+
width: '100%',
|
|
67
|
+
marginBottom: isPrevSameSpeaker ? '8px' : '8px',
|
|
68
|
+
marginTop: isPrevSameSpeaker ? '0px' : '16px',
|
|
69
|
+
position: 'relative',
|
|
70
|
+
alignItems: isUser ? 'flex-end' : 'flex-start',
|
|
71
|
+
};
|
|
72
|
+
var messageRowStyle = {
|
|
73
|
+
display: 'flex',
|
|
74
|
+
width: '100%',
|
|
75
|
+
justifyContent: isUser ? 'flex-end' : 'flex-start',
|
|
76
|
+
};
|
|
77
|
+
var labelContainerStyle = {
|
|
78
|
+
display: isPrevSameSpeaker ? 'none' : 'flex',
|
|
79
|
+
width: '80%',
|
|
80
|
+
justifyContent: isUser ? 'flex-end' : 'flex-start',
|
|
81
|
+
marginBottom: '4px',
|
|
82
|
+
};
|
|
83
|
+
var messageLabelStyle = {
|
|
84
|
+
fontSize: '12px',
|
|
85
|
+
color: '#757575',
|
|
86
|
+
lineHeight: 1,
|
|
87
|
+
};
|
|
88
|
+
var messageBubbleContainerStyle = {
|
|
89
|
+
display: 'flex',
|
|
90
|
+
flexDirection: 'column',
|
|
91
|
+
maxWidth: '80%',
|
|
92
|
+
};
|
|
93
|
+
var bubbleStyle = {
|
|
94
|
+
backgroundColor: isUser
|
|
95
|
+
? userBubbleBgColor
|
|
96
|
+
: agentBubbleBgColor,
|
|
97
|
+
color: isUser ? userTextColor : agentTextColor,
|
|
98
|
+
padding: '12px 16px',
|
|
99
|
+
borderRadius: '8px',
|
|
100
|
+
fontSize: fontSize,
|
|
101
|
+
fontFamily: fontFamily,
|
|
102
|
+
wordBreak: 'break-word',
|
|
103
|
+
lineHeight: 1.4,
|
|
104
|
+
maxWidth: '100%',
|
|
105
|
+
};
|
|
106
|
+
var timestampStyle = {
|
|
107
|
+
fontSize: '11px',
|
|
108
|
+
color: '#757575',
|
|
109
|
+
marginTop: '4px',
|
|
110
|
+
width: '80%',
|
|
111
|
+
textAlign: isUser ? 'right' : 'left',
|
|
112
|
+
display: isNextSameSpeaker ? 'none' : 'block', // Hide timestamp if next message is from same speaker
|
|
113
|
+
};
|
|
114
|
+
var welcomeTitleStyle = {
|
|
115
|
+
fontSize: '18px',
|
|
116
|
+
fontWeight: 'bold',
|
|
117
|
+
marginBottom: '4px',
|
|
118
|
+
color: '#000000',
|
|
119
|
+
};
|
|
120
|
+
var welcomeContentStyle = {
|
|
121
|
+
fontSize: '16px',
|
|
122
|
+
fontWeight: 'normal',
|
|
123
|
+
color: '#000000',
|
|
124
|
+
};
|
|
125
|
+
var speakerLabel = isUser ? 'You' : isSpecial ? 'System' : 'Agent';
|
|
126
|
+
var welcomeTitle = '';
|
|
127
|
+
var welcomeContent = '';
|
|
128
|
+
if (isWelcomeMessage) {
|
|
129
|
+
var messageText = message.text;
|
|
130
|
+
if (messageText.toLowerCase().startsWith('')) {
|
|
131
|
+
var parts = messageText.split(/[,.!?]/);
|
|
132
|
+
if (parts.length > 0) {
|
|
133
|
+
welcomeTitle = parts[0].trim();
|
|
134
|
+
welcomeContent = messageText.substring(welcomeTitle.length).trim();
|
|
135
|
+
// Remove any punctuation at the start
|
|
136
|
+
welcomeContent = welcomeContent.replace(/^[,.!?\s]+/, '');
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
else {
|
|
140
|
+
welcomeTitle = '';
|
|
141
|
+
welcomeContent = messageText;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
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;
|
|
145
|
+
// Handle special messages (like takeover indicators)
|
|
146
|
+
if (isSpecial) {
|
|
147
|
+
var specialMessageStyle = {
|
|
148
|
+
display: 'flex',
|
|
149
|
+
justifyContent: 'center',
|
|
150
|
+
alignItems: 'center',
|
|
151
|
+
margin: '16px 0',
|
|
152
|
+
width: '100%',
|
|
153
|
+
};
|
|
154
|
+
// Determine icon and style based on message content
|
|
155
|
+
var icon = void 0;
|
|
156
|
+
var backgroundColor = '#E3F2FD';
|
|
157
|
+
var textColor = '#1976D2';
|
|
158
|
+
if (message.text.toLowerCase().includes('offline') || message.text.toLowerCase().includes('inactive')) {
|
|
159
|
+
icon = _jsx(AlertCircle, { size: 18 });
|
|
160
|
+
backgroundColor = '#FFF3E0';
|
|
161
|
+
textColor = '#F57C00';
|
|
162
|
+
}
|
|
163
|
+
else if (message.text.toLowerCase().includes('took over') || message.text.toLowerCase().includes('takeover')) {
|
|
164
|
+
icon = _jsx(UserX, { size: 18 });
|
|
165
|
+
}
|
|
166
|
+
else {
|
|
167
|
+
icon = _jsx(User, { size: 18 });
|
|
168
|
+
}
|
|
169
|
+
var specialBubbleStyle = {
|
|
170
|
+
backgroundColor: backgroundColor,
|
|
171
|
+
color: textColor,
|
|
172
|
+
padding: '8px 16px',
|
|
173
|
+
borderRadius: '16px',
|
|
174
|
+
fontSize: '14px',
|
|
175
|
+
fontFamily: fontFamily,
|
|
176
|
+
fontWeight: '500',
|
|
177
|
+
display: 'flex',
|
|
178
|
+
alignItems: 'center',
|
|
179
|
+
gap: '8px',
|
|
180
|
+
};
|
|
181
|
+
return (_jsx("div", { style: specialMessageStyle, children: _jsxs("div", { style: specialBubbleStyle, children: [icon, message.text] }) }));
|
|
182
|
+
}
|
|
183
|
+
if (isWelcomeMessage) {
|
|
184
|
+
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
|
+
}
|
|
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 })] }));
|
|
187
|
+
};
|
|
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}; "]}
|