genassist-chat-react 1.0.21 → 1.0.23

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
@@ -62,10 +62,16 @@ function App() {
62
62
  tenant="your-tenant-id"
63
63
  userData={userData}
64
64
  theme={theme}
65
+ useWS={true}
66
+ reCaptchaKey={your-recaptcha-site-key}
65
67
  headerTitle="Customer Support"
66
68
  placeholder="Ask a question..."
67
69
  onError={() => {}}
68
70
  onTakeover={handleTakeover}
71
+ noColorAnimation={false}
72
+ useFile={false}
73
+ useAudio={false}
74
+ allowedExtensions={['image/*']}
69
75
  />
70
76
  </div>
71
77
  );
@@ -86,6 +92,13 @@ export default App;
86
92
  | theme | object | No | - | Custom theme options |
87
93
  | headerTitle | string | No | 'Chat' | Title displayed in the chat header |
88
94
  | placeholder | string | No | 'Type a message...' | Placeholder text for the input |
95
+ | useWS | boolean | false | true | 'Enable or disable websocket...' |
96
+ | useAudio | boolean | false | false | 'Enable or disable audio on chat input...' |
97
+ | useFile | boolean | false | false | 'Enable or disable file attachments on chat input...' |
98
+ | reCaptchaKey | string | false | undefined | 'Use google reCaptchaV3 site-key...' |
99
+ | allowedExtensions | string[] | false | undefined | 'Look for type AllowedExtension and see the supported list of extensions' |
100
+
101
+
89
102
 
90
103
  ## Theme Options
91
104
 
@@ -37,10 +37,15 @@ export var AttachmentPreview = function (_a) {
37
37
  overflow: 'hidden',
38
38
  whiteSpace: 'nowrap',
39
39
  textOverflow: 'ellipsis',
40
+ width: '100%',
40
41
  };
41
42
  var fileNameStyle = {
43
+ width: '90%',
42
44
  fontWeight: 'bold',
43
45
  fontSize: '14px',
46
+ overflow: 'hidden',
47
+ textOverflow: 'ellipsis',
48
+ whiteSpace: 'nowrap',
44
49
  };
45
50
  var fileSizeStyle = {
46
51
  fontSize: '12px',
@@ -67,6 +72,6 @@ export var AttachmentPreview = function (_a) {
67
72
  justifyContent: 'center',
68
73
  borderRadius: '8px',
69
74
  };
70
- 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(Spinner, { size: 24, color: "#000" }) }))] }));
75
+ 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 ? file.size / 1024 : 0).toFixed(2), " KB"] })] }), !uploading && (_jsx("button", { onClick: onRemove, style: removeButtonStyle, title: "Remove file", children: "\u2715" })), uploading && (_jsx("div", { style: uploadingOverlayStyle, children: _jsx(Spinner, { size: 24, color: "#000" }) }))] }));
71
76
  };
