@renxqoo/renx-code 0.0.2 → 0.0.4

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 (298) hide show
  1. package/README.md +59 -223
  2. package/bin/renx.cjs +34 -0
  3. package/package.json +27 -83
  4. package/src/App.tsx +297 -0
  5. package/src/agent/runtime/event-format.ts +258 -0
  6. package/src/agent/runtime/model-types.ts +13 -0
  7. package/src/agent/runtime/runtime.context-usage.test.ts +193 -0
  8. package/src/agent/runtime/runtime.error-handling.test.ts +236 -0
  9. package/src/agent/runtime/runtime.simple.test.ts +16 -0
  10. package/src/agent/runtime/runtime.test.ts +293 -0
  11. package/src/agent/runtime/runtime.ts +881 -0
  12. package/src/agent/runtime/runtime.usage-forwarding.test.ts +229 -0
  13. package/src/agent/runtime/source-modules.test.ts +57 -0
  14. package/src/agent/runtime/source-modules.ts +353 -0
  15. package/src/agent/runtime/tool-call-buffer.test.ts +65 -0
  16. package/src/agent/runtime/tool-call-buffer.ts +60 -0
  17. package/src/agent/runtime/tool-confirmation.test.ts +56 -0
  18. package/src/agent/runtime/tool-confirmation.ts +15 -0
  19. package/src/agent/runtime/types.ts +99 -0
  20. package/src/commands/slash-commands.test.ts +216 -0
  21. package/src/commands/slash-commands.ts +64 -0
  22. package/src/components/chat/assistant-reply.test.tsx +47 -0
  23. package/src/components/chat/assistant-reply.tsx +136 -0
  24. package/src/components/chat/assistant-segment.test.ts +99 -0
  25. package/src/components/chat/assistant-segment.tsx +125 -0
  26. package/src/components/chat/assistant-tool-group.tsx +900 -0
  27. package/src/components/chat/code-block.test.tsx +206 -0
  28. package/src/components/chat/code-block.tsx +313 -0
  29. package/src/components/chat/prompt-card.tsx +81 -0
  30. package/src/components/chat/segment-groups.test.ts +52 -0
  31. package/src/components/chat/segment-groups.ts +106 -0
  32. package/src/components/chat/turn-item.tsx +39 -0
  33. package/src/components/conversation-panel.tsx +43 -0
  34. package/src/components/file-mention-menu.tsx +77 -0
  35. package/src/components/file-picker-dialog.tsx +206 -0
  36. package/src/components/footer-hints.tsx +75 -0
  37. package/src/components/model-picker-dialog.tsx +248 -0
  38. package/src/components/prompt.tsx +233 -0
  39. package/src/components/slash-command-menu.tsx +65 -0
  40. package/src/components/tool-confirm-dialog-content.test.ts +103 -0
  41. package/src/components/tool-confirm-dialog-content.ts +186 -0
  42. package/src/components/tool-confirm-dialog.tsx +187 -0
  43. package/src/components/tool-display-config.ts +119 -0
  44. package/src/context-usage-regressions.test.ts +26 -0
  45. package/src/files/attachment-capabilities.test.ts +30 -0
  46. package/src/files/attachment-capabilities.ts +50 -0
  47. package/src/files/attachment-content.ts +153 -0
  48. package/src/files/file-mention-query.test.ts +34 -0
  49. package/src/files/file-mention-query.ts +32 -0
  50. package/src/files/prompt-display.ts +13 -0
  51. package/src/files/types.ts +5 -0
  52. package/src/files/workspace-files.ts +63 -0
  53. package/src/hooks/agent-event-handlers.test.ts +207 -0
  54. package/src/hooks/agent-event-handlers.ts +196 -0
  55. package/src/hooks/chat-local-replies.fixed.test.ts +119 -0
  56. package/src/hooks/chat-local-replies.test.ts +153 -0
  57. package/src/hooks/chat-local-replies.ts +63 -0
  58. package/src/hooks/turn-updater.test.ts +70 -0
  59. package/src/hooks/turn-updater.ts +166 -0
  60. package/src/hooks/use-agent-chat.context.test.ts +10 -0
  61. package/src/hooks/use-agent-chat.status.test.ts +14 -0
  62. package/src/hooks/use-agent-chat.test.ts +80 -0
  63. package/src/hooks/use-agent-chat.ts +621 -0
  64. package/src/hooks/use-file-mention-menu.ts +196 -0
  65. package/src/hooks/use-file-picker.ts +185 -0
  66. package/src/hooks/use-model-picker.ts +196 -0
  67. package/src/hooks/use-slash-command-menu.ts +154 -0
  68. package/src/index.tsx +55 -0
  69. package/src/runtime/clipboard.test.ts +43 -0
  70. package/src/runtime/clipboard.ts +89 -0
  71. package/src/runtime/exit.test.ts +177 -0
  72. package/src/runtime/exit.ts +98 -0
  73. package/src/runtime/runtime-support.test.ts +31 -0
  74. package/src/runtime/terminal-theme.test.ts +55 -0
  75. package/src/runtime/terminal-theme.ts +196 -0
  76. package/src/types/chat.ts +32 -0
  77. package/src/types/message-content.ts +48 -0
  78. package/src/ui/open-code-theme.ts +176 -0
  79. package/src/ui/opencode-markdown.ts +211 -0
  80. package/src/ui/theme.simple.test.ts +52 -0
  81. package/src/ui/theme.test.ts +151 -0
  82. package/src/ui/theme.ts +152 -0
  83. package/src/utils/time.test.ts +144 -0
  84. package/src/utils/time.ts +7 -0
  85. package/tsconfig.json +30 -0
  86. package/LICENSE +0 -21
  87. package/dist/App.d.ts +0 -2
  88. package/dist/App.d.ts.map +0 -1
  89. package/dist/App.js +0 -170
  90. package/dist/App.js.map +0 -1
  91. package/dist/agent/prompts/system.d.ts +0 -24
  92. package/dist/agent/prompts/system.d.ts.map +0 -1
  93. package/dist/agent/prompts/system.js +0 -222
  94. package/dist/agent/prompts/system.js.map +0 -1
  95. package/dist/agent/runtime/event-format.d.ts +0 -17
  96. package/dist/agent/runtime/event-format.d.ts.map +0 -1
  97. package/dist/agent/runtime/event-format.js +0 -194
  98. package/dist/agent/runtime/event-format.js.map +0 -1
  99. package/dist/agent/runtime/model-types.d.ts +0 -13
  100. package/dist/agent/runtime/model-types.d.ts.map +0 -1
  101. package/dist/agent/runtime/model-types.js +0 -1
  102. package/dist/agent/runtime/model-types.js.map +0 -1
  103. package/dist/agent/runtime/runtime.d.ts +0 -16
  104. package/dist/agent/runtime/runtime.d.ts.map +0 -1
  105. package/dist/agent/runtime/runtime.js +0 -691
  106. package/dist/agent/runtime/runtime.js.map +0 -1
  107. package/dist/agent/runtime/source-modules.d.ts +0 -176
  108. package/dist/agent/runtime/source-modules.d.ts.map +0 -1
  109. package/dist/agent/runtime/source-modules.js +0 -110
  110. package/dist/agent/runtime/source-modules.js.map +0 -1
  111. package/dist/agent/runtime/tool-call-buffer.d.ts +0 -12
  112. package/dist/agent/runtime/tool-call-buffer.d.ts.map +0 -1
  113. package/dist/agent/runtime/tool-call-buffer.js +0 -48
  114. package/dist/agent/runtime/tool-call-buffer.js.map +0 -1
  115. package/dist/agent/runtime/tool-confirmation.d.ts +0 -3
  116. package/dist/agent/runtime/tool-confirmation.d.ts.map +0 -1
  117. package/dist/agent/runtime/tool-confirmation.js +0 -9
  118. package/dist/agent/runtime/tool-confirmation.js.map +0 -1
  119. package/dist/agent/runtime/types.d.ts +0 -86
  120. package/dist/agent/runtime/types.d.ts.map +0 -1
  121. package/dist/agent/runtime/types.js +0 -1
  122. package/dist/agent/runtime/types.js.map +0 -1
  123. package/dist/cli.d.ts +0 -3
  124. package/dist/cli.d.ts.map +0 -1
  125. package/dist/cli.js +0 -43
  126. package/dist/cli.js.map +0 -1
  127. package/dist/commands/slash-commands.d.ts +0 -11
  128. package/dist/commands/slash-commands.d.ts.map +0 -1
  129. package/dist/commands/slash-commands.js +0 -48
  130. package/dist/commands/slash-commands.js.map +0 -1
  131. package/dist/components/chat/assistant-reply.d.ts +0 -13
  132. package/dist/components/chat/assistant-reply.d.ts.map +0 -1
  133. package/dist/components/chat/assistant-reply.js +0 -78
  134. package/dist/components/chat/assistant-reply.js.map +0 -1
  135. package/dist/components/chat/assistant-segment.d.ts +0 -8
  136. package/dist/components/chat/assistant-segment.d.ts.map +0 -1
  137. package/dist/components/chat/assistant-segment.js +0 -54
  138. package/dist/components/chat/assistant-segment.js.map +0 -1
  139. package/dist/components/chat/assistant-tool-group.d.ts +0 -7
  140. package/dist/components/chat/assistant-tool-group.d.ts.map +0 -1
  141. package/dist/components/chat/assistant-tool-group.js +0 -695
  142. package/dist/components/chat/assistant-tool-group.js.map +0 -1
  143. package/dist/components/chat/code-block.d.ts +0 -16
  144. package/dist/components/chat/code-block.d.ts.map +0 -1
  145. package/dist/components/chat/code-block.js +0 -194
  146. package/dist/components/chat/code-block.js.map +0 -1
  147. package/dist/components/chat/prompt-card.d.ts +0 -9
  148. package/dist/components/chat/prompt-card.d.ts.map +0 -1
  149. package/dist/components/chat/prompt-card.js +0 -18
  150. package/dist/components/chat/prompt-card.js.map +0 -1
  151. package/dist/components/chat/segment-groups.d.ts +0 -24
  152. package/dist/components/chat/segment-groups.d.ts.map +0 -1
  153. package/dist/components/chat/segment-groups.js +0 -69
  154. package/dist/components/chat/segment-groups.js.map +0 -1
  155. package/dist/components/chat/turn-item.d.ts +0 -9
  156. package/dist/components/chat/turn-item.d.ts.map +0 -1
  157. package/dist/components/chat/turn-item.js +0 -11
  158. package/dist/components/chat/turn-item.js.map +0 -1
  159. package/dist/components/conversation-panel.d.ts +0 -8
  160. package/dist/components/conversation-panel.d.ts.map +0 -1
  161. package/dist/components/conversation-panel.js +0 -8
  162. package/dist/components/conversation-panel.js.map +0 -1
  163. package/dist/components/file-mention-menu.d.ts +0 -11
  164. package/dist/components/file-mention-menu.d.ts.map +0 -1
  165. package/dist/components/file-mention-menu.js +0 -15
  166. package/dist/components/file-mention-menu.js.map +0 -1
  167. package/dist/components/file-picker-dialog.d.ts +0 -21
  168. package/dist/components/file-picker-dialog.d.ts.map +0 -1
  169. package/dist/components/file-picker-dialog.js +0 -48
  170. package/dist/components/file-picker-dialog.js.map +0 -1
  171. package/dist/components/footer-hints.d.ts +0 -7
  172. package/dist/components/footer-hints.d.ts.map +0 -1
  173. package/dist/components/footer-hints.js +0 -29
  174. package/dist/components/footer-hints.js.map +0 -1
  175. package/dist/components/model-picker-dialog.d.ts +0 -20
  176. package/dist/components/model-picker-dialog.d.ts.map +0 -1
  177. package/dist/components/model-picker-dialog.js +0 -72
  178. package/dist/components/model-picker-dialog.js.map +0 -1
  179. package/dist/components/prompt.d.ts +0 -18
  180. package/dist/components/prompt.d.ts.map +0 -1
  181. package/dist/components/prompt.js +0 -96
  182. package/dist/components/prompt.js.map +0 -1
  183. package/dist/components/slash-command-menu.d.ts +0 -9
  184. package/dist/components/slash-command-menu.d.ts.map +0 -1
  185. package/dist/components/slash-command-menu.js +0 -20
  186. package/dist/components/slash-command-menu.js.map +0 -1
  187. package/dist/components/tool-confirm-dialog-content.d.ts +0 -15
  188. package/dist/components/tool-confirm-dialog-content.d.ts.map +0 -1
  189. package/dist/components/tool-confirm-dialog-content.js +0 -143
  190. package/dist/components/tool-confirm-dialog-content.js.map +0 -1
  191. package/dist/components/tool-confirm-dialog.d.ts +0 -12
  192. package/dist/components/tool-confirm-dialog.d.ts.map +0 -1
  193. package/dist/components/tool-confirm-dialog.js +0 -21
  194. package/dist/components/tool-confirm-dialog.js.map +0 -1
  195. package/dist/components/tool-display-config.d.ts +0 -11
  196. package/dist/components/tool-display-config.d.ts.map +0 -1
  197. package/dist/components/tool-display-config.js +0 -94
  198. package/dist/components/tool-display-config.js.map +0 -1
  199. package/dist/config/paths.d.ts +0 -7
  200. package/dist/config/paths.d.ts.map +0 -1
  201. package/dist/config/paths.js +0 -24
  202. package/dist/config/paths.js.map +0 -1
  203. package/dist/files/attachment-capabilities.d.ts +0 -19
  204. package/dist/files/attachment-capabilities.d.ts.map +0 -1
  205. package/dist/files/attachment-capabilities.js +0 -26
  206. package/dist/files/attachment-capabilities.js.map +0 -1
  207. package/dist/files/attachment-content.d.ts +0 -5
  208. package/dist/files/attachment-content.d.ts.map +0 -1
  209. package/dist/files/attachment-content.js +0 -117
  210. package/dist/files/attachment-content.js.map +0 -1
  211. package/dist/files/file-mention-query.d.ts +0 -9
  212. package/dist/files/file-mention-query.d.ts.map +0 -1
  213. package/dist/files/file-mention-query.js +0 -23
  214. package/dist/files/file-mention-query.js.map +0 -1
  215. package/dist/files/prompt-display.d.ts +0 -3
  216. package/dist/files/prompt-display.d.ts.map +0 -1
  217. package/dist/files/prompt-display.js +0 -11
  218. package/dist/files/prompt-display.js.map +0 -1
  219. package/dist/files/types.d.ts +0 -6
  220. package/dist/files/types.d.ts.map +0 -1
  221. package/dist/files/types.js +0 -1
  222. package/dist/files/types.js.map +0 -1
  223. package/dist/files/workspace-files.d.ts +0 -3
  224. package/dist/files/workspace-files.d.ts.map +0 -1
  225. package/dist/files/workspace-files.js +0 -50
  226. package/dist/files/workspace-files.js.map +0 -1
  227. package/dist/hooks/agent-event-handlers.d.ts +0 -11
  228. package/dist/hooks/agent-event-handlers.d.ts.map +0 -1
  229. package/dist/hooks/agent-event-handlers.js +0 -137
  230. package/dist/hooks/agent-event-handlers.js.map +0 -1
  231. package/dist/hooks/chat-local-replies.d.ts +0 -9
  232. package/dist/hooks/chat-local-replies.d.ts.map +0 -1
  233. package/dist/hooks/chat-local-replies.js +0 -54
  234. package/dist/hooks/chat-local-replies.js.map +0 -1
  235. package/dist/hooks/turn-updater.d.ts +0 -9
  236. package/dist/hooks/turn-updater.d.ts.map +0 -1
  237. package/dist/hooks/turn-updater.js +0 -103
  238. package/dist/hooks/turn-updater.js.map +0 -1
  239. package/dist/hooks/use-agent-chat.d.ts +0 -29
  240. package/dist/hooks/use-agent-chat.d.ts.map +0 -1
  241. package/dist/hooks/use-agent-chat.js +0 -455
  242. package/dist/hooks/use-agent-chat.js.map +0 -1
  243. package/dist/hooks/use-file-mention-menu.d.ts +0 -22
  244. package/dist/hooks/use-file-mention-menu.d.ts.map +0 -1
  245. package/dist/hooks/use-file-mention-menu.js +0 -137
  246. package/dist/hooks/use-file-mention-menu.js.map +0 -1
  247. package/dist/hooks/use-file-picker.d.ts +0 -21
  248. package/dist/hooks/use-file-picker.d.ts.map +0 -1
  249. package/dist/hooks/use-file-picker.js +0 -145
  250. package/dist/hooks/use-file-picker.js.map +0 -1
  251. package/dist/hooks/use-model-picker.d.ts +0 -23
  252. package/dist/hooks/use-model-picker.d.ts.map +0 -1
  253. package/dist/hooks/use-model-picker.js +0 -151
  254. package/dist/hooks/use-model-picker.js.map +0 -1
  255. package/dist/hooks/use-slash-command-menu.d.ts +0 -19
  256. package/dist/hooks/use-slash-command-menu.d.ts.map +0 -1
  257. package/dist/hooks/use-slash-command-menu.js +0 -101
  258. package/dist/hooks/use-slash-command-menu.js.map +0 -1
  259. package/dist/index.d.ts +0 -2
  260. package/dist/index.d.ts.map +0 -1
  261. package/dist/index.js +0 -39
  262. package/dist/index.js.map +0 -1
  263. package/dist/runtime/clipboard.d.ts +0 -10
  264. package/dist/runtime/clipboard.d.ts.map +0 -1
  265. package/dist/runtime/clipboard.js +0 -64
  266. package/dist/runtime/clipboard.js.map +0 -1
  267. package/dist/runtime/exit.d.ts +0 -7
  268. package/dist/runtime/exit.d.ts.map +0 -1
  269. package/dist/runtime/exit.js +0 -85
  270. package/dist/runtime/exit.js.map +0 -1
  271. package/dist/runtime/terminal-theme.d.ts +0 -25
  272. package/dist/runtime/terminal-theme.d.ts.map +0 -1
  273. package/dist/runtime/terminal-theme.js +0 -148
  274. package/dist/runtime/terminal-theme.js.map +0 -1
  275. package/dist/types/chat.d.ts +0 -29
  276. package/dist/types/chat.d.ts.map +0 -1
  277. package/dist/types/chat.js +0 -1
  278. package/dist/types/chat.js.map +0 -1
  279. package/dist/types/message-content.d.ts +0 -38
  280. package/dist/types/message-content.d.ts.map +0 -1
  281. package/dist/types/message-content.js +0 -1
  282. package/dist/types/message-content.js.map +0 -1
  283. package/dist/ui/open-code-theme.d.ts +0 -58
  284. package/dist/ui/open-code-theme.d.ts.map +0 -1
  285. package/dist/ui/open-code-theme.js +0 -113
  286. package/dist/ui/open-code-theme.js.map +0 -1
  287. package/dist/ui/opencode-markdown.d.ts +0 -7
  288. package/dist/ui/opencode-markdown.d.ts.map +0 -1
  289. package/dist/ui/opencode-markdown.js +0 -169
  290. package/dist/ui/opencode-markdown.js.map +0 -1
  291. package/dist/ui/theme.d.ts +0 -68
  292. package/dist/ui/theme.d.ts.map +0 -1
  293. package/dist/ui/theme.js +0 -80
  294. package/dist/ui/theme.js.map +0 -1
  295. package/dist/utils/time.d.ts +0 -2
  296. package/dist/utils/time.d.ts.map +0 -1
  297. package/dist/utils/time.js +0 -7
  298. package/dist/utils/time.js.map +0 -1
