@nextclaw/ui 0.10.0 → 0.10.1

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.
Files changed (67) hide show
  1. package/CHANGELOG.md +18 -1
  2. package/dist/assets/{ChannelsList-VSRZzxx2.js → ChannelsList-BX7KqEk7.js} +4 -4
  3. package/dist/assets/ChatPage-zXLBKIAY.js +38 -0
  4. package/dist/assets/{DocBrowser-C65Hbvnb.js → DocBrowser-Cdbh4cVD.js} +1 -1
  5. package/dist/assets/{LogoBadge-4qtguXEJ.js → LogoBadge-4801esOJ.js} +1 -1
  6. package/dist/assets/MarketplacePage-GZgus0Or.js +49 -0
  7. package/dist/assets/{McpMarketplacePage-CHLkD8yX.js → McpMarketplacePage-CAGGvoMo.js} +1 -1
  8. package/dist/assets/{ModelConfig-CjsGdmZa.js → ModelConfig-CfLYjQM3.js} +1 -1
  9. package/dist/assets/ProvidersList-CEo1kdf-.js +1 -0
  10. package/dist/assets/{RemoteAccessPage-rOZCnH1x.js → RemoteAccessPage-6GYzD7cc.js} +1 -1
  11. package/dist/assets/{RuntimeConfig-CmJh6g0R.js → RuntimeConfig-BZdbp8mH.js} +1 -1
  12. package/dist/assets/{SearchConfig-C_hUuzR4.js → SearchConfig-ifvYKix-.js} +1 -1
  13. package/dist/assets/{SecretsConfig-Bu_zIRlQ.js → SecretsConfig-tDPbhTeR.js} +1 -1
  14. package/dist/assets/{SessionsConfig-DA_nqkM_.js → SessionsConfig-DhkAIzGm.js} +1 -1
  15. package/dist/assets/{chat-message-BOdA4h43.js → chat-message-C5Gl-dCH.js} +1 -1
  16. package/dist/assets/index-BTt_JlNV.css +1 -0
  17. package/dist/assets/index-JN3V84h_.js +8 -0
  18. package/dist/assets/{label-BYZ62ajO.js → label-D8zWKdqp.js} +1 -1
  19. package/dist/assets/{page-layout-UC-h92sU.js → page-layout-qAJ47LNQ.js} +1 -1
  20. package/dist/assets/{popover-DASCEr3G.js → popover-hyBGxpxS.js} +1 -1
  21. package/dist/assets/{security-config-Cvujq4fH.js → security-config-BJYZSnCA.js} +1 -1
  22. package/dist/assets/skeleton-CUQLsNsM.js +1 -0
  23. package/dist/assets/{status-dot-C1AvPwDD.js → status-dot-DKcoD-iY.js} +1 -1
  24. package/dist/assets/{switch-D3wVuCSh.js → switch-DtUdQxr_.js} +1 -1
  25. package/dist/assets/tabs-custom-Dj1BWHGK.js +1 -0
  26. package/dist/assets/useConfirmDialog-nZdrtETU.js +1 -0
  27. package/dist/assets/{vendor-DJt0Azq5.js → vendor-CNhxtHCf.js} +1 -1
  28. package/dist/index.html +3 -3
  29. package/package.json +5 -5
  30. package/src/components/chat/ChatSidebar.tsx +41 -69
  31. package/src/components/chat/adapters/chat-input-bar.adapter.test.ts +32 -1
  32. package/src/components/chat/adapters/chat-input-bar.adapter.ts +6 -3
  33. package/src/components/chat/adapters/chat-message.adapter.test.ts +141 -163
  34. package/src/components/chat/adapters/chat-message.adapter.ts +35 -0
  35. package/src/components/chat/chat-composer-state.ts +38 -0
  36. package/src/components/chat/chat-stream/types.ts +2 -0
  37. package/src/components/chat/containers/chat-input-bar.container.tsx +116 -55
  38. package/src/components/chat/containers/chat-message-list.container.tsx +2 -0
  39. package/src/components/chat/managers/chat-session-list.manager.test.ts +16 -1
  40. package/src/components/chat/managers/chat-session-list.manager.ts +0 -2
  41. package/src/components/chat/managers/chat-thread.manager.ts +0 -1
  42. package/src/components/chat/ncp/NcpChatPage.tsx +18 -18
  43. package/src/components/chat/ncp/ncp-app-client-fetch.test.ts +50 -33
  44. package/src/components/chat/ncp/ncp-app-client-fetch.ts +5 -123
  45. package/src/components/chat/ncp/ncp-chat-input.manager.ts +56 -1
  46. package/src/components/chat/ncp/ncp-chat-page-data.test.ts +8 -0
  47. package/src/components/chat/ncp/ncp-chat-thread.manager.ts +0 -1
  48. package/src/components/chat/presenter/chat-presenter-context.tsx +6 -0
  49. package/src/components/chat/stores/chat-input.store.ts +3 -0
  50. package/src/components/config/ChannelsList.test.tsx +2 -1
  51. package/src/components/config/weixin-channel-auth-section.test.tsx +2 -1
  52. package/src/components/layout/Sidebar.tsx +62 -102
  53. package/src/components/layout/sidebar-items.tsx +172 -0
  54. package/src/components/layout/sidebar.layout.test.tsx +11 -4
  55. package/src/lib/i18n.chat.ts +117 -0
  56. package/src/lib/i18n.remote.ts +1 -1
  57. package/src/lib/i18n.ts +2 -112
  58. package/src/transport/remote.transport.test.ts +135 -0
  59. package/src/transport/remote.transport.ts +11 -1
  60. package/dist/assets/ChatPage-CX0ZKE5i.js +0 -41
  61. package/dist/assets/MarketplacePage-DPCYptfD.js +0 -49
  62. package/dist/assets/ProvidersList-aXp_mo4J.js +0 -1
  63. package/dist/assets/index-C63mHRbE.css +0 -1
  64. package/dist/assets/index-DS7D1-KS.js +0 -8
  65. package/dist/assets/skeleton-DlYEKkkj.js +0 -1
  66. package/dist/assets/tabs-custom-CbgS7tu0.js +0 -1
  67. package/dist/assets/useConfirmDialog-BYbFEIbQ.js +0 -1