72
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQXR0YWNobWVudFByZXZpZXcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tcG9uZW50cy9BdHRhY2htZW50UHJldmlldy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBSyxNQUFNLE9BQU8sQ0FBQztBQUMxQixPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQ3BDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQVE3QyxNQUFNLENBQUMsSUFBTSxpQkFBaUIsR0FBcUMsVUFBQyxFQUFxQztRQUFuQyxJQUFJLFVBQUEsRUFBRSxRQUFRLGNBQUEsRUFBRSxpQkFBaUIsRUFBakIsU0FBUyxtQkFBRyxLQUFLLEtBQUE7SUFDckcsSUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztJQUMzQixJQUFNLE9BQU8sR0FBRyxRQUFRLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3hDLElBQUEsS0FBa0MsS0FBSyxDQUFDLFFBQVEsQ0FBZ0IsSUFBSSxDQUFDLEVBQXBFLFlBQVksUUFBQSxFQUFFLGVBQWUsUUFBdUMsQ0FBQztJQUU1RSxLQUFLLENBQUMsU0FBUyxDQUFDO1FBQ2QsSUFBSSxPQUFPLEVBQUUsQ0FBQztZQUNaLElBQU0sUUFBTSxHQUFHLElBQUksVUFBVSxFQUFFLENBQUM7WUFDaEMsUUFBTSxDQUFDLFNBQVMsR0FBRztnQkFDakIsZUFBZSxDQUFDLFFBQU0sQ0FBQyxNQUFnQixDQUFDLENBQUM7WUFDM0MsQ0FBQyxDQUFDO1lBQ0YsUUFBTSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM3QixDQUFDO0lBQ0gsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7SUFFcEIsSUFBTSxjQUFjLEdBQXdCO1FBQzFDLE9BQU8sRUFBRSxNQUFNO1FBQ2YsVUFBVSxFQUFFLFFBQVE7UUFDcEIsT0FBTyxFQUFFLEtBQUs7UUFDZCxlQUFlLEVBQUUsU0FBUztRQUMxQixZQUFZLEVBQUUsS0FBSztRQUNuQixRQUFRLEVBQUUsVUFBVTtRQUNwQixRQUFRLEVBQUUsT0FBTztLQUNsQixDQUFDO0lBRUYsSUFBTSxVQUFVLEdBQXdCO1FBQ3RDLEtBQUssRUFBRSxNQUFNO1FBQ2IsTUFBTSxFQUFFLE1BQU07UUFDZCxZQUFZLEVBQUUsS0FBSztRQUNuQixTQUFTLEVBQUUsT0FBTztRQUNsQixXQUFXLEVBQUUsTUFBTTtLQUNwQixDQUFDO0lBRUYsSUFBTSxhQUFhLEdBQXdCO1FBQ3pDLElBQUksRUFBRSxDQUFDO1FBQ1AsUUFBUSxFQUFFLFFBQVE7UUFDbEIsVUFBVSxFQUFFLFFBQVE7UUFDcEIsWUFBWSxFQUFFLFVBQVU7S0FDekIsQ0FBQztJQUVGLElBQU0sYUFBYSxHQUF3QjtRQUN6QyxVQUFVLEVBQUUsTUFBTTtRQUNsQixRQUFRLEVBQUUsTUFBTTtLQUNqQixDQUFDO0lBRUYsSUFBTSxhQUFhLEdBQXdCO1FBQ3pDLFFBQVEsRUFBRSxNQUFNO1FBQ2hCLEtBQUssRUFBRSxNQUFNO0tBQ2QsQ0FBQztJQUVGLElBQU0saUJBQWlCLEdBQXdCO1FBQzdDLFVBQVUsRUFBRSxNQUFNO1FBQ2xCLE1BQU0sRUFBRSxNQUFNO1FBQ2QsTUFBTSxFQUFFLFNBQVM7UUFDakIsT0FBTyxFQUFFLEtBQUs7UUFDZCxRQUFRLEVBQUUsVUFBVTtRQUNwQixHQUFHLEVBQUUsS0FBSztRQUNWLEtBQUssRUFBRSxLQUFLO0tBQ2IsQ0FBQztJQUVGLElBQU0scUJBQXFCLEdBQXdCO1FBQ2pELFFBQVEsRUFBRSxVQUFVO1FBQ3BCLEdBQUcsRUFBRSxDQUFDO1FBQ04sSUFBSSxFQUFFLENBQUM7UUFDUCxLQUFLLEVBQUUsQ0FBQztRQUNSLE1BQU0sRUFBRSxDQUFDO1FBQ1QsZUFBZSxFQUFFLDBCQUEwQjtRQUMzQyxPQUFPLEVBQUUsTUFBTTtRQUNmLFVBQVUsRUFBRSxRQUFRO1FBQ3BCLGNBQWMsRUFBRSxRQUFRO1FBQ3hCLFlBQVksRUFBRSxLQUFLO0tBQ3BCLENBQUM7SUFFRixPQUFPLENBQ0wsZUFBSyxLQUFLLEVBQUUsY0FBYyxhQUN2QixPQUFPLElBQUksWUFBWSxDQUFDLENBQUMsQ0FBQyxDQUN6QixjQUFLLEdBQUcsRUFBRSxZQUFZLEVBQUUsR0FBRyxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFLFVBQVUsR0FBSSxDQUM5RCxDQUFDLENBQUMsQ0FBQyxDQUNGLGNBQUssS0FBSyxFQUFFLEVBQUUsV0FBVyxFQUFFLE1BQU0sRUFBRSxZQUFHLFdBQVcsQ0FBQyxRQUFRLENBQUMsR0FBTyxDQUNuRSxFQUNELGVBQUssS0FBSyxFQUFFLGFBQWEsYUFDdkIsY0FBSyxLQUFLLEVBQUUsYUFBYSxFQUFFLEtBQUssRUFBRSxJQUFJLENBQUMsSUFBSSxZQUFHLElBQUksQ0FBQyxJQUFJLEdBQU8sRUFDOUQsZUFBSyxLQUFLLEVBQUUsYUFBYSxhQUFHLENBQUMsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLFdBQVUsSUFDL0QsRUFDTCxDQUFDLFNBQVMsSUFBSSxDQUNiLGlCQUFRLE9BQU8sRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLGlCQUFpQixFQUFFLEtBQUssRUFBQyxhQUFhLHVCQUUvRCxDQUNWLEVBQ0EsU0FBUyxJQUFJLENBQ1osY0FBSyxLQUFLLEVBQUUscUJBQXFCLFlBQy9CLEtBQUMsT0FBTyxJQUFDLElBQUksRUFBRSxFQUFFLEVBQUUsS0FBSyxFQUFDLE1BQU0sR0FBRyxHQUM5QixDQUNQLElBQ0csQ0FDUCxDQUFDO0FBQ0osQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcbmltcG9ydCB7IFNwaW5uZXIgfSBmcm9tICcuL1NwaW5uZXInO1xuaW1wb3J0IHsgZ2V0RmlsZUljb24gfSBmcm9tICcuL0ZpbGVUeXBlSWNvbic7XG5cbmludGVyZmFjZSBBdHRhY2htZW50UHJldmlld1Byb3BzIHtcbiAgZmlsZTogRmlsZTtcbiAgb25SZW1vdmU6ICgpID0+IHZvaWQ7XG4gIHVwbG9hZGluZz86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBBdHRhY2htZW50UHJldmlldzogUmVhY3QuRkM8QXR0YWNobWVudFByZXZpZXdQcm9wcz4gPSAoeyBmaWxlLCBvblJlbW92ZSwgdXBsb2FkaW5nID0gZmFsc2UgfSkgPT4ge1xuICBjb25zdCBmaWxlVHlwZSA9IGZpbGUudHlwZTtcbiAgY29uc3QgaXNJbWFnZSA9IGZpbGVUeXBlLnN0YXJ0c1dpdGgoJ2ltYWdlLycpO1xuICBjb25zdCBbaW1hZ2VQcmV2aWV3LCBzZXRJbWFnZVByZXZpZXddID0gUmVhY3QudXNlU3RhdGU8c3RyaW5nIHwgbnVsbD4obnVsbCk7XG5cbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoaXNJbWFnZSkge1xuICAgICAgY29uc3QgcmVhZGVyID0gbmV3IEZpbGVSZWFkZXIoKTtcbiAgICAgIHJlYWRlci5vbmxvYWRlbmQgPSAoKSA9PiB7XG4gICAgICAgIHNldEltYWdlUHJldmlldyhyZWFkZXIucmVzdWx0IGFzIHN0cmluZyk7XG4gICAgICB9O1xuICAgICAgcmVhZGVyLnJlYWRBc0RhdGFVUkwoZmlsZSk7XG4gICAgfVxuICB9LCBbZmlsZSwgaXNJbWFnZV0pO1xuXG4gIGNvbnN0IGNvbnRhaW5lclN0eWxlOiBSZWFjdC5DU1NQcm9wZXJ0aWVzID0ge1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBwYWRkaW5nOiAnOHB4JyxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICcjZjBmMGYwJyxcbiAgICBib3JkZXJSYWRpdXM6ICc4cHgnLFxuICAgIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICAgIG1heFdpZHRoOiAnMjUwcHgnLFxuICB9O1xuXG4gIGNvbnN0IGltYWdlU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgd2lkdGg6ICc0MHB4JyxcbiAgICBoZWlnaHQ6ICc0MHB4JyxcbiAgICBib3JkZXJSYWRpdXM6ICc0cHgnLFxuICAgIG9iamVjdEZpdDogJ2NvdmVyJyxcbiAgICBtYXJnaW5SaWdodDogJzEwcHgnLFxuICB9O1xuXG4gIGNvbnN0IGZpbGVJbmZvU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgZmxleDogMSxcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgd2hpdGVTcGFjZTogJ25vd3JhcCcsXG4gICAgdGV4dE92ZXJmbG93OiAnZWxsaXBzaXMnLFxuICB9O1xuXG4gIGNvbnN0IGZpbGVOYW1lU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgZm9udFdlaWdodDogJ2JvbGQnLFxuICAgIGZvbnRTaXplOiAnMTRweCcsXG4gIH07XG5cbiAgY29uc3QgZmlsZVNpemVTdHlsZTogUmVhY3QuQ1NTUHJvcGVydGllcyA9IHtcbiAgICBmb250U2l6ZTogJzEycHgnLFxuICAgIGNvbG9yOiAnIzY2NicsXG4gIH07XG5cbiAgY29uc3QgcmVtb3ZlQnV0dG9uU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgYmFja2dyb3VuZDogJ25vbmUnLFxuICAgIGJvcmRlcjogJ25vbmUnLFxuICAgIGN1cnNvcjogJ3BvaW50ZXInLFxuICAgIHBhZGRpbmc6ICc0cHgnLFxuICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgIHRvcDogJzRweCcsXG4gICAgcmlnaHQ6ICc0cHgnLFxuICB9O1xuXG4gIGNvbnN0IHVwbG9hZGluZ092ZXJsYXlTdHlsZTogUmVhY3QuQ1NTUHJvcGVydGllcyA9IHtcbiAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICB0b3A6IDAsXG4gICAgbGVmdDogMCxcbiAgICByaWdodDogMCxcbiAgICBib3R0b206IDAsXG4gICAgYmFja2dyb3VuZENvbG9yOiAncmdiYSgyNTUsIDI1NSwgMjU1LCAwLjcpJyxcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAganVzdGlmeUNvbnRlbnQ6ICdjZW50ZXInLFxuICAgIGJvcmRlclJhZGl1czogJzhweCcsXG4gIH07XG5cbiAgcmV0dXJuIChcbiAgICA8ZGl2IHN0eWxlPXtjb250YWluZXJTdHlsZX0+XG4gICAgICB7aXNJbWFnZSAmJiBpbWFnZVByZXZpZXcgPyAoXG4gICAgICAgIDxpbWcgc3JjPXtpbWFnZVByZXZpZXd9IGFsdD17ZmlsZS5uYW1lfSBzdHlsZT17aW1hZ2VTdHlsZX0gLz5cbiAgICAgICkgOiAoXG4gICAgICAgIDxkaXYgc3R5bGU9e3sgbWFyZ2luUmlnaHQ6ICcxMHB4JyB9fT57Z2V0RmlsZUljb24oZmlsZVR5cGUpfTwvZGl2PlxuICAgICAgKX1cbiAgICAgIDxkaXYgc3R5bGU9e2ZpbGVJbmZvU3R5bGV9PlxuICAgICAgICA8ZGl2IHN0eWxlPXtmaWxlTmFtZVN0eWxlfSB0aXRsZT17ZmlsZS5uYW1lfT57ZmlsZS5uYW1lfTwvZGl2PlxuICAgICAgICA8ZGl2IHN0eWxlPXtmaWxlU2l6ZVN0eWxlfT57KGZpbGUuc2l6ZSAvIDEwMjQpLnRvRml4ZWQoMil9IEtCPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICAgIHshdXBsb2FkaW5nICYmIChcbiAgICAgICAgPGJ1dHRvbiBvbkNsaWNrPXtvblJlbW92ZX0gc3R5bGU9e3JlbW92ZUJ1dHRvblN0eWxlfSB0aXRsZT1cIlJlbW92ZSBmaWxlXCI+XG4gICAgICAgICAgJiN4MjcxNTtcbiAgICAgICAgPC9idXR0b24+XG4gICAgICApfVxuICAgICAge3VwbG9hZGluZyAmJiAoXG4gICAgICAgIDxkaXYgc3R5bGU9e3VwbG9hZGluZ092ZXJsYXlTdHlsZX0+XG4gICAgICAgICAgPFNwaW5uZXIgc2l6ZT17MjR9IGNvbG9yPVwiIzAwMFwiIC8+XG4gICAgICAgIDwvZGl2PlxuICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKTtcbn07XG5cbiJdfQ==
77
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQXR0YWNobWVudFByZXZpZXcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tcG9uZW50cy9BdHRhY2htZW50UHJldmlldy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBSyxNQUFNLE9BQU8sQ0FBQztBQUMxQixPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQ3BDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQVE3QyxNQUFNLENBQUMsSUFBTSxpQkFBaUIsR0FBcUMsVUFBQyxFQUFxQztRQUFuQyxJQUFJLFVBQUEsRUFBRSxRQUFRLGNBQUEsRUFBRSxpQkFBaUIsRUFBakIsU0FBUyxtQkFBRyxLQUFLLEtBQUE7SUFDckcsSUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztJQUMzQixJQUFNLE9BQU8sR0FBRyxRQUFRLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3hDLElBQUEsS0FBa0MsS0FBSyxDQUFDLFFBQVEsQ0FBZ0IsSUFBSSxDQUFDLEVBQXBFLFlBQVksUUFBQSxFQUFFLGVBQWUsUUFBdUMsQ0FBQztJQUU1RSxLQUFLLENBQUMsU0FBUyxDQUFDO1FBQ2QsSUFBSSxPQUFPLEVBQUUsQ0FBQztZQUNaLElBQU0sUUFBTSxHQUFHLElBQUksVUFBVSxFQUFFLENBQUM7WUFDaEMsUUFBTSxDQUFDLFNBQVMsR0FBRztnQkFDakIsZUFBZSxDQUFDLFFBQU0sQ0FBQyxNQUFnQixDQUFDLENBQUM7WUFDM0MsQ0FBQyxDQUFDO1lBQ0YsUUFBTSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM3QixDQUFDO0lBQ0gsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7SUFFcEIsSUFBTSxjQUFjLEdBQXdCO1FBQzFDLE9BQU8sRUFBRSxNQUFNO1FBQ2YsVUFBVSxFQUFFLFFBQVE7UUFDcEIsT0FB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aW1wb3J0IFJlYWN0IGZyb20gJ3JlYWN0JztcbmltcG9ydCB7IFNwaW5uZXIgfSBmcm9tICcuL1NwaW5uZXInO1xuaW1wb3J0IHsgZ2V0RmlsZUljb24gfSBmcm9tICcuL0ZpbGVUeXBlSWNvbic7XG5cbmludGVyZmFjZSBBdHRhY2htZW50UHJldmlld1Byb3BzIHtcbiAgZmlsZTogRmlsZTtcbiAgb25SZW1vdmU6ICgpID0+IHZvaWQ7XG4gIHVwbG9hZGluZz86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBBdHRhY2htZW50UHJldmlldzogUmVhY3QuRkM8QXR0YWNobWVudFByZXZpZXdQcm9wcz4gPSAoeyBmaWxlLCBvblJlbW92ZSwgdXBsb2FkaW5nID0gZmFsc2UgfSkgPT4ge1xuICBjb25zdCBmaWxlVHlwZSA9IGZpbGUudHlwZTtcbiAgY29uc3QgaXNJbWFnZSA9IGZpbGVUeXBlLnN0YXJ0c1dpdGgoJ2ltYWdlLycpO1xuICBjb25zdCBbaW1hZ2VQcmV2aWV3LCBzZXRJbWFnZVByZXZpZXddID0gUmVhY3QudXNlU3RhdGU8c3RyaW5nIHwgbnVsbD4obnVsbCk7XG5cbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoaXNJbWFnZSkge1xuICAgICAgY29uc3QgcmVhZGVyID0gbmV3IEZpbGVSZWFkZXIoKTtcbiAgICAgIHJlYWRlci5vbmxvYWRlbmQgPSAoKSA9PiB7XG4gICAgICAgIHNldEltYWdlUHJldmlldyhyZWFkZXIucmVzdWx0IGFzIHN0cmluZyk7XG4gICAgICB9O1xuICAgICAgcmVhZGVyLnJlYWRBc0RhdGFVUkwoZmlsZSk7XG4gICAgfVxuICB9LCBbZmlsZSwgaXNJbWFnZV0pO1xuXG4gIGNvbnN0IGNvbnRhaW5lclN0eWxlOiBSZWFjdC5DU1NQcm9wZXJ0aWVzID0ge1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBwYWRkaW5nOiAnOHB4JyxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICcjZjBmMGYwJyxcbiAgICBib3JkZXJSYWRpdXM6ICc4cHgnLFxuICAgIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICAgIG1heFdpZHRoOiAnMjUwcHgnLFxuICB9O1xuXG4gIGNvbnN0IGltYWdlU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgd2lkdGg6ICc0MHB4JyxcbiAgICBoZWlnaHQ6ICc0MHB4JyxcbiAgICBib3JkZXJSYWRpdXM6ICc0cHgnLFxuICAgIG9iamVjdEZpdDogJ2NvdmVyJyxcbiAgICBtYXJnaW5SaWdodDogJzEwcHgnLFxuICB9O1xuXG4gIGNvbnN0IGZpbGVJbmZvU3R5bGU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gICAgZmxleDogMSxcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgd2hpdGVTcGFjZTogJ25vd3JhcCcsXG4gICAgdGV4dE92ZXJmbG93OiAnZWxsaXBzaXMnLFxuICAgIHdpZHRoOiAnMTAwJScsXG4gIH07XG5cbiAgY29uc3QgZmlsZU5hbWVTdHlsZTogUmVhY3QuQ1NTUHJvcGVydGllcyA9IHtcbiAgICB3aWR0aDogJzkwJScsXG4gICAgZm9udFdlaWdodDogJ2JvbGQnLFxuICAgIGZvbnRTaXplOiAnMTRweCcsXG4gICAgb3ZlcmZsb3c6ICdoaWRkZW4nLFxuICAgIHRleHRPdmVyZmxvdzogJ2VsbGlwc2lzJyxcbiAgICB3aGl0ZVNwYWNlOiAnbm93cmFwJyxcbiAgfTtcblxuICBjb25zdCBmaWxlU2l6ZVN0eWxlOiBSZWFjdC5DU1NQcm9wZXJ0aWVzID0ge1xuICAgIGZvbnRTaXplOiAnMTJweCcsXG4gICAgY29sb3I6ICcjNjY2JyxcbiAgfTtcblxuICBjb25zdCByZW1vdmVCdXR0b25TdHlsZTogUmVhY3QuQ1NTUHJvcGVydGllcyA9IHtcbiAgICBiYWNrZ3JvdW5kOiAnbm9uZScsXG4gICAgYm9yZGVyOiAnbm9uZScsXG4gICAgY3Vyc29yOiAncG9pbnRlcicsXG4gICAgcGFkZGluZzogJzRweCcsXG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgdG9wOiAnNHB4JyxcbiAgICByaWdodDogJzRweCcsXG4gIH07XG5cbiAgY29uc3QgdXBsb2FkaW5nT3ZlcmxheVN0eWxlOiBSZWFjdC5DU1NQcm9wZXJ0aWVzID0ge1xuICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgIHRvcDogMCxcbiAgICBsZWZ0OiAwLFxuICAgIHJpZ2h0OiAwLFxuICAgIGJvdHRvbTogMCxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6ICdyZ2JhKDI1NSwgMjU1LCAyNTUsIDAuNyknLFxuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBhbGlnbkl0ZW1zOiAnY2VudGVyJyxcbiAgICBqdXN0aWZ5Q29udGVudDogJ2NlbnRlcicsXG4gICAgYm9yZGVyUmFkaXVzOiAnOHB4JyxcbiAgfTtcblxuICByZXR1cm4gKFxuICAgIDxkaXYgc3R5bGU9e2NvbnRhaW5lclN0eWxlfT5cbiAgICAgIHtpc0ltYWdlICYmIGltYWdlUHJldmlldyA/IChcbiAgICAgICAgPGltZyBzcmM9e2ltYWdlUHJldmlld30gYWx0PXtmaWxlLm5hbWV9IHN0eWxlPXtpbWFnZVN0eWxlfSAvPlxuICAgICAgKSA6IChcbiAgICAgICAgPGRpdiBzdHlsZT17eyBtYXJnaW5SaWdodDogJzEwcHgnIH19PntnZXRGaWxlSWNvbihmaWxlVHlwZSl9PC9kaXY+XG4gICAgICApfVxuICAgICAgPGRpdiBzdHlsZT17ZmlsZUluZm9TdHlsZX0+XG4gICAgICAgIDxkaXYgc3R5bGU9e2ZpbGVOYW1lU3R5bGV9IHRpdGxlPXtmaWxlLm5hbWV9PntmaWxlLm5hbWV9PC9kaXY+XG4gICAgICAgIDxkaXYgc3R5bGU9e2ZpbGVTaXplU3R5bGV9PnsoZmlsZS5zaXplID8gZmlsZS5zaXplIC8gMTAyNCA6IDApLnRvRml4ZWQoMil9IEtCPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICAgIHshdXBsb2FkaW5nICYmIChcbiAgICAgICAgPGJ1dHRvbiBvbkNsaWNrPXtvblJlbW92ZX0gc3R5bGU9e3JlbW92ZUJ1dHRvblN0eWxlfSB0aXRsZT1cIlJlbW92ZSBmaWxlXCI+XG4gICAgICAgICAgJiN4MjcxNTtcbiAgICAgICAgPC9idXR0b24+XG4gICAgICApfVxuICAgICAge3VwbG9hZGluZyAmJiAoXG4gICAgICAgIDxkaXYgc3R5bGU9e3VwbG9hZGluZ092ZXJsYXlTdHlsZX0+XG4gICAgICAgICAgPFNwaW5uZXIgc2l6ZT17MjR9IGNvbG9yPVwiIzAwMFwiIC8+XG4gICAgICAgIDwvZGl2PlxuICAgICAgKX1cbiAgICA8L2Rpdj5cbiAgKTtcbn07XG5cbiJdfQ==
@@ -1,6 +1,5 @@
1
1
  import React from 'react';