@@ -0,0 +1,621 @@
1
+ import { useCallback, useEffect, useRef, useState } from 'react';
2
+
3
+ import { resolveSlashCommand } from '../commands/slash-commands';
4
+ import {
5
+ getAgentModelAttachmentCapabilities,
6
+ getAgentModelLabel,
7
+ runAgentPrompt,
8
+ } from '../agent/runtime/runtime';
9
+ import type {
10
+ AgentContextUsageEvent,
11
+ AgentToolConfirmDecision,
12
+ AgentToolConfirmEvent,
13
+ AgentUsageEvent,
14
+ } from '../agent/runtime/types';
15
+ import { requestExit } from '../runtime/exit';
16
+ import type { ChatTurn, ReplySegmentType } from '../types/chat';
17
+ import type { PromptFileSelection } from '../files/types';
18
+ import { buildAgentEventHandlers } from './agent-event-handlers';
19
+ import {
20
+ buildHelpSegments,
21
+ buildUnsupportedSegments,
22
+ extractErrorMessage,
23
+ } from './chat-local-replies';
24
+ import {
25
+ appendNoteLine,
26
+ appendToSegment,
27
+ createStreamingReply,
28
+ orderReplySegments,
29
+ patchTurn,
30
+ setReplyStatus,
31
+ } from './turn-updater';
32
+ import {
33
+ DEFAULT_ATTACHMENT_MODEL_CAPABILITIES,
34
+ type AttachmentModelCapabilities,
35
+ } from '../files/attachment-capabilities';
36
+ import { buildPromptContent } from '../files/attachment-content';
37
+ import { buildPromptDisplay } from '../files/prompt-display';
38
+
39
+ export type UseAgentChatResult = {
40
+ turns: ChatTurn[];
41
+ inputValue: string;
42
+ selectedFiles: PromptFileSelection[];
43
+ isThinking: boolean;
44
+ modelLabel: string;
45
+ contextUsagePercent: number | null;
46
+ pendingToolConfirm: (AgentToolConfirmEvent & { selectedAction: 'approve' | 'deny' }) | null;
47
+ setInputValue: (value: string) => void;
48
+ setSelectedFiles: (files: PromptFileSelection[]) => void;
49
+ appendSelectedFiles: (files: PromptFileSelection[]) => void;
50
+ removeSelectedFile: (absolutePath: string) => void;
51
+ submitInput: () => void;
52
+ stopActiveReply: () => void;
53
+ clearInput: () => void;
54
+ resetConversation: () => void;
55
+ setModelLabelDisplay: (label: string) => void;
56
+ setToolConfirmSelection: (selection: 'approve' | 'deny') => void;
57
+ submitToolConfirmSelection: () => void;
58
+ rejectPendingToolConfirm: () => void;
59
+ };
60
+
61
+ const INITIAL_MODEL_LABEL = process.env.AGENT_MODEL?.trim() || '';
62
+
63
+ const normalizeTokenCount = (value: number | undefined): number | undefined => {
64
+ if (typeof value !== 'number' || !Number.isFinite(value)) {
65
+ return undefined;
66
+ }
67
+ return Math.max(0, Math.round(value));
68
+ };
69
+
70
+ /** Normalizes context usage percent, ensuring it's a finite non-negative number. */
71
+ const normalizeContextUsagePercent = (value: unknown): number | null => {
72
+ if (typeof value === 'number' && Number.isFinite(value)) {
73
+ return Math.max(0, value);
74
+ }
75
+ return null;
76
+ };
77
+
78
+ const toReplyUsage = (
79
+ usage?: AgentUsageEvent
80
+ ):
81
+ | {
82
+ usagePromptTokens?: number;
83
+ usageCompletionTokens?: number;
84
+ usageTotalTokens?: number;
85
+ }
86
+ | undefined => {
87
+ if (!usage) {
88
+ return undefined;
89
+ }
90
+
91
+ const usagePromptTokens = normalizeTokenCount(usage.cumulativePromptTokens ?? usage.promptTokens);
92
+ const usageCompletionTokens = normalizeTokenCount(
93
+ usage.cumulativeCompletionTokens ?? usage.completionTokens
94
+ );
95
+ const usageTotalTokens = normalizeTokenCount(usage.cumulativeTotalTokens ?? usage.totalTokens);
96
+
97
+ if (
98
+ typeof usagePromptTokens !== 'number' &&
99
+ typeof usageCompletionTokens !== 'number' &&
100
+ typeof usageTotalTokens !== 'number'
101
+ ) {
102
+ return undefined;
103
+ }
104
+
105
+ return {
106
+ usagePromptTokens,
107
+ usageCompletionTokens,
108
+ usageTotalTokens,
109
+ };
110
+ };
111
+
112
+ export const resolveReplyStatus = (completionReason: string): 'done' | 'error' => {
113
+ return completionReason === 'error' ? 'error' : 'done';
114
+ };
115
+
116
+ export const useAgentChat = (): UseAgentChatResult => {
117
+ const [turns, setTurns] = useState<ChatTurn[]>([]);
118
+ const [inputValue, setInputValue] = useState('');
119
+ const [selectedFiles, setSelectedFiles] = useState<PromptFileSelection[]>([]);
120
+ const [isThinking, setIsThinking] = useState(false);
121
+ const [modelLabel, setModelLabel] = useState(INITIAL_MODEL_LABEL);
122
+ const [contextUsagePercent, setContextUsagePercent] = useState<number | null>(null);
123
+ const [attachmentCapabilities, setAttachmentCapabilities] = useState<AttachmentModelCapabilities>(
124
+ DEFAULT_ATTACHMENT_MODEL_CAPABILITIES
125
+ );
126
+ const [pendingToolConfirm, setPendingToolConfirm] = useState<
127
+ (AgentToolConfirmEvent & { selectedAction: 'approve' | 'deny' }) | null
128
+ >(null);
129
+
130
+ const removeSelectedFile = useCallback((absolutePath: string) => {
131
+ setSelectedFiles(current => current.filter(file => file.absolutePath !== absolutePath));
132
+ }, []);
133
+
134
+ const appendSelectedFiles = useCallback((files: PromptFileSelection[]) => {
135
+ if (files.length === 0) {
136
+ return;
137
+ }
138
+ setSelectedFiles(current => {
139
+ const seen = new Set(current.map(file => file.absolutePath));
140
+ const next = [...current];
141
+ for (const file of files) {
142
+ if (seen.has(file.absolutePath)) {
143
+ continue;
144
+ }
145
+ next.push(file);
146
+ seen.add(file.absolutePath);
147
+ }
148
+ return next;
149
+ });
150
+ }, []);
151
+
152
+ const turnIdRef = useRef(1);
153
+ const requestIdRef = useRef(0);
154
+ const activeTurnIdRef = useRef<number | null>(null);
155
+ const activeAbortControllerRef = useRef<AbortController | null>(null);
156
+ const activeRunPromiseRef = useRef<Promise<void> | null>(null);
157
+ const pendingToolConfirmResolverRef = useRef<
158
+ ((decision: AgentToolConfirmDecision) => void) | null
159
+ >(null);
160
+
161
+ const resolvePendingToolConfirm = useCallback((decision: AgentToolConfirmDecision) => {
162
+ const resolver = pendingToolConfirmResolverRef.current;
163
+ pendingToolConfirmResolverRef.current = null;
164
+ setPendingToolConfirm(null);
165
+ resolver?.(decision);
166
+ }, []);
167
+
168
+ useEffect(() => {
169
+ let disposed = false;
170
+ void getAgentModelLabel()
171
+ .then(label => {
172
+ if (!disposed) {
173
+ setModelLabel(label);
174
+ }
175
+ })
176
+ .catch(() => {});
177
+ void getAgentModelAttachmentCapabilities()
178
+ .then(capabilities => {
179
+ if (!disposed) {
180
+ setAttachmentCapabilities(capabilities);
181
+ }
182
+ })
183
+ .catch(() => {});
184
+
185
+ return () => {
186
+ disposed = true;
187
+ };
188
+ }, []);
189
+
190
+ useEffect(() => {
191
+ return () => {
192
+ // 清理活动的Promise以防止内存泄漏
193
+ const currentPromise = activeRunPromiseRef.current;
194
+ if (currentPromise) {
195
+ currentPromise.catch(() => {}); // 防止未处理的rejection
196
+ }
197
+ activeAbortControllerRef.current?.abort();
198
+ const resolver = pendingToolConfirmResolverRef.current;
199
+ pendingToolConfirmResolverRef.current = null;
200
+ resolver?.({
201
+ approved: false,
202
+ message: 'Tool confirmation cancelled because the UI was closed.',
203
+ });
204
+ };
205
+ }, []);
206
+
207
+ const appendSegment = useCallback(
208
+ (turnId: number, segmentId: string, type: ReplySegmentType, chunk: string, data?: unknown) => {
209
+ setTurns(prev =>
210
+ patchTurn(prev, turnId, turn => {
211
+ if (!turn.reply) {
212
+ return turn;
213
+ }
214
+ return {
215
+ ...turn,
216
+ reply: {
217
+ ...turn.reply,
218
+ segments: orderReplySegments(
219
+ appendToSegment(turn.reply.segments, segmentId, type, chunk, data)
220
+ ),
221
+ },
222
+ };
223
+ })
224
+ );
225
+ },
226
+ []
227
+ );
228
+
229
+ const appendEventLine = useCallback((turnId: number, text: string) => {
230
+ setTurns(prev =>
231
+ patchTurn(prev, turnId, turn => {
232
+ if (!turn.reply) {
233
+ return turn;
234
+ }
235
+ return {
236
+ ...turn,
237
+ reply: {
238
+ ...turn.reply,
239
+ segments: orderReplySegments(
240
+ appendNoteLine(turn.reply.segments, `${turnId}:events`, text)
241
+ ),
242
+ },
243
+ };
244
+ })
245
+ );
246
+ }, []);
247
+
248
+ const stopActiveReply = useCallback(() => {
249
+ if (!isThinking) {
250
+ return;
251
+ }
252
+
253
+ requestIdRef.current += 1;
254
+ activeAbortControllerRef.current?.abort();
255
+ activeAbortControllerRef.current = null;
256
+ setIsThinking(false);
257
+
258
+ const activeTurnId = activeTurnIdRef.current;
259
+ activeTurnIdRef.current = null;
260
+ if (typeof activeTurnId !== 'number') {
261
+ return;
262
+ }
263
+
264
+ appendEventLine(activeTurnId, '[stop] aborted by user');
265
+ setTurns(prev =>
266
+ setReplyStatus(prev, activeTurnId, 'done', {
267
+ completionReason: 'cancelled',
268
+ completionMessage: 'Stopped by user.',
269
+ })
270
+ );
271
+ }, [appendEventLine, isThinking]);
272
+
273
+ const resetConversation = useCallback(() => {
274
+ resolvePendingToolConfirm({
275
+ approved: false,
276
+ message: 'Tool confirmation cancelled because the conversation was reset.',
277
+ });
278
+ requestIdRef.current += 1;
279
+ activeAbortControllerRef.current?.abort();
280
+ activeAbortControllerRef.current = null;
281
+ activeTurnIdRef.current = null;
282
+ setIsThinking(false);
283
+ setTurns([]);
284
+ setSelectedFiles([]);
285
+ setContextUsagePercent(() => null);
286
+ }, [resolvePendingToolConfirm]);
287
+
288
+ const addTurn = useCallback(
289
+ (prompt: string, withStreamingReply = false, files: PromptFileSelection[] = []): number => {
290
+ const turnId = turnIdRef.current++;
291
+ const displayPrompt = buildPromptDisplay(prompt, files);
292
+ setTurns(prev => [
293
+ ...prev,
294
+ {
295
+ id: turnId,
296
+ prompt: displayPrompt,
297
+ files: files.map(file => file.relativePath),
298
+ createdAtMs: Date.now(),
299
+ reply: withStreamingReply ? createStreamingReply(modelLabel) : undefined,
300
+ },
301
+ ]);
302
+ return turnId;
303
+ },
304
+ [modelLabel]
305
+ );
306
+
307
+ const setImmediateReply = useCallback(
308
+ (
309
+ turnId: number,
310
+ segments: Array<{ id: string; type: 'thinking' | 'text'; content: string }>
311
+ ) => {
312
+ setTurns(prev =>
313
+ patchTurn(prev, turnId, turn => ({
314
+ ...turn,
315
+ reply: {
316
+ ...createStreamingReply(modelLabel),
317
+ status: 'done',
318
+ durationSeconds: 0,
319
+ segments,
320
+ },
321
+ }))
322
+ );
323
+ },
324
+ [modelLabel]
325
+ );
326
+
327
+ const runCommand = useCallback(
328
+ (commandText: string): boolean => {
329
+ const command = resolveSlashCommand(commandText);
330
+ if (!command) {
331
+ return false;
332
+ }
333
+
334
+ if (command.action === 'clear') {
335
+ resetConversation();
336
+ return true;
337
+ }
338
+
339
+ if (command.action === 'exit') {
340
+ requestExit(0);
341
+ return true;
342
+ }
343
+
344
+ if (command.action === 'help') {
345
+ const turnId = addTurn(commandText.trim(), true);
346
+ setImmediateReply(turnId, buildHelpSegments(turnId));
347
+ return true;
348
+ }
349
+
350
+ const turnId = addTurn(commandText.trim(), true);
351
+ setImmediateReply(turnId, buildUnsupportedSegments(turnId, command.name));
352
+ return true;
353
+ },
354
+ [addTurn, resetConversation, setImmediateReply]
355
+ );
356
+
357
+ const submitInput = useCallback(() => {
358
+ const text = inputValue.trim();
359
+ const attachedFiles = selectedFiles;
360
+ if ((text.length === 0 && attachedFiles.length === 0) || isThinking) {
361
+ return;
362
+ }
363
+
364
+ setInputValue('');
365
+
366
+ if (attachedFiles.length === 0 && text.startsWith('/') && runCommand(text)) {
367
+ return;
368
+ }
369
+
370
+ setSelectedFiles([]);
371
+
372
+ void (async () => {
373
+ const previousRun = activeRunPromiseRef.current;
374
+ if (previousRun) {
375
+ await previousRun.catch(error => {
376
+ console.debug(
377
+ 'Previous run failed:',
378
+ error instanceof Error ? error.message : String(error)
379
+ );
380
+ });
381
+ }
382
+
383
+ const turnId = addTurn(text, true, attachedFiles);
384
+ activeTurnIdRef.current = turnId;
385
+ const currentRequestId = ++requestIdRef.current;
386
+ const isCurrentRequest = () => currentRequestId === requestIdRef.current;
387
+ const abortController = new AbortController();
388
+ activeAbortControllerRef.current = abortController;
389
+
390
+ setIsThinking(true);
391
+
392
+ const baseHandlers = buildAgentEventHandlers({
393
+ turnId,
394
+ isCurrentRequest,
395
+ appendSegment,
396
+ appendEventLine,
397
+ });
398
+ const handlers = {
399
+ ...baseHandlers,
400
+ onToolConfirmRequest: (event: AgentToolConfirmEvent) => {
401
+ if (!isCurrentRequest()) {
402
+ return Promise.resolve({
403
+ approved: false,
404
+ message: 'Tool confirmation denied because the request is no longer active.',
405
+ });
406
+ }
407
+
408
+ if (pendingToolConfirmResolverRef.current) {
409
+ pendingToolConfirmResolverRef.current({
410
+ approved: false,
411
+ message: 'Superseded by a newer tool confirmation request.',
412
+ });
413
+ pendingToolConfirmResolverRef.current = null;
414
+ }
415
+
416
+ return new Promise<AgentToolConfirmDecision>(resolve => {
417
+ pendingToolConfirmResolverRef.current = resolve;
418
+ setPendingToolConfirm({
419
+ ...event,
420
+ selectedAction: 'approve',
421
+ });
422
+ });
423
+ },
424
+ onUsage: (event: AgentUsageEvent) => {
425
+ if (!isCurrentRequest()) {
426
+ return;
427
+ }
428
+ const normalized = normalizeContextUsagePercent(event.contextUsagePercent);
429
+ if (normalized !== null) {
430
+ setContextUsagePercent(normalized);
431
+ }
432
+ const replyUsage = toReplyUsage(event);
433
+ if (!replyUsage) {
434
+ return;
435
+ }
436
+ setTurns(prev =>
437
+ patchTurn(prev, turnId, turn => {
438
+ if (!turn.reply) {
439
+ return turn;
440
+ }
441
+ return {
442
+ ...turn,
443
+ reply: {
444
+ ...turn.reply,
445
+ ...replyUsage,
446
+ },
447
+ };
448
+ })
449
+ );
450
+ },
451
+ onContextUsage: (event: AgentContextUsageEvent) => {
452
+ if (!isCurrentRequest()) {
453
+ return;
454
+ }
455
+ const normalized = normalizeContextUsagePercent(event.contextUsagePercent);
456
+ if (normalized !== null) {
457
+ setContextUsagePercent(normalized);
458
+ }
459
+ },
460
+ };
461
+
462
+ const runPromise = buildPromptContent(text, attachedFiles, attachmentCapabilities)
463
+ .then(promptContent =>
464
+ runAgentPrompt(promptContent, handlers, {
465
+ abortSignal: abortController.signal,
466
+ })
467
+ )
468
+ .then(result => {
469
+ if (!isCurrentRequest()) {
470
+ return;
471
+ }
472
+
473
+ setModelLabel(result.modelLabel);
474
+ if (result.usage) {
475
+ const normalized = normalizeContextUsagePercent(result.usage.contextUsagePercent);
476
+ if (normalized !== null) {
477
+ setContextUsagePercent(normalized);
478
+ }
479
+ }
480
+ const replyUsage = toReplyUsage(result.usage);
481
+ setTurns(prev => {
482
+ const withFallbackText = patchTurn(prev, turnId, turn => {
483
+ if (!turn.reply || !result.text) {
484
+ return turn;
485
+ }
486
+
487
+ const hasAssistantText = turn.reply.segments.some(
488
+ segment =>
489
+ (segment.type === 'text' || segment.type === 'thinking') &&
490
+ segment.content.trim().length > 0
491
+ );
492
+ if (hasAssistantText) {
493
+ return turn;
494
+ }
495
+
496
+ return {
497
+ ...turn,
498
+ reply: {
499
+ ...turn.reply,
500
+ segments: orderReplySegments(
501
+ appendToSegment(turn.reply.segments, `${turnId}:text`, 'text', result.text)
502
+ ),
503
+ },
504
+ };
505
+ });
506
+
507
+ return setReplyStatus(
508
+ withFallbackText,
509
+ turnId,
510
+ resolveReplyStatus(result.completionReason),
511
+ {
512
+ durationSeconds: result.durationSeconds,
513
+ completionReason: result.completionReason,
514
+ completionMessage: result.completionMessage,
515
+ modelLabel: result.modelLabel,
516
+ ...(replyUsage ?? {}),
517
+ }
518
+ );
519
+ });
520
+ })
521
+ .catch(error => {
522
+ if (!isCurrentRequest()) {
523
+ return;
524
+ }
525
+ appendEventLine(turnId, `[error] ${extractErrorMessage(error)}`);
526
+ setTurns(prev => setReplyStatus(prev, turnId, 'error'));
527
+ })
528
+ .finally(() => {
529
+ if (activeAbortControllerRef.current === abortController) {
530
+ activeAbortControllerRef.current = null;
531
+ }
532
+ if (activeTurnIdRef.current === turnId) {
533
+ activeTurnIdRef.current = null;
534
+ }
535
+ if (!isCurrentRequest()) {
536
+ return;
537
+ }
538
+ setIsThinking(false);
539
+ });
540
+
541
+ const trackedRunPromise = runPromise.finally(() => {
542
+ if (activeRunPromiseRef.current === trackedRunPromise) {
543
+ activeRunPromiseRef.current = null;
544
+ }
545
+ });
546
+ activeRunPromiseRef.current = trackedRunPromise;
547
+ await trackedRunPromise;
548
+ })();
549
+ }, [
550
+ addTurn,
551
+ appendEventLine,
552
+ appendSegment,
553
+ attachmentCapabilities,
554
+ inputValue,
555
+ isThinking,
556
+ runCommand,
557
+ selectedFiles,
558
+ ]);
559
+
560
+ const clearInput = useCallback(() => {
561
+ setInputValue('');
562
+ setSelectedFiles([]);
563
+ }, []);
564
+
565
+ const setModelLabelDisplay = useCallback((label: string) => {
566
+ setModelLabel(label);
567
+ void getAgentModelAttachmentCapabilities()
568
+ .then(capabilities => {
569
+ setAttachmentCapabilities(capabilities);
570
+ })
571
+ .catch(() => {});
572
+ }, []);
573
+
574
+ const setToolConfirmSelection = useCallback((selection: 'approve' | 'deny') => {
575
+ setPendingToolConfirm(current =>
576
+ current ? { ...current, selectedAction: selection } : current
577
+ );
578
+ }, []);
579
+
580
+ const rejectPendingToolConfirm = useCallback(() => {
581
+ resolvePendingToolConfirm({
582
+ approved: false,
583
+ message: 'Tool call denied by user.',
584
+ });
585
+ }, [resolvePendingToolConfirm]);
586
+
587
+ const submitToolConfirmSelection = useCallback(() => {
588
+ if (!pendingToolConfirm) {
589
+ return;
590
+ }
591
+
592
+ if (pendingToolConfirm.selectedAction === 'deny') {
593
+ rejectPendingToolConfirm();
594
+ return;
595
+ }
596
+
597
+ resolvePendingToolConfirm({ approved: true });
598
+ }, [pendingToolConfirm, rejectPendingToolConfirm, resolvePendingToolConfirm]);
599
+
600
+ return {
601
+ turns,
602
+ inputValue,
603
+ selectedFiles,
604
+ isThinking,
605
+ modelLabel,
606
+ contextUsagePercent,
607
+ pendingToolConfirm,
608
+ setInputValue,
609
+ setSelectedFiles,
610
+ appendSelectedFiles,
611
+ removeSelectedFile,
612
+ submitInput,
613
+ stopActiveReply,
614
+ clearInput,
615
+ resetConversation,
616
+ setModelLabelDisplay,
617
+ setToolConfirmSelection,
618
+ submitToolConfirmSelection,
619
+ rejectPendingToolConfirm,
620
+ };
621
+ };