@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,248 @@
1
+ import type { KeyEvent, TextareaRenderable } from '@opentui/core';
2
+ import { TextAttributes } from '@opentui/core';
3
+ import { useEffect, useMemo, useRef } from 'react';
4
+
5
+ import type { AgentModelOption } from '../agent/runtime/model-types';
6
+ import { uiTheme } from '../ui/theme';
7
+
8
+ type ModelPickerDialogProps = {
9
+ visible: boolean;
10
+ viewportWidth: number;
11
+ viewportHeight: number;
12
+ loading: boolean;
13
+ switching: boolean;
14
+ error: string | null;
15
+ search: string;
16
+ options: AgentModelOption[];
17
+ selectedIndex: number;
18
+ onSearchChange: (value: string) => void;
19
+ onSelectIndex: (index: number) => void;
20
+ onConfirm: () => void;
21
+ onListKeyDown: (event: KeyEvent) => boolean;
22
+ };
23
+
24
+ const selectedBackground = '#f4b183';
25
+ const selectedForeground = '#050608';
26
+
27
+ const toProviderLabel = (provider: string) => {
28
+ if (!provider) {
29
+ return 'Other';
30
+ }
31
+ return provider.slice(0, 1).toUpperCase() + provider.slice(1);
32
+ };
33
+
34
+ export const ModelPickerDialog = ({
35
+ visible,
36
+ viewportWidth,
37
+ viewportHeight,
38
+ loading,
39
+ switching,
40
+ error,
41
+ search,
42
+ options,
43
+ selectedIndex,
44
+ onSearchChange,
45
+ onSelectIndex,
46
+ onConfirm,
47
+ onListKeyDown,
48
+ }: ModelPickerDialogProps) => {
49
+ const searchRef = useRef<TextareaRenderable | null>(null);
50
+
51
+ const panelWidth = Math.min(78, Math.max(40, viewportWidth - 8));
52
+ const panelHeight = Math.min(26, Math.max(14, viewportHeight - 4));
53
+ const left = Math.max(2, Math.floor((viewportWidth - panelWidth) / 2));
54
+ const top = Math.max(1, Math.floor((viewportHeight - panelHeight) / 2));
55
+
56
+ useEffect(() => {
57
+ if (!visible) {
58
+ return;
59
+ }
60
+
61
+ const searchInput = searchRef.current;
62
+ if (!searchInput) {
63
+ return;
64
+ }
65
+
66
+ searchInput.setText(search);
67
+ searchInput.cursorOffset = search.length;
68
+ searchInput.focus();
69
+ }, [search, visible]);
70
+
71
+ const rows = useMemo(() => {
72
+ const result: Array<{
73
+ type: 'section' | 'option';
74
+ label?: string;
75
+ option?: AgentModelOption;
76
+ index?: number;
77
+ }> = [];
78
+ let lastProvider = '';
79
+
80
+ options.forEach((option, index) => {
81
+ if (option.provider !== lastProvider) {
82
+ lastProvider = option.provider;
83
+ result.push({
84
+ type: 'section',
85
+ label: toProviderLabel(option.provider),
86
+ });
87
+ }
88
+
89
+ result.push({
90
+ type: 'option',
91
+ option,
92
+ index,
93
+ });
94
+ });
95
+
96
+ return result;
97
+ }, [options]);
98
+
99
+ if (!visible) {
100
+ return null;
101
+ }
102
+
103
+ return (
104
+ <box
105
+ position="absolute"
106
+ top={top}
107
+ left={left}
108
+ width={panelWidth}
109
+ height={panelHeight}
110
+ zIndex={140}
111
+ >
112
+ <box
113
+ width="100%"
114
+ height="100%"
115
+ flexDirection="column"
116
+ backgroundColor={uiTheme.surface}
117
+ border={['top', 'bottom', 'left', 'right']}
118
+ borderColor={uiTheme.divider}
119
+ >
120
+ <box justifyContent="space-between" paddingX={2} paddingTop={1} paddingBottom={0}>
121
+ <text fg={uiTheme.text} attributes={TextAttributes.BOLD}>
122
+ Select model
123
+ </text>
124
+ <text fg={uiTheme.muted}>
125
+ <strong>esc</strong>
126
+ </text>
127
+ </box>
128
+
129
+ <box paddingX={2} paddingTop={1} paddingBottom={1} flexDirection="column" gap={0}>
130
+ <text fg={uiTheme.muted}>Search</text>
131
+ <box backgroundColor={uiTheme.panel} paddingX={1}>
132
+ <textarea
133
+ ref={searchRef}
134
+ width="100%"
135
+ minHeight={1}
136
+ maxHeight={1}
137
+ initialValue={search}
138
+ textColor={uiTheme.text}
139
+ focusedTextColor={uiTheme.text}
140
+ backgroundColor={uiTheme.panel}
141
+ focusedBackgroundColor={uiTheme.panel}
142
+ onContentChange={() => onSearchChange(searchRef.current?.plainText ?? '')}
143
+ onKeyDown={event => {
144
+ if (onListKeyDown(event)) {
145
+ return;
146
+ }
147
+ }}
148
+ />
149
+ </box>
150
+ </box>
151
+
152
+ <box flexGrow={1} paddingX={1}>
153
+ <scrollbox
154
+ height="100%"
155
+ scrollY
156
+ stickyScroll
157
+ stickyStart="top"
158
+ scrollbarOptions={{ visible: false }}
159
+ viewportOptions={{ backgroundColor: uiTheme.surface }}
160
+ contentOptions={{ backgroundColor: uiTheme.surface }}
161
+ >
162
+ <box flexDirection="column" backgroundColor={uiTheme.surface}>
163
+ {loading ? (
164
+ <box paddingX={1}>
165
+ <text fg={uiTheme.muted}>Loading models...</text>
166
+ </box>
167
+ ) : rows.length === 0 ? (
168
+ <box paddingX={1}>
169
+ <text fg={uiTheme.muted}>No matching model</text>
170
+ </box>
171
+ ) : (
172
+ rows.map((row, idx) => {
173
+ if (row.type === 'section') {
174
+ return (
175
+ <box
176
+ key={`section:${row.label}:${idx}`}
177
+ paddingX={1}
178
+ paddingTop={idx === 0 ? 0 : 1}
179
+ >
180
+ <text fg="#b294ff" attributes={TextAttributes.BOLD}>
181
+ {row.label}
182
+ </text>
183
+ </box>
184
+ );
185
+ }
186
+
187
+ const option = row.option!;
188
+ const optionIndex = row.index ?? 0;
189
+ const isSelected = optionIndex === selectedIndex;
190
+ const suffix = option.current
191
+ ? 'Current'
192
+ : option.configured
193
+ ? 'Ready'
194
+ : 'No key';
195
+
196
+ return (
197
+ <box
198
+ key={option.id}
199
+ flexDirection="row"
200
+ justifyContent="space-between"
201
+ paddingX={1}
202
+ backgroundColor={isSelected ? selectedBackground : uiTheme.surface}
203
+ onMouseOver={() => onSelectIndex(optionIndex)}
204
+ onMouseUp={() => onConfirm()}
205
+ >
206
+ <text
207
+ fg={isSelected ? selectedForeground : uiTheme.text}
208
+ attributes={TextAttributes.BOLD}
209
+ wrapMode="none"
210
+ >
211
+ {option.name}
212
+ </text>
213
+ <text fg={isSelected ? selectedForeground : uiTheme.muted}>{suffix}</text>
214
+ </box>
215
+ );
216
+ })
217
+ )}
218
+ </box>
219
+ </scrollbox>
220
+ </box>
221
+
222
+ <box paddingX={2} paddingY={1} justifyContent="space-between">
223
+ <text fg={switching ? uiTheme.accent : uiTheme.muted}>
224
+ {switching ? 'Switching model...' : 'enter select up/down navigate'}
225
+ </text>
226
+ <text fg={uiTheme.muted}>
227
+ <strong>esc</strong> close
228
+ </text>
229
+ </box>
230
+
231
+ {error ? (
232
+ <box paddingX={2} paddingBottom={1}>
233
+ <text fg="#ff8d8d">{error}</text>
234
+ </box>
235
+ ) : null}
236
+
237
+ <box
238
+ position="absolute"
239
+ top={0}
240
+ left={0}
241
+ width={1}
242
+ height="100%"
243
+ backgroundColor={uiTheme.accent}
244
+ />
245
+ </box>
246
+ </box>
247
+ );
248
+ };
@@ -0,0 +1,233 @@
1
+ import type { KeyEvent, PasteEvent, TextareaRenderable } from '@opentui/core';
2
+ import { useCallback, useEffect, useMemo, useRef } from 'react';
3
+
4
+ import { FileMentionMenu } from './file-mention-menu';
5
+ import { FooterHints } from './footer-hints';
6
+ import { SlashCommandMenu } from './slash-command-menu';
7
+ import type { SlashCommandDefinition } from '../commands/slash-commands';
8
+ import {
9
+ isAudioSelection,
10
+ isImageSelection,
11
+ isVideoSelection,
12
+ } from '../files/attachment-capabilities';
13
+ import type { PromptFileSelection } from '../files/types';
14
+ import { useFileMentionMenu } from '../hooks/use-file-mention-menu';
15
+ import { useSlashCommandMenu } from '../hooks/use-slash-command-menu';
16
+ import { uiTheme } from '../ui/theme';
17
+
18
+ type PromptProps = {
19
+ isThinking: boolean;
20
+ disabled?: boolean;
21
+ modelLabel: string;
22
+ contextUsagePercent: number | null;
23
+ value: string;
24
+ selectedFiles: PromptFileSelection[];
25
+ onAddSelectedFiles: (files: PromptFileSelection[]) => void;
26
+ onValueChange: (value: string) => void;
27
+ onSlashCommandSelect?: (command: SlashCommandDefinition) => boolean;
28
+ onSlashMenuVisibilityChange?: (visible: boolean) => void;
29
+ onSubmit: () => void;
30
+ };
31
+
32
+ export const Prompt = ({
33
+ isThinking,
34
+ disabled = false,
35
+ modelLabel,
36
+ contextUsagePercent,
37
+ value,
38
+ selectedFiles,
39
+ onAddSelectedFiles,
40
+ onValueChange,
41
+ onSlashCommandSelect,
42
+ onSlashMenuVisibilityChange,
43
+ onSubmit,
44
+ }: PromptProps) => {
45
+ const textareaRef = useRef<TextareaRenderable | null>(null);
46
+ const mediaFiles = useMemo(
47
+ () =>
48
+ selectedFiles.filter(
49
+ file => isImageSelection(file) || isAudioSelection(file) || isVideoSelection(file)
50
+ ),
51
+ [selectedFiles]
52
+ );
53
+ const inputLocked = isThinking || disabled;
54
+ const promptAlignPaddingX =
55
+ uiTheme.layout.conversationPaddingX +
56
+ uiTheme.layout.conversationContentPaddingX +
57
+ uiTheme.layout.promptPaddingX;
58
+ const slashMenu = useSlashCommandMenu({
59
+ value,
60
+ onValueChange,
61
+ textareaRef,
62
+ onCommandSelected: onSlashCommandSelect,
63
+ disabled: inputLocked,
64
+ });
65
+ const fileMentionMenu = useFileMentionMenu({
66
+ value,
67
+ textareaRef,
68
+ selectedFiles,
69
+ onFilesSelected: onAddSelectedFiles,
70
+ onValueChange,
71
+ disabled: inputLocked,
72
+ });
73
+
74
+ useEffect(() => {
75
+ onSlashMenuVisibilityChange?.(slashMenu.visible || fileMentionMenu.visible);
76
+ }, [fileMentionMenu.visible, onSlashMenuVisibilityChange, slashMenu.visible]);
77
+
78
+ useEffect(() => {
79
+ const textarea = textareaRef.current;
80
+ if (!textarea) {
81
+ return;
82
+ }
83
+
84
+ if (textarea.plainText !== value) {
85
+ textarea.setText(value);
86
+ textarea.cursorOffset = value.length;
87
+ }
88
+ }, [value]);
89
+
90
+ useEffect(() => {
91
+ const textarea = textareaRef.current;
92
+ if (!textarea) {
93
+ return;
94
+ }
95
+
96
+ if (inputLocked) {
97
+ textarea.blur();
98
+ return;
99
+ }
100
+
101
+ textarea.focus();
102
+ }, [inputLocked]);
103
+
104
+ const submit = useCallback(() => {
105
+ if (inputLocked) {
106
+ return;
107
+ }
108
+ onSubmit();
109
+ }, [inputLocked, onSubmit]);
110
+
111
+ const handleContentChange = useCallback(() => {
112
+ onValueChange(textareaRef.current?.plainText ?? '');
113
+ }, [onValueChange]);
114
+
115
+ const handleKeyDown = useCallback(
116
+ (event: KeyEvent) => {
117
+ if (inputLocked) {
118
+ event.preventDefault();
119
+ return;
120
+ }
121
+
122
+ if (fileMentionMenu.handleKeyDown(event)) {
123
+ return;
124
+ }
125
+
126
+ if (slashMenu.handleKeyDown(event)) {
127
+ return;
128
+ }
129
+
130
+ const isEnter = event.name === 'return' || event.name === 'enter';
131
+ if (isEnter && !event.shift) {
132
+ event.preventDefault();
133
+ submit();
134
+ }
135
+ },
136
+ [fileMentionMenu, inputLocked, slashMenu, submit]
137
+ );
138
+
139
+ const handlePaste = useCallback((event: PasteEvent) => {
140
+ const normalized = event.text.replace(/\r\n/g, '\n').replace(/\r/g, '\n');
141
+ if (normalized === event.text) {
142
+ return;
143
+ }
144
+
145
+ event.preventDefault();
146
+ textareaRef.current?.insertText(normalized);
147
+ }, []);
148
+
149
+ return (
150
+ <box
151
+ flexDirection="column"
152
+ flexShrink={0}
153
+ width="100%"
154
+ gap={0}
155
+ paddingBottom={uiTheme.layout.promptPaddingBottom}
156
+ >
157
+ <box flexDirection="column" width="100%" gap={0} paddingX={promptAlignPaddingX}>
158
+ <FileMentionMenu
159
+ visible={fileMentionMenu.visible}
160
+ loading={fileMentionMenu.loading}
161
+ error={fileMentionMenu.error}
162
+ options={fileMentionMenu.options}
163
+ selectedIndex={fileMentionMenu.selectedIndex}
164
+ />
165
+ <SlashCommandMenu
166
+ visible={!fileMentionMenu.visible && slashMenu.visible}
167
+ options={slashMenu.options}
168
+ selectedIndex={slashMenu.selectedIndex}
169
+ />
170
+ <box width="100%" flexDirection="row" overflow="hidden">
171
+ <box width={1} backgroundColor={uiTheme.accent} />
172
+ <box
173
+ width="100%"
174
+ flexGrow={1}
175
+ paddingX={2}
176
+ paddingTop={1}
177
+ paddingBottom={0}
178
+ backgroundColor={uiTheme.inputBg}
179
+ >
180
+ {mediaFiles.length > 0 ? (
181
+ <box flexDirection="column" gap={0} paddingBottom={1}>
182
+ <text fg={uiTheme.muted}>Media files</text>
183
+ {mediaFiles.map(file => (
184
+ <text key={file.absolutePath} fg={uiTheme.accent} wrapMode="none">
185
+ {file.relativePath}
186
+ </text>
187
+ ))}
188
+ </box>
189
+ ) : null}
190
+ <textarea
191
+ ref={textareaRef}
192
+ buffered={false}
193
+ width="100%"
194
+ minWidth="100%"
195
+ maxWidth="100%"
196
+ minHeight={1}
197
+ maxHeight={4}
198
+ wrapMode="char"
199
+ initialValue={value}
200
+ textColor={uiTheme.userPromptText}
201
+ focusedTextColor={uiTheme.userPromptText}
202
+ backgroundColor="transparent"
203
+ focusedBackgroundColor="transparent"
204
+ cursorColor={uiTheme.inputCursor}
205
+ selectionBg={uiTheme.inputSelectionBg}
206
+ selectionFg={uiTheme.inputSelectionText}
207
+ placeholder={
208
+ isThinking
209
+ ? 'waiting for agent response...'
210
+ : disabled
211
+ ? 'command dialog active...'
212
+ : 'Type your message...'
213
+ }
214
+ placeholderColor={uiTheme.muted}
215
+ onContentChange={handleContentChange}
216
+ onKeyDown={handleKeyDown}
217
+ onPaste={handlePaste}
218
+ />
219
+ <box flexDirection="row" gap={1} paddingTop={1} paddingBottom={1}>
220
+ <text fg={uiTheme.text} attributes={uiTheme.typography.heading}>
221
+ {modelLabel}
222
+ </text>
223
+ <text fg={uiTheme.muted} attributes={uiTheme.typography.muted}>
224
+ Coding Agent
225
+ </text>
226
+ </box>
227
+ </box>
228
+ </box>
229
+ </box>
230
+ <FooterHints isThinking={isThinking} contextUsagePercent={contextUsagePercent} />
231
+ </box>
232
+ );
233
+ };
@@ -0,0 +1,65 @@
1
+ import { TextAttributes } from '@opentui/core';
2
+
3
+ import type { SlashCommandDefinition } from '../commands/slash-commands';
4
+ import { uiTheme } from '../ui/theme';
5
+
6
+ type SlashCommandMenuProps = {
7
+ visible: boolean;
8
+ options: SlashCommandDefinition[];
9
+ selectedIndex: number;
10
+ };
11
+
12
+ const selectedBackground = '#f4b183';
13
+ const selectedForeground = '#050608';
14
+
15
+ export const SlashCommandMenu = ({ visible, options, selectedIndex }: SlashCommandMenuProps) => {
16
+ if (!visible) {
17
+ return null;
18
+ }
19
+
20
+ const labelWidth = options.reduce((max, option) => {
21
+ const width = `/${option.name}`.length;
22
+ return Math.max(max, width);
23
+ }, 0);
24
+
25
+ return (
26
+ <box
27
+ width="100%"
28
+ flexShrink={0}
29
+ backgroundColor={uiTheme.panel}
30
+ border={['top', 'bottom', 'left', 'right']}
31
+ borderColor={uiTheme.divider}
32
+ marginBottom={0}
33
+ height={Math.min(11, options.length + 2)}
34
+ >
35
+ <scrollbox scrollY stickyScroll stickyStart="top" scrollbarOptions={{ visible: false }}>
36
+ <box flexDirection="column" backgroundColor={uiTheme.panel}>
37
+ {options.map((option, index) => {
38
+ const isSelected = index === selectedIndex;
39
+ const commandText = `/${option.name}`.padEnd(labelWidth + 2, ' ');
40
+
41
+ return (
42
+ <box
43
+ key={option.name}
44
+ flexDirection="row"
45
+ paddingX={1}
46
+ backgroundColor={isSelected ? selectedBackground : uiTheme.panel}
47
+ >
48
+ <text
49
+ fg={isSelected ? selectedForeground : uiTheme.text}
50
+ attributes={TextAttributes.BOLD}
51
+ flexShrink={0}
52
+ >
53
+ {commandText}
54
+ </text>
55
+ <text fg={isSelected ? selectedForeground : uiTheme.muted} wrapMode="word">
56
+ {option.description}
57
+ </text>
58
+ </box>
59
+ );
60
+ })}
61
+ </box>
62
+ </scrollbox>
63
+ </box>
64
+ );
65
+ };
@@ -0,0 +1,103 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { buildToolConfirmDialogContent } from './tool-confirm-dialog-content';
4
+
5
+ describe('buildToolConfirmDialogContent', () => {
6
+ it('formats outside-workspace glob confirmations with path details', () => {
7
+ const content = buildToolConfirmDialogContent({
8
+ toolCallId: 'call_1',
9
+ toolName: 'glob',
10
+ args: {
11
+ pattern: '**/*sandbox*',
12
+ path: '/Users/wrr/work/ironclaw',
13
+ },
14
+ rawArgs: {
15
+ pattern: '**/*sandbox*',
16
+ path: '/Users/wrr/work/ironclaw',
17
+ },
18
+ reason:
19
+ 'SEARCH_PATH_NOT_ALLOWED: /Users/wrr/work/ironclaw is outside allowed directories: /Users/wrr/work/coding-agent-v2',
20
+ metadata: {
21
+ requestedPath: '/Users/wrr/work/ironclaw',
22
+ allowedDirectories: ['/Users/wrr/work/coding-agent-v2'],
23
+ },
24
+ });
25
+
26
+ expect(content.summary).toBe('Glob **/*sandbox*');
27
+ expect(content.detail).toBe('Path: /Users/wrr/work/ironclaw');
28
+ expect(content.requestedPath).toBe('/Users/wrr/work/ironclaw');
29
+ expect(content.allowedDirectories).toEqual(['/Users/wrr/work/coding-agent-v2']);
30
+ expect(content.argumentItems).toEqual([]);
31
+ });
32
+
33
+ it('formats bash confirmations with command preview', () => {
34
+ const content = buildToolConfirmDialogContent({
35
+ toolCallId: 'call_2',
36
+ toolName: 'bash',
37
+ args: {
38
+ description: 'List repo files',
39
+ command: 'rg --files src',
40
+ },
41
+ rawArgs: {
42
+ description: 'List repo files',
43
+ command: 'rg --files src',
44
+ },
45
+ });
46
+
47
+ expect(content.summary).toBe('Run bash: List repo files');
48
+ expect(content.detail).toBe('$ rg --files src');
49
+ expect(content.reason).toBeUndefined();
50
+ expect(content.argumentItems).toEqual([]);
51
+ });
52
+
53
+ it('hides redundant file path arguments that are already surfaced elsewhere', () => {
54
+ const content = buildToolConfirmDialogContent({
55
+ toolCallId: 'call_3',
56
+ toolName: 'file_read',
57
+ args: {
58
+ path: '/Users/wrr/work/ironclaw/src/sandbox/config.rs',
59
+ },
60
+ rawArgs: {
61
+ path: '/Users/wrr/work/ironclaw/src/sandbox/config.rs',
62
+ },
63
+ reason:
64
+ 'PATH_NOT_ALLOWED: /Users/wrr/work/ironclaw/src/sandbox/config.rs is outside allowed directories: /Users/wrr/work/coding-agent-v2',
65
+ metadata: {
66
+ requestedPath: '/Users/wrr/work/ironclaw/src/sandbox/config.rs',
67
+ allowedDirectories: ['/Users/wrr/work/coding-agent-v2'],
68
+ },
69
+ });
70
+
71
+ expect(content.summary).toBe('Read /Users/wrr/work/ironclaw/src/sandbox/config.rs');
72
+ expect(content.argumentItems).toEqual([]);
73
+ });
74
+
75
+ it('parses json-like string arguments into readable structured values', () => {
76
+ const content = buildToolConfirmDialogContent({
77
+ toolCallId: 'call_4',
78
+ toolName: 'custom_tool',
79
+ args: {
80
+ payload: '{"path":"/tmp/project","recursive":true}',
81
+ retries: 3,
82
+ },
83
+ rawArgs: {
84
+ payload: '{"path":"/tmp/project","recursive":true}',
85
+ retries: 3,
86
+ },
87
+ });
88
+
89
+ expect(content.summary).toBe('Call custom_tool');
90
+ expect(content.argumentItems).toEqual([
91
+ {
92
+ label: 'Payload',
93
+ value: '{\n "path": "/tmp/project",\n "recursive": true\n}',
94
+ multiline: true,
95
+ },
96
+ {
97
+ label: 'Retries',
98
+ value: '3',
99
+ multiline: undefined,
100
+ },
101
+ ]);
102
+ });
103
+ });