2
2
  import { ChatMessage, ScheduleItem, Translations } from '../types';
3
- export { AttachmentPreview } from './AttachmentPreview';
4
3
  interface ChatMessageProps {
5
4
  message: ChatMessage;
6
5
  theme?: {
@@ -31,3 +30,4 @@ interface ChatMessageProps {
31
30
  isAgentTyping?: boolean;
32
31
  }
33
32
  export declare const ChatMessageComponent: React.FC<ChatMessageProps>;
33
+ export {};
@@ -14,11 +14,10 @@ import React from 'react';
14
14
  import { WelcomeCard } from './WelcomeCard';
15
15
  import { User, UserX, AlertCircle, ThumbsUp, ThumbsDown } from 'lucide-react';
16
16
  import { formatTimestamp } from '../utils/time';
17
- import { getFileIcon } from './FileTypeIcon';
18
17
  import { InteractiveContent } from './InteractiveContent';
19
18
  import { parseInteractiveContentBlocks } from '../utils/interactiveContent';
20
19
  import { defaultTranslations, getTranslationString, mergeTranslations } from '../utils/i18n';
21
- export { AttachmentPreview } from './AttachmentPreview';
20
+ import { UploadFilePreview } from './common/UploadFilePreview';
22
21
  export var ChatMessageComponent = function (_a) {
23
22
  var message = _a.message, theme = _a.theme, onPlayAudio = _a.onPlayAudio, _b = _a.isFirstMessage, isFirstMessage = _b === void 0 ? false : _b, _c = _a.isPrevSameSpeaker, isPrevSameSpeaker = _c === void 0 ? false : _c, onFeedback = _a.onFeedback, welcomeImageUrl = _a.welcomeImageUrl, welcomeTitleProp = _a.welcomeTitle, possibleQueries = _a.possibleQueries, onQuickQuery = _a.onQuickQuery, _d = _a.isLastMessage, isLastMessage = _d === void 0 ? false : _d, onQuickAction = _a.onQuickAction, onScheduleConfirm = _a.onScheduleConfirm, customTranslations = _a.translations, language = _a.language, agentName = _a.agentName, _e = _a.isAgentTyping, isAgentTyping = _e === void 0 ? false : _e;
24
23
  // Merge translations with defaults
@@ -33,7 +32,7 @@ export var ChatMessageComponent = function (_a) {
33
32
  var _f = React.useState(false), isHovered = _f[0], setIsHovered = _f[1];
34
33
  var editingFeedback = React.useState(false)[0];
35
34
  var displayText = React.useState(message.text)[0];
36
- var contentBlocks = React.useMemo(function () { return parseInteractiveContentBlocks(displayText); }, [displayText]);
35
+ var contentBlocks = React.useMemo(function () { return parseInteractiveContentBlocks(displayText, message === null || message === void 0 ? void 0 : message.type); }, [displayText]);
37
36
  // Format timestamp with translations
38
37
  var timestamp = React.useMemo(function () {
39
38
  var timeTranslations = (translations === null || translations === void 0 ? void 0 : translations.time) ? {
@@ -238,18 +237,7 @@ export var ChatMessageComponent = function (_a) {
238
237
  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 }) }) }));
239
238
  }
