@code0123/opencode-android-arm64 1.1.55 → 1.1.57

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 (359) hide show
  1. package/bin/opencode +3 -1
  2. package/package.json +1 -77
  3. package/runtime/highlights-eq9cgrbb.scm +604 -0
  4. package/runtime/highlights-ghv9g403.scm +205 -0
  5. package/runtime/highlights-hk7bwhj4.scm +284 -0
  6. package/runtime/highlights-r812a2qc.scm +150 -0
  7. package/runtime/highlights-x6tmsnaa.scm +115 -0
  8. package/runtime/index.js +287124 -0
  9. package/runtime/injections-73j83es3.scm +27 -0
  10. package/runtime/parser.worker.js +4081 -0
  11. package/runtime/tree-sitter-3jzf13jk.wasm +0 -0
  12. package/runtime/tree-sitter-bash-hq5s6fxb.wasm +0 -0
  13. package/runtime/tree-sitter-javascript-nd0q4pe9.wasm +0 -0
  14. package/runtime/tree-sitter-markdown-411r6y9b.wasm +0 -0
  15. package/runtime/tree-sitter-markdown_inline-j5349f42.wasm +0 -0
  16. package/runtime/tree-sitter-typescript-zxjzwt75.wasm +0 -0
  17. package/runtime/tree-sitter-zig-e78zbjpm.wasm +0 -0
  18. package/runtime/worker.js +208338 -0
  19. package/parsers-config.ts +0 -253
  20. package/src/acp/README.md +0 -164
  21. package/src/acp/agent.ts +0 -1676
  22. package/src/acp/session.ts +0 -117
  23. package/src/acp/types.ts +0 -23
  24. package/src/agent/agent.ts +0 -338
  25. package/src/agent/generate.txt +0 -75
  26. package/src/agent/prompt/compaction.txt +0 -12
  27. package/src/agent/prompt/explore.txt +0 -18
  28. package/src/agent/prompt/summary.txt +0 -11
  29. package/src/agent/prompt/title.txt +0 -44
  30. package/src/auth/index.ts +0 -70
  31. package/src/bun/index.ts +0 -137
  32. package/src/bun/registry.ts +0 -48
  33. package/src/bus/bus-event.ts +0 -43
  34. package/src/bus/global.ts +0 -10
  35. package/src/bus/index.ts +0 -105
  36. package/src/cli/bootstrap.ts +0 -17
  37. package/src/cli/cmd/acp.ts +0 -70
  38. package/src/cli/cmd/agent.ts +0 -257
  39. package/src/cli/cmd/auth.ts +0 -400
  40. package/src/cli/cmd/cmd.ts +0 -7
  41. package/src/cli/cmd/debug/agent.ts +0 -167
  42. package/src/cli/cmd/debug/config.ts +0 -16
  43. package/src/cli/cmd/debug/file.ts +0 -97
  44. package/src/cli/cmd/debug/index.ts +0 -48
  45. package/src/cli/cmd/debug/lsp.ts +0 -52
  46. package/src/cli/cmd/debug/ripgrep.ts +0 -87
  47. package/src/cli/cmd/debug/scrap.ts +0 -16
  48. package/src/cli/cmd/debug/skill.ts +0 -16
  49. package/src/cli/cmd/debug/snapshot.ts +0 -52
  50. package/src/cli/cmd/export.ts +0 -88
  51. package/src/cli/cmd/generate.ts +0 -38
  52. package/src/cli/cmd/github.ts +0 -1540
  53. package/src/cli/cmd/import.ts +0 -147
  54. package/src/cli/cmd/mcp.ts +0 -755
  55. package/src/cli/cmd/models.ts +0 -77
  56. package/src/cli/cmd/pr.ts +0 -112
  57. package/src/cli/cmd/run.ts +0 -598
  58. package/src/cli/cmd/serve.ts +0 -20
  59. package/src/cli/cmd/session.ts +0 -135
  60. package/src/cli/cmd/stats.ts +0 -426
  61. package/src/cli/cmd/tui/app.tsx +0 -801
  62. package/src/cli/cmd/tui/attach.ts +0 -52
  63. package/src/cli/cmd/tui/component/border.tsx +0 -21
  64. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  65. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -148
  66. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  67. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -234
  68. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -266
  69. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -108
  70. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  71. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -36
  72. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  73. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -177
  74. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  75. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  76. package/src/cli/cmd/tui/component/logo.tsx +0 -85
  77. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -666
  78. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -89
  79. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  80. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1132
  81. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  82. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  83. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  84. package/src/cli/cmd/tui/component/tips.tsx +0 -153
  85. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  86. package/src/cli/cmd/tui/context/args.tsx +0 -15
  87. package/src/cli/cmd/tui/context/directory.ts +0 -13
  88. package/src/cli/cmd/tui/context/exit.tsx +0 -52
  89. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  90. package/src/cli/cmd/tui/context/keybind.tsx +0 -100
  91. package/src/cli/cmd/tui/context/kv.tsx +0 -52
  92. package/src/cli/cmd/tui/context/local.tsx +0 -409
  93. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  94. package/src/cli/cmd/tui/context/route.tsx +0 -46
  95. package/src/cli/cmd/tui/context/sdk.tsx +0 -101
  96. package/src/cli/cmd/tui/context/sync.tsx +0 -470
  97. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  98. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  99. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  100. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
  101. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  102. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  103. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  104. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  105. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  106. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  107. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  108. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  109. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  110. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  111. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
  112. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  113. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  114. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  115. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  116. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  117. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  118. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  119. package/src/cli/cmd/tui/context/theme/opencode.json +0 -245
  120. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  121. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  122. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  123. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  124. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  125. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  126. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  127. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  128. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  129. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  130. package/src/cli/cmd/tui/context/theme.tsx +0 -1152
  131. package/src/cli/cmd/tui/event.ts +0 -48
  132. package/src/cli/cmd/tui/routes/home.tsx +0 -140
  133. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -64
  134. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -109
  135. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
  136. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  137. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  138. package/src/cli/cmd/tui/routes/session/header.tsx +0 -142
  139. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2126
  140. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -508
  141. package/src/cli/cmd/tui/routes/session/question.tsx +0 -466
  142. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -313
  143. package/src/cli/cmd/tui/thread.ts +0 -175
  144. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -68
  145. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -93
  146. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -215
  147. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -49
  148. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -88
  149. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -399
  150. package/src/cli/cmd/tui/ui/dialog.tsx +0 -167
  151. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  152. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  153. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  154. package/src/cli/cmd/tui/util/clipboard.ts +0 -159
  155. package/src/cli/cmd/tui/util/editor.ts +0 -32
  156. package/src/cli/cmd/tui/util/signal.ts +0 -7
  157. package/src/cli/cmd/tui/util/terminal.ts +0 -114
  158. package/src/cli/cmd/tui/util/transcript.ts +0 -98
  159. package/src/cli/cmd/tui/worker.ts +0 -152
  160. package/src/cli/cmd/uninstall.ts +0 -357
  161. package/src/cli/cmd/upgrade.ts +0 -73
  162. package/src/cli/cmd/web.ts +0 -81
  163. package/src/cli/error.ts +0 -57
  164. package/src/cli/logo.ts +0 -6
  165. package/src/cli/network.ts +0 -60
  166. package/src/cli/ui.ts +0 -113
  167. package/src/cli/upgrade.ts +0 -25
  168. package/src/command/index.ts +0 -150
  169. package/src/command/template/initialize.txt +0 -10
  170. package/src/command/template/review.txt +0 -99
  171. package/src/config/config.ts +0 -1477
  172. package/src/config/markdown.ts +0 -98
  173. package/src/env/index.ts +0 -28
  174. package/src/file/ignore.ts +0 -83
  175. package/src/file/index.ts +0 -583
  176. package/src/file/ripgrep.ts +0 -384
  177. package/src/file/time.ts +0 -69
  178. package/src/file/watcher.ts +0 -148
  179. package/src/flag/flag.ts +0 -93
  180. package/src/format/formatter.ts +0 -366
  181. package/src/format/index.ts +0 -137
  182. package/src/global/index.ts +0 -55
  183. package/src/id/id.ts +0 -83
  184. package/src/ide/index.ts +0 -76
  185. package/src/index.ts +0 -159
  186. package/src/installation/index.ts +0 -246
  187. package/src/lsp/client.ts +0 -252
  188. package/src/lsp/index.ts +0 -485
  189. package/src/lsp/language.ts +0 -119
  190. package/src/lsp/server.ts +0 -2046
  191. package/src/mcp/auth.ts +0 -132
  192. package/src/mcp/index.ts +0 -934
  193. package/src/mcp/oauth-callback.ts +0 -200
  194. package/src/mcp/oauth-provider.ts +0 -154
  195. package/src/patch/index.ts +0 -680
  196. package/src/permission/arity.ts +0 -163
  197. package/src/permission/index.ts +0 -210
  198. package/src/permission/next.ts +0 -280
  199. package/src/plugin/codex.ts +0 -624
  200. package/src/plugin/copilot.ts +0 -327
  201. package/src/plugin/index.ts +0 -138
  202. package/src/project/bootstrap.ts +0 -35
  203. package/src/project/instance.ts +0 -114
  204. package/src/project/project.ts +0 -371
  205. package/src/project/state.ts +0 -70
  206. package/src/project/vcs.ts +0 -76
  207. package/src/provider/auth.ts +0 -147
  208. package/src/provider/models-snapshot.ts +0 -38414
  209. package/src/provider/models.ts +0 -133
  210. package/src/provider/provider.ts +0 -1262
  211. package/src/provider/sdk/copilot/README.md +0 -5
  212. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -164
  213. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  214. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -17
  215. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  216. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -780
  217. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  218. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  219. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -87
  220. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  221. package/src/provider/sdk/copilot/index.ts +0 -2
  222. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  223. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -303
  224. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  225. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  226. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  227. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -207
  228. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1732
  229. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -177
  230. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  231. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -88
  232. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -128
  233. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -115
  234. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -65
  235. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -104
  236. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -103
  237. package/src/provider/transform.ts +0 -828
  238. package/src/pty/index.ts +0 -250
  239. package/src/question/index.ts +0 -171
  240. package/src/scheduler/index.ts +0 -61
  241. package/src/server/error.ts +0 -36
  242. package/src/server/event.ts +0 -7
  243. package/src/server/mdns.ts +0 -60
  244. package/src/server/routes/config.ts +0 -92
  245. package/src/server/routes/experimental.ts +0 -208
  246. package/src/server/routes/file.ts +0 -197
  247. package/src/server/routes/global.ts +0 -183
  248. package/src/server/routes/mcp.ts +0 -225
  249. package/src/server/routes/permission.ts +0 -68
  250. package/src/server/routes/project.ts +0 -82
  251. package/src/server/routes/provider.ts +0 -165
  252. package/src/server/routes/pty.ts +0 -169
  253. package/src/server/routes/question.ts +0 -98
  254. package/src/server/routes/session.ts +0 -939
  255. package/src/server/routes/tui.ts +0 -379
  256. package/src/server/server.ts +0 -613
  257. package/src/session/compaction.ts +0 -226
  258. package/src/session/index.ts +0 -517
  259. package/src/session/instruction.ts +0 -197
  260. package/src/session/llm.ts +0 -289
  261. package/src/session/message-v2.ts +0 -802
  262. package/src/session/message.ts +0 -189
  263. package/src/session/processor.ts +0 -407
  264. package/src/session/prompt/anthropic-20250930.txt +0 -166
  265. package/src/session/prompt/anthropic.txt +0 -105
  266. package/src/session/prompt/beast.txt +0 -147
  267. package/src/session/prompt/build-switch.txt +0 -5
  268. package/src/session/prompt/codex_header.txt +0 -79
  269. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  270. package/src/session/prompt/gemini.txt +0 -155
  271. package/src/session/prompt/max-steps.txt +0 -16
  272. package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
  273. package/src/session/prompt/plan.txt +0 -26
  274. package/src/session/prompt/qwen.txt +0 -109
  275. package/src/session/prompt/trinity.txt +0 -97
  276. package/src/session/prompt.ts +0 -1866
  277. package/src/session/retry.ts +0 -97
  278. package/src/session/revert.ts +0 -121
  279. package/src/session/status.ts +0 -76
  280. package/src/session/summary.ts +0 -217
  281. package/src/session/system.ts +0 -54
  282. package/src/session/todo.ts +0 -37
  283. package/src/share/share-next.ts +0 -200
  284. package/src/share/share.ts +0 -92
  285. package/src/shell/shell.ts +0 -67
  286. package/src/skill/discovery.ts +0 -97
  287. package/src/skill/index.ts +0 -1
  288. package/src/skill/skill.ts +0 -188
  289. package/src/snapshot/index.ts +0 -255
  290. package/src/storage/storage.ts +0 -227
  291. package/src/tool/apply_patch.ts +0 -281
  292. package/src/tool/apply_patch.txt +0 -33
  293. package/src/tool/bash.ts +0 -269
  294. package/src/tool/bash.txt +0 -115
  295. package/src/tool/batch.ts +0 -175
  296. package/src/tool/batch.txt +0 -24
  297. package/src/tool/codesearch.ts +0 -132
  298. package/src/tool/codesearch.txt +0 -12
  299. package/src/tool/edit.ts +0 -655
  300. package/src/tool/edit.txt +0 -10
  301. package/src/tool/external-directory.ts +0 -32
  302. package/src/tool/glob.ts +0 -78
  303. package/src/tool/glob.txt +0 -6
  304. package/src/tool/grep.ts +0 -147
  305. package/src/tool/grep.txt +0 -8
  306. package/src/tool/invalid.ts +0 -17
  307. package/src/tool/ls.ts +0 -121
  308. package/src/tool/ls.txt +0 -1
  309. package/src/tool/lsp.ts +0 -96
  310. package/src/tool/lsp.txt +0 -19
  311. package/src/tool/multiedit.ts +0 -46
  312. package/src/tool/multiedit.txt +0 -41
  313. package/src/tool/plan-enter.txt +0 -14
  314. package/src/tool/plan-exit.txt +0 -13
  315. package/src/tool/plan.ts +0 -130
  316. package/src/tool/question.ts +0 -33
  317. package/src/tool/question.txt +0 -10
  318. package/src/tool/read.ts +0 -211
  319. package/src/tool/read.txt +0 -12
  320. package/src/tool/registry.ts +0 -160
  321. package/src/tool/skill.ts +0 -123
  322. package/src/tool/task.ts +0 -165
  323. package/src/tool/task.txt +0 -60
  324. package/src/tool/todo.ts +0 -53
  325. package/src/tool/todoread.txt +0 -14
  326. package/src/tool/todowrite.txt +0 -167
  327. package/src/tool/tool.ts +0 -89
  328. package/src/tool/truncation.ts +0 -106
  329. package/src/tool/webfetch.ts +0 -186
  330. package/src/tool/webfetch.txt +0 -13
  331. package/src/tool/websearch.ts +0 -150
  332. package/src/tool/websearch.txt +0 -14
  333. package/src/tool/write.ts +0 -85
  334. package/src/tool/write.txt +0 -8
  335. package/src/util/abort.ts +0 -35
  336. package/src/util/archive.ts +0 -16
  337. package/src/util/color.ts +0 -19
  338. package/src/util/context.ts +0 -25
  339. package/src/util/defer.ts +0 -12
  340. package/src/util/eventloop.ts +0 -20
  341. package/src/util/filesystem.ts +0 -93
  342. package/src/util/fn.ts +0 -11
  343. package/src/util/format.ts +0 -20
  344. package/src/util/iife.ts +0 -3
  345. package/src/util/keybind.ts +0 -103
  346. package/src/util/lazy.ts +0 -18
  347. package/src/util/locale.ts +0 -81
  348. package/src/util/lock.ts +0 -98
  349. package/src/util/log.ts +0 -180
  350. package/src/util/proxied.ts +0 -3
  351. package/src/util/queue.ts +0 -32
  352. package/src/util/rpc.ts +0 -66
  353. package/src/util/scrap.ts +0 -10
  354. package/src/util/signal.ts +0 -12
  355. package/src/util/timeout.ts +0 -14
  356. package/src/util/token.ts +0 -7
  357. package/src/util/wildcard.ts +0 -56
  358. package/src/worktree/index.ts +0 -574
  359. package/tsconfig.json +0 -10
