innocode 1.0.0

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 (434) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/innocode +84 -0
  5. package/bin/opencode +84 -0
  6. package/bunfig.toml +5 -0
  7. package/package.json +126 -0
  8. package/parsers-config.ts +253 -0
  9. package/script/build.ts +198 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish.ts +186 -0
  12. package/script/schema.ts +47 -0
  13. package/script/seed-e2e.ts +50 -0
  14. package/src/acp/README.md +164 -0
  15. package/src/acp/agent.ts +1676 -0
  16. package/src/acp/session.ts +117 -0
  17. package/src/acp/types.ts +23 -0
  18. package/src/agent/agent.ts +338 -0
  19. package/src/agent/generate.txt +75 -0
  20. package/src/agent/prompt/compaction.txt +14 -0
  21. package/src/agent/prompt/explore.txt +18 -0
  22. package/src/agent/prompt/summary.txt +11 -0
  23. package/src/agent/prompt/title.txt +44 -0
  24. package/src/auth/index.ts +70 -0
  25. package/src/bun/index.ts +137 -0
  26. package/src/bun/registry.ts +48 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +70 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +400 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +167 -0
  36. package/src/cli/cmd/debug/config.ts +16 -0
  37. package/src/cli/cmd/debug/file.ts +97 -0
  38. package/src/cli/cmd/debug/index.ts +48 -0
  39. package/src/cli/cmd/debug/lsp.ts +52 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  41. package/src/cli/cmd/debug/scrap.ts +16 -0
  42. package/src/cli/cmd/debug/skill.ts +16 -0
  43. package/src/cli/cmd/debug/snapshot.ts +52 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +1540 -0
  47. package/src/cli/cmd/import.ts +147 -0
  48. package/src/cli/cmd/mcp.ts +765 -0
  49. package/src/cli/cmd/models.ts +77 -0
  50. package/src/cli/cmd/pr.ts +113 -0
  51. package/src/cli/cmd/run.ts +598 -0
  52. package/src/cli/cmd/serve.ts +20 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +426 -0
  55. package/src/cli/cmd/tui/app.tsx +812 -0
  56. package/src/cli/cmd/tui/attach.ts +60 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +243 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  66. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  67. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  68. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  69. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  72. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  73. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  74. package/src/cli/cmd/tui/component/prompt/index.tsx +1153 -0
  75. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  76. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  77. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  78. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  79. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  80. package/src/cli/cmd/tui/context/args.tsx +15 -0
  81. package/src/cli/cmd/tui/context/directory.ts +13 -0
  82. package/src/cli/cmd/tui/context/exit.tsx +54 -0
  83. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  84. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  85. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  86. package/src/cli/cmd/tui/context/local.tsx +409 -0
  87. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  88. package/src/cli/cmd/tui/context/route.tsx +46 -0
  89. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  90. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  91. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  92. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  93. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  97. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  98. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  99. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  100. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  101. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  103. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  104. package/src/cli/cmd/tui/context/theme/innocode.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  108. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  109. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  110. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  111. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  113. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  114. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  115. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  116. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  117. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  118. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  119. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  120. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  121. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  122. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  123. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  124. package/src/cli/cmd/tui/context/theme.tsx +1154 -0
  125. package/src/cli/cmd/tui/event.ts +48 -0
  126. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  131. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  132. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  133. package/src/cli/cmd/tui/routes/session/index.tsx +2139 -0
  134. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  135. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  136. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  137. package/src/cli/cmd/tui/thread.ts +188 -0
  138. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  139. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  140. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  141. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  142. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  143. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  144. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  145. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  146. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  147. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  148. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  149. package/src/cli/cmd/tui/util/editor.ts +32 -0
  150. package/src/cli/cmd/tui/util/signal.ts +7 -0
  151. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  152. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  153. package/src/cli/cmd/tui/win32.ts +129 -0
  154. package/src/cli/cmd/tui/worker.ts +152 -0
  155. package/src/cli/cmd/uninstall.ts +363 -0
  156. package/src/cli/cmd/upgrade.ts +73 -0
  157. package/src/cli/cmd/web.ts +81 -0
  158. package/src/cli/error.ts +57 -0
  159. package/src/cli/logo.ts +6 -0
  160. package/src/cli/network.ts +60 -0
  161. package/src/cli/ui.ts +113 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +150 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +101 -0
  166. package/src/config/config.ts +1517 -0
  167. package/src/config/markdown.ts +98 -0
  168. package/src/env/index.ts +28 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +583 -0
  171. package/src/file/ripgrep.ts +375 -0
  172. package/src/file/time.ts +69 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +148 -0
  175. package/src/format/formatter.ts +366 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +80 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +160 -0
  181. package/src/installation/index.ts +268 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +132 -0
  187. package/src/mcp/index.ts +937 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +280 -0
  194. package/src/plugin/codex.ts +624 -0
  195. package/src/plugin/copilot.ts +327 -0
  196. package/src/plugin/index.ts +138 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +114 -0
  199. package/src/project/project.ts +371 -0
  200. package/src/project/state.ts +70 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/error.ts +189 -0
  204. package/src/provider/models.ts +133 -0
  205. package/src/provider/provider.ts +1370 -0
  206. package/src/provider/sdk/copilot/README.md +5 -0
  207. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  208. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  209. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  211. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  212. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  213. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  214. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  215. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  216. package/src/provider/sdk/copilot/index.ts +2 -0
  217. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  218. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  219. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  220. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  221. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  222. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  223. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  224. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  225. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  226. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  227. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  228. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  229. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  230. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  231. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  232. package/src/provider/transform.ts +806 -0
  233. package/src/pty/index.ts +286 -0
  234. package/src/question/index.ts +171 -0
  235. package/src/scheduler/index.ts +61 -0
  236. package/src/server/error.ts +36 -0
  237. package/src/server/event.ts +7 -0
  238. package/src/server/mdns.ts +60 -0
  239. package/src/server/routes/config.ts +92 -0
  240. package/src/server/routes/experimental.ts +208 -0
  241. package/src/server/routes/file.ts +197 -0
  242. package/src/server/routes/global.ts +183 -0
  243. package/src/server/routes/mcp.ts +225 -0
  244. package/src/server/routes/permission.ts +68 -0
  245. package/src/server/routes/project.ts +82 -0
  246. package/src/server/routes/provider.ts +179 -0
  247. package/src/server/routes/pty.ts +176 -0
  248. package/src/server/routes/question.ts +98 -0
  249. package/src/server/routes/session.ts +939 -0
  250. package/src/server/routes/tui.ts +379 -0
  251. package/src/server/server.ts +621 -0
  252. package/src/session/compaction.ts +261 -0
  253. package/src/session/index.ts +543 -0
  254. package/src/session/instruction.ts +197 -0
  255. package/src/session/llm.ts +283 -0
  256. package/src/session/message-v2.ts +841 -0
  257. package/src/session/message.ts +189 -0
  258. package/src/session/processor.ts +410 -0
  259. package/src/session/prompt/anthropic-20250930.txt +166 -0
  260. package/src/session/prompt/anthropic.txt +105 -0
  261. package/src/session/prompt/beast.txt +147 -0
  262. package/src/session/prompt/build-switch.txt +5 -0
  263. package/src/session/prompt/codex_header.txt +79 -0
  264. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  265. package/src/session/prompt/gemini.txt +155 -0
  266. package/src/session/prompt/max-steps.txt +16 -0
  267. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  268. package/src/session/prompt/plan.txt +26 -0
  269. package/src/session/prompt/qwen.txt +109 -0
  270. package/src/session/prompt/trinity.txt +97 -0
  271. package/src/session/prompt.ts +1964 -0
  272. package/src/session/retry.ts +101 -0
  273. package/src/session/revert.ts +121 -0
  274. package/src/session/status.ts +76 -0
  275. package/src/session/summary.ts +203 -0
  276. package/src/session/system.ts +54 -0
  277. package/src/session/todo.ts +37 -0
  278. package/src/share/share-next.ts +200 -0
  279. package/src/share/share.ts +92 -0
  280. package/src/shell/shell.ts +67 -0
  281. package/src/skill/discovery.ts +97 -0
  282. package/src/skill/index.ts +1 -0
  283. package/src/skill/skill.ts +188 -0
  284. package/src/snapshot/index.ts +255 -0
  285. package/src/storage/storage.ts +227 -0
  286. package/src/tool/apply_patch.ts +281 -0
  287. package/src/tool/apply_patch.txt +33 -0
  288. package/src/tool/bash.ts +269 -0
  289. package/src/tool/bash.txt +115 -0
  290. package/src/tool/batch.ts +175 -0
  291. package/src/tool/batch.txt +24 -0
  292. package/src/tool/codesearch.ts +132 -0
  293. package/src/tool/codesearch.txt +12 -0
  294. package/src/tool/edit.ts +655 -0
  295. package/src/tool/edit.txt +10 -0
  296. package/src/tool/external-directory.ts +32 -0
  297. package/src/tool/glob.ts +80 -0
  298. package/src/tool/glob.txt +6 -0
  299. package/src/tool/grep.ts +150 -0
  300. package/src/tool/grep.txt +8 -0
  301. package/src/tool/invalid.ts +17 -0
  302. package/src/tool/ls.ts +121 -0
  303. package/src/tool/ls.txt +1 -0
  304. package/src/tool/lsp.ts +96 -0
  305. package/src/tool/lsp.txt +19 -0
  306. package/src/tool/multiedit.ts +46 -0
  307. package/src/tool/multiedit.txt +41 -0
  308. package/src/tool/plan-enter.txt +14 -0
  309. package/src/tool/plan-exit.txt +13 -0
  310. package/src/tool/plan.ts +130 -0
  311. package/src/tool/question.ts +33 -0
  312. package/src/tool/question.txt +10 -0
  313. package/src/tool/read.ts +261 -0
  314. package/src/tool/read.txt +14 -0
  315. package/src/tool/registry.ts +160 -0
  316. package/src/tool/skill.ts +123 -0
  317. package/src/tool/task.ts +165 -0
  318. package/src/tool/task.txt +60 -0
  319. package/src/tool/todo.ts +53 -0
  320. package/src/tool/todoread.txt +14 -0
  321. package/src/tool/todowrite.txt +167 -0
  322. package/src/tool/tool.ts +89 -0
  323. package/src/tool/truncation.ts +106 -0
  324. package/src/tool/webfetch.ts +186 -0
  325. package/src/tool/webfetch.txt +13 -0
  326. package/src/tool/websearch.ts +150 -0
  327. package/src/tool/websearch.txt +14 -0
  328. package/src/tool/write.ts +85 -0
  329. package/src/tool/write.txt +8 -0
  330. package/src/util/abort.ts +35 -0
  331. package/src/util/archive.ts +16 -0
  332. package/src/util/color.ts +19 -0
  333. package/src/util/context.ts +25 -0
  334. package/src/util/defer.ts +12 -0
  335. package/src/util/eventloop.ts +20 -0
  336. package/src/util/filesystem.ts +93 -0
  337. package/src/util/fn.ts +11 -0
  338. package/src/util/format.ts +20 -0
  339. package/src/util/iife.ts +3 -0
  340. package/src/util/keybind.ts +103 -0
  341. package/src/util/lazy.ts +18 -0
  342. package/src/util/locale.ts +81 -0
  343. package/src/util/lock.ts +98 -0
  344. package/src/util/log.ts +180 -0
  345. package/src/util/proxied.ts +3 -0
  346. package/src/util/queue.ts +32 -0
  347. package/src/util/rpc.ts +66 -0
  348. package/src/util/scrap.ts +10 -0
  349. package/src/util/signal.ts +12 -0
  350. package/src/util/timeout.ts +14 -0
  351. package/src/util/token.ts +7 -0
  352. package/src/util/wildcard.ts +56 -0
  353. package/src/worktree/index.ts +612 -0
  354. package/sst-env.d.ts +9 -0
  355. package/test/acp/agent-interface.test.ts +51 -0
  356. package/test/acp/event-subscription.test.ts +436 -0
  357. package/test/agent/agent.test.ts +675 -0
  358. package/test/bun.test.ts +53 -0
  359. package/test/cli/github-action.test.ts +161 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/import.test.ts +38 -0
  362. package/test/cli/tui/transcript.test.ts +322 -0
  363. package/test/config/agent-color.test.ts +71 -0
  364. package/test/config/config.test.ts +1802 -0
  365. package/test/config/fixtures/empty-frontmatter.md +4 -0
  366. package/test/config/fixtures/frontmatter.md +28 -0
  367. package/test/config/fixtures/markdown-header.md +11 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/fixtures/weird-model-id.md +13 -0
  370. package/test/config/markdown.test.ts +228 -0
  371. package/test/file/ignore.test.ts +10 -0
  372. package/test/file/path-traversal.test.ts +198 -0
  373. package/test/file/ripgrep.test.ts +39 -0
  374. package/test/fixture/fixture.ts +45 -0
  375. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  376. package/test/ide/ide.test.ts +82 -0
  377. package/test/keybind.test.ts +421 -0
  378. package/test/lsp/client.test.ts +95 -0
  379. package/test/mcp/headers.test.ts +153 -0
  380. package/test/mcp/oauth-browser.test.ts +249 -0
  381. package/test/memory/abort-leak.test.ts +136 -0
  382. package/test/patch/patch.test.ts +348 -0
  383. package/test/permission/arity.test.ts +33 -0
  384. package/test/permission/next.test.ts +690 -0
  385. package/test/permission-task.test.ts +319 -0
  386. package/test/plugin/auth-override.test.ts +44 -0
  387. package/test/plugin/codex.test.ts +123 -0
  388. package/test/preload.ts +63 -0
  389. package/test/project/project.test.ts +120 -0
  390. package/test/provider/amazon-bedrock.test.ts +445 -0
  391. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  392. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  393. package/test/provider/gitlab-duo.test.ts +262 -0
  394. package/test/provider/provider.test.ts +2129 -0
  395. package/test/provider/transform.test.ts +1928 -0
  396. package/test/question/question.test.ts +300 -0
  397. package/test/scheduler.test.ts +73 -0
  398. package/test/server/session-list.test.ts +39 -0
  399. package/test/server/session-select.test.ts +78 -0
  400. package/test/session/compaction.test.ts +423 -0
  401. package/test/session/instruction.test.ts +170 -0
  402. package/test/session/llm.test.ts +667 -0
  403. package/test/session/message-v2.test.ts +924 -0
  404. package/test/session/prompt-missing-file.test.ts +53 -0
  405. package/test/session/prompt-special-chars.test.ts +56 -0
  406. package/test/session/prompt-variant.test.ts +68 -0
  407. package/test/session/retry.test.ts +188 -0
  408. package/test/session/revert-compact.test.ts +285 -0
  409. package/test/session/session.test.ts +71 -0
  410. package/test/session/structured-output-integration.test.ts +233 -0
  411. package/test/session/structured-output.test.ts +385 -0
  412. package/test/skill/discovery.test.ts +60 -0
  413. package/test/skill/skill.test.ts +388 -0
  414. package/test/snapshot/snapshot.test.ts +1040 -0
  415. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  416. package/test/tool/apply_patch.test.ts +559 -0
  417. package/test/tool/bash.test.ts +399 -0
  418. package/test/tool/external-directory.test.ts +127 -0
  419. package/test/tool/fixtures/large-image.png +0 -0
  420. package/test/tool/fixtures/models-api.json +38413 -0
  421. package/test/tool/grep.test.ts +110 -0
  422. package/test/tool/question.test.ts +107 -0
  423. package/test/tool/read.test.ts +421 -0
  424. package/test/tool/registry.test.ts +122 -0
  425. package/test/tool/skill.test.ts +112 -0
  426. package/test/tool/truncation.test.ts +159 -0
  427. package/test/util/filesystem.test.ts +39 -0
  428. package/test/util/format.test.ts +59 -0
  429. package/test/util/iife.test.ts +36 -0
  430. package/test/util/lazy.test.ts +50 -0
  431. package/test/util/lock.test.ts +72 -0
  432. package/test/util/timeout.test.ts +21 -0
  433. package/test/util/wildcard.test.ts +75 -0
  434. package/tsconfig.json +16 -0
