@yushaw/sanqian-chat 0.2.3 → 0.2.6
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 +21 -0
- package/dist/core/index.d.mts +154 -1
- package/dist/core/index.d.ts +154 -1
- package/dist/main/index.d.mts +426 -2
- package/dist/main/index.d.ts +426 -2
- package/dist/main/index.js +1269 -4
- package/dist/main/index.mjs +1279 -3
- package/dist/preload/index.d.ts +49 -1
- package/dist/preload/index.js +42 -0
- package/dist/renderer/index.d.mts +300 -1
- package/dist/renderer/index.d.ts +300 -1
- package/dist/renderer/index.js +1068 -385
- package/dist/renderer/index.mjs +1003 -329
- package/package.json +1 -1
- package/src/renderer/styles/chat.css +37 -0
- package/src/renderer/styles/coreCss.ts +37 -0
- package/src/renderer/styles/safe.css +37 -0
package/dist/renderer/index.js
CHANGED
|
@@ -31,16 +31,23 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
|
|
|
31
31
|
var renderer_exports = {};
|
|
32
32
|
__export(renderer_exports, {
|
|
33
33
|
AlertBanner: () => AlertBanner,
|
|
34
|
+
AttachButton: () => AttachButton,
|
|
34
35
|
ChatInput: () => ChatInput,
|
|
35
36
|
CompactChat: () => CompactChat,
|
|
36
37
|
FloatingChat: () => FloatingChat,
|
|
37
38
|
HistoryList: () => HistoryList,
|
|
39
|
+
HistoryModal: () => HistoryModal,
|
|
38
40
|
HitlCard: () => HitlCard,
|
|
39
41
|
I18nProvider: () => I18nProvider,
|
|
40
42
|
IntermediateSteps: () => IntermediateSteps,
|
|
41
43
|
MarkdownRenderer: () => MarkdownRenderer,
|
|
42
44
|
MessageBubble: () => MessageBubble,
|
|
43
45
|
MessageList: () => MessageList,
|
|
46
|
+
ModeToggleButton: () => ModeToggleButton,
|
|
47
|
+
PanelControls: () => PanelControls,
|
|
48
|
+
PanelHeaderButtons: () => PanelHeaderButtons,
|
|
49
|
+
PanelResizer: () => PanelResizer,
|
|
50
|
+
Resizer: () => Resizer,
|
|
44
51
|
SanqianChat: () => SanqianChat,
|
|
45
52
|
SanqianChatMessage: () => SanqianChatMessage,
|
|
46
53
|
SanqianMessageList: () => SanqianMessageList,
|
|
@@ -56,7 +63,9 @@ __export(renderer_exports, {
|
|
|
56
63
|
ensureFullChatStyles: () => ensureFullChatStyles,
|
|
57
64
|
getTranslations: () => getTranslations,
|
|
58
65
|
resolveChatStrings: () => resolveChatStrings,
|
|
66
|
+
useAttachState: () => useAttachState,
|
|
59
67
|
useChat: () => useChat,
|
|
68
|
+
useChatPanel: () => useChatPanel,
|
|
60
69
|
useChatStyles: () => useChatStyles,
|
|
61
70
|
useConnection: () => useConnection,
|
|
62
71
|
useConversations: () => useConversations,
|
|
@@ -933,7 +942,14 @@ var CHAT_UI_STRINGS = {
|
|
|
933
942
|
hitlApprovalRequest: "Approval Request",
|
|
934
943
|
hitlInputRequest: "Input Request",
|
|
935
944
|
hitlApprovalRequired: "Approval Required",
|
|
936
|
-
hitlInputRequired: "Input Required"
|
|
945
|
+
hitlInputRequired: "Input Required",
|
|
946
|
+
// ChatPanel strings
|
|
947
|
+
attachWindow: "Dock",
|
|
948
|
+
detachWindow: "Undock",
|
|
949
|
+
floatWindow: "Float",
|
|
950
|
+
embedWindow: "Embed",
|
|
951
|
+
collapseSidebar: "Collapse",
|
|
952
|
+
history: "History"
|
|
937
953
|
},
|
|
938
954
|
zh: {
|
|
939
955
|
inputPlaceholder: "\u8F93\u5165\u6D88\u606F...",
|
|
@@ -983,7 +999,14 @@ var CHAT_UI_STRINGS = {
|
|
|
983
999
|
hitlApprovalRequest: "\u9700\u8981\u5BA1\u6279",
|
|
984
1000
|
hitlInputRequest: "\u9700\u8981\u8F93\u5165",
|
|
985
1001
|
hitlApprovalRequired: "\u9700\u8981\u5BA1\u6279",
|
|
986
|
-
hitlInputRequired: "\u9700\u8981\u8F93\u5165"
|
|
1002
|
+
hitlInputRequired: "\u9700\u8981\u8F93\u5165",
|
|
1003
|
+
// ChatPanel strings
|
|
1004
|
+
attachWindow: "\u5438\u9644",
|
|
1005
|
+
detachWindow: "\u5206\u79BB",
|
|
1006
|
+
floatWindow: "\u6D6E\u52A8",
|
|
1007
|
+
embedWindow: "\u5D4C\u5165",
|
|
1008
|
+
collapseSidebar: "\u6536\u8D77",
|
|
1009
|
+
history: "\u5386\u53F2"
|
|
987
1010
|
}
|
|
988
1011
|
};
|
|
989
1012
|
function resolveChatStrings(locale = "en", overrides) {
|
|
@@ -1027,6 +1050,12 @@ var toTranslations = (flat) => ({
|
|
|
1027
1050
|
untitled: flat.conversationUntitled,
|
|
1028
1051
|
delete: flat.delete,
|
|
1029
1052
|
deleteConfirm: flat.conversationDeleteConfirm
|
|
1053
|
+
},
|
|
1054
|
+
panel: {
|
|
1055
|
+
attachWindow: flat.attachWindow,
|
|
1056
|
+
detachWindow: flat.detachWindow,
|
|
1057
|
+
floatWindow: flat.floatWindow,
|
|
1058
|
+
embedWindow: flat.embedWindow
|
|
1030
1059
|
}
|
|
1031
1060
|
});
|
|
1032
1061
|
var I18nContext = (0, import_react3.createContext)(null);
|
|
@@ -1064,7 +1093,8 @@ function I18nProvider({
|
|
|
1064
1093
|
input: { ...base.input, ...custom.input },
|
|
1065
1094
|
message: { ...base.message, ...custom.message },
|
|
1066
1095
|
connection: { ...base.connection, ...custom.connection },
|
|
1067
|
-
conversation: { ...base.conversation, ...custom.conversation }
|
|
1096
|
+
conversation: { ...base.conversation, ...custom.conversation },
|
|
1097
|
+
panel: { ...base.panel, ...custom.panel }
|
|
1068
1098
|
};
|
|
1069
1099
|
}, [locale, customTranslations]);
|
|
1070
1100
|
return /* @__PURE__ */ (0, import_jsx_runtime2.jsx)(I18nContext.Provider, { value: { locale, setLocale, t }, children });
|
|
@@ -1500,6 +1530,43 @@ code {
|
|
|
1500
1530
|
animation: chat-fade-in 0.2s ease-out;
|
|
1501
1531
|
}
|
|
1502
1532
|
|
|
1533
|
+
/* Modal animations */
|
|
1534
|
+
@keyframes chat-modal-fade-in {
|
|
1535
|
+
from { opacity: 0; }
|
|
1536
|
+
to { opacity: 1; }
|
|
1537
|
+
}
|
|
1538
|
+
|
|
1539
|
+
@keyframes chat-modal-fade-out {
|
|
1540
|
+
from { opacity: 1; }
|
|
1541
|
+
to { opacity: 0; }
|
|
1542
|
+
}
|
|
1543
|
+
|
|
1544
|
+
@keyframes chat-modal-slide-in {
|
|
1545
|
+
from { opacity: 0; transform: scale(0.95); }
|
|
1546
|
+
to { opacity: 1; transform: scale(1); }
|
|
1547
|
+
}
|
|
1548
|
+
|
|
1549
|
+
@keyframes chat-modal-slide-out {
|
|
1550
|
+
from { opacity: 1; transform: scale(1); }
|
|
1551
|
+
to { opacity: 0; transform: scale(0.95); }
|
|
1552
|
+
}
|
|
1553
|
+
|
|
1554
|
+
.chat-modal-backdrop {
|
|
1555
|
+
animation: chat-modal-fade-in 0.15s ease-out forwards;
|
|
1556
|
+
}
|
|
1557
|
+
|
|
1558
|
+
.chat-modal-backdrop.chat-modal-closing {
|
|
1559
|
+
animation: chat-modal-fade-out 0.12s ease-in forwards;
|
|
1560
|
+
}
|
|
1561
|
+
|
|
1562
|
+
.chat-modal-content {
|
|
1563
|
+
animation: chat-modal-slide-in 0.15s ease-out forwards;
|
|
1564
|
+
}
|
|
1565
|
+
|
|
1566
|
+
.chat-modal-content.chat-modal-closing {
|
|
1567
|
+
animation: chat-modal-slide-out 0.12s ease-in forwards;
|
|
1568
|
+
}
|
|
1569
|
+
|
|
1503
1570
|
/* ========================================
|
|
1504
1571
|
Prose Overrides
|
|
1505
1572
|
======================================== */
|
|
@@ -5041,6 +5108,136 @@ function useResolvedUiConfig(config) {
|
|
|
5041
5108
|
}, [ipcConfig, config]);
|
|
5042
5109
|
}
|
|
5043
5110
|
|
|
5111
|
+
// src/renderer/hooks/useChatPanel.ts
|
|
5112
|
+
var import_react10 = require("react");
|
|
5113
|
+
var getChatPanelApi = () => {
|
|
5114
|
+
return window.chatPanel;
|
|
5115
|
+
};
|
|
5116
|
+
function useChatPanel() {
|
|
5117
|
+
const [mode, setModeState] = (0, import_react10.useState)("embedded");
|
|
5118
|
+
const [visible, setVisible] = (0, import_react10.useState)(false);
|
|
5119
|
+
const [width, setWidthState] = (0, import_react10.useState)(360);
|
|
5120
|
+
(0, import_react10.useEffect)(() => {
|
|
5121
|
+
const api = getChatPanelApi();
|
|
5122
|
+
if (!api) {
|
|
5123
|
+
console.warn("[useChatPanel] chatPanel API not available");
|
|
5124
|
+
return;
|
|
5125
|
+
}
|
|
5126
|
+
api.getMode?.().then(setModeState).catch(() => {
|
|
5127
|
+
});
|
|
5128
|
+
api.isVisible?.().then(setVisible).catch(() => {
|
|
5129
|
+
});
|
|
5130
|
+
api.getWidth?.().then(setWidthState).catch(() => {
|
|
5131
|
+
});
|
|
5132
|
+
const unsubMode = api.onModeChanged?.(setModeState);
|
|
5133
|
+
const unsubVisible = api.onVisibilityChanged?.(setVisible);
|
|
5134
|
+
return () => {
|
|
5135
|
+
unsubMode?.();
|
|
5136
|
+
unsubVisible?.();
|
|
5137
|
+
};
|
|
5138
|
+
}, []);
|
|
5139
|
+
const setMode = (0, import_react10.useCallback)(async (newMode) => {
|
|
5140
|
+
const api = getChatPanelApi();
|
|
5141
|
+
if (!api?.setMode) return;
|
|
5142
|
+
await api.setMode(newMode);
|
|
5143
|
+
setModeState(newMode);
|
|
5144
|
+
}, []);
|
|
5145
|
+
const toggleMode = (0, import_react10.useCallback)(async () => {
|
|
5146
|
+
const api = getChatPanelApi();
|
|
5147
|
+
if (!api?.toggleMode) return mode;
|
|
5148
|
+
const newMode = await api.toggleMode();
|
|
5149
|
+
setModeState(newMode);
|
|
5150
|
+
return newMode;
|
|
5151
|
+
}, [mode]);
|
|
5152
|
+
const show = (0, import_react10.useCallback)(async () => {
|
|
5153
|
+
const api = getChatPanelApi();
|
|
5154
|
+
if (!api?.show) return;
|
|
5155
|
+
await api.show();
|
|
5156
|
+
setVisible(true);
|
|
5157
|
+
}, []);
|
|
5158
|
+
const hide = (0, import_react10.useCallback)(async () => {
|
|
5159
|
+
const api = getChatPanelApi();
|
|
5160
|
+
if (!api?.hide) return;
|
|
5161
|
+
await api.hide();
|
|
5162
|
+
setVisible(false);
|
|
5163
|
+
}, []);
|
|
5164
|
+
const toggle = (0, import_react10.useCallback)(async () => {
|
|
5165
|
+
const api = getChatPanelApi();
|
|
5166
|
+
if (!api?.toggle) return;
|
|
5167
|
+
await api.toggle();
|
|
5168
|
+
setVisible((v) => !v);
|
|
5169
|
+
}, []);
|
|
5170
|
+
const setWidth = (0, import_react10.useCallback)(async (newWidth, animate) => {
|
|
5171
|
+
const api = getChatPanelApi();
|
|
5172
|
+
if (!api?.setWidth) return;
|
|
5173
|
+
await api.setWidth(newWidth, animate);
|
|
5174
|
+
setWidthState(newWidth);
|
|
5175
|
+
}, []);
|
|
5176
|
+
const onResizeEnd = (0, import_react10.useCallback)(async () => {
|
|
5177
|
+
const api = getChatPanelApi();
|
|
5178
|
+
if (!api?.onResizeEnd) return;
|
|
5179
|
+
await api.onResizeEnd();
|
|
5180
|
+
}, []);
|
|
5181
|
+
return {
|
|
5182
|
+
mode,
|
|
5183
|
+
visible,
|
|
5184
|
+
width,
|
|
5185
|
+
isEmbedded: mode === "embedded",
|
|
5186
|
+
isFloating: mode === "floating",
|
|
5187
|
+
setMode,
|
|
5188
|
+
toggleMode,
|
|
5189
|
+
show,
|
|
5190
|
+
hide,
|
|
5191
|
+
toggle,
|
|
5192
|
+
setWidth,
|
|
5193
|
+
onResizeEnd
|
|
5194
|
+
};
|
|
5195
|
+
}
|
|
5196
|
+
|
|
5197
|
+
// src/renderer/hooks/useAttachState.ts
|
|
5198
|
+
var import_react11 = require("react");
|
|
5199
|
+
var getChatPanelApi2 = () => {
|
|
5200
|
+
return window.chatPanel;
|
|
5201
|
+
};
|
|
5202
|
+
function useAttachState() {
|
|
5203
|
+
const [state, setState] = (0, import_react11.useState)("unavailable");
|
|
5204
|
+
const [isLoading, setIsLoading] = (0, import_react11.useState)(true);
|
|
5205
|
+
(0, import_react11.useEffect)(() => {
|
|
5206
|
+
const api = getChatPanelApi2();
|
|
5207
|
+
if (!api) {
|
|
5208
|
+
setIsLoading(false);
|
|
5209
|
+
return;
|
|
5210
|
+
}
|
|
5211
|
+
api.getAttachState?.().then((result) => {
|
|
5212
|
+
if (result.success) {
|
|
5213
|
+
setState(result.data);
|
|
5214
|
+
}
|
|
5215
|
+
}).catch(() => {
|
|
5216
|
+
}).finally(() => setIsLoading(false));
|
|
5217
|
+
const unsubscribe = api.onAttachStateChanged?.(setState);
|
|
5218
|
+
return () => {
|
|
5219
|
+
unsubscribe?.();
|
|
5220
|
+
};
|
|
5221
|
+
}, []);
|
|
5222
|
+
const toggle = (0, import_react11.useCallback)(async () => {
|
|
5223
|
+
const api = getChatPanelApi2();
|
|
5224
|
+
if (!api?.toggleAttach) return state;
|
|
5225
|
+
const result = await api.toggleAttach();
|
|
5226
|
+
if (result.success) {
|
|
5227
|
+
setState(result.data);
|
|
5228
|
+
return result.data;
|
|
5229
|
+
}
|
|
5230
|
+
return state;
|
|
5231
|
+
}, [state]);
|
|
5232
|
+
return {
|
|
5233
|
+
state,
|
|
5234
|
+
isAttached: state === "attached",
|
|
5235
|
+
isAvailable: state !== "unavailable",
|
|
5236
|
+
isLoading,
|
|
5237
|
+
toggle
|
|
5238
|
+
};
|
|
5239
|
+
}
|
|
5240
|
+
|
|
5044
5241
|
// src/core/history.ts
|
|
5045
5242
|
function safeParseArgs(value) {
|
|
5046
5243
|
if (!value) return void 0;
|
|
@@ -5800,38 +5997,38 @@ function createSdkAdapter(config) {
|
|
|
5800
5997
|
var import_browser = require("@yushaw/sanqian-sdk/browser");
|
|
5801
5998
|
|
|
5802
5999
|
// src/renderer/components/MessageList.tsx
|
|
5803
|
-
var
|
|
6000
|
+
var import_react12 = require("react");
|
|
5804
6001
|
var import_jsx_runtime3 = require("react/jsx-runtime");
|
|
5805
6002
|
var SCROLL_THRESHOLD = 100;
|
|
5806
|
-
var MessageList = (0,
|
|
6003
|
+
var MessageList = (0, import_react12.memo)(function MessageList2({
|
|
5807
6004
|
messages,
|
|
5808
6005
|
className = "",
|
|
5809
6006
|
renderMessage,
|
|
5810
6007
|
autoScroll = true,
|
|
5811
6008
|
scrollBehavior = "smooth"
|
|
5812
6009
|
}) {
|
|
5813
|
-
const containerRef = (0,
|
|
5814
|
-
const isNearBottomRef = (0,
|
|
5815
|
-
const checkIfNearBottom = (0,
|
|
6010
|
+
const containerRef = (0, import_react12.useRef)(null);
|
|
6011
|
+
const isNearBottomRef = (0, import_react12.useRef)(true);
|
|
6012
|
+
const checkIfNearBottom = (0, import_react12.useCallback)(() => {
|
|
5816
6013
|
const container = containerRef.current;
|
|
5817
6014
|
if (!container) return true;
|
|
5818
6015
|
return container.scrollTop <= SCROLL_THRESHOLD;
|
|
5819
6016
|
}, []);
|
|
5820
|
-
const handleScroll = (0,
|
|
6017
|
+
const handleScroll = (0, import_react12.useCallback)(() => {
|
|
5821
6018
|
isNearBottomRef.current = checkIfNearBottom();
|
|
5822
6019
|
}, [checkIfNearBottom]);
|
|
5823
|
-
const scrollToBottom = (0,
|
|
6020
|
+
const scrollToBottom = (0, import_react12.useCallback)(
|
|
5824
6021
|
(behavior = scrollBehavior) => {
|
|
5825
6022
|
containerRef.current?.scrollTo({ top: 0, behavior });
|
|
5826
6023
|
},
|
|
5827
6024
|
[scrollBehavior]
|
|
5828
6025
|
);
|
|
5829
|
-
(0,
|
|
6026
|
+
(0, import_react12.useEffect)(() => {
|
|
5830
6027
|
if (autoScroll && isNearBottomRef.current) {
|
|
5831
6028
|
scrollToBottom();
|
|
5832
6029
|
}
|
|
5833
6030
|
}, [messages, autoScroll, scrollToBottom]);
|
|
5834
|
-
(0,
|
|
6031
|
+
(0, import_react12.useEffect)(() => {
|
|
5835
6032
|
scrollToBottom("instant");
|
|
5836
6033
|
isNearBottomRef.current = true;
|
|
5837
6034
|
}, [scrollToBottom]);
|
|
@@ -5849,9 +6046,9 @@ var MessageList = (0, import_react10.memo)(function MessageList2({
|
|
|
5849
6046
|
});
|
|
5850
6047
|
|
|
5851
6048
|
// src/renderer/components/MessageBubble.tsx
|
|
5852
|
-
var
|
|
6049
|
+
var import_react13 = require("react");
|
|
5853
6050
|
var import_jsx_runtime4 = require("react/jsx-runtime");
|
|
5854
|
-
var MessageBubble = (0,
|
|
6051
|
+
var MessageBubble = (0, import_react13.memo)(function MessageBubble2({
|
|
5855
6052
|
message,
|
|
5856
6053
|
className = "",
|
|
5857
6054
|
children,
|
|
@@ -5868,20 +6065,20 @@ var MessageBubble = (0, import_react11.memo)(function MessageBubble2({
|
|
|
5868
6065
|
});
|
|
5869
6066
|
|
|
5870
6067
|
// src/renderer/components/SanqianChat.tsx
|
|
5871
|
-
var
|
|
6068
|
+
var import_react24 = require("react");
|
|
5872
6069
|
|
|
5873
6070
|
// src/renderer/utils/chatConfig.ts
|
|
5874
|
-
var
|
|
6071
|
+
var import_react14 = require("react");
|
|
5875
6072
|
var getSystemTheme = () => {
|
|
5876
6073
|
if (typeof window === "undefined") return "light";
|
|
5877
6074
|
return window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
|
|
5878
6075
|
};
|
|
5879
6076
|
function useResolvedTheme(mode = "auto") {
|
|
5880
|
-
const [resolvedTheme, setResolvedTheme] = (0,
|
|
6077
|
+
const [resolvedTheme, setResolvedTheme] = (0, import_react14.useState)(() => {
|
|
5881
6078
|
if (mode === "auto") return getSystemTheme();
|
|
5882
6079
|
return mode === "dark" ? "dark" : "light";
|
|
5883
6080
|
});
|
|
5884
|
-
(0,
|
|
6081
|
+
(0, import_react14.useEffect)(() => {
|
|
5885
6082
|
if (mode !== "auto") {
|
|
5886
6083
|
setResolvedTheme(mode === "dark" ? "dark" : "light");
|
|
5887
6084
|
return;
|
|
@@ -5943,7 +6140,7 @@ function getBaseColors(isDark) {
|
|
|
5943
6140
|
};
|
|
5944
6141
|
}
|
|
5945
6142
|
function useAccentStyle(accentColor) {
|
|
5946
|
-
return (0,
|
|
6143
|
+
return (0, import_react14.useMemo)(() => {
|
|
5947
6144
|
if (!accentColor) return void 0;
|
|
5948
6145
|
const rgb = parseColorToRgb(accentColor);
|
|
5949
6146
|
if (!rgb) {
|
|
@@ -5985,7 +6182,7 @@ function resolveOnPin(onPin) {
|
|
|
5985
6182
|
}
|
|
5986
6183
|
|
|
5987
6184
|
// src/renderer/utils/useChatHeader.tsx
|
|
5988
|
-
var
|
|
6185
|
+
var import_react15 = require("react");
|
|
5989
6186
|
|
|
5990
6187
|
// src/renderer/assets/sanqianLogo.ts
|
|
5991
6188
|
var SANQIAN_LOGO_DATA_URI = "";
|
|
@@ -6028,11 +6225,11 @@ var resolveLogoNode = (logo, size, alt = SANQIAN_LOGO_ALT) => {
|
|
|
6028
6225
|
|
|
6029
6226
|
// src/renderer/utils/useChatHeader.tsx
|
|
6030
6227
|
var useChatHeader = (config) => {
|
|
6031
|
-
const resolvedOnClose = (0,
|
|
6032
|
-
const resolvedOnPin = (0,
|
|
6033
|
-
const [isPinned, setIsPinned] = (0,
|
|
6034
|
-
const isFirstRender = (0,
|
|
6035
|
-
(0,
|
|
6228
|
+
const resolvedOnClose = (0, import_react15.useMemo)(() => resolveOnClose(config?.onClose), [config?.onClose]);
|
|
6229
|
+
const resolvedOnPin = (0, import_react15.useMemo)(() => resolveOnPin(config?.onPin), [config?.onPin]);
|
|
6230
|
+
const [isPinned, setIsPinned] = (0, import_react15.useState)(config?.alwaysOnTop ?? false);
|
|
6231
|
+
const isFirstRender = (0, import_react15.useRef)(true);
|
|
6232
|
+
(0, import_react15.useEffect)(() => {
|
|
6036
6233
|
if (isFirstRender.current) {
|
|
6037
6234
|
isFirstRender.current = false;
|
|
6038
6235
|
return;
|
|
@@ -6044,8 +6241,8 @@ var useChatHeader = (config) => {
|
|
|
6044
6241
|
}, [config?.alwaysOnTop, resolvedOnPin]);
|
|
6045
6242
|
const showPin = !!resolvedOnPin || typeof config?.alwaysOnTop === "boolean";
|
|
6046
6243
|
const showClose = !!resolvedOnClose;
|
|
6047
|
-
const logoNode = (0,
|
|
6048
|
-
const togglePin = (0,
|
|
6244
|
+
const logoNode = (0, import_react15.useMemo)(() => resolveLogoNode(config?.logo, "header"), [config?.logo]);
|
|
6245
|
+
const togglePin = (0, import_react15.useCallback)(() => {
|
|
6049
6246
|
setIsPinned((prev) => {
|
|
6050
6247
|
const nextPinned = !prev;
|
|
6051
6248
|
resolvedOnPin?.(nextPinned);
|
|
@@ -6063,27 +6260,27 @@ var useChatHeader = (config) => {
|
|
|
6063
6260
|
};
|
|
6064
6261
|
|
|
6065
6262
|
// src/renderer/components/SanqianMessageList.tsx
|
|
6066
|
-
var
|
|
6263
|
+
var import_react20 = require("react");
|
|
6067
6264
|
var import_react_virtuoso = require("react-virtuoso");
|
|
6068
6265
|
|
|
6069
6266
|
// src/renderer/components/SanqianChatMessage.tsx
|
|
6070
|
-
var
|
|
6267
|
+
var import_react19 = require("react");
|
|
6071
6268
|
|
|
6072
6269
|
// src/renderer/renderers/MarkdownRenderer.tsx
|
|
6073
|
-
var
|
|
6270
|
+
var import_react16 = require("react");
|
|
6074
6271
|
var import_streamdown = require("streamdown");
|
|
6075
6272
|
var import_rehype_harden = require("rehype-harden");
|
|
6076
6273
|
var import_remark_gfm = __toESM(require("remark-gfm"));
|
|
6077
6274
|
var import_jsx_runtime6 = require("react/jsx-runtime");
|
|
6078
6275
|
var PROSE_CLASSES = "prose prose-chat max-w-none";
|
|
6079
|
-
var MarkdownRenderer = (0,
|
|
6276
|
+
var MarkdownRenderer = (0, import_react16.memo)(function MarkdownRenderer2({
|
|
6080
6277
|
content,
|
|
6081
6278
|
isStreaming = false,
|
|
6082
6279
|
className = "",
|
|
6083
6280
|
components,
|
|
6084
6281
|
onLinkClick
|
|
6085
6282
|
}) {
|
|
6086
|
-
const rehypePlugins = (0,
|
|
6283
|
+
const rehypePlugins = (0, import_react16.useMemo)(() => {
|
|
6087
6284
|
const origin = typeof window !== "undefined" ? window.location.origin : "http://localhost";
|
|
6088
6285
|
return [
|
|
6089
6286
|
[import_rehype_harden.harden, {
|
|
@@ -6097,10 +6294,10 @@ var MarkdownRenderer = (0, import_react14.memo)(function MarkdownRenderer2({
|
|
|
6097
6294
|
import_streamdown.defaultRehypePlugins.katex
|
|
6098
6295
|
];
|
|
6099
6296
|
}, []);
|
|
6100
|
-
const remarkPlugins = (0,
|
|
6297
|
+
const remarkPlugins = (0, import_react16.useMemo)(() => {
|
|
6101
6298
|
return [import_remark_gfm.default];
|
|
6102
6299
|
}, []);
|
|
6103
|
-
const customComponents = (0,
|
|
6300
|
+
const customComponents = (0, import_react16.useMemo)(() => {
|
|
6104
6301
|
const comps = {};
|
|
6105
6302
|
comps.p = ({ children }) => {
|
|
6106
6303
|
if (isStreaming && children && typeof children === "string" && children.endsWith("\u258C")) {
|
|
@@ -6157,10 +6354,10 @@ var MarkdownRenderer = (0, import_react14.memo)(function MarkdownRenderer2({
|
|
|
6157
6354
|
});
|
|
6158
6355
|
|
|
6159
6356
|
// src/renderer/components/IntermediateSteps.tsx
|
|
6160
|
-
var
|
|
6357
|
+
var import_react18 = require("react");
|
|
6161
6358
|
|
|
6162
6359
|
// src/renderer/renderers/ToolArgumentsDisplay.tsx
|
|
6163
|
-
var
|
|
6360
|
+
var import_react17 = require("react");
|
|
6164
6361
|
var import_jsx_runtime7 = require("react/jsx-runtime");
|
|
6165
6362
|
function formatValue(value, indent = 0) {
|
|
6166
6363
|
const indentStr = " ".repeat(indent);
|
|
@@ -6220,7 +6417,7 @@ function formatValue(value, indent = 0) {
|
|
|
6220
6417
|
}
|
|
6221
6418
|
return /* @__PURE__ */ (0, import_jsx_runtime7.jsx)("span", { className: "text-[var(--chat-text)]", children: String(value) });
|
|
6222
6419
|
}
|
|
6223
|
-
var ToolArgumentsDisplay = (0,
|
|
6420
|
+
var ToolArgumentsDisplay = (0, import_react17.memo)(function ToolArgumentsDisplay2({
|
|
6224
6421
|
args,
|
|
6225
6422
|
className = ""
|
|
6226
6423
|
}) {
|
|
@@ -6286,7 +6483,7 @@ function ExpandableText({
|
|
|
6286
6483
|
style,
|
|
6287
6484
|
maxLines = 3
|
|
6288
6485
|
}) {
|
|
6289
|
-
const [isExpanded, setIsExpanded] = (0,
|
|
6486
|
+
const [isExpanded, setIsExpanded] = (0, import_react18.useState)(false);
|
|
6290
6487
|
const trimmed = content.trim();
|
|
6291
6488
|
const lines = trimmed.split("\n");
|
|
6292
6489
|
const needsExpand = lines.length > maxLines;
|
|
@@ -6307,7 +6504,7 @@ function ExpandableText({
|
|
|
6307
6504
|
);
|
|
6308
6505
|
}
|
|
6309
6506
|
function ToolCallItem({ toolCall, toolResult }) {
|
|
6310
|
-
const [expanded, setExpanded] = (0,
|
|
6507
|
+
const [expanded, setExpanded] = (0, import_react18.useState)(false);
|
|
6311
6508
|
const hasArgs = toolCall.toolArgs && Object.keys(toolCall.toolArgs).length > 0;
|
|
6312
6509
|
const rawName = cleanToolName(toolCall.toolName);
|
|
6313
6510
|
const displayName = rawName && !/^[{\[]/.test(rawName.trim()) ? rawName : toolCall.toolCallId || "tool";
|
|
@@ -6345,7 +6542,7 @@ function StreamingToolCallItem({
|
|
|
6345
6542
|
isActive,
|
|
6346
6543
|
strings
|
|
6347
6544
|
}) {
|
|
6348
|
-
const [expanded, setExpanded] = (0,
|
|
6545
|
+
const [expanded, setExpanded] = (0, import_react18.useState)(false);
|
|
6349
6546
|
const hasArgs = toolCall.toolArgs && Object.keys(toolCall.toolArgs).length > 0;
|
|
6350
6547
|
const rawName = cleanToolName(toolCall.toolName);
|
|
6351
6548
|
const displayName = rawName && !/^[{\[]/.test(rawName.trim()) ? rawName : toolCall.toolCallId || "tool";
|
|
@@ -6413,7 +6610,7 @@ function groupBlocksImpl(blocks) {
|
|
|
6413
6610
|
}
|
|
6414
6611
|
return rounds;
|
|
6415
6612
|
}
|
|
6416
|
-
var ThinkingSection = (0,
|
|
6613
|
+
var ThinkingSection = (0, import_react18.memo)(function ThinkingSection2({
|
|
6417
6614
|
thinking,
|
|
6418
6615
|
currentThinking,
|
|
6419
6616
|
isStreaming,
|
|
@@ -6422,16 +6619,16 @@ var ThinkingSection = (0, import_react16.memo)(function ThinkingSection2({
|
|
|
6422
6619
|
className = "",
|
|
6423
6620
|
strings = {}
|
|
6424
6621
|
}) {
|
|
6425
|
-
const contentRef = (0,
|
|
6426
|
-
const isUserScrollingRef = (0,
|
|
6427
|
-
const [loadingSymbolIndex, setLoadingSymbolIndex] = (0,
|
|
6428
|
-
const [manualExpanded, setManualExpanded] = (0,
|
|
6622
|
+
const contentRef = (0, import_react18.useRef)(null);
|
|
6623
|
+
const isUserScrollingRef = (0, import_react18.useRef)(false);
|
|
6624
|
+
const [loadingSymbolIndex, setLoadingSymbolIndex] = (0, import_react18.useState)(0);
|
|
6625
|
+
const [manualExpanded, setManualExpanded] = (0, import_react18.useState)(false);
|
|
6429
6626
|
const effectiveIsStreaming = isComplete ? false : isStreaming;
|
|
6430
6627
|
const effectiveIsPaused = isComplete ? false : isPaused;
|
|
6431
6628
|
const isExpanded = effectiveIsStreaming || effectiveIsPaused || manualExpanded;
|
|
6432
6629
|
const displayContent = effectiveIsStreaming || effectiveIsPaused ? currentThinking || thinking : thinking;
|
|
6433
6630
|
const hasContent = displayContent && displayContent.split("\n").filter((line) => line.trim() && !line.trim().match(/^─+$/)).length > 0;
|
|
6434
|
-
(0,
|
|
6631
|
+
(0, import_react18.useEffect)(() => {
|
|
6435
6632
|
if (effectiveIsStreaming) {
|
|
6436
6633
|
const interval = setInterval(() => {
|
|
6437
6634
|
setLoadingSymbolIndex((prev) => (prev + 1) % LOADING_SYMBOLS.length);
|
|
@@ -6439,13 +6636,13 @@ var ThinkingSection = (0, import_react16.memo)(function ThinkingSection2({
|
|
|
6439
6636
|
return () => clearInterval(interval);
|
|
6440
6637
|
}
|
|
6441
6638
|
}, [effectiveIsStreaming]);
|
|
6442
|
-
(0,
|
|
6639
|
+
(0, import_react18.useEffect)(() => {
|
|
6443
6640
|
if (effectiveIsStreaming && isExpanded && contentRef.current && !isUserScrollingRef.current) {
|
|
6444
6641
|
contentRef.current.scrollTop = contentRef.current.scrollHeight;
|
|
6445
6642
|
}
|
|
6446
6643
|
}, [displayContent, effectiveIsStreaming, isExpanded]);
|
|
6447
|
-
const prevManualExpandedRef = (0,
|
|
6448
|
-
(0,
|
|
6644
|
+
const prevManualExpandedRef = (0, import_react18.useRef)(manualExpanded);
|
|
6645
|
+
(0, import_react18.useEffect)(() => {
|
|
6449
6646
|
const wasCollapsed = !prevManualExpandedRef.current;
|
|
6450
6647
|
const isNowExpanded = manualExpanded;
|
|
6451
6648
|
if (wasCollapsed && isNowExpanded && !effectiveIsStreaming && contentRef.current) {
|
|
@@ -6453,13 +6650,13 @@ var ThinkingSection = (0, import_react16.memo)(function ThinkingSection2({
|
|
|
6453
6650
|
}
|
|
6454
6651
|
prevManualExpandedRef.current = manualExpanded;
|
|
6455
6652
|
}, [manualExpanded, effectiveIsStreaming]);
|
|
6456
|
-
const handleScroll = (0,
|
|
6653
|
+
const handleScroll = (0, import_react18.useCallback)(() => {
|
|
6457
6654
|
if (!contentRef.current || !effectiveIsStreaming) return;
|
|
6458
6655
|
const { scrollTop, scrollHeight, clientHeight } = contentRef.current;
|
|
6459
6656
|
const isAtBottom = scrollHeight - scrollTop - clientHeight < 20;
|
|
6460
6657
|
isUserScrollingRef.current = !isAtBottom;
|
|
6461
6658
|
}, [effectiveIsStreaming]);
|
|
6462
|
-
(0,
|
|
6659
|
+
(0, import_react18.useEffect)(() => {
|
|
6463
6660
|
if (!effectiveIsStreaming) {
|
|
6464
6661
|
isUserScrollingRef.current = false;
|
|
6465
6662
|
}
|
|
@@ -6498,14 +6695,14 @@ var ThinkingSection = (0, import_react16.memo)(function ThinkingSection2({
|
|
|
6498
6695
|
isExpanded && /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { ref: contentRef, onScroll: handleScroll, className: "mt-1.5 ml-2 max-h-60 overflow-y-auto", children: /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("pre", { className: "whitespace-pre-wrap m-0 font-sans italic", style: { color: "var(--chat-text)", opacity: 0.7, fontSize: "0.8125rem", lineHeight: "1.3" }, children: displayContent?.trim() }) })
|
|
6499
6696
|
] });
|
|
6500
6697
|
});
|
|
6501
|
-
var IntermediateSteps = (0,
|
|
6698
|
+
var IntermediateSteps = (0, import_react18.memo)(function IntermediateSteps2({
|
|
6502
6699
|
blocks,
|
|
6503
6700
|
className = "",
|
|
6504
6701
|
defaultExpanded = false,
|
|
6505
6702
|
strings = {}
|
|
6506
6703
|
}) {
|
|
6507
|
-
const [isExpanded, setIsExpanded] = (0,
|
|
6508
|
-
const rounds = (0,
|
|
6704
|
+
const [isExpanded, setIsExpanded] = (0, import_react18.useState)(defaultExpanded);
|
|
6705
|
+
const rounds = (0, import_react18.useMemo)(() => groupBlocksIntoRounds(blocks), [blocks]);
|
|
6509
6706
|
if (rounds.length === 0) return null;
|
|
6510
6707
|
const stepCount = rounds.filter((round) => round.toolCalls.length > 0).length;
|
|
6511
6708
|
if (stepCount === 0) return null;
|
|
@@ -6587,7 +6784,7 @@ var IntermediateSteps = (0, import_react16.memo)(function IntermediateSteps2({
|
|
|
6587
6784
|
}) })
|
|
6588
6785
|
] });
|
|
6589
6786
|
});
|
|
6590
|
-
var StreamingTimeline = (0,
|
|
6787
|
+
var StreamingTimeline = (0, import_react18.memo)(function StreamingTimeline2({
|
|
6591
6788
|
blocks,
|
|
6592
6789
|
currentThinking = "",
|
|
6593
6790
|
isThinkingStreaming = false,
|
|
@@ -6596,14 +6793,14 @@ var StreamingTimeline = (0, import_react16.memo)(function StreamingTimeline2({
|
|
|
6596
6793
|
className = "",
|
|
6597
6794
|
strings = {}
|
|
6598
6795
|
}) {
|
|
6599
|
-
const [isExpanded, setIsExpanded] = (0,
|
|
6600
|
-
const isUserScrollingRef = (0,
|
|
6601
|
-
const timelineRef = (0,
|
|
6796
|
+
const [isExpanded, setIsExpanded] = (0, import_react18.useState)(true);
|
|
6797
|
+
const isUserScrollingRef = (0, import_react18.useRef)(false);
|
|
6798
|
+
const timelineRef = (0, import_react18.useRef)(null);
|
|
6602
6799
|
const isThinkingActive = isThinkingStreaming;
|
|
6603
6800
|
const isToolsActive = isToolCallsStreaming;
|
|
6604
|
-
const rounds = (0,
|
|
6801
|
+
const rounds = (0, import_react18.useMemo)(() => groupAllBlocksIntoRounds(blocks), [blocks]);
|
|
6605
6802
|
const activeThinking = currentThinking;
|
|
6606
|
-
const displayRounds = (0,
|
|
6803
|
+
const displayRounds = (0, import_react18.useMemo)(() => {
|
|
6607
6804
|
const result = [];
|
|
6608
6805
|
for (let roundIdx = 0; roundIdx < rounds.length; roundIdx++) {
|
|
6609
6806
|
const round = rounds[roundIdx];
|
|
@@ -6665,18 +6862,18 @@ var StreamingTimeline = (0, import_react16.memo)(function StreamingTimeline2({
|
|
|
6665
6862
|
}, [rounds, activeThinking, isThinkingActive, isToolsActive]);
|
|
6666
6863
|
const stepCount = rounds.filter((round) => round.toolCalls.length > 0).length;
|
|
6667
6864
|
const summary = `${stepCount} ${strings.steps || "\u6B65"}`;
|
|
6668
|
-
const handleScroll = (0,
|
|
6865
|
+
const handleScroll = (0, import_react18.useCallback)(() => {
|
|
6669
6866
|
if (!timelineRef.current) return;
|
|
6670
6867
|
const { scrollTop, scrollHeight, clientHeight } = timelineRef.current;
|
|
6671
6868
|
const isAtBottom = scrollHeight - scrollTop - clientHeight < 20;
|
|
6672
6869
|
isUserScrollingRef.current = !isAtBottom;
|
|
6673
6870
|
}, []);
|
|
6674
|
-
(0,
|
|
6871
|
+
(0, import_react18.useEffect)(() => {
|
|
6675
6872
|
if (isExpanded && timelineRef.current && !isUserScrollingRef.current) {
|
|
6676
6873
|
timelineRef.current.scrollTop = timelineRef.current.scrollHeight;
|
|
6677
6874
|
}
|
|
6678
6875
|
}, [displayRounds, isExpanded]);
|
|
6679
|
-
(0,
|
|
6876
|
+
(0, import_react18.useEffect)(() => {
|
|
6680
6877
|
if (!isThinkingActive && !isToolsActive) {
|
|
6681
6878
|
isUserScrollingRef.current = false;
|
|
6682
6879
|
}
|
|
@@ -6764,7 +6961,7 @@ var StreamingTimeline = (0, import_react16.memo)(function StreamingTimeline2({
|
|
|
6764
6961
|
|
|
6765
6962
|
// src/renderer/components/SanqianChatMessage.tsx
|
|
6766
6963
|
var import_jsx_runtime9 = require("react/jsx-runtime");
|
|
6767
|
-
var SanqianChatMessage = (0,
|
|
6964
|
+
var SanqianChatMessage = (0, import_react19.memo)(function SanqianChatMessage2({ message }) {
|
|
6768
6965
|
if (message.role === "tool") return null;
|
|
6769
6966
|
const isUser = message.role === "user";
|
|
6770
6967
|
const hasToolCalls = (message.toolCalls?.length ?? 0) > 0;
|
|
@@ -6824,9 +7021,9 @@ var import_jsx_runtime10 = require("react/jsx-runtime");
|
|
|
6824
7021
|
var AT_BOTTOM_THRESHOLD = 50;
|
|
6825
7022
|
var INCREASE_VIEWPORT_BY = { top: 1500, bottom: 1500 };
|
|
6826
7023
|
function ResizeAwareMessage({ message, onHeightChange }) {
|
|
6827
|
-
const containerRef = (0,
|
|
6828
|
-
const prevHeightRef = (0,
|
|
6829
|
-
(0,
|
|
7024
|
+
const containerRef = (0, import_react20.useRef)(null);
|
|
7025
|
+
const prevHeightRef = (0, import_react20.useRef)(0);
|
|
7026
|
+
(0, import_react20.useEffect)(() => {
|
|
6830
7027
|
const container = containerRef.current;
|
|
6831
7028
|
if (!container || !onHeightChange) return;
|
|
6832
7029
|
let rafId = null;
|
|
@@ -6848,16 +7045,16 @@ function ResizeAwareMessage({ message, onHeightChange }) {
|
|
|
6848
7045
|
}, [onHeightChange]);
|
|
6849
7046
|
return /* @__PURE__ */ (0, import_jsx_runtime10.jsx)("div", { ref: containerRef, children: /* @__PURE__ */ (0, import_jsx_runtime10.jsx)(SanqianChatMessage, { message }) });
|
|
6850
7047
|
}
|
|
6851
|
-
var SanqianMessageList = (0,
|
|
7048
|
+
var SanqianMessageList = (0, import_react20.forwardRef)(
|
|
6852
7049
|
function SanqianMessageList2({ messages, isLoading, className = "", emptyState, onClickEmpty, onAtBottomChange }, ref) {
|
|
6853
|
-
const virtuosoRef = (0,
|
|
6854
|
-
const [, setAtBottomState] = (0,
|
|
6855
|
-
const atBottomRef = (0,
|
|
6856
|
-
const isManualScrollingRef = (0,
|
|
6857
|
-
const manualScrollTimeoutRef = (0,
|
|
6858
|
-
const isFollowOutputScrollingRef = (0,
|
|
6859
|
-
const followOutputScrollTimeoutRef = (0,
|
|
6860
|
-
const setAtBottom = (0,
|
|
7050
|
+
const virtuosoRef = (0, import_react20.useRef)(null);
|
|
7051
|
+
const [, setAtBottomState] = (0, import_react20.useState)(true);
|
|
7052
|
+
const atBottomRef = (0, import_react20.useRef)(true);
|
|
7053
|
+
const isManualScrollingRef = (0, import_react20.useRef)(false);
|
|
7054
|
+
const manualScrollTimeoutRef = (0, import_react20.useRef)(null);
|
|
7055
|
+
const isFollowOutputScrollingRef = (0, import_react20.useRef)(false);
|
|
7056
|
+
const followOutputScrollTimeoutRef = (0, import_react20.useRef)(null);
|
|
7057
|
+
const setAtBottom = (0, import_react20.useCallback)(
|
|
6861
7058
|
(value) => {
|
|
6862
7059
|
if (isManualScrollingRef.current) return;
|
|
6863
7060
|
if (isFollowOutputScrollingRef.current && !value) return;
|
|
@@ -6867,18 +7064,18 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6867
7064
|
},
|
|
6868
7065
|
[onAtBottomChange]
|
|
6869
7066
|
);
|
|
6870
|
-
const visibleMessages = (0,
|
|
7067
|
+
const visibleMessages = (0, import_react20.useMemo)(
|
|
6871
7068
|
() => messages.filter((message) => message.role !== "tool"),
|
|
6872
7069
|
[messages]
|
|
6873
7070
|
);
|
|
6874
7071
|
const lastMessage = visibleMessages.length > 0 ? visibleMessages[visibleMessages.length - 1] : null;
|
|
6875
7072
|
const hasStreamingActivity = isLoading || (lastMessage?.isStreaming ?? false);
|
|
6876
|
-
const allItems = (0,
|
|
7073
|
+
const allItems = (0, import_react20.useMemo)(() => {
|
|
6877
7074
|
const items = [...visibleMessages];
|
|
6878
7075
|
items.push({ id: "footer-spacer", type: "footer" });
|
|
6879
7076
|
return items;
|
|
6880
7077
|
}, [visibleMessages]);
|
|
6881
|
-
const scrollToBottom = (0,
|
|
7078
|
+
const scrollToBottom = (0, import_react20.useCallback)(
|
|
6882
7079
|
(behavior = "smooth", source) => {
|
|
6883
7080
|
if (!virtuosoRef.current || allItems.length === 0) return;
|
|
6884
7081
|
const isManualScroll = source === "scroll-button";
|
|
@@ -6904,14 +7101,14 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6904
7101
|
},
|
|
6905
7102
|
[allItems.length, onAtBottomChange]
|
|
6906
7103
|
);
|
|
6907
|
-
(0,
|
|
7104
|
+
(0, import_react20.useImperativeHandle)(
|
|
6908
7105
|
ref,
|
|
6909
7106
|
() => ({
|
|
6910
7107
|
scrollToBottom
|
|
6911
7108
|
}),
|
|
6912
7109
|
[scrollToBottom]
|
|
6913
7110
|
);
|
|
6914
|
-
(0,
|
|
7111
|
+
(0, import_react20.useEffect)(() => {
|
|
6915
7112
|
return () => {
|
|
6916
7113
|
if (manualScrollTimeoutRef.current) {
|
|
6917
7114
|
clearTimeout(manualScrollTimeoutRef.current);
|
|
@@ -6921,13 +7118,13 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6921
7118
|
}
|
|
6922
7119
|
};
|
|
6923
7120
|
}, []);
|
|
6924
|
-
const handleMessageHeightChange = (0,
|
|
7121
|
+
const handleMessageHeightChange = (0, import_react20.useCallback)(() => {
|
|
6925
7122
|
if (isManualScrollingRef.current) return;
|
|
6926
7123
|
if (atBottomRef.current && hasStreamingActivity) {
|
|
6927
7124
|
scrollToBottom("auto", "height-change");
|
|
6928
7125
|
}
|
|
6929
7126
|
}, [scrollToBottom, hasStreamingActivity]);
|
|
6930
|
-
const handleClick = (0,
|
|
7127
|
+
const handleClick = (0, import_react20.useCallback)(
|
|
6931
7128
|
(e) => {
|
|
6932
7129
|
const selection = window.getSelection();
|
|
6933
7130
|
if (!selection || selection.isCollapsed) {
|
|
@@ -6938,7 +7135,7 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6938
7135
|
},
|
|
6939
7136
|
[onClickEmpty]
|
|
6940
7137
|
);
|
|
6941
|
-
const handleFollowOutput = (0,
|
|
7138
|
+
const handleFollowOutput = (0, import_react20.useCallback)((isAtBottom) => {
|
|
6942
7139
|
if (isAtBottom) {
|
|
6943
7140
|
if (followOutputScrollTimeoutRef.current) {
|
|
6944
7141
|
clearTimeout(followOutputScrollTimeoutRef.current);
|
|
@@ -6952,7 +7149,7 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6952
7149
|
}
|
|
6953
7150
|
return false;
|
|
6954
7151
|
}, []);
|
|
6955
|
-
const itemContent = (0,
|
|
7152
|
+
const itemContent = (0, import_react20.useCallback)(
|
|
6956
7153
|
(_index, item) => {
|
|
6957
7154
|
if ("type" in item && item.type === "footer") {
|
|
6958
7155
|
return /* @__PURE__ */ (0, import_jsx_runtime10.jsx)("div", { className: "h-28" });
|
|
@@ -6986,10 +7183,10 @@ var SanqianMessageList = (0, import_react18.forwardRef)(
|
|
|
6986
7183
|
);
|
|
6987
7184
|
|
|
6988
7185
|
// src/renderer/components/ChatInput.tsx
|
|
6989
|
-
var
|
|
7186
|
+
var import_react21 = require("react");
|
|
6990
7187
|
var import_jsx_runtime11 = require("react/jsx-runtime");
|
|
6991
|
-
var ChatInput = (0,
|
|
6992
|
-
(0,
|
|
7188
|
+
var ChatInput = (0, import_react21.memo)(
|
|
7189
|
+
(0, import_react21.forwardRef)(function ChatInput2({
|
|
6993
7190
|
onSend,
|
|
6994
7191
|
onStop,
|
|
6995
7192
|
placeholder = "Type a message...",
|
|
@@ -7004,12 +7201,12 @@ var ChatInput = (0, import_react19.memo)(
|
|
|
7004
7201
|
className = "",
|
|
7005
7202
|
style
|
|
7006
7203
|
}, ref) {
|
|
7007
|
-
const [text, setText] = (0,
|
|
7008
|
-
const textareaRef = (0,
|
|
7204
|
+
const [text, setText] = (0, import_react21.useState)("");
|
|
7205
|
+
const textareaRef = (0, import_react21.useRef)(null);
|
|
7009
7206
|
const canSend = text.trim().length > 0 && !disabled && !isLoading;
|
|
7010
7207
|
const showStopButton = isStreaming && !!onStop;
|
|
7011
7208
|
const showSpinner = isLoading && !showStopButton;
|
|
7012
|
-
(0,
|
|
7209
|
+
(0, import_react21.useImperativeHandle)(
|
|
7013
7210
|
ref,
|
|
7014
7211
|
() => ({
|
|
7015
7212
|
focus: () => textareaRef.current?.focus(),
|
|
@@ -7019,13 +7216,13 @@ var ChatInput = (0, import_react19.memo)(
|
|
|
7019
7216
|
}),
|
|
7020
7217
|
[text]
|
|
7021
7218
|
);
|
|
7022
|
-
(0,
|
|
7219
|
+
(0, import_react21.useEffect)(() => {
|
|
7023
7220
|
if (autoFocus) {
|
|
7024
7221
|
const timer = setTimeout(() => textareaRef.current?.focus(), 100);
|
|
7025
7222
|
return () => clearTimeout(timer);
|
|
7026
7223
|
}
|
|
7027
7224
|
}, [autoFocus]);
|
|
7028
|
-
const handleSubmit = (0,
|
|
7225
|
+
const handleSubmit = (0, import_react21.useCallback)(
|
|
7029
7226
|
(e) => {
|
|
7030
7227
|
e?.preventDefault();
|
|
7031
7228
|
if (!canSend) return;
|
|
@@ -7034,7 +7231,7 @@ var ChatInput = (0, import_react19.memo)(
|
|
|
7034
7231
|
},
|
|
7035
7232
|
[text, canSend, onSend]
|
|
7036
7233
|
);
|
|
7037
|
-
const handleKeyDown = (0,
|
|
7234
|
+
const handleKeyDown = (0, import_react21.useCallback)(
|
|
7038
7235
|
(e) => {
|
|
7039
7236
|
if (e.key === "Enter" && !e.shiftKey && !e.nativeEvent.isComposing) {
|
|
7040
7237
|
e.preventDefault();
|
|
@@ -7102,7 +7299,7 @@ var ChatInput = (0, import_react19.memo)(
|
|
|
7102
7299
|
);
|
|
7103
7300
|
|
|
7104
7301
|
// src/renderer/components/HitlCard.tsx
|
|
7105
|
-
var
|
|
7302
|
+
var import_react22 = require("react");
|
|
7106
7303
|
var import_jsx_runtime12 = require("react/jsx-runtime");
|
|
7107
7304
|
var defaultStrings = {
|
|
7108
7305
|
approve: "Approve",
|
|
@@ -7141,7 +7338,7 @@ var riskColors = {
|
|
|
7141
7338
|
icon: "!!"
|
|
7142
7339
|
}
|
|
7143
7340
|
};
|
|
7144
|
-
var HitlCard = (0,
|
|
7341
|
+
var HitlCard = (0, import_react22.memo)(function HitlCard2({
|
|
7145
7342
|
interrupt,
|
|
7146
7343
|
onApprove,
|
|
7147
7344
|
onReject,
|
|
@@ -7153,16 +7350,16 @@ var HitlCard = (0, import_react20.memo)(function HitlCard2({
|
|
|
7153
7350
|
const t = { ...defaultStrings, ...strings };
|
|
7154
7351
|
const isApproval = interrupt.type === "approval_request";
|
|
7155
7352
|
const isUserInput = interrupt.type === "user_input_request";
|
|
7156
|
-
const [answer, setAnswer] = (0,
|
|
7157
|
-
const [selectedIndices, setSelectedIndices] = (0,
|
|
7158
|
-
const [isComposing, setIsComposing] = (0,
|
|
7159
|
-
const [timeLeft, setTimeLeft] = (0,
|
|
7160
|
-
const [rememberChoice, setRememberChoice] = (0,
|
|
7161
|
-
const inputRef = (0,
|
|
7162
|
-
const textareaRef = (0,
|
|
7353
|
+
const [answer, setAnswer] = (0, import_react22.useState)(interrupt.default || "");
|
|
7354
|
+
const [selectedIndices, setSelectedIndices] = (0, import_react22.useState)([]);
|
|
7355
|
+
const [isComposing, setIsComposing] = (0, import_react22.useState)(false);
|
|
7356
|
+
const [timeLeft, setTimeLeft] = (0, import_react22.useState)(interrupt.timeout ?? null);
|
|
7357
|
+
const [rememberChoice, setRememberChoice] = (0, import_react22.useState)(false);
|
|
7358
|
+
const inputRef = (0, import_react22.useRef)(null);
|
|
7359
|
+
const textareaRef = (0, import_react22.useRef)(null);
|
|
7163
7360
|
const riskLevel = interrupt.risk_level || "medium";
|
|
7164
7361
|
const riskStyle = riskColors[riskLevel];
|
|
7165
|
-
(0,
|
|
7362
|
+
(0, import_react22.useEffect)(() => {
|
|
7166
7363
|
if (isUserInput) {
|
|
7167
7364
|
if (!interrupt.options || interrupt.options.length === 0) {
|
|
7168
7365
|
inputRef.current?.focus();
|
|
@@ -7170,7 +7367,7 @@ var HitlCard = (0, import_react20.memo)(function HitlCard2({
|
|
|
7170
7367
|
}
|
|
7171
7368
|
}
|
|
7172
7369
|
}, [isUserInput, interrupt.options]);
|
|
7173
|
-
(0,
|
|
7370
|
+
(0, import_react22.useEffect)(() => {
|
|
7174
7371
|
if (timeLeft === null || timeLeft <= 0) return;
|
|
7175
7372
|
const timer = setInterval(() => {
|
|
7176
7373
|
setTimeLeft((prev) => {
|
|
@@ -7368,12 +7565,12 @@ var HitlCard = (0, import_react20.memo)(function HitlCard2({
|
|
|
7368
7565
|
});
|
|
7369
7566
|
|
|
7370
7567
|
// src/renderer/primitives/AlertBanner.tsx
|
|
7371
|
-
var
|
|
7568
|
+
var import_react23 = require("react");
|
|
7372
7569
|
var import_jsx_runtime13 = (
|
|
7373
7570
|
// Error icon (circle with X)
|
|
7374
7571
|
require("react/jsx-runtime")
|
|
7375
7572
|
);
|
|
7376
|
-
var AlertBanner = (0,
|
|
7573
|
+
var AlertBanner = (0, import_react23.memo)(function AlertBanner2({
|
|
7377
7574
|
type,
|
|
7378
7575
|
message,
|
|
7379
7576
|
action,
|
|
@@ -7383,7 +7580,7 @@ var AlertBanner = (0, import_react21.memo)(function AlertBanner2({
|
|
|
7383
7580
|
maxLines = 3,
|
|
7384
7581
|
strings
|
|
7385
7582
|
}) {
|
|
7386
|
-
const [isExpanded, setIsExpanded] = (0,
|
|
7583
|
+
const [isExpanded, setIsExpanded] = (0, import_react23.useState)(false);
|
|
7387
7584
|
const defaultIcon = type === "error" ? /* @__PURE__ */ (0, import_jsx_runtime13.jsxs)("svg", { className: "size-4 shrink-0", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2", children: [
|
|
7388
7585
|
/* @__PURE__ */ (0, import_jsx_runtime13.jsx)("circle", { cx: "12", cy: "12", r: "10" }),
|
|
7389
7586
|
/* @__PURE__ */ (0, import_jsx_runtime13.jsx)("path", { d: "M15 9l-6 6M9 9l6 6", strokeLinecap: "round" })
|
|
@@ -7446,7 +7643,7 @@ var AlertBanner = (0, import_react21.memo)(function AlertBanner2({
|
|
|
7446
7643
|
|
|
7447
7644
|
// src/renderer/components/SanqianChat.tsx
|
|
7448
7645
|
var import_jsx_runtime14 = require("react/jsx-runtime");
|
|
7449
|
-
var SanqianChat = (0,
|
|
7646
|
+
var SanqianChat = (0, import_react24.memo)(function SanqianChat2({
|
|
7450
7647
|
adapter,
|
|
7451
7648
|
placeholder,
|
|
7452
7649
|
autoConnect = true,
|
|
@@ -7458,20 +7655,20 @@ var SanqianChat = (0, import_react22.memo)(function SanqianChat2({
|
|
|
7458
7655
|
config,
|
|
7459
7656
|
logo
|
|
7460
7657
|
}) {
|
|
7461
|
-
const chatContainerRef = (0,
|
|
7462
|
-
const chatInputRef = (0,
|
|
7463
|
-
const messageListRef = (0,
|
|
7464
|
-
const onStateChangeRef = (0,
|
|
7658
|
+
const chatContainerRef = (0, import_react24.useRef)(null);
|
|
7659
|
+
const chatInputRef = (0, import_react24.useRef)(null);
|
|
7660
|
+
const messageListRef = (0, import_react24.useRef)(null);
|
|
7661
|
+
const onStateChangeRef = (0, import_react24.useRef)(onStateChange);
|
|
7465
7662
|
onStateChangeRef.current = onStateChange;
|
|
7466
7663
|
const resolvedConfig = useResolvedUiConfig(config);
|
|
7467
7664
|
const { themeClass, isDarkMode } = useResolvedTheme(resolvedConfig?.theme ?? "auto");
|
|
7468
7665
|
const accentStyle = useAccentStyle(resolvedConfig?.accentColor);
|
|
7469
7666
|
const resolvedLogo = logo ?? resolvedConfig?.logo;
|
|
7470
|
-
const strings = (0,
|
|
7667
|
+
const strings = (0, import_react24.useMemo)(
|
|
7471
7668
|
() => resolveChatStrings(resolvedConfig?.locale, resolvedConfig?.strings),
|
|
7472
7669
|
[resolvedConfig?.locale, resolvedConfig?.strings]
|
|
7473
7670
|
);
|
|
7474
|
-
const headerConfig = (0,
|
|
7671
|
+
const headerConfig = (0, import_react24.useMemo)(
|
|
7475
7672
|
() => {
|
|
7476
7673
|
if (!resolvedConfig) {
|
|
7477
7674
|
return resolvedLogo ? { logo: resolvedLogo } : void 0;
|
|
@@ -7484,7 +7681,7 @@ var SanqianChat = (0, import_react22.memo)(function SanqianChat2({
|
|
|
7484
7681
|
[resolvedConfig, resolvedLogo]
|
|
7485
7682
|
);
|
|
7486
7683
|
const { logoNode, showPin, showClose, isPinned, togglePin, resolvedOnClose } = useChatHeader(headerConfig);
|
|
7487
|
-
const emptyLogoNode = (0,
|
|
7684
|
+
const emptyLogoNode = (0, import_react24.useMemo)(() => resolveLogoNode(resolvedLogo, "empty"), [resolvedLogo]);
|
|
7488
7685
|
const showHeader = !!(logoNode || showClose || showPin);
|
|
7489
7686
|
useChatStyles();
|
|
7490
7687
|
useConnection({ adapter, autoConnect });
|
|
@@ -7505,7 +7702,7 @@ var SanqianChat = (0, import_react22.memo)(function SanqianChat2({
|
|
|
7505
7702
|
chat.submitHitlInput({ cancelled: true });
|
|
7506
7703
|
}
|
|
7507
7704
|
};
|
|
7508
|
-
(0,
|
|
7705
|
+
(0, import_react24.useEffect)(() => {
|
|
7509
7706
|
onStateChangeRef.current?.({
|
|
7510
7707
|
messages: chat.messages,
|
|
7511
7708
|
conversationId: chat.conversationId
|
|
@@ -7624,10 +7821,187 @@ var SanqianChat = (0, import_react22.memo)(function SanqianChat2({
|
|
|
7624
7821
|
});
|
|
7625
7822
|
|
|
7626
7823
|
// src/renderer/components/FloatingChat.tsx
|
|
7627
|
-
var
|
|
7824
|
+
var import_react29 = require("react");
|
|
7825
|
+
|
|
7826
|
+
// src/renderer/components/ModeToggleButton.tsx
|
|
7827
|
+
var import_react25 = require("react");
|
|
7828
|
+
var import_jsx_runtime15 = (
|
|
7829
|
+
// Float out icon - two overlapping panels (macOS style)
|
|
7830
|
+
require("react/jsx-runtime")
|
|
7831
|
+
);
|
|
7832
|
+
function ModeToggleButton({ className, locale = "en" }) {
|
|
7833
|
+
const { isEmbedded, toggleMode } = useChatPanel();
|
|
7834
|
+
const strings = (0, import_react25.useMemo)(() => resolveChatStrings(locale), [locale]);
|
|
7835
|
+
const handleClick = async () => {
|
|
7836
|
+
await toggleMode();
|
|
7837
|
+
};
|
|
7838
|
+
const tooltipText = isEmbedded ? strings.floatWindow : strings.embedWindow;
|
|
7839
|
+
return /* @__PURE__ */ (0, import_jsx_runtime15.jsxs)("div", { className: `chat-tooltip-wrapper ${className || ""}`, children: [
|
|
7840
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)(
|
|
7841
|
+
"button",
|
|
7842
|
+
{
|
|
7843
|
+
onClick: handleClick,
|
|
7844
|
+
className: headerIconButtonClass,
|
|
7845
|
+
"aria-label": tooltipText,
|
|
7846
|
+
children: isEmbedded ? /* @__PURE__ */ (0, import_jsx_runtime15.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: [
|
|
7847
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)("rect", { x: "8", y: "3", width: "13", height: "13", rx: "2", strokeLinecap: "round", strokeLinejoin: "round" }),
|
|
7848
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)("path", { d: "M5 8H4a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-1", strokeLinecap: "round", strokeLinejoin: "round" })
|
|
7849
|
+
] }) : (
|
|
7850
|
+
// Embed icon - panel with sidebar (dock to sidebar)
|
|
7851
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: [
|
|
7852
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)("rect", { x: "3", y: "4", width: "18", height: "16", rx: "2", strokeLinecap: "round", strokeLinejoin: "round" }),
|
|
7853
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)("path", { d: "M15 4v16", strokeLinecap: "round" })
|
|
7854
|
+
] })
|
|
7855
|
+
)
|
|
7856
|
+
}
|
|
7857
|
+
),
|
|
7858
|
+
/* @__PURE__ */ (0, import_jsx_runtime15.jsx)("div", { className: "chat-tooltip", children: tooltipText })
|
|
7859
|
+
] });
|
|
7860
|
+
}
|
|
7861
|
+
|
|
7862
|
+
// src/renderer/components/AttachButton.tsx
|
|
7863
|
+
var import_react26 = require("react");
|
|
7864
|
+
var import_jsx_runtime16 = (
|
|
7865
|
+
// Attached state: filled icon
|
|
7866
|
+
require("react/jsx-runtime")
|
|
7867
|
+
);
|
|
7868
|
+
function AttachButton({ className, locale = "en" }) {
|
|
7869
|
+
const { isAttached, isAvailable, toggle } = useAttachState();
|
|
7870
|
+
const { isFloating } = useChatPanel();
|
|
7871
|
+
const strings = (0, import_react26.useMemo)(() => resolveChatStrings(locale), [locale]);
|
|
7872
|
+
if (!isFloating || !isAvailable) return null;
|
|
7873
|
+
const handleClick = async () => {
|
|
7874
|
+
await toggle();
|
|
7875
|
+
};
|
|
7876
|
+
const tooltipText = isAttached ? strings.detachWindow : strings.attachWindow;
|
|
7877
|
+
return /* @__PURE__ */ (0, import_jsx_runtime16.jsxs)("div", { className: `chat-tooltip-wrapper ${className || ""}`, children: [
|
|
7878
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)(
|
|
7879
|
+
"button",
|
|
7880
|
+
{
|
|
7881
|
+
onClick: handleClick,
|
|
7882
|
+
className: headerIconButtonClass,
|
|
7883
|
+
"aria-label": tooltipText,
|
|
7884
|
+
children: isAttached ? /* @__PURE__ */ (0, import_jsx_runtime16.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 16 16", fill: "currentColor", children: [
|
|
7885
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)("path", { d: "M4.5 2A2.5 2.5 0 0 0 2 4.5v7A2.5 2.5 0 0 0 4.5 14h7a2.5 2.5 0 0 0 2.5-2.5v-7A2.5 2.5 0 0 0 11.5 2h-7z" }),
|
|
7886
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)("path", { d: "M9.5 6.5h3v1h-3v-1z", fill: "var(--bg-color, #fff)" })
|
|
7887
|
+
] }) : (
|
|
7888
|
+
// Detached state: outline icon
|
|
7889
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 16 16", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: [
|
|
7890
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)("rect", { x: "2.5", y: "2.5", width: "11", height: "11", rx: "2" }),
|
|
7891
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)("path", { d: "M9.5 6.5h3", strokeLinecap: "round" })
|
|
7892
|
+
] })
|
|
7893
|
+
)
|
|
7894
|
+
}
|
|
7895
|
+
),
|
|
7896
|
+
/* @__PURE__ */ (0, import_jsx_runtime16.jsx)("div", { className: "chat-tooltip", children: tooltipText })
|
|
7897
|
+
] });
|
|
7898
|
+
}
|
|
7899
|
+
|
|
7900
|
+
// src/renderer/components/Resizer.tsx
|
|
7901
|
+
var import_react27 = require("react");
|
|
7902
|
+
var import_jsx_runtime17 = require("react/jsx-runtime");
|
|
7903
|
+
var DEFAULT_WIDTH = 360;
|
|
7904
|
+
function Resizer({
|
|
7905
|
+
position = "left",
|
|
7906
|
+
minWidth = 240,
|
|
7907
|
+
maxWidth = 600,
|
|
7908
|
+
defaultWidth = DEFAULT_WIDTH,
|
|
7909
|
+
className = ""
|
|
7910
|
+
}) {
|
|
7911
|
+
const { isEmbedded, width, setWidth, onResizeEnd } = useChatPanel();
|
|
7912
|
+
const [isHovered, setIsHovered] = (0, import_react27.useState)(false);
|
|
7913
|
+
const [isDragging, setIsDragging] = (0, import_react27.useState)(false);
|
|
7914
|
+
const startX = (0, import_react27.useRef)(0);
|
|
7915
|
+
const startWidth = (0, import_react27.useRef)(0);
|
|
7916
|
+
const handleDoubleClick = (0, import_react27.useCallback)(() => {
|
|
7917
|
+
setWidth(defaultWidth, true);
|
|
7918
|
+
}, [defaultWidth, setWidth]);
|
|
7919
|
+
const handleMouseDown = (0, import_react27.useCallback)((e) => {
|
|
7920
|
+
e.preventDefault();
|
|
7921
|
+
setIsDragging(true);
|
|
7922
|
+
startX.current = e.clientX;
|
|
7923
|
+
startWidth.current = width;
|
|
7924
|
+
document.body.style.cursor = "col-resize";
|
|
7925
|
+
document.body.style.userSelect = "none";
|
|
7926
|
+
}, [width]);
|
|
7927
|
+
(0, import_react27.useEffect)(() => {
|
|
7928
|
+
if (!isDragging) return;
|
|
7929
|
+
const handleMouseMove = (e) => {
|
|
7930
|
+
const deltaX = e.clientX - startX.current;
|
|
7931
|
+
const newWidth = position === "left" ? startWidth.current - deltaX : startWidth.current + deltaX;
|
|
7932
|
+
const clampedWidth = Math.min(maxWidth, Math.max(minWidth, newWidth));
|
|
7933
|
+
setWidth(clampedWidth);
|
|
7934
|
+
};
|
|
7935
|
+
const handleMouseUp = () => {
|
|
7936
|
+
setIsDragging(false);
|
|
7937
|
+
document.body.style.cursor = "";
|
|
7938
|
+
document.body.style.userSelect = "";
|
|
7939
|
+
onResizeEnd();
|
|
7940
|
+
};
|
|
7941
|
+
document.addEventListener("mousemove", handleMouseMove);
|
|
7942
|
+
document.addEventListener("mouseup", handleMouseUp);
|
|
7943
|
+
return () => {
|
|
7944
|
+
document.removeEventListener("mousemove", handleMouseMove);
|
|
7945
|
+
document.removeEventListener("mouseup", handleMouseUp);
|
|
7946
|
+
};
|
|
7947
|
+
}, [isDragging, position, minWidth, maxWidth, setWidth, onResizeEnd]);
|
|
7948
|
+
if (!isEmbedded) return null;
|
|
7949
|
+
const isActive = isHovered || isDragging;
|
|
7950
|
+
return /* @__PURE__ */ (0, import_jsx_runtime17.jsxs)(
|
|
7951
|
+
"div",
|
|
7952
|
+
{
|
|
7953
|
+
className: `chat-resizer ${className}`,
|
|
7954
|
+
onMouseDown: handleMouseDown,
|
|
7955
|
+
onDoubleClick: handleDoubleClick,
|
|
7956
|
+
onMouseEnter: () => setIsHovered(true),
|
|
7957
|
+
onMouseLeave: () => setIsHovered(false),
|
|
7958
|
+
style: {
|
|
7959
|
+
position: "absolute",
|
|
7960
|
+
top: 0,
|
|
7961
|
+
bottom: 0,
|
|
7962
|
+
[position]: 0,
|
|
7963
|
+
width: 6,
|
|
7964
|
+
cursor: "col-resize",
|
|
7965
|
+
zIndex: 100,
|
|
7966
|
+
// Ensure it's above other content and not affected by drag region
|
|
7967
|
+
WebkitAppRegion: "no-drag"
|
|
7968
|
+
},
|
|
7969
|
+
children: [
|
|
7970
|
+
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)(
|
|
7971
|
+
"div",
|
|
7972
|
+
{
|
|
7973
|
+
style: {
|
|
7974
|
+
position: "absolute",
|
|
7975
|
+
top: 0,
|
|
7976
|
+
bottom: 0,
|
|
7977
|
+
left: position === "left" ? 0 : "auto",
|
|
7978
|
+
right: position === "right" ? 0 : "auto",
|
|
7979
|
+
width: 2,
|
|
7980
|
+
backgroundColor: isActive ? "var(--chat-accent, #2563EB)" : "transparent",
|
|
7981
|
+
transition: "background-color 0.15s ease-out, opacity 0.15s ease-out",
|
|
7982
|
+
opacity: isDragging ? 1 : isHovered ? 0.7 : 0
|
|
7983
|
+
}
|
|
7984
|
+
}
|
|
7985
|
+
),
|
|
7986
|
+
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)(
|
|
7987
|
+
"div",
|
|
7988
|
+
{
|
|
7989
|
+
style: {
|
|
7990
|
+
position: "absolute",
|
|
7991
|
+
top: 0,
|
|
7992
|
+
bottom: 0,
|
|
7993
|
+
left: -2,
|
|
7994
|
+
right: -2
|
|
7995
|
+
}
|
|
7996
|
+
}
|
|
7997
|
+
)
|
|
7998
|
+
]
|
|
7999
|
+
}
|
|
8000
|
+
);
|
|
8001
|
+
}
|
|
7628
8002
|
|
|
7629
8003
|
// src/renderer/hooks/useWindowBackgroundSync.ts
|
|
7630
|
-
var
|
|
8004
|
+
var import_react28 = require("react");
|
|
7631
8005
|
var cachedPlatform = null;
|
|
7632
8006
|
var getPlatform = () => {
|
|
7633
8007
|
if (cachedPlatform !== null) return cachedPlatform;
|
|
@@ -7645,7 +8019,7 @@ var setBackgroundColor = (color) => {
|
|
|
7645
8019
|
}
|
|
7646
8020
|
};
|
|
7647
8021
|
function useWindowBackgroundSync(isDarkMode) {
|
|
7648
|
-
(0,
|
|
8022
|
+
(0, import_react28.useEffect)(() => {
|
|
7649
8023
|
if (getPlatform() !== "win32") return;
|
|
7650
8024
|
const colors = getBaseColors(isDarkMode);
|
|
7651
8025
|
setBackgroundColor(colors.bg);
|
|
@@ -7653,8 +8027,8 @@ function useWindowBackgroundSync(isDarkMode) {
|
|
|
7653
8027
|
}
|
|
7654
8028
|
|
|
7655
8029
|
// src/renderer/components/FloatingChat.tsx
|
|
7656
|
-
var
|
|
7657
|
-
var FloatingChat = (0,
|
|
8030
|
+
var import_jsx_runtime18 = require("react/jsx-runtime");
|
|
8031
|
+
var FloatingChat = (0, import_react29.memo)(function FloatingChat2({
|
|
7658
8032
|
messages,
|
|
7659
8033
|
isLoading,
|
|
7660
8034
|
isStreaming,
|
|
@@ -7675,8 +8049,8 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7675
8049
|
header,
|
|
7676
8050
|
footer
|
|
7677
8051
|
}) {
|
|
7678
|
-
const chatContainerRef = (0,
|
|
7679
|
-
const chatInputRef = (0,
|
|
8052
|
+
const chatContainerRef = (0, import_react29.useRef)(null);
|
|
8053
|
+
const chatInputRef = (0, import_react29.useRef)(null);
|
|
7680
8054
|
useChatStyles();
|
|
7681
8055
|
useFocusPersistence({
|
|
7682
8056
|
containerRef: chatContainerRef,
|
|
@@ -7688,18 +8062,18 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7688
8062
|
const { themeClass, isDarkMode } = useResolvedTheme(themeMode);
|
|
7689
8063
|
const accentStyle = useAccentStyle(resolvedConfig?.accentColor);
|
|
7690
8064
|
useWindowBackgroundSync(isDarkMode);
|
|
7691
|
-
const rootStyle = (0,
|
|
8065
|
+
const rootStyle = (0, import_react29.useMemo)(() => ({
|
|
7692
8066
|
...accentStyle || {},
|
|
7693
8067
|
minHeight: "100vh",
|
|
7694
8068
|
minWidth: "100vw"
|
|
7695
8069
|
}), [accentStyle]);
|
|
7696
8070
|
const resolvedLogo = logo ?? resolvedConfig?.logo;
|
|
7697
8071
|
const resolvedLocale = resolvedConfig?.locale ?? locale;
|
|
7698
|
-
const resolvedStrings = (0,
|
|
8072
|
+
const resolvedStrings = (0, import_react29.useMemo)(
|
|
7699
8073
|
() => resolveChatStrings(resolvedLocale, resolvedConfig?.strings),
|
|
7700
8074
|
[resolvedLocale, resolvedConfig?.strings]
|
|
7701
8075
|
);
|
|
7702
|
-
const headerConfig = (0,
|
|
8076
|
+
const headerConfig = (0, import_react29.useMemo)(
|
|
7703
8077
|
() => {
|
|
7704
8078
|
if (!resolvedConfig) {
|
|
7705
8079
|
return resolvedLogo ? { logo: resolvedLogo } : void 0;
|
|
@@ -7712,71 +8086,77 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7712
8086
|
[resolvedConfig, resolvedLogo]
|
|
7713
8087
|
);
|
|
7714
8088
|
const { logoNode, showPin, showClose, isPinned, togglePin, resolvedOnClose } = useChatHeader(headerConfig);
|
|
7715
|
-
const emptyLogoNode = (0,
|
|
8089
|
+
const emptyLogoNode = (0, import_react29.useMemo)(() => resolveLogoNode(resolvedLogo, "empty"), [resolvedLogo]);
|
|
7716
8090
|
const inputPlaceholder = placeholder ?? resolvedStrings.inputPlaceholder;
|
|
7717
8091
|
const showHeader = !!(header || logoNode || showPin || showClose);
|
|
7718
|
-
const defaultHeader = /* @__PURE__ */ (0,
|
|
7719
|
-
/* @__PURE__ */ (0,
|
|
8092
|
+
const defaultHeader = /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("header", { className: "flex h-9 flex-shrink-0 items-center px-2 border-b chat-divider-border", children: [
|
|
8093
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "flex items-center gap-2", children: [
|
|
7720
8094
|
logoNode,
|
|
7721
|
-
/* @__PURE__ */ (0,
|
|
8095
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("span", { className: "text-sm font-medium text-[var(--chat-text)]", children: resolvedStrings.chat })
|
|
7722
8096
|
] }),
|
|
7723
|
-
/* @__PURE__ */ (0,
|
|
7724
|
-
/* @__PURE__ */ (0,
|
|
7725
|
-
|
|
7726
|
-
|
|
7727
|
-
|
|
7728
|
-
|
|
7729
|
-
|
|
7730
|
-
|
|
7731
|
-
|
|
7732
|
-
|
|
7733
|
-
/* @__PURE__ */ (0,
|
|
7734
|
-
|
|
7735
|
-
|
|
7736
|
-
/* @__PURE__ */ (0,
|
|
7737
|
-
|
|
7738
|
-
|
|
7739
|
-
|
|
7740
|
-
|
|
7741
|
-
|
|
7742
|
-
|
|
7743
|
-
|
|
7744
|
-
{
|
|
7745
|
-
|
|
7746
|
-
|
|
7747
|
-
|
|
7748
|
-
|
|
7749
|
-
|
|
7750
|
-
|
|
8097
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-1" }),
|
|
8098
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "flex items-center gap-1", children: [
|
|
8099
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(ModeToggleButton, { locale: resolvedLocale }),
|
|
8100
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(AttachButton, { locale: resolvedLocale }),
|
|
8101
|
+
showPin && /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8102
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
8103
|
+
"button",
|
|
8104
|
+
{
|
|
8105
|
+
onClick: togglePin,
|
|
8106
|
+
className: headerIconButtonClass,
|
|
8107
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", strokeLinecap: "round", strokeLinejoin: "round", children: isPinned ? /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)(import_jsx_runtime18.Fragment, { children: [
|
|
8108
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M12 17v5" }),
|
|
8109
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M9 10.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-.76a2 2 0 0 0-1.11-1.79l-1.78-.9A2 2 0 0 1 15 10.76V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H8a2 2 0 0 0 0 4 1 1 0 0 1 1 1z" })
|
|
8110
|
+
] }) : /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)(import_jsx_runtime18.Fragment, { children: [
|
|
8111
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M12 17v5" }),
|
|
8112
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M15 9.34V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H7.89" }),
|
|
8113
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "m2 2 20 20" }),
|
|
8114
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M9 9v1.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h11" })
|
|
8115
|
+
] }) })
|
|
8116
|
+
}
|
|
8117
|
+
),
|
|
8118
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "chat-tooltip", children: isPinned ? resolvedStrings.unpin : resolvedStrings.pin })
|
|
8119
|
+
] }),
|
|
8120
|
+
showClose && /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8121
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
8122
|
+
"button",
|
|
8123
|
+
{
|
|
8124
|
+
onClick: resolvedOnClose,
|
|
8125
|
+
className: headerIconButtonClass,
|
|
8126
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("path", { d: "M6 6l12 12M18 6l-12 12", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8127
|
+
}
|
|
8128
|
+
),
|
|
8129
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "chat-tooltip", children: resolvedStrings.close })
|
|
8130
|
+
] })
|
|
7751
8131
|
] })
|
|
7752
8132
|
] });
|
|
7753
8133
|
const resolvedHeader = header ?? (showHeader ? defaultHeader : null);
|
|
7754
|
-
const defaultRenderMessage = (0,
|
|
8134
|
+
const defaultRenderMessage = (0, import_react29.useCallback)(
|
|
7755
8135
|
(message) => {
|
|
7756
8136
|
if (message.role === "tool") return null;
|
|
7757
8137
|
const isUser = message.role === "user";
|
|
7758
|
-
return /* @__PURE__ */ (0,
|
|
8138
|
+
return /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: `flex ${isUser ? "justify-end" : "justify-start"} mb-4`, children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: `text-left ${isUser ? "max-w-[80%]" : "w-full"}`, children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: `relative group ${isUser ? "inline-block max-w-full" : "block w-full"}`, children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7759
8139
|
"div",
|
|
7760
8140
|
{
|
|
7761
8141
|
className: `message-bubble-wrapper ${isUser ? "rounded-2xl shadow-sm bg-[var(--chat-accent)] text-white px-4 py-3 whitespace-pre-wrap break-words" : "text-[var(--chat-text)] break-words"}`,
|
|
7762
|
-
children: isUser ? /* @__PURE__ */ (0,
|
|
8142
|
+
children: isUser ? /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("span", { className: "whitespace-pre-wrap", children: message.content }) : message.content && (renderContent ? /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "prose prose-chat dark:prose-invert max-w-none px-3", children: renderContent(message.content, message.isStreaming ?? false) }) : /* @__PURE__ */ (0, import_jsx_runtime18.jsx)(MarkdownRenderer, { content: message.content, isStreaming: message.isStreaming, className: "px-3" }))
|
|
7763
8143
|
}
|
|
7764
8144
|
) }) }) });
|
|
7765
8145
|
},
|
|
7766
8146
|
[renderContent]
|
|
7767
8147
|
);
|
|
7768
|
-
const defaultRenderHitl = (0,
|
|
7769
|
-
(interrupt, onApprove, onReject) => /* @__PURE__ */ (0,
|
|
7770
|
-
/* @__PURE__ */ (0,
|
|
7771
|
-
interrupt.tool && /* @__PURE__ */ (0,
|
|
8148
|
+
const defaultRenderHitl = (0, import_react29.useCallback)(
|
|
8149
|
+
(interrupt, onApprove, onReject) => /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "p-4 bg-yellow-50 dark:bg-yellow-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg m-2", children: [
|
|
8150
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("p", { className: "font-medium mb-2", children: interrupt.type === "approval_request" ? resolvedStrings.hitlApprovalRequired : resolvedStrings.hitlInputRequired }),
|
|
8151
|
+
interrupt.tool && /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("p", { className: "text-sm mb-2", children: [
|
|
7772
8152
|
resolvedStrings.hitlToolLabel,
|
|
7773
8153
|
": ",
|
|
7774
8154
|
interrupt.tool
|
|
7775
8155
|
] }),
|
|
7776
|
-
interrupt.reason && /* @__PURE__ */ (0,
|
|
7777
|
-
interrupt.question && /* @__PURE__ */ (0,
|
|
7778
|
-
/* @__PURE__ */ (0,
|
|
7779
|
-
/* @__PURE__ */ (0,
|
|
8156
|
+
interrupt.reason && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("p", { className: "text-sm mb-2", children: interrupt.reason }),
|
|
8157
|
+
interrupt.question && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("p", { className: "text-sm mb-2", children: interrupt.question }),
|
|
8158
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { className: "flex gap-2 mt-3", children: [
|
|
8159
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7780
8160
|
"button",
|
|
7781
8161
|
{
|
|
7782
8162
|
onClick: onApprove,
|
|
@@ -7784,7 +8164,7 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7784
8164
|
children: resolvedStrings.hitlApprove
|
|
7785
8165
|
}
|
|
7786
8166
|
),
|
|
7787
|
-
/* @__PURE__ */ (0,
|
|
8167
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7788
8168
|
"button",
|
|
7789
8169
|
{
|
|
7790
8170
|
onClick: onReject,
|
|
@@ -7796,9 +8176,10 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7796
8176
|
] }),
|
|
7797
8177
|
[resolvedStrings]
|
|
7798
8178
|
);
|
|
7799
|
-
return /* @__PURE__ */ (0,
|
|
7800
|
-
|
|
7801
|
-
|
|
8179
|
+
return /* @__PURE__ */ (0, import_jsx_runtime18.jsxs)("div", { ref: chatContainerRef, className: `relative flex flex-col h-full bg-[var(--chat-bg)] text-[var(--chat-text)] ${themeClass} ${className}`, style: rootStyle, "data-chat-font-size": resolvedConfig?.fontSize || "normal", children: [
|
|
8180
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)(Resizer, { position: "left", minWidth: 240, maxWidth: 600 }),
|
|
8181
|
+
resolvedHeader && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-shrink-0", children: resolvedHeader }),
|
|
8182
|
+
messages.length === 0 ? /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex flex-1 items-center justify-center p-4", children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex items-center justify-center", children: emptyLogoNode }) }) : /* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7802
8183
|
MessageList,
|
|
7803
8184
|
{
|
|
7804
8185
|
messages,
|
|
@@ -7806,12 +8187,12 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7806
8187
|
renderMessage: renderMessage || defaultRenderMessage
|
|
7807
8188
|
}
|
|
7808
8189
|
),
|
|
7809
|
-
pendingInterrupt && /* @__PURE__ */ (0,
|
|
8190
|
+
pendingInterrupt && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-shrink-0", children: (renderHitl || defaultRenderHitl)(
|
|
7810
8191
|
pendingInterrupt,
|
|
7811
8192
|
() => onApproveHitl?.(),
|
|
7812
8193
|
() => onRejectHitl?.()
|
|
7813
8194
|
) }),
|
|
7814
|
-
error && /* @__PURE__ */ (0,
|
|
8195
|
+
error && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-shrink-0 px-3 pt-2", children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7815
8196
|
AlertBanner,
|
|
7816
8197
|
{
|
|
7817
8198
|
type: "error",
|
|
@@ -7819,7 +8200,7 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7819
8200
|
className: `flex items-center gap-2 rounded-xl px-3 py-2 text-sm ${isDarkMode ? "bg-red-900/60 text-red-200" : "bg-red-100 text-red-800"}`
|
|
7820
8201
|
}
|
|
7821
8202
|
) }),
|
|
7822
|
-
/* @__PURE__ */ (0,
|
|
8203
|
+
/* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-shrink-0 px-3 pb-3 pt-1 border-t chat-divider-border", children: /* @__PURE__ */ (0, import_jsx_runtime18.jsx)(
|
|
7823
8204
|
ChatInput,
|
|
7824
8205
|
{
|
|
7825
8206
|
ref: chatInputRef,
|
|
@@ -7834,16 +8215,16 @@ var FloatingChat = (0, import_react24.memo)(function FloatingChat2({
|
|
|
7834
8215
|
autoFocus: true
|
|
7835
8216
|
}
|
|
7836
8217
|
) }),
|
|
7837
|
-
footer && /* @__PURE__ */ (0,
|
|
8218
|
+
footer && /* @__PURE__ */ (0, import_jsx_runtime18.jsx)("div", { className: "flex-shrink-0", children: footer })
|
|
7838
8219
|
] });
|
|
7839
8220
|
});
|
|
7840
8221
|
|
|
7841
8222
|
// src/renderer/components/HistoryList.tsx
|
|
7842
|
-
var
|
|
7843
|
-
var
|
|
8223
|
+
var import_react30 = require("react");
|
|
8224
|
+
var import_jsx_runtime19 = require("react/jsx-runtime");
|
|
7844
8225
|
function DeleteButton({ onClick, title, colors }) {
|
|
7845
|
-
const [isHovered, setIsHovered] = (0,
|
|
7846
|
-
return /* @__PURE__ */ (0,
|
|
8226
|
+
const [isHovered, setIsHovered] = (0, import_react30.useState)(false);
|
|
8227
|
+
return /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
7847
8228
|
"button",
|
|
7848
8229
|
{
|
|
7849
8230
|
onClick,
|
|
@@ -7864,7 +8245,7 @@ function DeleteButton({ onClick, title, colors }) {
|
|
|
7864
8245
|
alignItems: "center",
|
|
7865
8246
|
justifyContent: "center"
|
|
7866
8247
|
},
|
|
7867
|
-
children: /* @__PURE__ */ (0,
|
|
8248
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
7868
8249
|
"svg",
|
|
7869
8250
|
{
|
|
7870
8251
|
style: { width: "0.875rem", height: "0.875rem" },
|
|
@@ -7872,7 +8253,7 @@ function DeleteButton({ onClick, title, colors }) {
|
|
|
7872
8253
|
fill: "none",
|
|
7873
8254
|
stroke: "currentColor",
|
|
7874
8255
|
strokeWidth: "1.5",
|
|
7875
|
-
children: /* @__PURE__ */ (0,
|
|
8256
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
7876
8257
|
"path",
|
|
7877
8258
|
{
|
|
7878
8259
|
d: "M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16",
|
|
@@ -7886,8 +8267,8 @@ function DeleteButton({ onClick, title, colors }) {
|
|
|
7886
8267
|
);
|
|
7887
8268
|
}
|
|
7888
8269
|
function LoadMoreButton({ onClick, colors, children }) {
|
|
7889
|
-
const [isHovered, setIsHovered] = (0,
|
|
7890
|
-
return /* @__PURE__ */ (0,
|
|
8270
|
+
const [isHovered, setIsHovered] = (0, import_react30.useState)(false);
|
|
8271
|
+
return /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
7891
8272
|
"button",
|
|
7892
8273
|
{
|
|
7893
8274
|
onClick,
|
|
@@ -7936,7 +8317,7 @@ function formatRelativeTime(dateStr, strings) {
|
|
|
7936
8317
|
return date.toLocaleDateString(void 0, { month: "short", day: "numeric" });
|
|
7937
8318
|
}
|
|
7938
8319
|
}
|
|
7939
|
-
var HistoryList = (0,
|
|
8320
|
+
var HistoryList = (0, import_react30.memo)(function HistoryList2({
|
|
7940
8321
|
conversations,
|
|
7941
8322
|
selectedId,
|
|
7942
8323
|
isLoading,
|
|
@@ -7948,11 +8329,11 @@ var HistoryList = (0, import_react25.memo)(function HistoryList2({
|
|
|
7948
8329
|
isDarkMode = false,
|
|
7949
8330
|
strings = {}
|
|
7950
8331
|
}) {
|
|
7951
|
-
const [hoveredId, setHoveredId] = (0,
|
|
7952
|
-
const loadMoreRef = (0,
|
|
7953
|
-
const isLoadingRef = (0,
|
|
8332
|
+
const [hoveredId, setHoveredId] = (0, import_react30.useState)(null);
|
|
8333
|
+
const loadMoreRef = (0, import_react30.useRef)(null);
|
|
8334
|
+
const isLoadingRef = (0, import_react30.useRef)(isLoading);
|
|
7954
8335
|
isLoadingRef.current = isLoading;
|
|
7955
|
-
(0,
|
|
8336
|
+
(0, import_react30.useEffect)(() => {
|
|
7956
8337
|
if (!hasMore || loadError || !onLoadMore) return;
|
|
7957
8338
|
const sentinel = loadMoreRef.current;
|
|
7958
8339
|
if (!sentinel) return;
|
|
@@ -7976,7 +8357,7 @@ var HistoryList = (0, import_react25.memo)(function HistoryList2({
|
|
|
7976
8357
|
deleteText: "#ef4444",
|
|
7977
8358
|
loadingDot: isDarkMode ? "rgba(255, 255, 255, 0.3)" : "rgba(0, 0, 0, 0.2)"
|
|
7978
8359
|
};
|
|
7979
|
-
const loadingDots = /* @__PURE__ */ (0,
|
|
8360
|
+
const loadingDots = /* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { style: { display: "flex", justifyContent: "center", padding: "1.5rem 0" }, children: /* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { style: { display: "flex", gap: "0.25rem" }, children: [0, 150, 300].map((delay) => /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
7980
8361
|
"span",
|
|
7981
8362
|
{
|
|
7982
8363
|
style: {
|
|
@@ -7994,13 +8375,13 @@ var HistoryList = (0, import_react25.memo)(function HistoryList2({
|
|
|
7994
8375
|
return loadingDots;
|
|
7995
8376
|
}
|
|
7996
8377
|
if (conversations.length === 0) {
|
|
7997
|
-
return /* @__PURE__ */ (0,
|
|
8378
|
+
return /* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { style: { textAlign: "center", color: colors.muted, padding: "1.5rem 0", fontSize: "0.875rem" }, children: strings.noHistory || "No conversations yet" });
|
|
7998
8379
|
}
|
|
7999
|
-
return /* @__PURE__ */ (0,
|
|
8380
|
+
return /* @__PURE__ */ (0, import_jsx_runtime19.jsxs)("div", { style: { padding: "0.25rem 0.75rem 0" }, children: [
|
|
8000
8381
|
conversations.map((conv) => {
|
|
8001
8382
|
const isSelected = conv.id === selectedId;
|
|
8002
8383
|
const isHovered = conv.id === hoveredId;
|
|
8003
|
-
return /* @__PURE__ */ (0,
|
|
8384
|
+
return /* @__PURE__ */ (0, import_jsx_runtime19.jsxs)(
|
|
8004
8385
|
"div",
|
|
8005
8386
|
{
|
|
8006
8387
|
style: {
|
|
@@ -8018,17 +8399,17 @@ var HistoryList = (0, import_react25.memo)(function HistoryList2({
|
|
|
8018
8399
|
onMouseEnter: () => setHoveredId(conv.id),
|
|
8019
8400
|
onMouseLeave: () => setHoveredId(null),
|
|
8020
8401
|
children: [
|
|
8021
|
-
/* @__PURE__ */ (0,
|
|
8022
|
-
/* @__PURE__ */ (0,
|
|
8402
|
+
/* @__PURE__ */ (0, import_jsx_runtime19.jsxs)("div", { style: { minWidth: 0, flex: 1 }, children: [
|
|
8403
|
+
/* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { style: {
|
|
8023
8404
|
fontSize: "0.875rem",
|
|
8024
8405
|
color: colors.text,
|
|
8025
8406
|
overflow: "hidden",
|
|
8026
8407
|
textOverflow: "ellipsis",
|
|
8027
8408
|
whiteSpace: "nowrap"
|
|
8028
8409
|
}, children: conv.title || "Untitled" }),
|
|
8029
|
-
/* @__PURE__ */ (0,
|
|
8410
|
+
/* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { style: { fontSize: "0.75rem", color: colors.muted }, children: formatRelativeTime(conv.updatedAt, strings) })
|
|
8030
8411
|
] }),
|
|
8031
|
-
isHovered && /* @__PURE__ */ (0,
|
|
8412
|
+
isHovered && /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(
|
|
8032
8413
|
DeleteButton,
|
|
8033
8414
|
{
|
|
8034
8415
|
onClick: (e) => {
|
|
@@ -8044,15 +8425,234 @@ var HistoryList = (0, import_react25.memo)(function HistoryList2({
|
|
|
8044
8425
|
conv.id
|
|
8045
8426
|
);
|
|
8046
8427
|
}),
|
|
8047
|
-
hasMore && /* @__PURE__ */ (0,
|
|
8428
|
+
hasMore && /* @__PURE__ */ (0, import_jsx_runtime19.jsx)("div", { ref: loadMoreRef, style: { paddingTop: "0.5rem", textAlign: "center", minHeight: "2rem" }, children: isLoading ? loadingDots : loadError ? /* @__PURE__ */ (0, import_jsx_runtime19.jsx)(LoadMoreButton, { onClick: onLoadMore, colors, children: strings.loadMore || "Load more" }) : null })
|
|
8048
8429
|
] });
|
|
8049
8430
|
});
|
|
8050
8431
|
|
|
8432
|
+
// src/renderer/components/HistoryModal.tsx
|
|
8433
|
+
var import_react31 = require("react");
|
|
8434
|
+
var import_jsx_runtime20 = require("react/jsx-runtime");
|
|
8435
|
+
var ANIMATION_DURATION = 120;
|
|
8436
|
+
var HistoryModal = (0, import_react31.memo)(function HistoryModal2({
|
|
8437
|
+
isOpen,
|
|
8438
|
+
onClose,
|
|
8439
|
+
title,
|
|
8440
|
+
closeLabel = "Close",
|
|
8441
|
+
children,
|
|
8442
|
+
isDarkMode = false
|
|
8443
|
+
}) {
|
|
8444
|
+
const modalRef = (0, import_react31.useRef)(null);
|
|
8445
|
+
const [isClosing, setIsClosing] = (0, import_react31.useState)(false);
|
|
8446
|
+
const [shouldRender, setShouldRender] = (0, import_react31.useState)(isOpen);
|
|
8447
|
+
(0, import_react31.useEffect)(() => {
|
|
8448
|
+
if (isOpen) {
|
|
8449
|
+
setShouldRender(true);
|
|
8450
|
+
setIsClosing(false);
|
|
8451
|
+
}
|
|
8452
|
+
}, [isOpen]);
|
|
8453
|
+
const handleClose = (0, import_react31.useCallback)(() => {
|
|
8454
|
+
setIsClosing(true);
|
|
8455
|
+
setTimeout(() => {
|
|
8456
|
+
setShouldRender(false);
|
|
8457
|
+
setIsClosing(false);
|
|
8458
|
+
onClose();
|
|
8459
|
+
}, ANIMATION_DURATION);
|
|
8460
|
+
}, [onClose]);
|
|
8461
|
+
(0, import_react31.useEffect)(() => {
|
|
8462
|
+
if (!shouldRender || isClosing) return;
|
|
8463
|
+
const handleKeyDown = (e) => {
|
|
8464
|
+
if (e.key === "Escape") {
|
|
8465
|
+
e.preventDefault();
|
|
8466
|
+
handleClose();
|
|
8467
|
+
}
|
|
8468
|
+
};
|
|
8469
|
+
document.addEventListener("keydown", handleKeyDown);
|
|
8470
|
+
return () => document.removeEventListener("keydown", handleKeyDown);
|
|
8471
|
+
}, [shouldRender, isClosing, handleClose]);
|
|
8472
|
+
(0, import_react31.useEffect)(() => {
|
|
8473
|
+
if (!isOpen || !modalRef.current) return;
|
|
8474
|
+
modalRef.current.focus();
|
|
8475
|
+
}, [isOpen]);
|
|
8476
|
+
const handleBackdropClick = (0, import_react31.useCallback)(
|
|
8477
|
+
(e) => {
|
|
8478
|
+
if (e.target === e.currentTarget && !isClosing) {
|
|
8479
|
+
handleClose();
|
|
8480
|
+
}
|
|
8481
|
+
},
|
|
8482
|
+
[handleClose, isClosing]
|
|
8483
|
+
);
|
|
8484
|
+
if (!shouldRender) return null;
|
|
8485
|
+
const closingClass = isClosing ? " chat-modal-closing" : "";
|
|
8486
|
+
return /* @__PURE__ */ (0, import_jsx_runtime20.jsx)(
|
|
8487
|
+
"div",
|
|
8488
|
+
{
|
|
8489
|
+
className: `chat-modal-backdrop${closingClass}`,
|
|
8490
|
+
style: {
|
|
8491
|
+
position: "absolute",
|
|
8492
|
+
inset: 0,
|
|
8493
|
+
zIndex: 50,
|
|
8494
|
+
display: "flex",
|
|
8495
|
+
alignItems: "center",
|
|
8496
|
+
justifyContent: "center",
|
|
8497
|
+
background: "rgba(0, 0, 0, 0.5)"
|
|
8498
|
+
},
|
|
8499
|
+
onClick: handleBackdropClick,
|
|
8500
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime20.jsxs)(
|
|
8501
|
+
"div",
|
|
8502
|
+
{
|
|
8503
|
+
ref: modalRef,
|
|
8504
|
+
role: "dialog",
|
|
8505
|
+
"aria-modal": "true",
|
|
8506
|
+
"aria-labelledby": "history-modal-title",
|
|
8507
|
+
tabIndex: -1,
|
|
8508
|
+
className: `chat-modal-content${closingClass}`,
|
|
8509
|
+
style: {
|
|
8510
|
+
width: "80%",
|
|
8511
|
+
height: "70%",
|
|
8512
|
+
maxWidth: "600px",
|
|
8513
|
+
minHeight: "300px",
|
|
8514
|
+
background: "var(--chat-bg)",
|
|
8515
|
+
borderRadius: "12px",
|
|
8516
|
+
border: "1px solid var(--chat-border)",
|
|
8517
|
+
boxShadow: isDarkMode ? "0 25px 50px -12px rgba(0, 0, 0, 0.5)" : "0 25px 50px -12px rgba(0, 0, 0, 0.25)",
|
|
8518
|
+
display: "flex",
|
|
8519
|
+
flexDirection: "column",
|
|
8520
|
+
overflow: "hidden",
|
|
8521
|
+
outline: "none"
|
|
8522
|
+
},
|
|
8523
|
+
onClick: (e) => e.stopPropagation(),
|
|
8524
|
+
children: [
|
|
8525
|
+
/* @__PURE__ */ (0, import_jsx_runtime20.jsxs)(
|
|
8526
|
+
"div",
|
|
8527
|
+
{
|
|
8528
|
+
style: {
|
|
8529
|
+
display: "flex",
|
|
8530
|
+
alignItems: "center",
|
|
8531
|
+
justifyContent: "space-between",
|
|
8532
|
+
padding: "8px 12px",
|
|
8533
|
+
borderBottom: "1px solid var(--chat-border)",
|
|
8534
|
+
flexShrink: 0
|
|
8535
|
+
},
|
|
8536
|
+
children: [
|
|
8537
|
+
/* @__PURE__ */ (0, import_jsx_runtime20.jsx)(
|
|
8538
|
+
"span",
|
|
8539
|
+
{
|
|
8540
|
+
id: "history-modal-title",
|
|
8541
|
+
style: {
|
|
8542
|
+
fontSize: "0.8125rem",
|
|
8543
|
+
fontWeight: 500,
|
|
8544
|
+
color: "var(--chat-text)"
|
|
8545
|
+
},
|
|
8546
|
+
children: title
|
|
8547
|
+
}
|
|
8548
|
+
),
|
|
8549
|
+
/* @__PURE__ */ (0, import_jsx_runtime20.jsx)(
|
|
8550
|
+
"button",
|
|
8551
|
+
{
|
|
8552
|
+
onClick: handleClose,
|
|
8553
|
+
"aria-label": closeLabel,
|
|
8554
|
+
style: {
|
|
8555
|
+
display: "flex",
|
|
8556
|
+
alignItems: "center",
|
|
8557
|
+
justifyContent: "center",
|
|
8558
|
+
width: "22px",
|
|
8559
|
+
height: "22px",
|
|
8560
|
+
borderRadius: "4px",
|
|
8561
|
+
border: "none",
|
|
8562
|
+
background: "transparent",
|
|
8563
|
+
color: "var(--chat-muted)",
|
|
8564
|
+
cursor: "pointer",
|
|
8565
|
+
transition: "background 0.15s, color 0.15s"
|
|
8566
|
+
},
|
|
8567
|
+
onMouseEnter: (e) => {
|
|
8568
|
+
e.currentTarget.style.background = "var(--chat-hover)";
|
|
8569
|
+
e.currentTarget.style.color = "var(--chat-text)";
|
|
8570
|
+
},
|
|
8571
|
+
onMouseLeave: (e) => {
|
|
8572
|
+
e.currentTarget.style.background = "transparent";
|
|
8573
|
+
e.currentTarget.style.color = "var(--chat-muted)";
|
|
8574
|
+
},
|
|
8575
|
+
title: closeLabel,
|
|
8576
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime20.jsx)(
|
|
8577
|
+
"svg",
|
|
8578
|
+
{
|
|
8579
|
+
width: "14",
|
|
8580
|
+
height: "14",
|
|
8581
|
+
viewBox: "0 0 24 24",
|
|
8582
|
+
fill: "none",
|
|
8583
|
+
stroke: "currentColor",
|
|
8584
|
+
strokeWidth: "2",
|
|
8585
|
+
strokeLinecap: "round",
|
|
8586
|
+
strokeLinejoin: "round",
|
|
8587
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime20.jsx)("path", { d: "M18 6L6 18M6 6l12 12" })
|
|
8588
|
+
}
|
|
8589
|
+
)
|
|
8590
|
+
}
|
|
8591
|
+
)
|
|
8592
|
+
]
|
|
8593
|
+
}
|
|
8594
|
+
),
|
|
8595
|
+
/* @__PURE__ */ (0, import_jsx_runtime20.jsx)(
|
|
8596
|
+
"div",
|
|
8597
|
+
{
|
|
8598
|
+
style: {
|
|
8599
|
+
flex: 1,
|
|
8600
|
+
overflow: "auto",
|
|
8601
|
+
minHeight: 0
|
|
8602
|
+
},
|
|
8603
|
+
className: "chat-scrollbar",
|
|
8604
|
+
children
|
|
8605
|
+
}
|
|
8606
|
+
)
|
|
8607
|
+
]
|
|
8608
|
+
}
|
|
8609
|
+
)
|
|
8610
|
+
}
|
|
8611
|
+
);
|
|
8612
|
+
});
|
|
8613
|
+
|
|
8051
8614
|
// src/renderer/components/CompactChat.tsx
|
|
8052
|
-
var
|
|
8615
|
+
var import_react32 = require("react");
|
|
8053
8616
|
var import_react_dom = require("react-dom");
|
|
8054
|
-
|
|
8055
|
-
|
|
8617
|
+
|
|
8618
|
+
// src/renderer/components/PanelControls.tsx
|
|
8619
|
+
var import_jsx_runtime21 = require("react/jsx-runtime");
|
|
8620
|
+
function PanelHeaderButtons({
|
|
8621
|
+
showModeToggle = true,
|
|
8622
|
+
showAttachToggle = true,
|
|
8623
|
+
locale = "en"
|
|
8624
|
+
}) {
|
|
8625
|
+
return /* @__PURE__ */ (0, import_jsx_runtime21.jsxs)(import_jsx_runtime21.Fragment, { children: [
|
|
8626
|
+
showModeToggle && /* @__PURE__ */ (0, import_jsx_runtime21.jsx)(ModeToggleButton, { locale }),
|
|
8627
|
+
showAttachToggle && /* @__PURE__ */ (0, import_jsx_runtime21.jsx)(AttachButton, { locale })
|
|
8628
|
+
] });
|
|
8629
|
+
}
|
|
8630
|
+
function PanelResizer({
|
|
8631
|
+
showResizer = true,
|
|
8632
|
+
resizerPosition = "left",
|
|
8633
|
+
resizerMinWidth = 280,
|
|
8634
|
+
resizerMaxWidth = 600
|
|
8635
|
+
}) {
|
|
8636
|
+
if (!showResizer) return null;
|
|
8637
|
+
return /* @__PURE__ */ (0, import_jsx_runtime21.jsx)(
|
|
8638
|
+
Resizer,
|
|
8639
|
+
{
|
|
8640
|
+
position: resizerPosition,
|
|
8641
|
+
minWidth: resizerMinWidth,
|
|
8642
|
+
maxWidth: resizerMaxWidth
|
|
8643
|
+
}
|
|
8644
|
+
);
|
|
8645
|
+
}
|
|
8646
|
+
var PanelControls = {
|
|
8647
|
+
HeaderButtons: PanelHeaderButtons,
|
|
8648
|
+
Resizer: PanelResizer,
|
|
8649
|
+
ModeToggleButton,
|
|
8650
|
+
AttachButton
|
|
8651
|
+
};
|
|
8652
|
+
|
|
8653
|
+
// src/renderer/components/CompactChat.tsx
|
|
8654
|
+
var import_jsx_runtime22 = require("react/jsx-runtime");
|
|
8655
|
+
var CompactChat = (0, import_react32.memo)(function CompactChat2({
|
|
8056
8656
|
adapter,
|
|
8057
8657
|
config,
|
|
8058
8658
|
logo,
|
|
@@ -8087,12 +8687,12 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8087
8687
|
const { themeClass, isDarkMode: resolvedIsDarkMode } = useResolvedTheme(themeMode);
|
|
8088
8688
|
const accentStyle = useAccentStyle(resolvedConfig?.accentColor);
|
|
8089
8689
|
const resolvedLogo = logo ?? resolvedConfig?.logo;
|
|
8090
|
-
const baseStrings = (0,
|
|
8690
|
+
const baseStrings = (0, import_react32.useMemo)(
|
|
8091
8691
|
() => resolveChatStrings(resolvedConfig?.locale, resolvedConfig?.strings),
|
|
8092
8692
|
[resolvedConfig?.locale, resolvedConfig?.strings]
|
|
8093
8693
|
);
|
|
8094
|
-
const mergedStrings = (0,
|
|
8095
|
-
const headerConfig = (0,
|
|
8694
|
+
const mergedStrings = (0, import_react32.useMemo)(() => ({ ...baseStrings, ...strings }), [baseStrings, strings]);
|
|
8695
|
+
const headerConfig = (0, import_react32.useMemo)(
|
|
8096
8696
|
() => {
|
|
8097
8697
|
if (!resolvedConfig) {
|
|
8098
8698
|
return resolvedLogo ? { logo: resolvedLogo } : void 0;
|
|
@@ -8105,16 +8705,17 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8105
8705
|
[resolvedConfig, resolvedLogo]
|
|
8106
8706
|
);
|
|
8107
8707
|
const { logoNode, showPin, showClose, isPinned, togglePin, resolvedOnClose } = useChatHeader(headerConfig);
|
|
8108
|
-
const emptyLogoNode = (0,
|
|
8109
|
-
const
|
|
8110
|
-
const
|
|
8111
|
-
const
|
|
8112
|
-
const [
|
|
8113
|
-
const
|
|
8708
|
+
const emptyLogoNode = (0, import_react32.useMemo)(() => resolveLogoNode(resolvedLogo, "empty"), [resolvedLogo]);
|
|
8709
|
+
const { isEmbedded, toggleMode, hide: hidePanel } = useChatPanel();
|
|
8710
|
+
const chatContainerRef = (0, import_react32.useRef)(null);
|
|
8711
|
+
const chatInputRef = (0, import_react32.useRef)(null);
|
|
8712
|
+
const [showHistory, setShowHistory] = (0, import_react32.useState)(false);
|
|
8713
|
+
const [connectionAlert, setConnectionAlert] = (0, import_react32.useState)(null);
|
|
8714
|
+
const portalContainerRef = (0, import_react32.useRef)(inputPortalContainer ?? null);
|
|
8114
8715
|
portalContainerRef.current = inputPortalContainer ?? null;
|
|
8115
8716
|
const shouldRenderInputExternally = !!inputPortalContainer;
|
|
8116
8717
|
const inputPlaceholder = placeholder ?? mergedStrings.inputPlaceholder;
|
|
8117
|
-
const headerTitle =
|
|
8718
|
+
const headerTitle = mergedStrings.chat || "Chat";
|
|
8118
8719
|
const connection = useConnection({
|
|
8119
8720
|
adapter,
|
|
8120
8721
|
autoConnect,
|
|
@@ -8148,27 +8749,27 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8148
8749
|
adapter,
|
|
8149
8750
|
onError
|
|
8150
8751
|
});
|
|
8151
|
-
(0,
|
|
8752
|
+
(0, import_react32.useEffect)(() => {
|
|
8152
8753
|
if (sendMessageRef) {
|
|
8153
8754
|
sendMessageRef.current = chat.sendMessage;
|
|
8154
8755
|
}
|
|
8155
8756
|
}, [sendMessageRef, chat.sendMessage]);
|
|
8156
|
-
(0,
|
|
8757
|
+
(0, import_react32.useEffect)(() => {
|
|
8157
8758
|
if (newConversationRef) {
|
|
8158
8759
|
newConversationRef.current = chat.newConversation;
|
|
8159
8760
|
}
|
|
8160
8761
|
}, [newConversationRef, chat.newConversation]);
|
|
8161
|
-
(0,
|
|
8762
|
+
(0, import_react32.useEffect)(() => {
|
|
8162
8763
|
if (parentFocusInputRef) {
|
|
8163
8764
|
parentFocusInputRef.current = () => chatInputRef.current?.focus();
|
|
8164
8765
|
}
|
|
8165
8766
|
}, [parentFocusInputRef]);
|
|
8166
|
-
(0,
|
|
8767
|
+
(0, import_react32.useEffect)(() => {
|
|
8167
8768
|
if (parentSetTextRef) {
|
|
8168
8769
|
parentSetTextRef.current = (text) => chatInputRef.current?.setValue(text);
|
|
8169
8770
|
}
|
|
8170
8771
|
}, [parentSetTextRef]);
|
|
8171
|
-
(0,
|
|
8772
|
+
(0, import_react32.useEffect)(() => {
|
|
8172
8773
|
if (onMessageReceived && chat.messages.length > 0) {
|
|
8173
8774
|
const lastMessage = chat.messages[chat.messages.length - 1];
|
|
8174
8775
|
if (lastMessage.role === "assistant" && !lastMessage.isStreaming) {
|
|
@@ -8176,12 +8777,12 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8176
8777
|
}
|
|
8177
8778
|
}
|
|
8178
8779
|
}, [chat.messages, onMessageReceived]);
|
|
8179
|
-
(0,
|
|
8780
|
+
(0, import_react32.useEffect)(() => {
|
|
8180
8781
|
if (onLoadingChange) {
|
|
8181
8782
|
onLoadingChange(chat.isLoading);
|
|
8182
8783
|
}
|
|
8183
8784
|
}, [chat.isLoading, onLoadingChange]);
|
|
8184
|
-
(0,
|
|
8785
|
+
(0, import_react32.useEffect)(() => {
|
|
8185
8786
|
if (onStateChange) {
|
|
8186
8787
|
onStateChange({
|
|
8187
8788
|
messages: chat.messages,
|
|
@@ -8189,19 +8790,19 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8189
8790
|
});
|
|
8190
8791
|
}
|
|
8191
8792
|
}, [chat.messages, chat.conversationId, onStateChange]);
|
|
8192
|
-
(0,
|
|
8793
|
+
(0, import_react32.useEffect)(() => {
|
|
8193
8794
|
if (showHistory && connection.isConnected) {
|
|
8194
8795
|
conversations.loadConversations();
|
|
8195
8796
|
}
|
|
8196
8797
|
}, [showHistory, connection.isConnected]);
|
|
8197
|
-
const handleSelectConversation = (0,
|
|
8798
|
+
const handleSelectConversation = (0, import_react32.useCallback)(
|
|
8198
8799
|
async (id) => {
|
|
8199
8800
|
await chat.loadConversation(id);
|
|
8200
8801
|
setShowHistory(false);
|
|
8201
8802
|
},
|
|
8202
8803
|
[chat]
|
|
8203
8804
|
);
|
|
8204
|
-
const handleDeleteConversation = (0,
|
|
8805
|
+
const handleDeleteConversation = (0, import_react32.useCallback)(
|
|
8205
8806
|
async (id) => {
|
|
8206
8807
|
await conversations.deleteConversation(id);
|
|
8207
8808
|
if (id === chat.conversationId) {
|
|
@@ -8210,14 +8811,14 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8210
8811
|
},
|
|
8211
8812
|
[conversations, chat]
|
|
8212
8813
|
);
|
|
8213
|
-
const handleNewChat = (0,
|
|
8814
|
+
const handleNewChat = (0, import_react32.useCallback)(() => {
|
|
8214
8815
|
chat.newConversation();
|
|
8215
8816
|
setShowHistory(false);
|
|
8216
8817
|
setTimeout(() => {
|
|
8217
8818
|
chatInputRef.current?.focus();
|
|
8218
8819
|
}, 0);
|
|
8219
8820
|
}, [chat]);
|
|
8220
|
-
(0,
|
|
8821
|
+
(0, import_react32.useEffect)(() => {
|
|
8221
8822
|
const handleKeyDown = (e) => {
|
|
8222
8823
|
const isMac2 = navigator.platform.includes("Mac");
|
|
8223
8824
|
const modifierKey = isMac2 ? e.metaKey : e.ctrlKey;
|
|
@@ -8231,7 +8832,7 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8231
8832
|
}, [handleNewChat]);
|
|
8232
8833
|
const isMac = typeof navigator !== "undefined" && navigator.platform.includes("Mac");
|
|
8233
8834
|
const shortcutKey = isMac ? "\u2318N" : "Ctrl+N";
|
|
8234
|
-
const defaultRenderMessage = (0,
|
|
8835
|
+
const defaultRenderMessage = (0, import_react32.useCallback)((message) => {
|
|
8235
8836
|
if (message.role === "tool") return null;
|
|
8236
8837
|
const isUser = message.role === "user";
|
|
8237
8838
|
const hasToolCalls = message.toolCalls && message.toolCalls.length > 0;
|
|
@@ -8243,9 +8844,9 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8243
8844
|
const hasThinking = message.thinking || message.isThinkingStreaming || message.currentThinking;
|
|
8244
8845
|
const showThinkingSection = !isUser && !hasToolActivity && !showIntermediateSteps && hasThinking;
|
|
8245
8846
|
const isToolCallsStreaming = message.isToolCallsStreaming ?? ((message.toolCalls?.some((tc) => tc.status === "running") ?? false) || (message.blocks?.some((b) => b.type === "tool_call" && b.toolStatus === "running") ?? false));
|
|
8246
|
-
return /* @__PURE__ */ (0,
|
|
8247
|
-
!isUser && showIntermediateSteps && message.blocks && /* @__PURE__ */ (0,
|
|
8248
|
-
showThinkingSection && /* @__PURE__ */ (0,
|
|
8847
|
+
return /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: `flex ${isUser ? "justify-end" : "justify-start"} mb-4`, children: /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: `text-left ${isUser ? "max-w-[80%]" : "w-full"}`, children: [
|
|
8848
|
+
!isUser && showIntermediateSteps && message.blocks && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(IntermediateSteps, { blocks: message.blocks, strings: mergedStrings }),
|
|
8849
|
+
showThinkingSection && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8249
8850
|
ThinkingSection,
|
|
8250
8851
|
{
|
|
8251
8852
|
thinking: message.thinking || "",
|
|
@@ -8256,7 +8857,7 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8256
8857
|
strings: mergedStrings
|
|
8257
8858
|
}
|
|
8258
8859
|
),
|
|
8259
|
-
showStreamingTimeline && message.blocks && /* @__PURE__ */ (0,
|
|
8860
|
+
showStreamingTimeline && message.blocks && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8260
8861
|
StreamingTimeline,
|
|
8261
8862
|
{
|
|
8262
8863
|
blocks: message.blocks,
|
|
@@ -8267,23 +8868,22 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8267
8868
|
strings: mergedStrings
|
|
8268
8869
|
}
|
|
8269
8870
|
),
|
|
8270
|
-
/* @__PURE__ */ (0,
|
|
8871
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: `relative group ${isUser ? "inline-block max-w-full" : "block w-full"}`, style: { fontSize: "0.875rem" }, children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8271
8872
|
"div",
|
|
8272
8873
|
{
|
|
8273
8874
|
className: `message-bubble-wrapper ${isUser ? "rounded-2xl shadow-sm bg-[var(--chat-accent)] text-white px-4 py-3 whitespace-pre-wrap break-words" : "text-[var(--chat-text)] break-words"}`,
|
|
8274
|
-
children: isUser ? /* @__PURE__ */ (0,
|
|
8875
|
+
children: isUser ? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("span", { className: "whitespace-pre-wrap", children: message.content }) : message.content && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(MarkdownRenderer, { content: message.content, isStreaming: message.isStreaming, className: "px-3" })
|
|
8275
8876
|
}
|
|
8276
8877
|
) })
|
|
8277
8878
|
] }) });
|
|
8278
8879
|
}, [mergedStrings]);
|
|
8279
|
-
const containerClass = floating ? `chat-window-container ${themeClass} ${className}` : `flex h-full flex-col bg-[var(--chat-bg)] text-[var(--chat-text)] ${themeClass} ${className}`;
|
|
8280
|
-
const baseColors = (0,
|
|
8880
|
+
const containerClass = floating ? `relative chat-window-container ${themeClass} ${className}` : `relative flex h-full flex-col bg-[var(--chat-bg)] text-[var(--chat-text)] ${themeClass} ${className}`;
|
|
8881
|
+
const baseColors = (0, import_react32.useMemo)(() => {
|
|
8281
8882
|
if (!floating) return null;
|
|
8282
8883
|
return getBaseColors(resolvedIsDarkMode);
|
|
8283
8884
|
}, [floating, resolvedIsDarkMode]);
|
|
8284
|
-
const containerStyle = (0,
|
|
8285
|
-
|
|
8286
|
-
return {
|
|
8885
|
+
const containerStyle = (0, import_react32.useMemo)(() => {
|
|
8886
|
+
const baseStyle = floating && baseColors ? {
|
|
8287
8887
|
...accentStyle,
|
|
8288
8888
|
// Override CSS variables with tinted colors for consistency with app
|
|
8289
8889
|
"--chat-bg": baseColors.bg,
|
|
@@ -8291,9 +8891,17 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8291
8891
|
"--chat-card": baseColors.card,
|
|
8292
8892
|
"--chat-window-bg": baseColors.bg,
|
|
8293
8893
|
"--chat-window-border": resolvedIsDarkMode ? "rgba(255, 255, 255, 0.1)" : "rgba(0, 0, 0, 0.1)"
|
|
8294
|
-
};
|
|
8295
|
-
|
|
8296
|
-
|
|
8894
|
+
} : accentStyle;
|
|
8895
|
+
if (isEmbedded) {
|
|
8896
|
+
return {
|
|
8897
|
+
...baseStyle,
|
|
8898
|
+
borderRadius: 0,
|
|
8899
|
+
boxShadow: "none"
|
|
8900
|
+
};
|
|
8901
|
+
}
|
|
8902
|
+
return baseStyle;
|
|
8903
|
+
}, [floating, baseColors, accentStyle, resolvedIsDarkMode, isEmbedded]);
|
|
8904
|
+
return /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)(
|
|
8297
8905
|
"div",
|
|
8298
8906
|
{
|
|
8299
8907
|
ref: chatContainerRef,
|
|
@@ -8301,103 +8909,172 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8301
8909
|
style: containerStyle,
|
|
8302
8910
|
"data-chat-font-size": resolvedConfig?.fontSize || "normal",
|
|
8303
8911
|
children: [
|
|
8304
|
-
|
|
8305
|
-
|
|
8306
|
-
|
|
8307
|
-
|
|
8308
|
-
|
|
8309
|
-
|
|
8310
|
-
|
|
8311
|
-
|
|
8312
|
-
|
|
8313
|
-
|
|
8314
|
-
|
|
8315
|
-
|
|
8316
|
-
|
|
8317
|
-
|
|
8318
|
-
|
|
8319
|
-
|
|
8320
|
-
|
|
8321
|
-
|
|
8322
|
-
/* @__PURE__ */ (0,
|
|
8323
|
-
/* @__PURE__ */ (0,
|
|
8912
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(PanelResizer, {}),
|
|
8913
|
+
!hideHeader && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("header", { className: "flex h-9 flex-shrink-0 items-center px-2 border-b chat-divider-border", style: { WebkitAppRegion: "drag" }, children: isEmbedded ? (
|
|
8914
|
+
// Embedded mode: left-aligned buttons (collapse, float, history, new chat)
|
|
8915
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "flex items-center gap-0.5", style: { WebkitAppRegion: "no-drag" }, children: [
|
|
8916
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8917
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8918
|
+
"button",
|
|
8919
|
+
{
|
|
8920
|
+
onClick: hidePanel,
|
|
8921
|
+
className: headerIconButtonClass,
|
|
8922
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: [
|
|
8923
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("rect", { x: "3", y: "4", width: "18", height: "16", rx: "2", strokeLinecap: "round", strokeLinejoin: "round" }),
|
|
8924
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M15 4v16", strokeLinecap: "round" })
|
|
8925
|
+
] })
|
|
8926
|
+
}
|
|
8927
|
+
),
|
|
8928
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: mergedStrings.collapseSidebar })
|
|
8929
|
+
] }),
|
|
8930
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8931
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8932
|
+
"button",
|
|
8933
|
+
{
|
|
8934
|
+
onClick: toggleMode,
|
|
8935
|
+
className: headerIconButtonClass,
|
|
8936
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: [
|
|
8937
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("rect", { x: "8", y: "3", width: "13", height: "13", rx: "2", strokeLinecap: "round", strokeLinejoin: "round" }),
|
|
8938
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M5 8H4a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-1", strokeLinecap: "round", strokeLinejoin: "round" })
|
|
8939
|
+
] })
|
|
8940
|
+
}
|
|
8941
|
+
),
|
|
8942
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: mergedStrings.floatWindow })
|
|
8943
|
+
] }),
|
|
8944
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8945
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8946
|
+
"button",
|
|
8947
|
+
{
|
|
8948
|
+
onClick: () => setShowHistory(true),
|
|
8949
|
+
className: headerIconButtonClass,
|
|
8950
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8951
|
+
}
|
|
8952
|
+
),
|
|
8953
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: mergedStrings.history || mergedStrings.recentChats })
|
|
8954
|
+
] }),
|
|
8955
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8956
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8324
8957
|
"button",
|
|
8325
8958
|
{
|
|
8326
8959
|
onClick: handleNewChat,
|
|
8327
8960
|
className: headerIconButtonClass,
|
|
8328
|
-
children: /* @__PURE__ */ (0,
|
|
8961
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 5v14m-7-7h14", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8329
8962
|
}
|
|
8330
8963
|
),
|
|
8331
|
-
/* @__PURE__ */ (0,
|
|
8332
|
-
/* @__PURE__ */ (0,
|
|
8333
|
-
/* @__PURE__ */ (0,
|
|
8964
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip", children: [
|
|
8965
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { children: mergedStrings.newChat || "New Chat" }),
|
|
8966
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip-shortcut", children: shortcutKey })
|
|
8334
8967
|
] })
|
|
8335
8968
|
] }),
|
|
8336
|
-
!showHistory && /* @__PURE__ */ (0, import_jsx_runtime17.jsx)(
|
|
8337
|
-
"button",
|
|
8338
|
-
{
|
|
8339
|
-
onClick: () => setShowHistory(true),
|
|
8340
|
-
className: headerIconButtonClass,
|
|
8341
|
-
title: mergedStrings.recentChats || "Recent Chats",
|
|
8342
|
-
children: /* @__PURE__ */ (0, import_jsx_runtime17.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8343
|
-
}
|
|
8344
|
-
),
|
|
8345
|
-
showPin && /* @__PURE__ */ (0, import_jsx_runtime17.jsx)(
|
|
8346
|
-
"button",
|
|
8347
|
-
{
|
|
8348
|
-
onClick: togglePin,
|
|
8349
|
-
className: headerIconButtonClass,
|
|
8350
|
-
title: isPinned ? mergedStrings.unpin : mergedStrings.pin,
|
|
8351
|
-
children: /* @__PURE__ */ (0, import_jsx_runtime17.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", strokeLinecap: "round", strokeLinejoin: "round", children: isPinned ? (
|
|
8352
|
-
// Pin (active state) - lucide pin
|
|
8353
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsxs)(import_jsx_runtime17.Fragment, { children: [
|
|
8354
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M12 17v5" }),
|
|
8355
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M9 10.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-.76a2 2 0 0 0-1.11-1.79l-1.78-.9A2 2 0 0 1 15 10.76V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H8a2 2 0 0 0 0 4 1 1 0 0 1 1 1z" })
|
|
8356
|
-
] })
|
|
8357
|
-
) : (
|
|
8358
|
-
// PinOff (inactive state) - lucide pin-off
|
|
8359
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsxs)(import_jsx_runtime17.Fragment, { children: [
|
|
8360
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M12 17v5" }),
|
|
8361
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M15 9.34V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H7.89" }),
|
|
8362
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "m2 2 20 20" }),
|
|
8363
|
-
/* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M9 9v1.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h11" })
|
|
8364
|
-
] })
|
|
8365
|
-
) })
|
|
8366
|
-
}
|
|
8367
|
-
),
|
|
8368
|
-
showClose && /* @__PURE__ */ (0, import_jsx_runtime17.jsx)(
|
|
8369
|
-
"button",
|
|
8370
|
-
{
|
|
8371
|
-
onClick: resolvedOnClose,
|
|
8372
|
-
className: headerIconButtonClass,
|
|
8373
|
-
title: mergedStrings.close,
|
|
8374
|
-
children: /* @__PURE__ */ (0, import_jsx_runtime17.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime17.jsx)("path", { d: "M6 6l12 12M18 6l-12 12", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8375
|
-
}
|
|
8376
|
-
),
|
|
8377
8969
|
headerRight
|
|
8378
8970
|
] })
|
|
8379
|
-
|
|
8380
|
-
|
|
8381
|
-
|
|
8971
|
+
) : (
|
|
8972
|
+
// Floating mode: original layout with logo, title, and right-aligned buttons
|
|
8973
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)(import_jsx_runtime22.Fragment, { children: [
|
|
8974
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "flex items-center gap-1", style: { WebkitAppRegion: "no-drag" }, children: headerLeft || /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "flex items-center gap-2", children: [
|
|
8975
|
+
logoNode,
|
|
8976
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("span", { className: "text-sm font-medium text-[var(--chat-text)]", children: headerTitle })
|
|
8977
|
+
] }) }),
|
|
8978
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "flex-1" }),
|
|
8979
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "flex items-center gap-0.5", style: { WebkitAppRegion: "no-drag" }, children: [
|
|
8980
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(PanelHeaderButtons, { locale: resolvedConfig?.locale }),
|
|
8981
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8982
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8983
|
+
"button",
|
|
8984
|
+
{
|
|
8985
|
+
onClick: handleNewChat,
|
|
8986
|
+
className: headerIconButtonClass,
|
|
8987
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 5v14m-7-7h14", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
8988
|
+
}
|
|
8989
|
+
),
|
|
8990
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip", children: [
|
|
8991
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { children: mergedStrings.newChat || "New Chat" }),
|
|
8992
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip-shortcut", children: shortcutKey })
|
|
8993
|
+
] })
|
|
8994
|
+
] }),
|
|
8995
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
8996
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8997
|
+
"button",
|
|
8998
|
+
{
|
|
8999
|
+
onClick: () => setShowHistory(true),
|
|
9000
|
+
className: headerIconButtonClass,
|
|
9001
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
9002
|
+
}
|
|
9003
|
+
),
|
|
9004
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: mergedStrings.recentChats || "Recent Chats" })
|
|
9005
|
+
] }),
|
|
9006
|
+
showPin && /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
9007
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9008
|
+
"button",
|
|
9009
|
+
{
|
|
9010
|
+
onClick: togglePin,
|
|
9011
|
+
className: headerIconButtonClass,
|
|
9012
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", strokeLinecap: "round", strokeLinejoin: "round", children: isPinned ? (
|
|
9013
|
+
// Pin (active state) - lucide pin
|
|
9014
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)(import_jsx_runtime22.Fragment, { children: [
|
|
9015
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 17v5" }),
|
|
9016
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M9 10.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-.76a2 2 0 0 0-1.11-1.79l-1.78-.9A2 2 0 0 1 15 10.76V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H8a2 2 0 0 0 0 4 1 1 0 0 1 1 1z" })
|
|
9017
|
+
] })
|
|
9018
|
+
) : (
|
|
9019
|
+
// PinOff (inactive state) - lucide pin-off
|
|
9020
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsxs)(import_jsx_runtime22.Fragment, { children: [
|
|
9021
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M12 17v5" }),
|
|
9022
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M15 9.34V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H7.89" }),
|
|
9023
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "m2 2 20 20" }),
|
|
9024
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M9 9v1.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h11" })
|
|
9025
|
+
] })
|
|
9026
|
+
) })
|
|
9027
|
+
}
|
|
9028
|
+
),
|
|
9029
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: isPinned ? mergedStrings.unpin : mergedStrings.pin })
|
|
9030
|
+
] }),
|
|
9031
|
+
showClose && /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "chat-tooltip-wrapper", children: [
|
|
9032
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9033
|
+
"button",
|
|
9034
|
+
{
|
|
9035
|
+
onClick: resolvedOnClose,
|
|
9036
|
+
className: headerIconButtonClass,
|
|
9037
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("svg", { className: headerIconClass, viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.5", children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("path", { d: "M6 6l12 12M18 6l-12 12", strokeLinecap: "round", strokeLinejoin: "round" }) })
|
|
9038
|
+
}
|
|
9039
|
+
),
|
|
9040
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "chat-tooltip", children: mergedStrings.close })
|
|
9041
|
+
] }),
|
|
9042
|
+
headerRight
|
|
9043
|
+
] })
|
|
9044
|
+
] })
|
|
9045
|
+
) }),
|
|
9046
|
+
/* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9047
|
+
HistoryModal,
|
|
8382
9048
|
{
|
|
8383
|
-
|
|
8384
|
-
|
|
8385
|
-
|
|
8386
|
-
|
|
8387
|
-
loadError: conversations.loadError,
|
|
8388
|
-
onSelect: handleSelectConversation,
|
|
8389
|
-
onDelete: handleDeleteConversation,
|
|
8390
|
-
onLoadMore: conversations.loadMore,
|
|
9049
|
+
isOpen: showHistory,
|
|
9050
|
+
onClose: () => setShowHistory(false),
|
|
9051
|
+
title: mergedStrings.recentChats || "Recent Chats",
|
|
9052
|
+
closeLabel: mergedStrings.close || "Close",
|
|
8391
9053
|
isDarkMode: resolvedIsDarkMode,
|
|
8392
|
-
|
|
8393
|
-
|
|
8394
|
-
|
|
8395
|
-
|
|
8396
|
-
|
|
8397
|
-
|
|
8398
|
-
|
|
9054
|
+
children: /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9055
|
+
HistoryList,
|
|
9056
|
+
{
|
|
9057
|
+
conversations: conversations.conversations,
|
|
9058
|
+
selectedId: chat.conversationId,
|
|
9059
|
+
isLoading: conversations.isLoading,
|
|
9060
|
+
hasMore: conversations.hasMore,
|
|
9061
|
+
loadError: conversations.loadError,
|
|
9062
|
+
onSelect: handleSelectConversation,
|
|
9063
|
+
onDelete: handleDeleteConversation,
|
|
9064
|
+
onLoadMore: conversations.loadMore,
|
|
9065
|
+
isDarkMode: resolvedIsDarkMode,
|
|
9066
|
+
strings: {
|
|
9067
|
+
noHistory: mergedStrings.noHistory,
|
|
9068
|
+
loadMore: mergedStrings.loadMore,
|
|
9069
|
+
today: mergedStrings.today,
|
|
9070
|
+
yesterday: mergedStrings.yesterday,
|
|
9071
|
+
delete: mergedStrings.delete
|
|
9072
|
+
}
|
|
9073
|
+
}
|
|
9074
|
+
)
|
|
8399
9075
|
}
|
|
8400
|
-
)
|
|
9076
|
+
),
|
|
9077
|
+
chat.messages.length === 0 ? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "flex flex-1 items-center justify-center p-4", children: emptyState ?? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)("div", { className: "flex items-center justify-center", children: emptyLogoNode }) }) : /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8401
9078
|
MessageList,
|
|
8402
9079
|
{
|
|
8403
9080
|
messages: chat.messages,
|
|
@@ -8407,83 +9084,80 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8407
9084
|
),
|
|
8408
9085
|
(() => {
|
|
8409
9086
|
if (hideInput && !shouldRenderInputExternally) return null;
|
|
8410
|
-
const disableInput =
|
|
8411
|
-
const inputElement = /* @__PURE__ */ (0,
|
|
9087
|
+
const disableInput = !!chat.pendingInterrupt || chat.isLoading && !chat.isStreaming;
|
|
9088
|
+
const inputElement = /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
8412
9089
|
ChatInput,
|
|
8413
9090
|
{
|
|
8414
9091
|
ref: chatInputRef,
|
|
8415
9092
|
onSend: chat.sendMessage,
|
|
8416
9093
|
onStop: chat.stopStreaming,
|
|
8417
|
-
placeholder:
|
|
9094
|
+
placeholder: inputPlaceholder,
|
|
8418
9095
|
sendLabel: mergedStrings.inputSend,
|
|
8419
9096
|
stopLabel: mergedStrings.inputStop,
|
|
8420
9097
|
disabled: disableInput,
|
|
8421
9098
|
isStreaming: chat.isStreaming,
|
|
8422
9099
|
isLoading: chat.isLoading,
|
|
8423
|
-
autoFocus:
|
|
8424
|
-
className: showHistory ? "opacity-50" : ""
|
|
9100
|
+
autoFocus: true
|
|
8425
9101
|
}
|
|
8426
9102
|
);
|
|
8427
9103
|
if (shouldRenderInputExternally && portalContainerRef.current) {
|
|
8428
9104
|
return (0, import_react_dom.createPortal)(inputElement, portalContainerRef.current);
|
|
8429
9105
|
}
|
|
8430
|
-
return /* @__PURE__ */ (0,
|
|
8431
|
-
|
|
8432
|
-
|
|
8433
|
-
|
|
8434
|
-
|
|
8435
|
-
|
|
8436
|
-
|
|
8437
|
-
|
|
8438
|
-
|
|
8439
|
-
|
|
8440
|
-
|
|
8441
|
-
|
|
8442
|
-
|
|
8443
|
-
|
|
8444
|
-
|
|
8445
|
-
|
|
8446
|
-
|
|
8447
|
-
|
|
8448
|
-
|
|
8449
|
-
|
|
8450
|
-
|
|
8451
|
-
|
|
8452
|
-
|
|
8453
|
-
|
|
8454
|
-
|
|
8455
|
-
|
|
8456
|
-
|
|
8457
|
-
|
|
8458
|
-
|
|
8459
|
-
|
|
8460
|
-
|
|
8461
|
-
|
|
8462
|
-
|
|
8463
|
-
|
|
8464
|
-
|
|
8465
|
-
|
|
8466
|
-
|
|
8467
|
-
|
|
8468
|
-
|
|
8469
|
-
|
|
8470
|
-
|
|
8471
|
-
|
|
8472
|
-
|
|
8473
|
-
|
|
8474
|
-
|
|
8475
|
-
|
|
8476
|
-
|
|
8477
|
-
|
|
8478
|
-
|
|
8479
|
-
|
|
8480
|
-
|
|
8481
|
-
|
|
8482
|
-
inputRequest: mergedStrings.hitlInputRequest
|
|
8483
|
-
}
|
|
9106
|
+
return /* @__PURE__ */ (0, import_jsx_runtime22.jsxs)("div", { className: "flex flex-col gap-2 px-3 pb-3 pt-1", children: [
|
|
9107
|
+
alert ? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9108
|
+
AlertBanner,
|
|
9109
|
+
{
|
|
9110
|
+
type: alert.type,
|
|
9111
|
+
message: alert.message,
|
|
9112
|
+
action: alert.action,
|
|
9113
|
+
onDismiss: alert.dismissible ? onAlertDismiss : void 0,
|
|
9114
|
+
className: `flex items-center gap-2 rounded-xl px-3 py-2 text-sm ${alert.type === "error" ? resolvedIsDarkMode ? "bg-red-900/60 text-red-200" : "bg-red-100 text-red-800" : resolvedIsDarkMode ? "bg-amber-900/60 text-amber-200" : "bg-amber-100 text-amber-800"}`
|
|
9115
|
+
}
|
|
9116
|
+
) : connectionAlert ? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9117
|
+
AlertBanner,
|
|
9118
|
+
{
|
|
9119
|
+
type: connectionAlert.type,
|
|
9120
|
+
message: connectionAlert.message,
|
|
9121
|
+
action: connectionAlert.action,
|
|
9122
|
+
onDismiss: connectionAlert.dismissible ? () => setConnectionAlert(null) : void 0,
|
|
9123
|
+
className: `flex items-center gap-2 rounded-xl px-3 py-2 text-sm ${connectionAlert.type === "error" ? resolvedIsDarkMode ? "bg-red-900/60 text-red-200" : "bg-red-100 text-red-800" : resolvedIsDarkMode ? "bg-amber-900/60 text-amber-200" : "bg-amber-100 text-amber-800"}`
|
|
9124
|
+
}
|
|
9125
|
+
) : chat.error ? /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9126
|
+
AlertBanner,
|
|
9127
|
+
{
|
|
9128
|
+
type: "error",
|
|
9129
|
+
message: chat.error,
|
|
9130
|
+
onDismiss: () => chat.setError(null),
|
|
9131
|
+
className: `flex items-center gap-2 rounded-xl px-3 py-2 text-sm ${resolvedIsDarkMode ? "bg-red-900/60 text-red-200" : "bg-red-100 text-red-800"}`
|
|
9132
|
+
}
|
|
9133
|
+
) : null,
|
|
9134
|
+
chat.pendingInterrupt && /* @__PURE__ */ (0, import_jsx_runtime22.jsx)(
|
|
9135
|
+
HitlCard,
|
|
9136
|
+
{
|
|
9137
|
+
interrupt: chat.pendingInterrupt,
|
|
9138
|
+
onApprove: chat.approveHitl,
|
|
9139
|
+
onReject: chat.rejectHitl,
|
|
9140
|
+
onSubmit: chat.submitHitlInput,
|
|
9141
|
+
isDarkMode: resolvedIsDarkMode,
|
|
9142
|
+
strings: {
|
|
9143
|
+
approve: mergedStrings.hitlApprove,
|
|
9144
|
+
reject: mergedStrings.hitlReject,
|
|
9145
|
+
submit: mergedStrings.hitlSubmit,
|
|
9146
|
+
cancel: mergedStrings.hitlCancel,
|
|
9147
|
+
rememberChoice: mergedStrings.hitlRememberChoice,
|
|
9148
|
+
requiredField: mergedStrings.hitlRequiredField,
|
|
9149
|
+
timeoutIn: mergedStrings.hitlTimeoutIn,
|
|
9150
|
+
seconds: mergedStrings.hitlSeconds,
|
|
9151
|
+
executeTool: mergedStrings.hitlExecuteTool,
|
|
9152
|
+
toolLabel: mergedStrings.hitlToolLabel,
|
|
9153
|
+
argsLabel: mergedStrings.hitlArgsLabel,
|
|
9154
|
+
defaultPrefix: mergedStrings.hitlDefaultPrefix,
|
|
9155
|
+
enterResponse: mergedStrings.hitlEnterResponse,
|
|
9156
|
+
approvalRequest: mergedStrings.hitlApprovalRequest,
|
|
9157
|
+
inputRequest: mergedStrings.hitlInputRequest
|
|
8484
9158
|
}
|
|
8485
|
-
|
|
8486
|
-
|
|
9159
|
+
}
|
|
9160
|
+
),
|
|
8487
9161
|
inputElement
|
|
8488
9162
|
] });
|
|
8489
9163
|
})()
|
|
@@ -8494,16 +9168,23 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8494
9168
|
// Annotate the CommonJS export names for ESM import in node:
|
|
8495
9169
|
0 && (module.exports = {
|
|
8496
9170
|
AlertBanner,
|
|
9171
|
+
AttachButton,
|
|
8497
9172
|
ChatInput,
|
|
8498
9173
|
CompactChat,
|
|
8499
9174
|
FloatingChat,
|
|
8500
9175
|
HistoryList,
|
|
9176
|
+
HistoryModal,
|
|
8501
9177
|
HitlCard,
|
|
8502
9178
|
I18nProvider,
|
|
8503
9179
|
IntermediateSteps,
|
|
8504
9180
|
MarkdownRenderer,
|
|
8505
9181
|
MessageBubble,
|
|
8506
9182
|
MessageList,
|
|
9183
|
+
ModeToggleButton,
|
|
9184
|
+
PanelControls,
|
|
9185
|
+
PanelHeaderButtons,
|
|
9186
|
+
PanelResizer,
|
|
9187
|
+
Resizer,
|
|
8507
9188
|
SanqianChat,
|
|
8508
9189
|
SanqianChatMessage,
|
|
8509
9190
|
SanqianMessageList,
|
|
@@ -8519,7 +9200,9 @@ var CompactChat = (0, import_react26.memo)(function CompactChat2({
|
|
|
8519
9200
|
ensureFullChatStyles,
|
|
8520
9201
|
getTranslations,
|
|
8521
9202
|
resolveChatStrings,
|
|
9203
|
+
useAttachState,
|
|
8522
9204
|
useChat,
|
|
9205
|
+
useChatPanel,
|
|
8523
9206
|
useChatStyles,
|
|
8524
9207
|
useConnection,
|
|
8525
9208
|
useConversations,
|