@@ -1,1132 +0,0 @@
1
- import { BoxRenderable, TextareaRenderable, MouseEvent, PasteEvent, t, dim, fg } from "@opentui/core"
2
- import { createEffect, createMemo, type JSX, onMount, createSignal, onCleanup, Show, Switch, Match } from "solid-js"
3
- import "opentui-spinner/solid"
4
- import { useLocal } from "@tui/context/local"
5
- import { useTheme } from "@tui/context/theme"
6
- import { EmptyBorder } from "@tui/component/border"
7
- import { useSDK } from "@tui/context/sdk"
8
- import { useRoute } from "@tui/context/route"
9
- import { useSync } from "@tui/context/sync"
10
- import { Identifier } from "@/id/id"
11
- import { createStore, produce } from "solid-js/store"
12
- import { useKeybind } from "@tui/context/keybind"
13
- import { usePromptHistory, type PromptInfo } from "./history"
14
- import { usePromptStash } from "./stash"
15
- import { DialogStash } from "../dialog-stash"
16
- import { type AutocompleteRef, Autocomplete } from "./autocomplete"
17
- import { useCommandDialog } from "../dialog-command"
18
- import { useRenderer } from "@opentui/solid"
19
- import { Editor } from "@tui/util/editor"
20
- import { useExit } from "../../context/exit"
21
- import { Clipboard } from "../../util/clipboard"
22
- import type { FilePart } from "@opencode-ai/sdk/v2"
23
- import { TuiEvent } from "../../event"
24
- import { iife } from "@/util/iife"
25
- import { Locale } from "@/util/locale"
26
- import { formatDuration } from "@/util/format"
27
- import { createColors, createFrames } from "../../ui/spinner.ts"
28
- import { useDialog } from "@tui/ui/dialog"
29
- import { DialogProvider as DialogProviderConnect } from "../dialog-provider"
30
- import { DialogAlert } from "../../ui/dialog-alert"
31
- import { useToast } from "../../ui/toast"
32
- import { useKV } from "../../context/kv"
33
- import { useTextareaKeybindings } from "../textarea-keybindings"
34
- import { DialogSkill } from "../dialog-skill"
35
-
36
- export type PromptProps = {
37
- sessionID?: string
38
- visible?: boolean
39
- disabled?: boolean
40
- onSubmit?: () => void
41
- ref?: (ref: PromptRef) => void
42
- hint?: JSX.Element
43
- showPlaceholder?: boolean
44
- }
45
-
46
- export type PromptRef = {
47
- focused: boolean
48
- current: PromptInfo
49
- set(prompt: PromptInfo): void
50
- reset(): void
51
- blur(): void
52
- focus(): void
53
- submit(): void
54
- }
55
-
56
- const PLACEHOLDERS = ["Fix a TODO in the codebase", "What is the tech stack of this project?", "Fix broken tests"]
57
-
58
- export function Prompt(props: PromptProps) {
59
- let input: TextareaRenderable
60
- let anchor: BoxRenderable
61
- let autocomplete: AutocompleteRef
62
-
63
- const keybind = useKeybind()
64
- const local = useLocal()
65
- const sdk = useSDK()
66
- const route = useRoute()
67
- const sync = useSync()
68
- const dialog = useDialog()
69
- const toast = useToast()
70
- const status = createMemo(() => sync.data.session_status?.[props.sessionID ?? ""] ?? { type: "idle" })
71
- const history = usePromptHistory()
72
- const stash = usePromptStash()
73
- const command = useCommandDialog()
74
- const renderer = useRenderer()
75
- const { theme, syntax } = useTheme()
76
- const kv = useKV()
77
-
78
- function promptModelWarning() {
79
- toast.show({
80
- variant: "warning",
81
- message: "Connect a provider to send prompts",
82
- duration: 3000,
83
- })
84
- if (sync.data.provider.length === 0) {
85
- dialog.replace(() => <DialogProviderConnect />)
86
- }
87
- }
88
-
89
- const textareaKeybindings = useTextareaKeybindings()
90
-
91
- const fileStyleId = syntax().getStyleId("extmark.file")!
92
- const agentStyleId = syntax().getStyleId("extmark.agent")!
93
- const pasteStyleId = syntax().getStyleId("extmark.paste")!
94
- let promptPartTypeId = 0
95
-
96
- sdk.event.on(TuiEvent.PromptAppend.type, (evt) => {
97
- if (!input || input.isDestroyed) return
98
- input.insertText(evt.properties.text)
99
- setTimeout(() => {
100
- // setTimeout is a workaround and needs to be addressed properly
101
- if (!input || input.isDestroyed) return
102
- input.getLayoutNode().markDirty()
103
- input.gotoBufferEnd()
104
- renderer.requestRender()
105
- }, 0)
106
- })
107
-
108
- createEffect(() => {
109
- if (props.disabled) input.cursorColor = theme.backgroundElement
110
- if (!props.disabled) input.cursorColor = theme.text
111
- })
112
-
113
- const lastUserMessage = createMemo(() => {
114
- if (!props.sessionID) return undefined
115
- const messages = sync.data.message[props.sessionID]
116
- if (!messages) return undefined
117
- return messages.findLast((m) => m.role === "user")
118
- })
119
-
120
- const [store, setStore] = createStore<{
121
- prompt: PromptInfo
122
- mode: "normal" | "shell"
123
- extmarkToPartIndex: Map<number, number>
124
- interrupt: number
125
- placeholder: number
126
- }>({
127
- placeholder: Math.floor(Math.random() * PLACEHOLDERS.length),
128
- prompt: {
129
- input: "",
130
- parts: [],
131
- },
132
- mode: "normal",
133
- extmarkToPartIndex: new Map(),
134
- interrupt: 0,
135
- })
136
-
137
- // Initialize agent/model/variant from last user message when session changes
138
- let syncedSessionID: string | undefined
139
- createEffect(() => {
140
- const sessionID = props.sessionID
141
- const msg = lastUserMessage()
142
-
143
- if (sessionID !== syncedSessionID) {
144
- if (!sessionID || !msg) return
145
-
146
- syncedSessionID = sessionID
147
-
148
- // Only set agent if it's a primary agent (not a subagent)
149
- const isPrimaryAgent = local.agent.list().some((x) => x.name === msg.agent)
150
- if (msg.agent && isPrimaryAgent) {
151
- local.agent.set(msg.agent)
152
- if (msg.model) local.model.set(msg.model)
153
- if (msg.variant) local.model.variant.set(msg.variant)
154
- }
155
- }
156
- })
157
-
158
- command.register(() => {
159
- return [
160
- {
161
- title: "Clear prompt",
162
- value: "prompt.clear",
163
- category: "Prompt",
164
- hidden: true,
165
- onSelect: (dialog) => {
166
- input.extmarks.clear()
167
- input.clear()
168
- dialog.clear()
169
- },
170
- },
171
- {
172
- title: "Submit prompt",
173
- value: "prompt.submit",
174
- keybind: "input_submit",
175
- category: "Prompt",
176
- hidden: true,
177
- onSelect: (dialog) => {
178
- if (!input.focused) return
179
- submit()
180
- dialog.clear()
181
- },
182
- },
183
- {
184
- title: "Paste",
185
- value: "prompt.paste",
186
- keybind: "input_paste",
187
- category: "Prompt",
188
- hidden: true,
189
- onSelect: async () => {
190
- const content = await Clipboard.read()
191
- if (content?.mime.startsWith("image/")) {
192
- await pasteImage({
193
- filename: "clipboard",
194
- mime: content.mime,
195
- content: content.data,
196
- })
197
- }
198
- },
199
- },
200
- {
201
- title: "Interrupt session",
202
- value: "session.interrupt",
203
- keybind: "session_interrupt",
204
- category: "Session",
205
- hidden: true,
206
- enabled: status().type !== "idle",
207
- onSelect: (dialog) => {
208
- if (autocomplete.visible) return
209
- if (!input.focused) return
210
- // TODO: this should be its own command
211
- if (store.mode === "shell") {
212
- setStore("mode", "normal")
213
- return
214
- }
215
- if (!props.sessionID) return
216
-
217
- setStore("interrupt", store.interrupt + 1)
218
-
219
- setTimeout(() => {
220
- setStore("interrupt", 0)
221
- }, 5000)
222
-
223
- if (store.interrupt >= 2) {
224
- sdk.client.session.abort({
225
- sessionID: props.sessionID,
226
- })
227
- setStore("interrupt", 0)
228
- }
229
- dialog.clear()
230
- },
231
- },
232
- {
233
- title: "Open editor",
234
- category: "Session",
235
- keybind: "editor_open",
236
- value: "prompt.editor",
237
- slash: {
238
- name: "editor",
239
- },
240
- onSelect: async (dialog) => {
241
- dialog.clear()
242
-
243
- // replace summarized text parts with the actual text
244
- const text = store.prompt.parts
245
- .filter((p) => p.type === "text")
246
- .reduce((acc, p) => {
247
- if (!p.source) return acc
248
- return acc.replace(p.source.text.value, p.text)
249
- }, store.prompt.input)
250
-
251
- const nonTextParts = store.prompt.parts.filter((p) => p.type !== "text")
252
-
253
- const value = text
254
- const content = await Editor.open({ value, renderer })
255
- if (!content) return
256
-
257
- input.setText(content)
258
-
259
- // Update positions for nonTextParts based on their location in new content
260
- // Filter out parts whose virtual text was deleted
261
- // this handles a case where the user edits the text in the editor
262
- // such that the virtual text moves around or is deleted
263
- const updatedNonTextParts = nonTextParts
264
- .map((part) => {
265
- let virtualText = ""
266
- if (part.type === "file" && part.source?.text) {
267
- virtualText = part.source.text.value
268
- } else if (part.type === "agent" && part.source) {
269
- virtualText = part.source.value
270
- }
271
-
272
- if (!virtualText) return part
273
-
274
- const newStart = content.indexOf(virtualText)
275
- // if the virtual text is deleted, remove the part
276
- if (newStart === -1) return null
277
-
278
- const newEnd = newStart + virtualText.length
279
-
280
- if (part.type === "file" && part.source?.text) {
281
- return {
282
- ...part,
283
- source: {
284
- ...part.source,
285
- text: {
286
- ...part.source.text,
287
- start: newStart,
288
- end: newEnd,
289
- },
290
- },
291
- }
292
- }
293
-
294
- if (part.type === "agent" && part.source) {
295
- return {
296
- ...part,
297
- source: {
298
- ...part.source,
299
- start: newStart,
300
- end: newEnd,
301
- },
302
- }
303
- }
304
-
305
- return part
306
- })
307
- .filter((part) => part !== null)
308
-
309
- setStore("prompt", {
310
- input: content,
311
- // keep only the non-text parts because the text parts were
312
- // already expanded inline
313
- parts: updatedNonTextParts,
314
- })
315
- restoreExtmarksFromParts(updatedNonTextParts)
316
- input.cursorOffset = Bun.stringWidth(content)
317
- },
318
- },
319
- {
320
- title: "Skills",
321
- value: "prompt.skills",
322
- category: "Prompt",
323
- slash: {
324
- name: "skills",
325
- },
326
- onSelect: () => {
327
- dialog.replace(() => (
328
- <DialogSkill
329
- onSelect={(skill) => {
330
- input.setText(`/${skill} `)
331
- setStore("prompt", {
332
- input: `/${skill} `,
333
- parts: [],
334
- })
335
- input.gotoBufferEnd()
336
- }}
337
- />
338
- ))
339
- },
340
- },
341
- ]
342
- })
343
-
344
- const ref: PromptRef = {
345
- get focused() {
346
- return input.focused
347
- },
348
- get current() {
349
- return store.prompt
350
- },
351
- focus() {
352
- input.focus()
353
- },
354
- blur() {
355
- input.blur()
356
- },
357
- set(prompt) {
358
- input.setText(prompt.input)
359
- setStore("prompt", prompt)
360
- restoreExtmarksFromParts(prompt.parts)
361
- input.gotoBufferEnd()
362
- },
363
- reset() {
364
- input.clear()
365
- input.extmarks.clear()
366
- setStore("prompt", {
367
- input: "",
368
- parts: [],
369
- })
370
- setStore("extmarkToPartIndex", new Map())
371
- },
372
- submit() {
373
- submit()
374
- },
375
- }
376
-
377
- createEffect(() => {
378
- if (props.visible !== false) input?.focus()
379
- if (props.visible === false) input?.blur()
380
- })
381
-
382
- function restoreExtmarksFromParts(parts: PromptInfo["parts"]) {
383
- input.extmarks.clear()
384
- setStore("extmarkToPartIndex", new Map())
385
-
386
- parts.forEach((part, partIndex) => {
387
- let start = 0
388
- let end = 0
389
- let virtualText = ""
390
- let styleId: number | undefined
391
-
392
- if (part.type === "file" && part.source?.text) {
393
- start = part.source.text.start
394
- end = part.source.text.end
395
- virtualText = part.source.text.value
396
- styleId = fileStyleId
397
- } else if (part.type === "agent" && part.source) {
398
- start = part.source.start
399
- end = part.source.end
400
- virtualText = part.source.value
401
- styleId = agentStyleId
402
- } else if (part.type === "text" && part.source?.text) {
403
- start = part.source.text.start
404
- end = part.source.text.end
405
- virtualText = part.source.text.value
406
- styleId = pasteStyleId
407
- }
408
-
409
- if (virtualText) {
410
- const extmarkId = input.extmarks.create({
411
- start,
412
- end,
413
- virtual: true,
414
- styleId,
415
- typeId: promptPartTypeId,
416
- })
417
- setStore("extmarkToPartIndex", (map: Map<number, number>) => {
418
- const newMap = new Map(map)
419
- newMap.set(extmarkId, partIndex)
420
- return newMap
421
- })
422
- }
423
- })
424
- }
425
-
426
- function syncExtmarksWithPromptParts() {
427
- const allExtmarks = input.extmarks.getAllForTypeId(promptPartTypeId)
428
- setStore(
429
- produce((draft) => {
430
- const newMap = new Map<number, number>()
431
- const newParts: typeof draft.prompt.parts = []
432
-
433
- for (const extmark of allExtmarks) {
434
- const partIndex = draft.extmarkToPartIndex.get(extmark.id)
435
- if (partIndex !== undefined) {
436
- const part = draft.prompt.parts[partIndex]
437
- if (part) {
438
- if (part.type === "agent" && part.source) {
439
- part.source.start = extmark.start
440
- part.source.end = extmark.end
441
- } else if (part.type === "file" && part.source?.text) {
442
- part.source.text.start = extmark.start
443
- part.source.text.end = extmark.end
444
- } else if (part.type === "text" && part.source?.text) {
445
- part.source.text.start = extmark.start
446
- part.source.text.end = extmark.end
447
- }
448
- newMap.set(extmark.id, newParts.length)
449
- newParts.push(part)
450
- }
451
- }
452
- }
453
-
454
- draft.extmarkToPartIndex = newMap
455
- draft.prompt.parts = newParts
456
- }),
457
- )
458
- }
459
-
460
- command.register(() => [
461
- {
462
- title: "Stash prompt",
463
- value: "prompt.stash",
464
- category: "Prompt",
465
- enabled: !!store.prompt.input,
466
- onSelect: (dialog) => {
467
- if (!store.prompt.input) return
468
- stash.push({
469
- input: store.prompt.input,
470
- parts: store.prompt.parts,
471
- })
472
- input.extmarks.clear()
473
- input.clear()
474
- setStore("prompt", { input: "", parts: [] })
475
- setStore("extmarkToPartIndex", new Map())
476
- dialog.clear()
477
- },
478
- },
479
- {
480
- title: "Stash pop",
481
- value: "prompt.stash.pop",
482
- category: "Prompt",
483
- enabled: stash.list().length > 0,
484
- onSelect: (dialog) => {
485
- const entry = stash.pop()
486
- if (entry) {
487
- input.setText(entry.input)
488
- setStore("prompt", { input: entry.input, parts: entry.parts })
489
- restoreExtmarksFromParts(entry.parts)
490
- input.gotoBufferEnd()
491
- }
492
- dialog.clear()
493
- },
494
- },
495
- {
496
- title: "Stash list",
497
- value: "prompt.stash.list",
498
- category: "Prompt",
499
- enabled: stash.list().length > 0,
500
- onSelect: (dialog) => {
501
- dialog.replace(() => (
502
- <DialogStash
503
- onSelect={(entry) => {
504
- input.setText(entry.input)
505
- setStore("prompt", { input: entry.input, parts: entry.parts })
506
- restoreExtmarksFromParts(entry.parts)
507
- input.gotoBufferEnd()
508
- }}
509
- />
510
- ))
511
- },
512
- },
513
- ])
514
-
515
- async function submit() {
516
- if (props.disabled) return
517
- if (autocomplete?.visible) return
518
- if (!store.prompt.input) return
519
- const trimmed = store.prompt.input.trim()
520
- if (trimmed === "exit" || trimmed === "quit" || trimmed === ":q") {
521
- exit()
522
- return
523
- }
524
- const selectedModel = local.model.current()
525
- if (!selectedModel) {
526
- promptModelWarning()
527
- return
528
- }
529
- const sessionID = props.sessionID
530
- ? props.sessionID
531
- : await (async () => {
532
- const sessionID = await sdk.client.session.create({}).then((x) => x.data!.id)
533
- return sessionID
534
- })()
535
- const messageID = Identifier.ascending("message")
536
- let inputText = store.prompt.input
537
-
538
- // Expand pasted text inline before submitting
539
- const allExtmarks = input.extmarks.getAllForTypeId(promptPartTypeId)
540
- const sortedExtmarks = allExtmarks.sort((a: { start: number }, b: { start: number }) => b.start - a.start)
541
-
542
- for (const extmark of sortedExtmarks) {
543
- const partIndex = store.extmarkToPartIndex.get(extmark.id)
544
- if (partIndex !== undefined) {
545
- const part = store.prompt.parts[partIndex]
546
- if (part?.type === "text" && part.text) {
547
- const before = inputText.slice(0, extmark.start)
548
- const after = inputText.slice(extmark.end)
549
- inputText = before + part.text + after
550
- }
551
- }
552
- }
553
-
554
- // Filter out text parts (pasted content) since they're now expanded inline
555
- const nonTextParts = store.prompt.parts.filter((part) => part.type !== "text")
556
-
557
- // Capture mode before it gets reset
558
- const currentMode = store.mode
559
- const variant = local.model.variant.current()
560
-
561
- if (store.mode === "shell") {
562
- sdk.client.session.shell({
563
- sessionID,
564
- agent: local.agent.current().name,
565
- model: {
566
- providerID: selectedModel.providerID,
567
- modelID: selectedModel.modelID,
568
- },
569
- command: inputText,
570
- })
571
- setStore("mode", "normal")
572
- } else if (
573
- inputText.startsWith("/") &&
574
- iife(() => {
575
- const firstLine = inputText.split("\n")[0]
576
- const command = firstLine.split(" ")[0].slice(1)
577
- return sync.data.command.some((x) => x.name === command)
578
- })
579
- ) {
580
- // Parse command from first line, preserve multi-line content in arguments
581
- const firstLineEnd = inputText.indexOf("\n")
582
- const firstLine = firstLineEnd === -1 ? inputText : inputText.slice(0, firstLineEnd)
583
- const [command, ...firstLineArgs] = firstLine.split(" ")
584
- const restOfInput = firstLineEnd === -1 ? "" : inputText.slice(firstLineEnd + 1)
585
- const args = firstLineArgs.join(" ") + (restOfInput ? "\n" + restOfInput : "")
586
-
587
- sdk.client.session.command({
588
- sessionID,
589
- command: command.slice(1),
590
- arguments: args,
591
- agent: local.agent.current().name,
592
- model: `${selectedModel.providerID}/${selectedModel.modelID}`,
593
- messageID,
594
- variant,
595
- parts: nonTextParts
596
- .filter((x) => x.type === "file")
597
- .map((x) => ({
598
- id: Identifier.ascending("part"),
599
- ...x,
600
- })),
601
- })
602
- } else {
603
- sdk.client.session
604
- .prompt({
605
- sessionID,
606
- ...selectedModel,
607
- messageID,
608
- agent: local.agent.current().name,
609
- model: selectedModel,
610
- variant,
611
- parts: [
612
- {
613
- id: Identifier.ascending("part"),
614
- type: "text",
615
- text: inputText,
616
- },
617
- ...nonTextParts.map((x) => ({
618
- id: Identifier.ascending("part"),
619
- ...x,
620
- })),
621
- ],
622
- })
623
- .catch(() => {})
624
- }
625
- history.append({
626
- ...store.prompt,
627
- mode: currentMode,
628
- })
629
- input.extmarks.clear()
630
- setStore("prompt", {
631
- input: "",
632
- parts: [],
633
- })
634
- setStore("extmarkToPartIndex", new Map())
635
- props.onSubmit?.()
636
-
637
- // temporary hack to make sure the message is sent
638
- if (!props.sessionID)
639
- setTimeout(() => {
640
- route.navigate({
641
- type: "session",
642
- sessionID,
643
- })
644
- }, 50)
645
- input.clear()
646
- }
647
- const exit = useExit()
648
-
649
- function pasteText(text: string, virtualText: string) {
650
- const currentOffset = input.visualCursor.offset
651
- const extmarkStart = currentOffset
652
- const extmarkEnd = extmarkStart + virtualText.length
653
-
654
- input.insertText(virtualText + " ")
655
-
656
- const extmarkId = input.extmarks.create({
657
- start: extmarkStart,
658
- end: extmarkEnd,
659
- virtual: true,
660
- styleId: pasteStyleId,
661
- typeId: promptPartTypeId,
662
- })
663
-
664
- setStore(
665
- produce((draft) => {
666
- const partIndex = draft.prompt.parts.length
667
- draft.prompt.parts.push({
668
- type: "text" as const,
669
- text,
670
- source: {
671
- text: {
672
- start: extmarkStart,
673
- end: extmarkEnd,
674
- value: virtualText,
675
- },
676
- },
677
- })
678
- draft.extmarkToPartIndex.set(extmarkId, partIndex)
679
- }),
680
- )
681
- }
682
-
683
- async function pasteImage(file: { filename?: string; content: string; mime: string }) {
684
- const currentOffset = input.visualCursor.offset
685
- const extmarkStart = currentOffset
686
- const count = store.prompt.parts.filter((x) => x.type === "file").length
687
- const virtualText = `[Image ${count + 1}]`
688
- const extmarkEnd = extmarkStart + virtualText.length
689
- const textToInsert = virtualText + " "
690
-
691
- input.insertText(textToInsert)
692
-
693
- const extmarkId = input.extmarks.create({
694
- start: extmarkStart,
695
- end: extmarkEnd,
696
- virtual: true,
697
- styleId: pasteStyleId,
698
- typeId: promptPartTypeId,
699
- })
700
-
701
- const part: Omit<FilePart, "id" | "messageID" | "sessionID"> = {
702
- type: "file" as const,
703
- mime: file.mime,
704
- filename: file.filename,
705
- url: `data:${file.mime};base64,${file.content}`,
706
- source: {
707
- type: "file",
708
- path: file.filename ?? "",
709
- text: {
710
- start: extmarkStart,
711
- end: extmarkEnd,
712
- value: virtualText,
713
- },
714
- },
715
- }
716
- setStore(
717
- produce((draft) => {
718
- const partIndex = draft.prompt.parts.length
719
- draft.prompt.parts.push(part)
720
- draft.extmarkToPartIndex.set(extmarkId, partIndex)
721
- }),
722
- )
723
- return
724
- }
725
-
726
- const highlight = createMemo(() => {
727
- if (keybind.leader) return theme.border
728
- if (store.mode === "shell") return theme.primary
729
- return local.agent.color(local.agent.current().name)
730
- })
731
-
732
- const showVariant = createMemo(() => {
733
- const variants = local.model.variant.list()
734
- if (variants.length === 0) return false
735
- const current = local.model.variant.current()
736
- return !!current
737
- })
738
-
739
- const spinnerDef = createMemo(() => {
740
- const color = local.agent.color(local.agent.current().name)
741
- return {
742
- frames: createFrames({
743
- color,
744
- style: "blocks",
745
- inactiveFactor: 0.6,
746
- // enableFading: false,
747
- minAlpha: 0.3,
748
- }),
749
- color: createColors({
750
- color,
751
- style: "blocks",
752
- inactiveFactor: 0.6,
753
- // enableFading: false,
754
- minAlpha: 0.3,
755
- }),
756
- }
757
- })
758
-
759
- return (
760
- <>
761
- <Autocomplete
762
- sessionID={props.sessionID}
763
- ref={(r) => (autocomplete = r)}
764
- anchor={() => anchor}
765
- input={() => input}
766
- setPrompt={(cb) => {
767
- setStore("prompt", produce(cb))
768
- }}
769
- setExtmark={(partIndex, extmarkId) => {
770
- setStore("extmarkToPartIndex", (map: Map<number, number>) => {
771
- const newMap = new Map(map)
772
- newMap.set(extmarkId, partIndex)
773
- return newMap
774
- })
775
- }}
776
- value={store.prompt.input}
777
- fileStyleId={fileStyleId}
778
- agentStyleId={agentStyleId}
779
- promptPartTypeId={() => promptPartTypeId}
780
- />
781
- <box ref={(r) => (anchor = r)} visible={props.visible !== false}>
782
- <box
783
- border={["left"]}
784
- borderColor={highlight()}
785
- customBorderChars={{
786
- ...EmptyBorder,
787
- vertical: "┃",
788
- bottomLeft: "╹",
789
- }}
790
- >
791
- <box
792
- paddingLeft={2}
793
- paddingRight={2}
794
- paddingTop={1}
795
- flexShrink={0}
796
- backgroundColor={theme.backgroundElement}
797
- flexGrow={1}
798
- >
799
- <textarea
800
- placeholder={props.sessionID ? undefined : `Ask anything... "${PLACEHOLDERS[store.placeholder]}"`}
801
- textColor={keybind.leader ? theme.textMuted : theme.text}
802
- focusedTextColor={keybind.leader ? theme.textMuted : theme.text}
803
- minHeight={1}
804
- maxHeight={6}
805
- onContentChange={() => {
806
- const value = input.plainText
807
- setStore("prompt", "input", value)
808
- autocomplete.onInput(value)
809
- syncExtmarksWithPromptParts()
810
- }}
811
- keyBindings={textareaKeybindings()}
812
- onKeyDown={async (e) => {
813
- if (props.disabled) {
814
- e.preventDefault()
815
- return
816
- }
817
- // Handle clipboard paste (Ctrl+V) - check for images first on Windows
818
- // This is needed because Windows terminal doesn't properly send image data
819
- // through bracketed paste, so we need to intercept the keypress and
820
- // directly read from clipboard before the terminal handles it
821
- if (keybind.match("input_paste", e)) {
822
- const content = await Clipboard.read()
823
- if (content?.mime.startsWith("image/")) {
824
- e.preventDefault()
825
- await pasteImage({
826
- filename: "clipboard",
827
- mime: content.mime,
828
- content: content.data,
829
- })
830
- return
831
- }
832
- // If no image, let the default paste behavior continue
833
- }
834
- if (keybind.match("input_clear", e) && store.prompt.input !== "") {
835
- input.clear()
836
- input.extmarks.clear()
837
- setStore("prompt", {
838
- input: "",
839
- parts: [],
840
- })
841
- setStore("extmarkToPartIndex", new Map())
842
- return
843
- }
844
- if (keybind.match("app_exit", e)) {
845
- if (store.prompt.input === "") {
846
- await exit()
847
- // Don't preventDefault - let textarea potentially handle the event
848
- e.preventDefault()
849
- return
850
- }
851
- }
852
- if (e.name === "!" && input.visualCursor.offset === 0) {
853
- setStore("mode", "shell")
854
- e.preventDefault()
855
- return
856
- }
857
- if (store.mode === "shell") {
858
- if ((e.name === "backspace" && input.visualCursor.offset === 0) || e.name === "escape") {
859
- setStore("mode", "normal")
860
- e.preventDefault()
861
- return
862
- }
863
- }
864
- if (store.mode === "normal") autocomplete.onKeyDown(e)
865
- if (!autocomplete.visible) {
866
- if (
867
- (keybind.match("history_previous", e) && input.cursorOffset === 0) ||
868
- (keybind.match("history_next", e) && input.cursorOffset === input.plainText.length)
869
- ) {
870
- const direction = keybind.match("history_previous", e) ? -1 : 1
871
- const item = history.move(direction, input.plainText)
872
-
873
- if (item) {
874
- input.setText(item.input)
875
- setStore("prompt", item)
876
- setStore("mode", item.mode ?? "normal")
877
- restoreExtmarksFromParts(item.parts)
878
- e.preventDefault()
879
- if (direction === -1) input.cursorOffset = 0
880
- if (direction === 1) input.cursorOffset = input.plainText.length
881
- }
882
- return
883
- }
884
-
885
- if (keybind.match("history_previous", e) && input.visualCursor.visualRow === 0) input.cursorOffset = 0
886
- if (keybind.match("history_next", e) && input.visualCursor.visualRow === input.height - 1)
887
- input.cursorOffset = input.plainText.length
888
- }
889
- }}
890
- onSubmit={submit}
891
- onPaste={async (event: PasteEvent) => {
892
- if (props.disabled) {
893
- event.preventDefault()
894
- return
895
- }
896
-
897
- // Normalize line endings at the boundary
898
- // Windows ConPTY/Terminal often sends CR-only newlines in bracketed paste
899
- // Replace CRLF first, then any remaining CR
900
- const normalizedText = event.text.replace(/\r\n/g, "\n").replace(/\r/g, "\n")
901
- const pastedContent = normalizedText.trim()
902
- if (!pastedContent) {
903
- command.trigger("prompt.paste")
904
- return
905
- }
906
-
907
- // trim ' from the beginning and end of the pasted content. just
908
- // ' and nothing else
909
- const filepath = pastedContent.replace(/^'+|'+$/g, "").replace(/\\ /g, " ")
910
- const isUrl = /^(https?):\/\//.test(filepath)
911
- if (!isUrl) {
912
- try {
913
- const file = Bun.file(filepath)
914
- // Handle SVG as raw text content, not as base64 image
915
- if (file.type === "image/svg+xml") {
916
- event.preventDefault()
917
- const content = await file.text().catch(() => {})
918
- if (content) {
919
- pasteText(content, `[SVG: ${file.name ?? "image"}]`)
920
- return
921
- }
922
- }
923
- if (file.type.startsWith("image/")) {
924
- event.preventDefault()
925
- const content = await file
926
- .arrayBuffer()
927
- .then((buffer) => Buffer.from(buffer).toString("base64"))
928
- .catch(() => {})
929
- if (content) {
930
- await pasteImage({
931
- filename: file.name,
932
- mime: file.type,
933
- content,
934
- })
935
- return
936
- }
937
- }
938
- } catch {}
939
- }
940
-
941
- const lineCount = (pastedContent.match(/\n/g)?.length ?? 0) + 1
942
- if (
943
- (lineCount >= 3 || pastedContent.length > 150) &&
944
- !sync.data.config.experimental?.disable_paste_summary
945
- ) {
946
- event.preventDefault()
947
- pasteText(pastedContent, `[Pasted ~${lineCount} lines]`)
948
- return
949
- }
950
-
951
- // Force layout update and render for the pasted content
952
- setTimeout(() => {
953
- // setTimeout is a workaround and needs to be addressed properly
954
- if (!input || input.isDestroyed) return
955
- input.getLayoutNode().markDirty()
956
- renderer.requestRender()
957
- }, 0)
958
- }}
959
- ref={(r: TextareaRenderable) => {
960
- input = r
961
- if (promptPartTypeId === 0) {
962
- promptPartTypeId = input.extmarks.registerType("prompt-part")
963
- }
964
- props.ref?.(ref)
965
- setTimeout(() => {
966
- // setTimeout is a workaround and needs to be addressed properly
967
- if (!input || input.isDestroyed) return
968
- input.cursorColor = theme.text
969
- }, 0)
970
- }}
971
- onMouseDown={(r: MouseEvent) => r.target?.focus()}
972
- focusedBackgroundColor={theme.backgroundElement}
973
- cursorColor={theme.text}
974
- syntaxStyle={syntax()}
975
- />
976
- <box flexDirection="row" flexShrink={0} paddingTop={1} gap={1}>
977
- <text fg={highlight()}>
978
- {store.mode === "shell" ? "Shell" : Locale.titlecase(local.agent.current().name)}{" "}
979
- </text>
980
- <Show when={store.mode === "normal"}>
981
- <box flexDirection="row" gap={1}>
982
- <text flexShrink={0} fg={keybind.leader ? theme.textMuted : theme.text}>
983
- {local.model.parsed().model}
984
- </text>
985
- <text fg={theme.textMuted}>{local.model.parsed().provider}</text>
986
- <Show when={showVariant()}>
987
- <text fg={theme.textMuted}>·</text>
988
- <text>
989
- <span style={{ fg: theme.warning, bold: true }}>{local.model.variant.current()}</span>
990
- </text>
991
- </Show>
992
- </box>
993
- </Show>
994
- </box>
995
- </box>
996
- </box>
997
- <box
998
- height={1}
999
- border={["left"]}
1000
- borderColor={highlight()}
1001
- customBorderChars={{
1002
- ...EmptyBorder,
1003
- vertical: theme.backgroundElement.a !== 0 ? "╹" : " ",
1004
- }}
1005
- >
1006
- <box
1007
- height={1}
1008
- border={["bottom"]}
1009
- borderColor={theme.backgroundElement}
1010
- customBorderChars={
1011
- theme.backgroundElement.a !== 0
1012
- ? {
1013
- ...EmptyBorder,
1014
- horizontal: "▀",
1015
- }
1016
- : {
1017
- ...EmptyBorder,
1018
- horizontal: " ",
1019
- }
1020
- }
1021
- />
1022
- </box>
1023
- <box flexDirection="row" justifyContent="space-between">
1024
- <Show when={status().type !== "idle"} fallback={<text />}>
1025
- <box
1026
- flexDirection="row"
1027
- gap={1}
1028
- flexGrow={1}
1029
- justifyContent={status().type === "retry" ? "space-between" : "flex-start"}
1030
- >
1031
- <box flexShrink={0} flexDirection="row" gap={1}>
1032
- <box marginLeft={1}>
1033
- <Show when={kv.get("animations_enabled", true)} fallback={<text fg={theme.textMuted}>[⋯]</text>}>
1034
- <spinner color={spinnerDef().color} frames={spinnerDef().frames} interval={40} />
1035
- </Show>
1036
- </box>
1037
- <box flexDirection="row" gap={1} flexShrink={0}>
1038
- {(() => {
1039
- const retry = createMemo(() => {
1040
- const s = status()
1041
- if (s.type !== "retry") return
1042
- return s
1043
- })
1044
- const message = createMemo(() => {
1045
- const r = retry()
1046
- if (!r) return
1047
- if (r.message.includes("exceeded your current quota") && r.message.includes("gemini"))
1048
- return "gemini is way too hot right now"
1049
- if (r.message.length > 80) return r.message.slice(0, 80) + "..."
1050
- return r.message
1051
- })
1052
- const isTruncated = createMemo(() => {
1053
- const r = retry()
1054
- if (!r) return false
1055
- return r.message.length > 120
1056
- })
1057
- const [seconds, setSeconds] = createSignal(0)
1058
- onMount(() => {
1059
- const timer = setInterval(() => {
1060
- const next = retry()?.next
1061
- if (next) setSeconds(Math.round((next - Date.now()) / 1000))
1062
- }, 1000)
1063
-
1064
- onCleanup(() => {
1065
- clearInterval(timer)
1066
- })
1067
- })
1068
- const handleMessageClick = () => {
1069
- const r = retry()
1070
- if (!r) return
1071
- if (isTruncated()) {
1072
- DialogAlert.show(dialog, "Retry Error", r.message)
1073
- }
1074
- }
1075
-
1076
- const retryText = () => {
1077
- const r = retry()
1078
- if (!r) return ""
1079
- const baseMessage = message()
1080
- const truncatedHint = isTruncated() ? " (click to expand)" : ""
1081
- const duration = formatDuration(seconds())
1082
- const retryInfo = ` [retrying ${duration ? `in ${duration} ` : ""}attempt #${r.attempt}]`
1083
- return baseMessage + truncatedHint + retryInfo
1084
- }
1085
-
1086
- return (
1087
- <Show when={retry()}>
1088
- <box onMouseUp={handleMessageClick}>
1089
- <text fg={theme.error}>{retryText()}</text>
1090
- </box>
1091
- </Show>
1092
- )
1093
- })()}
1094
- </box>
1095
- </box>
1096
- <text fg={store.interrupt > 0 ? theme.primary : theme.text}>
1097
- esc{" "}
1098
- <span style={{ fg: store.interrupt > 0 ? theme.primary : theme.textMuted }}>
1099
- {store.interrupt > 0 ? "again to interrupt" : "interrupt"}
1100
- </span>
1101
- </text>
1102
- </box>
1103
- </Show>
1104
- <Show when={status().type !== "retry"}>
1105
- <box gap={2} flexDirection="row">
1106
- <Switch>
1107
- <Match when={store.mode === "normal"}>
1108
- <Show when={local.model.variant.list().length > 0}>
1109
- <text fg={theme.text}>
1110
- {keybind.print("variant_cycle")} <span style={{ fg: theme.textMuted }}>variants</span>
1111
- </text>
1112
- </Show>
1113
- <text fg={theme.text}>
1114
- {keybind.print("agent_cycle")} <span style={{ fg: theme.textMuted }}>agents</span>
1115
- </text>
1116
- <text fg={theme.text}>
1117
- {keybind.print("command_list")} <span style={{ fg: theme.textMuted }}>commands</span>
1118
- </text>
1119
- </Match>
1120
- <Match when={store.mode === "shell"}>
1121
- <text fg={theme.text}>
1122
- esc <span style={{ fg: theme.textMuted }}>exit shell mode</span>
1123
- </text>
1124
- </Match>
1125
- </Switch>
1126
- </box>
1127
- </Show>
1128
- </box>
1129
- </box>
1130
- </>
1131
- )
1132
- }