@@ -0,0 +1,313 @@
1
+ import { useSync } from "@tui/context/sync"
2
+ import { createMemo, For, Show, Switch, Match } from "solid-js"
3
+ import { createStore } from "solid-js/store"
4
+ import { useTheme } from "../../context/theme"
5
+ import { Locale } from "@/util/locale"
6
+ import path from "path"
7
+ import type { AssistantMessage } from "@opencode-ai/sdk/v2"
8
+ import { Global } from "@/global"
9
+ import { Installation } from "@/installation"
10
+ import { useKeybind } from "../../context/keybind"
11
+ import { useDirectory } from "../../context/directory"
12
+ import { useKV } from "../../context/kv"
13
+ import { TodoItem } from "../../component/todo-item"
14
+
15
+ export function Sidebar(props: { sessionID: string; overlay?: boolean }) {
16
+ const sync = useSync()
17
+ const { theme } = useTheme()
18
+ const session = createMemo(() => sync.session.get(props.sessionID)!)
19
+ const diff = createMemo(() => sync.data.session_diff[props.sessionID] ?? [])
20
+ const todo = createMemo(() => sync.data.todo[props.sessionID] ?? [])
21
+ const messages = createMemo(() => sync.data.message[props.sessionID] ?? [])
22
+
23
+ const [expanded, setExpanded] = createStore({
24
+ mcp: true,
25
+ diff: true,
26
+ todo: true,
27
+ lsp: true,
28
+ })
29
+
30
+ // Sort MCP servers alphabetically for consistent display order
31
+ const mcpEntries = createMemo(() => Object.entries(sync.data.mcp).sort(([a], [b]) => a.localeCompare(b)))
32
+
33
+ // Count connected and error MCP servers for collapsed header display
34
+ const connectedMcpCount = createMemo(() => mcpEntries().filter(([_, item]) => item.status === "connected").length)
35
+ const errorMcpCount = createMemo(
36
+ () =>
37
+ mcpEntries().filter(
38
+ ([_, item]) =>
39
+ item.status === "failed" || item.status === "needs_auth" || item.status === "needs_client_registration",
40
+ ).length,
41
+ )
42
+
43
+ const cost = createMemo(() => {
44
+ const total = messages().reduce((sum, x) => sum + (x.role === "assistant" ? x.cost : 0), 0)
45
+ return new Intl.NumberFormat("en-US", {
46
+ style: "currency",
47
+ currency: "USD",
48
+ }).format(total)
49
+ })
50
+
51
+ const context = createMemo(() => {
52
+ const last = messages().findLast((x) => x.role === "assistant" && x.tokens.output > 0) as AssistantMessage
53
+ if (!last) return
54
+ const total =
55
+ last.tokens.input + last.tokens.output + last.tokens.reasoning + last.tokens.cache.read + last.tokens.cache.write
56
+ const model = sync.data.provider.find((x) => x.id === last.providerID)?.models[last.modelID]
57
+ return {
58
+ tokens: total.toLocaleString(),
59
+ percentage: model?.limit.context ? Math.round((total / model.limit.context) * 100) : null,
60
+ }
61
+ })
62
+
63
+ const directory = useDirectory()
64
+ const kv = useKV()
65
+
66
+ const hasProviders = createMemo(() =>
67
+ sync.data.provider.some((x) => x.id !== "opencode" || Object.values(x.models).some((y) => y.cost?.input !== 0)),
68
+ )
69
+ const gettingStartedDismissed = createMemo(() => kv.get("dismissed_getting_started", false))
70
+
71
+ return (
72
+ <Show when={session()}>
73
+ <box
74
+ backgroundColor={theme.backgroundPanel}
75
+ width={42}
76
+ height="100%"
77
+ paddingTop={1}
78
+ paddingBottom={1}
79
+ paddingLeft={2}
80
+ paddingRight={2}
81
+ position={props.overlay ? "absolute" : "relative"}
82
+ >
83
+ <scrollbox flexGrow={1}>
84
+ <box flexShrink={0} gap={1} paddingRight={1}>
85
+ <box paddingRight={1}>
86
+ <text fg={theme.text}>
87
+ <b>{session().title}</b>
88
+ </text>
89
+ <Show when={session().share?.url}>
90
+ <text fg={theme.textMuted}>{session().share!.url}</text>
91
+ </Show>
92
+ </box>
93
+ <box>
94
+ <text fg={theme.text}>
95
+ <b>Context</b>
96
+ </text>
97
+ <text fg={theme.textMuted}>{context()?.tokens ?? 0} tokens</text>
98
+ <text fg={theme.textMuted}>{context()?.percentage ?? 0}% used</text>
99
+ <text fg={theme.textMuted}>{cost()} spent</text>
100
+ </box>
101
+ <Show when={mcpEntries().length > 0}>
102
+ <box>
103
+ <box
104
+ flexDirection="row"
105
+ gap={1}
106
+ onMouseDown={() => mcpEntries().length > 2 && setExpanded("mcp", !expanded.mcp)}
107
+ >
108
+ <Show when={mcpEntries().length > 2}>
109
+ <text fg={theme.text}>{expanded.mcp ? "▼" : "▶"}</text>
110
+ </Show>
111
+ <text fg={theme.text}>
112
+ <b>MCP</b>
113
+ <Show when={!expanded.mcp}>
114
+ <span style={{ fg: theme.textMuted }}>
115
+ {" "}
116
+ ({connectedMcpCount()} active
117
+ {errorMcpCount() > 0 ? `, ${errorMcpCount()} error${errorMcpCount() > 1 ? "s" : ""}` : ""})
118
+ </span>
119
+ </Show>
120
+ </text>
121
+ </box>
122
+ <Show when={mcpEntries().length <= 2 || expanded.mcp}>
123
+ <For each={mcpEntries()}>
124
+ {([key, item]) => (
125
+ <box flexDirection="row" gap={1}>
126
+ <text
127
+ flexShrink={0}
128
+ style={{
129
+ fg: (
130
+ {
131
+ connected: theme.success,
132
+ failed: theme.error,
133
+ disabled: theme.textMuted,
134
+ needs_auth: theme.warning,
135
+ needs_client_registration: theme.error,
136
+ } as Record<string, typeof theme.success>
137
+ )[item.status],
138
+ }}
139
+ >
140
+
141
+ </text>
142
+ <text fg={theme.text} wrapMode="word">
143
+ {key}{" "}
144
+ <span style={{ fg: theme.textMuted }}>
145
+ <Switch fallback={item.status}>
146
+ <Match when={item.status === "connected"}>Connected</Match>
147
+ <Match when={item.status === "failed" && item}>{(val) => <i>{val().error}</i>}</Match>
148
+ <Match when={item.status === "disabled"}>Disabled</Match>
149
+ <Match when={(item.status as string) === "needs_auth"}>Needs auth</Match>
150
+ <Match when={(item.status as string) === "needs_client_registration"}>
151
+ Needs client ID
152
+ </Match>
153
+ </Switch>
154
+ </span>
155
+ </text>
156
+ </box>
157
+ )}
158
+ </For>
159
+ </Show>
160
+ </box>
161
+ </Show>
162
+ <box>
163
+ <box
164
+ flexDirection="row"
165
+ gap={1}
166
+ onMouseDown={() => sync.data.lsp.length > 2 && setExpanded("lsp", !expanded.lsp)}
167
+ >
168
+ <Show when={sync.data.lsp.length > 2}>
169
+ <text fg={theme.text}>{expanded.lsp ? "▼" : "▶"}</text>
170
+ </Show>
171
+ <text fg={theme.text}>
172
+ <b>LSP</b>
173
+ </text>
174
+ </box>
175
+ <Show when={sync.data.lsp.length <= 2 || expanded.lsp}>
176
+ <Show when={sync.data.lsp.length === 0}>
177
+ <text fg={theme.textMuted}>
178
+ {sync.data.config.lsp === false
179
+ ? "LSPs have been disabled in settings"
180
+ : "LSPs will activate as files are read"}
181
+ </text>
182
+ </Show>
183
+ <For each={sync.data.lsp}>
184
+ {(item) => (
185
+ <box flexDirection="row" gap={1}>
186
+ <text
187
+ flexShrink={0}
188
+ style={{
189
+ fg: {
190
+ connected: theme.success,
191
+ error: theme.error,
192
+ }[item.status],
193
+ }}
194
+ >
195
+
196
+ </text>
197
+ <text fg={theme.textMuted}>
198
+ {item.id} {item.root}
199
+ </text>
200
+ </box>
201
+ )}
202
+ </For>
203
+ </Show>
204
+ </box>
205
+ <Show when={todo().length > 0 && todo().some((t) => t.status !== "completed")}>
206
+ <box>
207
+ <box
208
+ flexDirection="row"
209
+ gap={1}
210
+ onMouseDown={() => todo().length > 2 && setExpanded("todo", !expanded.todo)}
211
+ >
212
+ <Show when={todo().length > 2}>
213
+ <text fg={theme.text}>{expanded.todo ? "▼" : "▶"}</text>
214
+ </Show>
215
+ <text fg={theme.text}>
216
+ <b>Todo</b>
217
+ </text>
218
+ </box>
219
+ <Show when={todo().length <= 2 || expanded.todo}>
220
+ <For each={todo()}>{(todo) => <TodoItem status={todo.status} content={todo.content} />}</For>
221
+ </Show>
222
+ </box>
223
+ </Show>
224
+ <Show when={diff().length > 0}>
225
+ <box>
226
+ <box
227
+ flexDirection="row"
228
+ gap={1}
229
+ onMouseDown={() => diff().length > 2 && setExpanded("diff", !expanded.diff)}
230
+ >
231
+ <Show when={diff().length > 2}>
232
+ <text fg={theme.text}>{expanded.diff ? "▼" : "▶"}</text>
233
+ </Show>
234
+ <text fg={theme.text}>
235
+ <b>Modified Files</b>
236
+ </text>
237
+ </box>
238
+ <Show when={diff().length <= 2 || expanded.diff}>
239
+ <For each={diff() || []}>
240
+ {(item) => {
241
+ return (
242
+ <box flexDirection="row" gap={1} justifyContent="space-between">
243
+ <text fg={theme.textMuted} wrapMode="none">
244
+ {item.file}
245
+ </text>
246
+ <box flexDirection="row" gap={1} flexShrink={0}>
247
+ <Show when={item.additions}>
248
+ <text fg={theme.diffAdded}>+{item.additions}</text>
249
+ </Show>
250
+ <Show when={item.deletions}>
251
+ <text fg={theme.diffRemoved}>-{item.deletions}</text>
252
+ </Show>
253
+ </box>
254
+ </box>
255
+ )
256
+ }}
257
+ </For>
258
+ </Show>
259
+ </box>
260
+ </Show>
261
+ </box>
262
+ </scrollbox>
263
+
264
+ <box flexShrink={0} gap={1} paddingTop={1}>
265
+ <Show when={!hasProviders() && !gettingStartedDismissed()}>
266
+ <box
267
+ backgroundColor={theme.backgroundElement}
268
+ paddingTop={1}
269
+ paddingBottom={1}
270
+ paddingLeft={2}
271
+ paddingRight={2}
272
+ flexDirection="row"
273
+ gap={1}
274
+ >
275
+ <text flexShrink={0} fg={theme.text}>
276
+
277
+ </text>
278
+ <box flexGrow={1} gap={1}>
279
+ <box flexDirection="row" justifyContent="space-between">
280
+ <text fg={theme.text}>
281
+ <b>Getting started</b>
282
+ </text>
283
+ <text fg={theme.textMuted} onMouseDown={() => kv.set("dismissed_getting_started", true)}>
284
+
285
+ </text>
286
+ </box>
287
+ <text fg={theme.textMuted}>InnoCode includes free models so you can start immediately.</text>
288
+ <text fg={theme.textMuted}>
289
+ Connect from 75+ providers to use other models, including Claude, GPT, Gemini etc
290
+ </text>
291
+ <box flexDirection="row" gap={1} justifyContent="space-between">
292
+ <text fg={theme.text}>Connect provider</text>
293
+ <text fg={theme.textMuted}>/connect</text>
294
+ </box>
295
+ </box>
296
+ </box>
297
+ </Show>
298
+ <text>
299
+ <span style={{ fg: theme.textMuted }}>{directory().split("/").slice(0, -1).join("/")}/</span>
300
+ <span style={{ fg: theme.text }}>{directory().split("/").at(-1)}</span>
301
+ </text>
302
+ <text fg={theme.textMuted}>
303
+ <span style={{ fg: theme.success }}>•</span> <b>Open</b>
304
+ <span style={{ fg: theme.text }}>
305
+ <b>Code</b>
306
+ </span>{" "}
307
+ <span>{Installation.VERSION}</span>
308
+ </text>
309
+ </box>
310
+ </box>
311
+ </Show>
312
+ )
313
+ }
@@ -0,0 +1,188 @@
1
+ import { cmd } from "@/cli/cmd/cmd"
2
+ import { tui } from "./app"
3
+ import { Rpc } from "@/util/rpc"
4
+ import { type rpc } from "./worker"
5
+ import path from "path"
6
+ import { UI } from "@/cli/ui"
7
+ import { iife } from "@/util/iife"
8
+ import { Log } from "@/util/log"
9
+ import { withNetworkOptions, resolveNetworkOptions } from "@/cli/network"
10
+ import type { Event } from "@opencode-ai/sdk/v2"
11
+ import type { EventSource } from "./context/sdk"
12
+ import { win32DisableProcessedInput, win32InstallCtrlCGuard } from "./win32"
13
+
14
+ declare global {
15
+ const OPENCODE_WORKER_PATH: string
16
+ }
17
+
18
+ type RpcClient = ReturnType<typeof Rpc.client<typeof rpc>>
19
+
20
+ function createWorkerFetch(client: RpcClient): typeof fetch {
21
+ const fn = async (input: RequestInfo | URL, init?: RequestInit): Promise<Response> => {
22
+ const request = new Request(input, init)
23
+ const body = request.body ? await request.text() : undefined
24
+ const result = await client.call("fetch", {
25
+ url: request.url,
26
+ method: request.method,
27
+ headers: Object.fromEntries(request.headers.entries()),
28
+ body,
29
+ })
30
+ return new Response(result.body, {
31
+ status: result.status,
32
+ headers: result.headers,
33
+ })
34
+ }
35
+ return fn as typeof fetch
36
+ }
37
+
38
+ function createEventSource(client: RpcClient): EventSource {
39
+ return {
40
+ on: (handler) => client.on<Event>("event", handler),
41
+ }
42
+ }
43
+
44
+ export const TuiThreadCommand = cmd({
45
+ command: "$0 [project]",
46
+ describe: "start innocode tui",
47
+ builder: (yargs) =>
48
+ withNetworkOptions(yargs)
49
+ .positional("project", {
50
+ type: "string",
51
+ describe: "path to start innocode in",
52
+ })
53
+ .option("model", {
54
+ type: "string",
55
+ alias: ["m"],
56
+ describe: "model to use in the format of provider/model",
57
+ })
58
+ .option("continue", {
59
+ alias: ["c"],
60
+ describe: "continue the last session",
61
+ type: "boolean",
62
+ })
63
+ .option("session", {
64
+ alias: ["s"],
65
+ type: "string",
66
+ describe: "session id to continue",
67
+ })
68
+ .option("fork", {
69
+ type: "boolean",
70
+ describe: "fork the session when continuing (use with --continue or --session)",
71
+ })
72
+ .option("prompt", {
73
+ type: "string",
74
+ describe: "prompt to use",
75
+ })
76
+ .option("agent", {
77
+ type: "string",
78
+ describe: "agent to use",
79
+ }),
80
+ handler: async (args) => {
81
+ // Keep ENABLE_PROCESSED_INPUT cleared even if other code flips it.
82
+ // (Important when running under `bun run` wrappers on Windows.)
83
+ const unguard = win32InstallCtrlCGuard()
84
+ try {
85
+ // Must be the very first thing — disables CTRL_C_EVENT before any Worker
86
+ // spawn or async work so the OS cannot kill the process group.
87
+ win32DisableProcessedInput()
88
+
89
+ if (args.fork && !args.continue && !args.session) {
90
+ UI.error("--fork requires --continue or --session")
91
+ process.exitCode = 1
92
+ return
93
+ }
94
+
95
+ // Resolve relative paths against PWD to preserve behavior when using --cwd flag
96
+ const baseCwd = process.env.PWD ?? process.cwd()
97
+ const cwd = args.project ? path.resolve(baseCwd, args.project) : process.cwd()
98
+ const localWorker = new URL("./worker.ts", import.meta.url)
99
+ const distWorker = new URL("./cli/cmd/tui/worker.js", import.meta.url)
100
+ const workerPath = await iife(async () => {
101
+ if (typeof OPENCODE_WORKER_PATH !== "undefined") return OPENCODE_WORKER_PATH
102
+ if (await Bun.file(distWorker).exists()) return distWorker
103
+ return localWorker
104
+ })
105
+ try {
106
+ process.chdir(cwd)
107
+ } catch (e) {
108
+ UI.error("Failed to change directory to " + cwd)
109
+ return
110
+ }
111
+
112
+ const worker = new Worker(workerPath, {
113
+ env: Object.fromEntries(
114
+ Object.entries(process.env).filter((entry): entry is [string, string] => entry[1] !== undefined),
115
+ ),
116
+ })
117
+ worker.onerror = (e) => {
118
+ Log.Default.error(e)
119
+ }
120
+ const client = Rpc.client<typeof rpc>(worker)
121
+ process.on("uncaughtException", (e) => {
122
+ Log.Default.error(e)
123
+ })
124
+ process.on("unhandledRejection", (e) => {
125
+ Log.Default.error(e)
126
+ })
127
+ process.on("SIGUSR2", async () => {
128
+ await client.call("reload", undefined)
129
+ })
130
+
131
+ const prompt = await iife(async () => {
132
+ const piped = !process.stdin.isTTY ? await Bun.stdin.text() : undefined
133
+ if (!args.prompt) return piped
134
+ return piped ? piped + "\n" + args.prompt : args.prompt
135
+ })
136
+
137
+ // Check if server should be started (port or hostname explicitly set in CLI or config)
138
+ const networkOpts = await resolveNetworkOptions(args)
139
+ const shouldStartServer =
140
+ process.argv.includes("--port") ||
141
+ process.argv.includes("--hostname") ||
142
+ process.argv.includes("--mdns") ||
143
+ networkOpts.mdns ||
144
+ networkOpts.port !== 0 ||
145
+ networkOpts.hostname !== "127.0.0.1"
146
+
147
+ let url: string
148
+ let customFetch: typeof fetch | undefined
149
+ let events: EventSource | undefined
150
+
151
+ if (shouldStartServer) {
152
+ // Start HTTP server for external access
153
+ const server = await client.call("server", networkOpts)
154
+ url = server.url
155
+ } else {
156
+ // Use direct RPC communication (no HTTP)
157
+ url = "http://opencode.internal"
158
+ customFetch = createWorkerFetch(client)
159
+ events = createEventSource(client)
160
+ }
161
+
162
+ const tuiPromise = tui({
163
+ url,
164
+ fetch: customFetch,
165
+ events,
166
+ args: {
167
+ continue: args.continue,
168
+ sessionID: args.session,
169
+ agent: args.agent,
170
+ model: args.model,
171
+ prompt,
172
+ fork: args.fork,
173
+ },
174
+ onExit: async () => {
175
+ await client.call("shutdown", undefined)
176
+ },
177
+ })
178
+
179
+ setTimeout(() => {
180
+ client.call("checkUpgrade", { directory: cwd }).catch(() => {})
181
+ }, 1000)
182
+
183
+ await tuiPromise
184
+ } finally {
185
+ unguard?.()
186
+ }
187
+ },
188
+ })
@@ -0,0 +1,59 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "../context/theme"
3
+ import { useDialog, type DialogContext } from "./dialog"
4
+ import { useKeyboard } from "@opentui/solid"
5
+
6
+ export type DialogAlertProps = {
7
+ title: string
8
+ message: string
9
+ onConfirm?: () => void
10
+ }
11
+
12
+ export function DialogAlert(props: DialogAlertProps) {
13
+ const dialog = useDialog()
14
+ const { theme } = useTheme()
15
+
16
+ useKeyboard((evt) => {
17
+ if (evt.name === "return") {
18
+ props.onConfirm?.()
19
+ dialog.clear()
20
+ }
21
+ })
22
+ return (
23
+ <box paddingLeft={2} paddingRight={2} gap={1}>
24
+ <box flexDirection="row" justifyContent="space-between">
25
+ <text attributes={TextAttributes.BOLD} fg={theme.text}>
26
+ {props.title}
27
+ </text>
28
+ <text fg={theme.textMuted} onMouseUp={() => dialog.clear()}>
29
+ esc
30
+ </text>
31
+ </box>
32
+ <box paddingBottom={1}>
33
+ <text fg={theme.textMuted}>{props.message}</text>
34
+ </box>
35
+ <box flexDirection="row" justifyContent="flex-end" paddingBottom={1}>
36
+ <box
37
+ paddingLeft={3}
38
+ paddingRight={3}
39
+ backgroundColor={theme.primary}
40
+ onMouseUp={() => {
41
+ props.onConfirm?.()
42
+ dialog.clear()
43
+ }}
44
+ >
45
+ <text fg={theme.selectedListItemText}>ok</text>
46
+ </box>
47
+ </box>
48
+ </box>
49
+ )
50
+ }
51
+
52
+ DialogAlert.show = (dialog: DialogContext, title: string, message: string) => {
53
+ return new Promise<void>((resolve) => {
54
+ dialog.replace(
55
+ () => <DialogAlert title={title} message={message} onConfirm={() => resolve()} />,
56
+ () => resolve(),
57
+ )
58
+ })
59
+ }
@@ -0,0 +1,85 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { useTheme } from "../context/theme"
3
+ import { useDialog, type DialogContext } from "./dialog"
4
+ import { createStore } from "solid-js/store"
5
+ import { For } from "solid-js"
6
+ import { useKeyboard } from "@opentui/solid"
7
+ import { Locale } from "@/util/locale"
8
+
9
+ export type DialogConfirmProps = {
10
+ title: string
11
+ message: string
12
+ onConfirm?: () => void
13
+ onCancel?: () => void
14
+ }
15
+
16
+ export function DialogConfirm(props: DialogConfirmProps) {
17
+ const dialog = useDialog()
18
+ const { theme } = useTheme()
19
+ const [store, setStore] = createStore({
20
+ active: "confirm" as "confirm" | "cancel",
21
+ })
22
+
23
+ useKeyboard((evt) => {
24
+ if (evt.name === "return") {
25
+ if (store.active === "confirm") props.onConfirm?.()
26
+ if (store.active === "cancel") props.onCancel?.()
27
+ dialog.clear()
28
+ }
29
+
30
+ if (evt.name === "left" || evt.name === "right") {
31
+ setStore("active", store.active === "confirm" ? "cancel" : "confirm")
32
+ }
33
+ })
34
+ return (
35
+ <box paddingLeft={2} paddingRight={2} gap={1}>
36
+ <box flexDirection="row" justifyContent="space-between">
37
+ <text attributes={TextAttributes.BOLD} fg={theme.text}>
38
+ {props.title}
39
+ </text>
40
+ <text fg={theme.textMuted} onMouseUp={() => dialog.clear()}>
41
+ esc
42
+ </text>
43
+ </box>
44
+ <box paddingBottom={1}>
45
+ <text fg={theme.textMuted}>{props.message}</text>
46
+ </box>
47
+ <box flexDirection="row" justifyContent="flex-end" paddingBottom={1}>
48
+ <For each={["cancel", "confirm"]}>
49
+ {(key) => (
50
+ <box
51
+ paddingLeft={1}
52
+ paddingRight={1}
53
+ backgroundColor={key === store.active ? theme.primary : undefined}
54
+ onMouseUp={(evt) => {
55
+ if (key === "confirm") props.onConfirm?.()
56
+ if (key === "cancel") props.onCancel?.()
57
+ dialog.clear()
58
+ }}
59
+ >
60
+ <text fg={key === store.active ? theme.selectedListItemText : theme.textMuted}>
61
+ {Locale.titlecase(key)}
62
+ </text>
63
+ </box>
64
+ )}
65
+ </For>
66
+ </box>
67
+ </box>
68
+ )
69
+ }
70
+
71
+ DialogConfirm.show = (dialog: DialogContext, title: string, message: string) => {
72
+ return new Promise<boolean>((resolve) => {
73
+ dialog.replace(
74
+ () => (
75
+ <DialogConfirm
76
+ title={title}
77
+ message={message}
78
+ onConfirm={() => resolve(true)}
79
+ onCancel={() => resolve(false)}
80
+ />
81
+ ),
82
+ () => resolve(false),
83
+ )
84
+ })
85
+ }