@@ -0,0 +1,117 @@
1
+ export const CHAT_LABELS: Record<string, { zh: string; en: string }> = {
2
+ chatPageTitle: { zh: 'Agent 对话', en: 'Agent Chat' },
3
+ chatPageDescription: {
4
+ zh: '在 UI 内直接与 Agent 交互,支持多会话与多 Agent 切换。',
5
+ en: 'Chat with your agent directly in UI with multi-session and multi-agent switching.'
6
+ },
7
+ chatRefresh: { zh: '刷新', en: 'Refresh' },
8
+ chatNewSession: { zh: '新会话', en: 'New Session' },
9
+ chatSearchSessionPlaceholder: { zh: '搜索会话 key / 标签', en: 'Search session key / label' },
10
+ chatAgentLabel: { zh: '目标 Agent', en: 'Target Agent' },
11
+ chatSelectAgent: { zh: '选择 Agent', en: 'Select Agent' },
12
+ chatModelLabel: { zh: '对话模型', en: 'Chat Model' },
13
+ chatSelectModel: { zh: '选择模型', en: 'Select model' },
14
+ chatThinkingLevelOff: { zh: '思考关闭', en: 'Thinking Off' },
15
+ chatThinkingLevelMinimal: { zh: '思考 Minimal', en: 'Thinking Minimal' },
16
+ chatThinkingLevelLow: { zh: '思考 Low', en: 'Thinking Low' },
17
+ chatThinkingLevelMedium: { zh: '思考 Medium', en: 'Thinking Medium' },
18
+ chatThinkingLevelHigh: { zh: '思考 High', en: 'Thinking High' },
19
+ chatThinkingLevelAdaptive: { zh: '思考 Adaptive', en: 'Thinking Adaptive' },
20
+ chatThinkingLevelXhigh: { zh: '思考 XHigh', en: 'Thinking XHigh' },
21
+ chatSessionTypeLabel: { zh: '会话类型', en: 'Session Type' },
22
+ chatSessionTypeNative: { zh: '原生', en: 'Native' },
23
+ chatSessionTypeCodex: { zh: 'Codex', en: 'Codex' },
24
+ chatSessionTypeClaude: { zh: 'Claude Code', en: 'Claude Code' },
25
+ chatSessionTypeUnavailableSuffix: {
26
+ zh: '当前不可用,请启用对应插件或新建 Native 会话。',
27
+ en: 'is unavailable now. Re-enable the plugin or create a native session.'
28
+ },
29
+ chatModelNoOptions: { zh: '暂无可用模型,请先配置提供商。', en: 'No available models. Configure a provider first.' },
30
+ chatGoConfigureProvider: { zh: '去配置提供商', en: 'Go to Providers' },
31
+ chatProviderSetupTitle: { zh: '开始前先配置提供商', en: 'Configure a Provider First' },
32
+ chatProviderSetupDescription: {
33
+ zh: '你还没有可用模型。先在提供商页面配置并保存至少一个 Provider 后,再回来开始对话。',
34
+ en: 'No models are available yet. Configure and save at least one provider, then return to start chatting.'
35
+ },
36
+ chatSessionLabel: { zh: '当前会话', en: 'Current Session' },
37
+ chatNoSession: { zh: '未选择会话', en: 'No session selected' },
38
+ chatNoSessionHint: { zh: '创建一个会话并发送第一条消息。', en: 'Create a session and send your first message.' },
39
+ chatHistoryLoading: { zh: '加载会话历史中...', en: 'Loading session history...' },
40
+ chatNoMessages: { zh: '暂无消息,发送一条开始对话。', en: 'No messages yet. Send one to start.' },
41
+ chatTyping: { zh: 'Agent 正在思考...', en: 'Agent is thinking...' },
42
+ chatInputPlaceholder: { zh: '输入消息,输入 / 选择技能,Enter 发送,Shift + Enter 换行', en: 'Type a message, type / to select skills, Enter to send, Shift + Enter for newline' },
43
+ chatInputHint: { zh: '支持多轮上下文,默认走当前会话。', en: 'Multi-turn context is preserved in the current session.' },
44
+ chatSlashSectionCommands: { zh: '命令', en: 'Commands' },
45
+ chatSlashSectionSkills: { zh: '技能', en: 'Skills' },
46
+ chatSlashTypeCommand: { zh: '命令', en: 'Command' },
47
+ chatSlashTypeSkill: { zh: '技能', en: 'Skill' },
48
+ chatSlashSkillSpec: { zh: '标识', en: 'Spec' },
49
+ chatSlashLoading: { zh: '加载命令与技能中…', en: 'Loading commands and skills…' },
50
+ chatSlashNoResult: { zh: '无匹配项', en: 'No matches' },
51
+ chatSlashHint: { zh: '输入 / 触发命令或技能选择', en: 'Type / to access commands and skills' },
52
+ chatSlashCommandHint: { zh: '回车插入命令,继续输入参数后发送。', en: 'Press Enter to insert command, then add args and send.' },
53
+ chatSlashSkillHint: { zh: '回车把该技能加入本轮请求。', en: 'Press Enter to add this skill for the next turn.' },
54
+ chatSend: { zh: '发送', en: 'Send' },
55
+ chatStop: { zh: '停止', en: 'Stop' },
56
+ chatStopPreparing: { zh: '正在建立可停止会话,请稍候…', en: 'Preparing stoppable run…' },
57
+ chatStopUnavailable: { zh: '当前后端引擎不支持手动停止。', en: 'Manual stop is not supported by the current backend engine.' },
58
+ chatSending: { zh: '发送中...', en: 'Sending...' },
59
+ chatQueueSend: { zh: '排队发送', en: 'Queue' },
60
+ chatQueuedHintPrefix: { zh: '当前有', en: 'Queued' },
61
+ chatQueuedHintSuffix: { zh: '条消息待发送。', en: 'pending messages.' },
62
+ chatQueueMoveFirst: { zh: '置顶到下一条', en: 'Move to Next' },
63
+ chatDeleteSession: { zh: '删除会话', en: 'Delete Session' },
64
+ chatDeleteSessionConfirm: { zh: '确认删除当前会话?', en: 'Delete the current session?' },
65
+ chatSendFailed: { zh: '发送消息失败', en: 'Failed to send message' },
66
+ chatRoleUser: { zh: '你', en: 'You' },
67
+ chatRoleAssistant: { zh: '助手', en: 'Assistant' },
68
+ chatRoleTool: { zh: '工具', en: 'Tool' },
69
+ chatRoleSystem: { zh: '系统', en: 'System' },
70
+ chatRoleMessage: { zh: '消息', en: 'Message' },
71
+ chatToolCall: { zh: '工具调用', en: 'Tool Call' },
72
+ chatToolResult: { zh: '工具结果', en: 'Tool Result' },
73
+ chatToolWorkflow: { zh: '工具工作流', en: 'Tool Workflow' },
74
+ chatToolWorkflowDetails: { zh: '展开查看参数和结果', en: 'Expand to view params and results' },
75
+ chatToolOutput: { zh: '查看输出', en: 'View Output' },
76
+ chatToolNoOutput: { zh: '无输出(执行完成)', en: 'No output (completed)' },
77
+ chatReasoning: { zh: '推理过程', en: 'Reasoning' },
78
+ chatImageAttachment: { zh: '图片附件', en: 'Image attachment' },
79
+ chatFileAttachment: { zh: '文件附件', en: 'File attachment' },
80
+ chatUnknownPart: { zh: '未知消息片段', en: 'Unknown message part' },
81
+ chatCodeCopy: { zh: '复制代码', en: 'Copy' },
82
+ chatCodeCopied: { zh: '已复制', en: 'Copied' },
83
+ chatSidebarNewTask: { zh: '新任务', en: 'New Task' },
84
+ chatSidebarSearchPlaceholder: { zh: '搜索对话...', en: 'Search conversations...' },
85
+ chatSidebarScheduledTasks: { zh: '定时任务', en: 'Scheduled Tasks' },
86
+ chatSidebarSkills: { zh: '技能', en: 'Skills' },
87
+ chatSidebarTaskRecords: { zh: '会话记录', en: 'Sessions' },
88
+ chatSidebarToday: { zh: '今天', en: 'Today' },
89
+ chatSidebarYesterday: { zh: '昨天', en: 'Yesterday' },
90
+ chatSidebarPrevious7Days: { zh: '近 7 天', en: 'Previous 7 Days' },
91
+ chatSidebarOlder: { zh: '更早', en: 'Older' },
92
+ chatWelcomeTitle: { zh: '你好,有什么可以帮你的吗?', en: 'Hello, how can I help you?' },
93
+ chatWelcomeSubtitle: { zh: '开始一个新任务或选择已有对话', en: 'Start a new task or select an existing conversation' },
94
+ chatWelcomeCapability1Title: { zh: '智能对话', en: 'Smart Conversations' },
95
+ chatWelcomeCapability1Desc: { zh: '多轮上下文对话,支持多种 AI 模型', en: 'Multi-turn context conversations with multiple AI models' },
96
+ chatWelcomeCapability2Title: { zh: '技能扩展', en: 'Skill Extensions' },
97
+ chatWelcomeCapability2Desc: { zh: '通过安装技能扩展 Agent 能力', en: 'Extend Agent capabilities by installing skills' },
98
+ chatWelcomeCapability3Title: { zh: '定时任务', en: 'Scheduled Tasks' },
99
+ chatWelcomeCapability3Desc: { zh: '设置定时执行的自动化任务', en: 'Set up scheduled automated tasks' },
100
+ chatSkillsPickerTitle: { zh: '技能', en: 'Skills' },
101
+ chatSkillsPickerEmpty: { zh: '暂无已安装技能', en: 'No skills installed' },
102
+ chatSkillsPickerSearchPlaceholder: { zh: '搜索技能', en: 'Search skills' },
103
+ chatSkillsPickerNoDescription: { zh: '暂无描述', en: 'No description' },
104
+ chatSkillsPickerOfficial: { zh: '官方', en: 'Official' },
105
+ chatSkillsPickerManage: { zh: '管理技能', en: 'Manage Skills' },
106
+ chatInputAttach: { zh: '添加附件', en: 'Attach file' },
107
+ chatInputImageUnsupported: {
108
+ zh: '当前仅支持 PNG、JPEG、WEBP、GIF 图片。',
109
+ en: 'Only PNG, JPEG, WEBP, and GIF images are supported.'
110
+ },
111
+ chatInputImageTooLarge: {
112
+ zh: '图片不能超过 {maxMb} MB。',
113
+ en: 'Images must be {maxMb} MB or smaller.'
114
+ },
115
+ chatInputImageReadFailed: { zh: '读取图片失败,请重试。', en: 'Failed to read the image. Please try again.' },
116
+ chatInputAttachComingSoon: { zh: '即将支持', en: 'Coming soon' }
117
+ };
@@ -96,7 +96,7 @@ export const REMOTE_LABELS: Record<string, { zh: string; en: string }> = {
96
96
  remoteAccountEntryTitle: { zh: 'NextClaw 账号', en: 'NextClaw Account' },
97
97
  remoteAccountEntryDisconnected: { zh: '未登录,点击连接', en: 'Not signed in. Click to connect.' },
98
98
  remoteAccountEntryConnected: { zh: '已连接到 NextClaw', en: 'Connected to NextClaw' },
99
- remoteAccountEntryManage: { zh: '账号与设备入口', en: 'Account and Device Entry' },
99
+ remoteAccountEntryManage: { zh: '账号', en: 'Account' },
100
100
  accountPanelTitle: { zh: 'NextClaw 账号', en: 'NextClaw Account' },
101
101
  accountPanelDescription: {
102
102
  zh: '远程访问依赖这个账号登录。后续 token、授权和更多云端能力也会基于它展开。',
package/src/lib/i18n.ts CHANGED
@@ -1,5 +1,6 @@
1
1
  import { CHANNEL_LABELS } from './i18n.channels';
2
2
  import { CHANNEL_AUTH_LABELS } from './i18n.channel-auth';
3
+ import { CHAT_LABELS } from './i18n.chat';
3
4
  import { MARKETPLACE_LABELS } from './i18n.marketplace';
4
5
  import { REMOTE_LABELS } from './i18n.remote';
5
6
  export type I18nLanguage = 'zh' | 'en';
@@ -557,118 +558,7 @@ export const LABELS: Record<string, { zh: string; en: string }> = {
557
558
  },
558
559
 
559
560
  // Chat
560
- chatPageTitle: { zh: 'Agent 对话', en: 'Agent Chat' },
561
- chatPageDescription: {
562
- zh: '在 UI 内直接与 Agent 交互,支持多会话与多 Agent 切换。',
563
- en: 'Chat with your agent directly in UI with multi-session and multi-agent switching.'
564
- },
565
- chatRefresh: { zh: '刷新', en: 'Refresh' },
566
- chatNewSession: { zh: '新会话', en: 'New Session' },
567
- chatSearchSessionPlaceholder: { zh: '搜索会话 key / 标签', en: 'Search session key / label' },
568
- chatAgentLabel: { zh: '目标 Agent', en: 'Target Agent' },
569
- chatSelectAgent: { zh: '选择 Agent', en: 'Select Agent' },
570
- chatModelLabel: { zh: '对话模型', en: 'Chat Model' },
571
- chatSelectModel: { zh: '选择模型', en: 'Select model' },
572
- chatThinkingLevelOff: { zh: '思考关闭', en: 'Thinking Off' },
573
- chatThinkingLevelMinimal: { zh: '思考 Minimal', en: 'Thinking Minimal' },
574
- chatThinkingLevelLow: { zh: '思考 Low', en: 'Thinking Low' },
575
- chatThinkingLevelMedium: { zh: '思考 Medium', en: 'Thinking Medium' },
576
- chatThinkingLevelHigh: { zh: '思考 High', en: 'Thinking High' },
577
- chatThinkingLevelAdaptive: { zh: '思考 Adaptive', en: 'Thinking Adaptive' },
578
- chatThinkingLevelXhigh: { zh: '思考 XHigh', en: 'Thinking XHigh' },
579
- chatSessionTypeLabel: { zh: '会话类型', en: 'Session Type' },
580
- chatSessionTypeNative: { zh: '原生', en: 'Native' },
581
- chatSessionTypeCodex: { zh: 'Codex', en: 'Codex' },
582
- chatSessionTypeClaude: { zh: 'Claude Code', en: 'Claude Code' },
583
- chatSessionTypeUnavailableSuffix: {
584
- zh: '当前不可用,请启用对应插件或新建 Native 会话。',
585
- en: 'is unavailable now. Re-enable the plugin or create a native session.'
586
- },
587
- chatModelNoOptions: { zh: '暂无可用模型,请先配置提供商。', en: 'No available models. Configure a provider first.' },
588
- chatGoConfigureProvider: { zh: '去配置提供商', en: 'Go to Providers' },
589
- chatProviderSetupTitle: { zh: '开始前先配置提供商', en: 'Configure a Provider First' },
590
- chatProviderSetupDescription: {
591
- zh: '你还没有可用模型。先在提供商页面配置并保存至少一个 Provider 后,再回来开始对话。',
592
- en: 'No models are available yet. Configure and save at least one provider, then return to start chatting.'
593
- },
594
- chatSessionLabel: { zh: '当前会话', en: 'Current Session' },
595
- chatNoSession: { zh: '未选择会话', en: 'No session selected' },
596
- chatNoSessionHint: { zh: '创建一个会话并发送第一条消息。', en: 'Create a session and send your first message.' },
597
- chatHistoryLoading: { zh: '加载会话历史中...', en: 'Loading session history...' },
598
- chatNoMessages: { zh: '暂无消息,发送一条开始对话。', en: 'No messages yet. Send one to start.' },
599
- chatTyping: { zh: 'Agent 正在思考...', en: 'Agent is thinking...' },
600
- chatInputPlaceholder: { zh: '输入消息,输入 / 选择技能,Enter 发送,Shift + Enter 换行', en: 'Type a message, type / to select skills, Enter to send, Shift + Enter for newline' },
601
- chatInputHint: { zh: '支持多轮上下文,默认走当前会话。', en: 'Multi-turn context is preserved in the current session.' },
602
- chatSlashSectionCommands: { zh: '命令', en: 'Commands' },
603
- chatSlashSectionSkills: { zh: '技能', en: 'Skills' },
604
- chatSlashTypeCommand: { zh: '命令', en: 'Command' },
605
- chatSlashTypeSkill: { zh: '技能', en: 'Skill' },
606
- chatSlashSkillSpec: { zh: '标识', en: 'Spec' },
607
- chatSlashLoading: { zh: '加载命令与技能中…', en: 'Loading commands and skills…' },
608
- chatSlashNoResult: { zh: '无匹配项', en: 'No matches' },
609
- chatSlashHint: { zh: '输入 / 触发命令或技能选择', en: 'Type / to access commands and skills' },
610
- chatSlashCommandHint: { zh: '回车插入命令,继续输入参数后发送。', en: 'Press Enter to insert command, then add args and send.' },
611
- chatSlashSkillHint: { zh: '回车把该技能加入本轮请求。', en: 'Press Enter to add this skill for the next turn.' },
612
- chatSend: { zh: '发送', en: 'Send' },
613
- chatStop: { zh: '停止', en: 'Stop' },
614
- chatStopPreparing: { zh: '正在建立可停止会话,请稍候…', en: 'Preparing stoppable run…' },
615
- chatStopUnavailable: { zh: '当前后端引擎不支持手动停止。', en: 'Manual stop is not supported by the current backend engine.' },
616
- chatSending: { zh: '发送中...', en: 'Sending...' },
617
- chatQueueSend: { zh: '排队发送', en: 'Queue' },
618
- chatQueuedHintPrefix: { zh: '当前有', en: 'Queued' },
619
- chatQueuedHintSuffix: { zh: '条消息待发送。', en: 'pending messages.' },
620
- chatQueueMoveFirst: { zh: '置顶到下一条', en: 'Move to Next' },
621
- chatDeleteSession: { zh: '删除会话', en: 'Delete Session' },
622
- chatDeleteSessionConfirm: { zh: '确认删除当前会话?', en: 'Delete the current session?' },
623
- chatSendFailed: { zh: '发送消息失败', en: 'Failed to send message' },
624
- chatRoleUser: { zh: '你', en: 'You' },
625
- chatRoleAssistant: { zh: '助手', en: 'Assistant' },
626
- chatRoleTool: { zh: '工具', en: 'Tool' },
627
- chatRoleSystem: { zh: '系统', en: 'System' },
628
- chatRoleMessage: { zh: '消息', en: 'Message' },
629
- chatToolCall: { zh: '工具调用', en: 'Tool Call' },
630
- chatToolResult: { zh: '工具结果', en: 'Tool Result' },
631
- chatToolWorkflow: { zh: '工具工作流', en: 'Tool Workflow' },
632
- chatToolWorkflowDetails: { zh: '展开查看参数和结果', en: 'Expand to view params and results' },
633
- chatToolOutput: { zh: '查看输出', en: 'View Output' },
634
- chatToolNoOutput: { zh: '无输出(执行完成)', en: 'No output (completed)' },
635
- chatReasoning: { zh: '推理过程', en: 'Reasoning' },
636
- chatUnknownPart: { zh: '未知消息片段', en: 'Unknown message part' },
637
- chatCodeCopy: { zh: '复制代码', en: 'Copy' },
638
- chatCodeCopied: { zh: '已复制', en: 'Copied' },
639
-
640
- // Chat Sidebar (unified)
641
- chatSidebarNewTask: { zh: '新任务', en: 'New Task' },
642
- chatSidebarSearchPlaceholder: { zh: '搜索对话...', en: 'Search conversations...' },
643
- chatSidebarScheduledTasks: { zh: '定时任务', en: 'Scheduled Tasks' },
644
- chatSidebarSkills: { zh: '技能', en: 'Skills' },
645
- chatSidebarTaskRecords: { zh: '会话记录', en: 'Sessions' },
646
- chatSidebarToday: { zh: '今天', en: 'Today' },
647
- chatSidebarYesterday: { zh: '昨天', en: 'Yesterday' },
648
- chatSidebarPrevious7Days: { zh: '近 7 天', en: 'Previous 7 Days' },
649
- chatSidebarOlder: { zh: '更早', en: 'Older' },
650
-
651
- // Welcome page
652
- chatWelcomeTitle: { zh: '你好,有什么可以帮你的吗?', en: 'Hello, how can I help you?' },
653
- chatWelcomeSubtitle: { zh: '开始一个新任务或选择已有对话', en: 'Start a new task or select an existing conversation' },
654
- chatWelcomeCapability1Title: { zh: '智能对话', en: 'Smart Conversations' },
655
- chatWelcomeCapability1Desc: { zh: '多轮上下文对话,支持多种 AI 模型', en: 'Multi-turn context conversations with multiple AI models' },
656
- chatWelcomeCapability2Title: { zh: '技能扩展', en: 'Skill Extensions' },
657
- chatWelcomeCapability2Desc: { zh: '通过安装技能扩展 Agent 能力', en: 'Extend Agent capabilities by installing skills' },
658
- chatWelcomeCapability3Title: { zh: '定时任务', en: 'Scheduled Tasks' },
659
- chatWelcomeCapability3Desc: { zh: '设置定时执行的自动化任务', en: 'Set up scheduled automated tasks' },
660
-
661
- // Skills picker
662
- chatSkillsPickerTitle: { zh: '技能', en: 'Skills' },
663
- chatSkillsPickerEmpty: { zh: '暂无已安装技能', en: 'No skills installed' },
664
- chatSkillsPickerSearchPlaceholder: { zh: '搜索技能', en: 'Search skills' },
665
- chatSkillsPickerNoDescription: { zh: '暂无描述', en: 'No description' },
666
- chatSkillsPickerOfficial: { zh: '官方', en: 'Official' },
667
- chatSkillsPickerManage: { zh: '管理技能', en: 'Manage Skills' },
668
-
669
- // Input bar
670
- chatInputAttach: { zh: '添加附件', en: 'Attach file' },
671
- chatInputAttachComingSoon: { zh: '即将支持', en: 'Coming soon' },
561
+ ...CHAT_LABELS,
672
562
 
673
563
  // Cron
674
564
  cronPageTitle: { zh: '定时任务', en: 'Cron Jobs' },
@@ -0,0 +1,135 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
2
+ import { RemoteSessionMultiplexTransport } from '@/transport/remote.transport';
3
+
4
+ class MockWebSocket {
5
+ static readonly CONNECTING = 0;
6
+ static readonly OPEN = 1;
7
+ static readonly CLOSING = 2;
8
+ static readonly CLOSED = 3;
9
+ static instances: MockWebSocket[] = [];
10
+
11
+ readonly sent: string[] = [];
12
+ readyState = MockWebSocket.CONNECTING;
13
+ onopen: ((event: Event) => void) | null = null;
14
+ onmessage: ((event: MessageEvent) => void) | null = null;
15
+ onerror: ((event: Event) => void) | null = null;
16
+ onclose: ((event: CloseEvent) => void) | null = null;
17
+
18
+ constructor(public readonly url: string) {
19
+ MockWebSocket.instances.push(this);
20
+ }
21
+
22
+ send(data: string): void {
23
+ this.sent.push(data);
24
+ }
25
+
26
+ close(): void {
27
+ this.readyState = MockWebSocket.CLOSED;
28
+ this.onclose?.({} as CloseEvent);
29
+ }
30
+
31
+ open(): void {
32
+ this.readyState = MockWebSocket.OPEN;
33
+ this.onopen?.({} as Event);
34
+ }
35
+
36
+ receive(frame: unknown): void {
37
+ this.onmessage?.({
38
+ data: JSON.stringify(frame)
39
+ } as MessageEvent);
40
+ }
41
+
42
+ static reset(): void {
43
+ MockWebSocket.instances = [];
44
+ }
45
+ }
46
+
47
+ function lastSentRequestFrame(socket: MockWebSocket): { type: 'request'; id: string } {
48
+ const raw = socket.sent.at(-1);
49
+ if (!raw) {
50
+ throw new Error('Expected a sent request frame.');
51
+ }
52
+ return JSON.parse(raw) as { type: 'request'; id: string };
53
+ }
54
+
55
+ describe('RemoteSessionMultiplexTransport request path', () => {
56
+ beforeEach(() => {
57
+ MockWebSocket.reset();
58
+ vi.useFakeTimers();
59
+ vi.stubGlobal('WebSocket', MockWebSocket as unknown as typeof WebSocket);
60
+ });
61
+
62
+ afterEach(() => {
63
+ vi.useRealTimers();
64
+ vi.unstubAllGlobals();
65
+ });
66
+
67
+ it('keeps ordinary requests on the remote websocket multiplex channel', async () => {
68
+ const transport = new RemoteSessionMultiplexTransport({
69
+ mode: 'remote',
70
+ protocolVersion: 1,
71
+ wsPath: '/_remote/ws'
72
+ }, 'https://remote.claw.cool');
73
+
74
+ const requestPromise = transport.request<{ sessions: unknown[]; total: number }>({
75
+ method: 'GET',
76
+ path: '/api/sessions'
77
+ });
78
+
79
+ const socket = MockWebSocket.instances[0];
80
+ if (!socket) {
81
+ throw new Error('Expected remote websocket to be created.');
82
+ }
83
+ socket.open();
84
+ await Promise.resolve();
85
+ await Promise.resolve();
86
+
87
+ const requestFrame = lastSentRequestFrame(socket);
88
+ socket.receive({
89
+ type: 'response',
90
+ id: requestFrame.id,
91
+ status: 200,
92
+ body: {
93
+ ok: true,
94
+ data: {
95
+ sessions: [],
96
+ total: 0
97
+ }
98
+ }
99
+ });
100
+
101
+ await expect(requestPromise).resolves.toEqual({
102
+ sessions: [],
103
+ total: 0
104
+ });
105
+ expect(socket.url).toBe('wss://remote.claw.cool/_remote/ws');
106
+ });
107
+
108
+ it('fails predictably when a remote request frame never receives a response', async () => {
109
+ const transport = new RemoteSessionMultiplexTransport({
110
+ mode: 'remote',
111
+ protocolVersion: 1,
112
+ wsPath: '/_remote/ws'
113
+ }, 'https://remote.claw.cool');
114
+
115
+ const requestPromise = transport.request({
116
+ method: 'GET',
117
+ path: '/api/sessions'
118
+ });
119
+ const timeoutExpectation = expect(requestPromise).rejects.toThrow(
120
+ 'Timed out waiting for remote request response: GET /api/sessions'
121
+ );
122
+
123
+ const socket = MockWebSocket.instances[0];
124
+ if (!socket) {
125
+ throw new Error('Expected remote websocket to be created.');
126
+ }
127
+ socket.open();
128
+ await Promise.resolve();
129
+ await Promise.resolve();
130
+
131
+ await vi.advanceTimersByTimeAsync(15_000);
132
+
133
+ await timeoutExpectation;
134
+ });
135
+ });
@@ -27,6 +27,7 @@ type RemoteBrowserCommand =
27
27
  type PendingRequest = {
28
28
  resolve: (value: unknown) => void;
29
29
  reject: (error: Error) => void;
30
+ timeoutId: number;
30
31
  };
31
32
 
32
33
  type PendingStream = {
@@ -69,6 +70,8 @@ function createId(prefix: string): string {
69
70
  return `${prefix}-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 10)}`;
70
71
  }
71
72
 
73
+ const REMOTE_REQUEST_TIMEOUT_MS = 15_000;
74
+
72
75
  export class RemoteSessionMultiplexTransport implements AppTransport {
73
76
  private socket: WebSocket | null = null;
74
77
  private connectPromise: Promise<void> | null = null;
@@ -88,9 +91,14 @@ export class RemoteSessionMultiplexTransport implements AppTransport {
88
91
  await this.ensureSocket();
89
92
  const id = createId('req');
90
93
  return await new Promise<T>((resolve, reject) => {
94
+ const timeoutId = window.setTimeout(() => {
95
+ this.pendingRequests.delete(id);
96
+ reject(new Error(`Timed out waiting for remote request response: ${input.method} ${input.path}`));
97
+ }, REMOTE_REQUEST_TIMEOUT_MS);
91
98
  this.pendingRequests.set(id, {
92
99
  resolve: (value) => resolve(value as T),
93
- reject
100
+ reject,
101
+ timeoutId
94
102
  });
95
103
  this.send({
96
104
  type: 'request',
@@ -297,6 +305,7 @@ export class RemoteSessionMultiplexTransport implements AppTransport {
297
305
 
298
306
  private failPendingWork(error: Error): void {
299
307
  for (const pending of this.pendingRequests.values()) {
308
+ window.clearTimeout(pending.timeoutId);
300
309
  pending.reject(error);
301
310
  }
302
311
  this.pendingRequests.clear();
@@ -331,6 +340,7 @@ export class RemoteSessionMultiplexTransport implements AppTransport {
331
340
  return;
332
341
  }
333
342
  this.pendingRequests.delete(frame.id);
343
+ window.clearTimeout(pending.timeoutId);
334
344
  if (frame.type === 'request.error') {
335
345
  pending.reject(new Error(frame.message));
336
346
  return;