240
239
  return (_jsxs("div", { style: messageContainerStyle, onMouseEnter: function () { return setIsHovered(true); }, onMouseLeave: function () { if (!editingFeedback)
241
- setIsHovered(false); }, children: [_jsx("div", { style: labelRowStyle, children: isUser ? (_jsxs(_Fragment, { children: [_jsx("div", { style: topTimestampStyle, children: timestamp }), _jsx("div", { style: messageLabelStyle, children: t('labels.you') })] })) : (_jsxs(_Fragment, { children: [_jsx("div", { style: messageLabelStyle, children: agentName || t('labels.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: {
242
- display: 'flex',
243
- alignItems: 'center',
244
- gap: '12px',
245
- padding: '12px 14px',
246
- backgroundColor: isUser ? 'rgba(255,255,255,0.12)' : '#f5f5f5',
247
- border: isUser ? '1px solid rgba(255,255,255,0.18)' : '1px solid #e5e5e5',
248
- borderRadius: '12px',
249
- minWidth: '260px',
250
- maxWidth: '360px',
251
- pointerEvents: 'none',
252
- }, 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: _jsx("div", { style: { display: 'flex', flexDirection: 'column', gap: 12 }, children: _jsx(InteractiveContent, { blocks: contentBlocks, primaryColor: primaryColor, textColor: bubbleTextColor, isActionable: !isUser && isLastMessage && !isAgentTyping, onQuickAction: onQuickAction, onScheduleConfirm: onScheduleConfirm }) }) }) })), 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: t('feedback.thumbsUp'), onClick: function () {
240
+ setIsHovered(false); }, children: [_jsx("div", { style: labelRowStyle, children: isUser ? (_jsxs(_Fragment, { children: [_jsx("div", { style: topTimestampStyle, children: timestamp }), _jsx("div", { style: messageLabelStyle, children: t('labels.you') })] })) : (_jsxs(_Fragment, { children: [_jsx("div", { style: messageLabelStyle, children: agentName || t('labels.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 (_jsx(UploadFilePreview, { file: attachment }, 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: _jsx("div", { style: { display: 'flex', flexDirection: 'column', gap: 12 }, children: _jsx(InteractiveContent, { blocks: contentBlocks, primaryColor: primaryColor, textColor: bubbleTextColor, isActionable: !isUser && isLastMessage && !isAgentTyping, onQuickAction: onQuickAction, onScheduleConfirm: onScheduleConfirm }) }) }) })), 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: t('feedback.thumbsUp'), onClick: function () {
253
241
  // Use message_id from socket, fallback to id if message_id doesn't exist
254
242
  var msgId = message.message_id || message.id;
255
243
  if (msgId && onFeedback) {
@@ -269,4 +257,4 @@ export var ChatMessageComponent = function (_a) {
269
257
  }
270
258
  }, children: _jsx(ThumbsDown, { size: 18 }) })] })) }))] }));
271
259
  };
272
- //# 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 { getFileIcon } from './FileTypeIcon';\nimport { InteractiveContent } from './InteractiveContent';\nimport { parseInteractiveContentBlocks } from '../utils/interactiveContent';\nimport { defaultTranslations, getTranslationString, mergeTranslations } from '../utils/i18n';\nexport { AttachmentPreview } from './AttachmentPreview';\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),\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            <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            <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"]}
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"]}