opencode-v2 1.1.53

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 (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,666 @@
1
+ import type { BoxRenderable, TextareaRenderable, KeyEvent, ScrollBoxRenderable } from "@opentui/core"
2
+ import { pathToFileURL } from "bun"
3
+ import fuzzysort from "fuzzysort"
4
+ import { firstBy } from "remeda"
5
+ import { createMemo, createResource, createEffect, onMount, onCleanup, Index, Show, createSignal } from "solid-js"
6
+ import { createStore } from "solid-js/store"
7
+ import { useSDK } from "@tui/context/sdk"
8
+ import { useSync } from "@tui/context/sync"
9
+ import { useTheme, selectedForeground } from "@tui/context/theme"
10
+ import { SplitBorder } from "@tui/component/border"
11
+ import { useCommandDialog } from "@tui/component/dialog-command"
12
+ import { useTerminalDimensions } from "@opentui/solid"
13
+ import { Locale } from "@/util/locale"
14
+ import type { PromptInfo } from "./history"
15
+ import { useFrecency } from "./frecency"
16
+
17
+ function removeLineRange(input: string) {
18
+ const hashIndex = input.lastIndexOf("#")
19
+ return hashIndex !== -1 ? input.substring(0, hashIndex) : input
20
+ }
21
+
22
+ function extractLineRange(input: string) {
23
+ const hashIndex = input.lastIndexOf("#")
24
+ if (hashIndex === -1) {
25
+ return { baseQuery: input }
26
+ }
27
+
28
+ const baseName = input.substring(0, hashIndex)
29
+ const linePart = input.substring(hashIndex + 1)
30
+ const lineMatch = linePart.match(/^(\d+)(?:-(\d*))?$/)
31
+
32
+ if (!lineMatch) {
33
+ return { baseQuery: baseName }
34
+ }
35
+
36
+ const startLine = Number(lineMatch[1])
37
+ const endLine = lineMatch[2] && startLine < Number(lineMatch[2]) ? Number(lineMatch[2]) : undefined
38
+
39
+ return {
40
+ lineRange: {
41
+ baseName,
42
+ startLine,
43
+ endLine,
44
+ },
45
+ baseQuery: baseName,
46
+ }
47
+ }
48
+
49
+ export type AutocompleteRef = {
50
+ onInput: (value: string) => void
51
+ onKeyDown: (e: KeyEvent) => void
52
+ visible: false | "@" | "/"
53
+ }
54
+
55
+ export type AutocompleteOption = {
56
+ display: string
57
+ value?: string
58
+ aliases?: string[]
59
+ disabled?: boolean
60
+ description?: string
61
+ isDirectory?: boolean
62
+ onSelect?: () => void
63
+ path?: string
64
+ }
65
+
66
+ export function Autocomplete(props: {
67
+ value: string
68
+ sessionID?: string
69
+ setPrompt: (input: (prompt: PromptInfo) => void) => void
70
+ setExtmark: (partIndex: number, extmarkId: number) => void
71
+ anchor: () => BoxRenderable
72
+ input: () => TextareaRenderable
73
+ ref: (ref: AutocompleteRef) => void
74
+ fileStyleId: number
75
+ agentStyleId: number
76
+ promptPartTypeId: () => number
77
+ }) {
78
+ const sdk = useSDK()
79
+ const sync = useSync()
80
+ const command = useCommandDialog()
81
+ const { theme } = useTheme()
82
+ const dimensions = useTerminalDimensions()
83
+ const frecency = useFrecency()
84
+
85
+ const [store, setStore] = createStore({
86
+ index: 0,
87
+ selected: 0,
88
+ visible: false as AutocompleteRef["visible"],
89
+ input: "keyboard" as "keyboard" | "mouse",
90
+ })
91
+
92
+ const [positionTick, setPositionTick] = createSignal(0)
93
+
94
+ createEffect(() => {
95
+ if (store.visible) {
96
+ let lastPos = { x: 0, y: 0, width: 0 }
97
+ const interval = setInterval(() => {
98
+ const anchor = props.anchor()
99
+ if (anchor.x !== lastPos.x || anchor.y !== lastPos.y || anchor.width !== lastPos.width) {
100
+ lastPos = { x: anchor.x, y: anchor.y, width: anchor.width }
101
+ setPositionTick((t) => t + 1)
102
+ }
103
+ }, 50)
104
+
105
+ onCleanup(() => clearInterval(interval))
106
+ }
107
+ })
108
+
109
+ const position = createMemo(() => {
110
+ if (!store.visible) return { x: 0, y: 0, width: 0 }
111
+ const dims = dimensions()
112
+ positionTick()
113
+ const anchor = props.anchor()
114
+ const parent = anchor.parent
115
+ const parentX = parent?.x ?? 0
116
+ const parentY = parent?.y ?? 0
117
+
118
+ return {
119
+ x: anchor.x - parentX,
120
+ y: anchor.y - parentY,
121
+ width: anchor.width,
122
+ }
123
+ })
124
+
125
+ const filter = createMemo(() => {
126
+ if (!store.visible) return
127
+ // Track props.value to make memo reactive to text changes
128
+ props.value // <- there surely is a better way to do this, like making .input() reactive
129
+
130
+ return props.input().getTextRange(store.index + 1, props.input().cursorOffset)
131
+ })
132
+
133
+ // filter() reads reactive props.value plus non-reactive cursor/text state.
134
+ // On keypress those can be briefly out of sync, so filter() may return an empty/partial string.
135
+ // Copy it into search in an effect because effects run after reactive updates have been rendered and painted
136
+ // so the input has settled and all consumers read the same stable value.
137
+ const [search, setSearch] = createSignal("")
138
+ createEffect(() => {
139
+ const next = filter()
140
+ setSearch(next ? next : "")
141
+ })
142
+
143
+ // When the filter changes due to how TUI works, the mousemove might still be triggered
144
+ // via a synthetic event as the layout moves underneath the cursor. This is a workaround to make sure the input mode remains keyboard so
145
+ // that the mouseover event doesn't trigger when filtering.
146
+ createEffect(() => {
147
+ filter()
148
+ setStore("input", "keyboard")
149
+ })
150
+
151
+ function insertPart(text: string, part: PromptInfo["parts"][number]) {
152
+ const input = props.input()
153
+ const currentCursorOffset = input.cursorOffset
154
+
155
+ const charAfterCursor = props.value.at(currentCursorOffset)
156
+ const needsSpace = charAfterCursor !== " "
157
+ const append = "@" + text + (needsSpace ? " " : "")
158
+
159
+ input.cursorOffset = store.index
160
+ const startCursor = input.logicalCursor
161
+ input.cursorOffset = currentCursorOffset
162
+ const endCursor = input.logicalCursor
163
+
164
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
165
+ input.insertText(append)
166
+
167
+ const virtualText = "@" + text
168
+ const extmarkStart = store.index
169
+ const extmarkEnd = extmarkStart + Bun.stringWidth(virtualText)
170
+
171
+ const styleId = part.type === "file" ? props.fileStyleId : part.type === "agent" ? props.agentStyleId : undefined
172
+
173
+ const extmarkId = input.extmarks.create({
174
+ start: extmarkStart,
175
+ end: extmarkEnd,
176
+ virtual: true,
177
+ styleId,
178
+ typeId: props.promptPartTypeId(),
179
+ })
180
+
181
+ props.setPrompt((draft) => {
182
+ if (part.type === "file") {
183
+ const existingIndex = draft.parts.findIndex((p) => p.type === "file" && "url" in p && p.url === part.url)
184
+ if (existingIndex !== -1) {
185
+ const existing = draft.parts[existingIndex]
186
+ if (
187
+ part.source?.text &&
188
+ existing &&
189
+ "source" in existing &&
190
+ existing.source &&
191
+ "text" in existing.source &&
192
+ existing.source.text
193
+ ) {
194
+ existing.source.text.start = extmarkStart
195
+ existing.source.text.end = extmarkEnd
196
+ existing.source.text.value = virtualText
197
+ }
198
+ return
199
+ }
200
+ }
201
+
202
+ if (part.type === "file" && part.source?.text) {
203
+ part.source.text.start = extmarkStart
204
+ part.source.text.end = extmarkEnd
205
+ part.source.text.value = virtualText
206
+ } else if (part.type === "agent" && part.source) {
207
+ part.source.start = extmarkStart
208
+ part.source.end = extmarkEnd
209
+ part.source.value = virtualText
210
+ }
211
+ const partIndex = draft.parts.length
212
+ draft.parts.push(part)
213
+ props.setExtmark(partIndex, extmarkId)
214
+ })
215
+
216
+ if (part.type === "file" && part.source && part.source.type === "file") {
217
+ frecency.updateFrecency(part.source.path)
218
+ }
219
+ }
220
+
221
+ const [files] = createResource(
222
+ () => search(),
223
+ async (query) => {
224
+ if (!store.visible || store.visible === "/") return []
225
+
226
+ const { lineRange, baseQuery } = extractLineRange(query ?? "")
227
+
228
+ // Get files from SDK
229
+ const result = await sdk.client.find.files({
230
+ query: baseQuery,
231
+ })
232
+
233
+ const options: AutocompleteOption[] = []
234
+
235
+ // Add file options
236
+ if (!result.error && result.data) {
237
+ const sortedFiles = result.data.sort((a, b) => {
238
+ const aScore = frecency.getFrecency(a)
239
+ const bScore = frecency.getFrecency(b)
240
+ if (aScore !== bScore) return bScore - aScore
241
+ const aDepth = a.split("/").length
242
+ const bDepth = b.split("/").length
243
+ if (aDepth !== bDepth) return aDepth - bDepth
244
+ return a.localeCompare(b)
245
+ })
246
+
247
+ const width = props.anchor().width - 4
248
+ options.push(
249
+ ...sortedFiles.map((item): AutocompleteOption => {
250
+ const fullPath = `${process.cwd()}/${item}`
251
+ const urlObj = pathToFileURL(fullPath)
252
+ let filename = item
253
+ if (lineRange && !item.endsWith("/")) {
254
+ filename = `${item}#${lineRange.startLine}${lineRange.endLine ? `-${lineRange.endLine}` : ""}`
255
+ urlObj.searchParams.set("start", String(lineRange.startLine))
256
+ if (lineRange.endLine !== undefined) {
257
+ urlObj.searchParams.set("end", String(lineRange.endLine))
258
+ }
259
+ }
260
+ const url = urlObj.href
261
+
262
+ const isDir = item.endsWith("/")
263
+ return {
264
+ display: Locale.truncateMiddle(filename, width),
265
+ value: filename,
266
+ isDirectory: isDir,
267
+ path: item,
268
+ onSelect: () => {
269
+ insertPart(filename, {
270
+ type: "file",
271
+ mime: "text/plain",
272
+ filename,
273
+ url,
274
+ source: {
275
+ type: "file",
276
+ text: {
277
+ start: 0,
278
+ end: 0,
279
+ value: "",
280
+ },
281
+ path: item,
282
+ },
283
+ })
284
+ },
285
+ }
286
+ }),
287
+ )
288
+ }
289
+
290
+ return options
291
+ },
292
+ {
293
+ initialValue: [],
294
+ },
295
+ )
296
+
297
+ const mcpResources = createMemo(() => {
298
+ if (!store.visible || store.visible === "/") return []
299
+
300
+ const options: AutocompleteOption[] = []
301
+ const width = props.anchor().width - 4
302
+
303
+ for (const res of Object.values(sync.data.mcp_resource)) {
304
+ const text = `${res.name} (${res.uri})`
305
+ options.push({
306
+ display: Locale.truncateMiddle(text, width),
307
+ value: text,
308
+ description: res.description,
309
+ onSelect: () => {
310
+ insertPart(res.name, {
311
+ type: "file",
312
+ mime: res.mimeType ?? "text/plain",
313
+ filename: res.name,
314
+ url: res.uri,
315
+ source: {
316
+ type: "resource",
317
+ text: {
318
+ start: 0,
319
+ end: 0,
320
+ value: "",
321
+ },
322
+ clientName: res.client,
323
+ uri: res.uri,
324
+ },
325
+ })
326
+ },
327
+ })
328
+ }
329
+
330
+ return options
331
+ })
332
+
333
+ const agents = createMemo(() => {
334
+ const agents = sync.data.agent
335
+ return agents
336
+ .filter((agent) => !agent.hidden && agent.mode !== "primary")
337
+ .map(
338
+ (agent): AutocompleteOption => ({
339
+ display: "@" + agent.name,
340
+ onSelect: () => {
341
+ insertPart(agent.name, {
342
+ type: "agent",
343
+ name: agent.name,
344
+ source: {
345
+ start: 0,
346
+ end: 0,
347
+ value: "",
348
+ },
349
+ })
350
+ },
351
+ }),
352
+ )
353
+ })
354
+
355
+ const commands = createMemo((): AutocompleteOption[] => {
356
+ const results: AutocompleteOption[] = [...command.slashes()]
357
+
358
+ for (const serverCommand of sync.data.command) {
359
+ if (serverCommand.source === "skill") continue
360
+ const label = serverCommand.source === "mcp" ? ":mcp" : ""
361
+ results.push({
362
+ display: "/" + serverCommand.name + label,
363
+ description: serverCommand.description,
364
+ onSelect: () => {
365
+ const newText = "/" + serverCommand.name + " "
366
+ const cursor = props.input().logicalCursor
367
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
368
+ props.input().insertText(newText)
369
+ props.input().cursorOffset = Bun.stringWidth(newText)
370
+ },
371
+ })
372
+ }
373
+
374
+ results.sort((a, b) => a.display.localeCompare(b.display))
375
+
376
+ const max = firstBy(results, [(x) => x.display.length, "desc"])?.display.length
377
+ if (!max) return results
378
+ return results.map((item) => ({
379
+ ...item,
380
+ display: item.display.padEnd(max + 2),
381
+ }))
382
+ })
383
+
384
+ const options = createMemo((prev: AutocompleteOption[] | undefined) => {
385
+ const filesValue = files()
386
+ const agentsValue = agents()
387
+ const commandsValue = commands()
388
+
389
+ const mixed: AutocompleteOption[] =
390
+ store.visible === "@" ? [...agentsValue, ...(filesValue || []), ...mcpResources()] : [...commandsValue]
391
+
392
+ const searchValue = search()
393
+
394
+ if (!searchValue) {
395
+ return mixed
396
+ }
397
+
398
+ if (files.loading && prev && prev.length > 0) {
399
+ return prev
400
+ }
401
+
402
+ const result = fuzzysort.go(removeLineRange(searchValue), mixed, {
403
+ keys: [
404
+ (obj) => removeLineRange((obj.value ?? obj.display).trimEnd()),
405
+ "description",
406
+ (obj) => obj.aliases?.join(" ") ?? "",
407
+ ],
408
+ limit: 10,
409
+ scoreFn: (objResults) => {
410
+ const displayResult = objResults[0]
411
+ let score = objResults.score
412
+ if (displayResult && displayResult.target.startsWith(store.visible + searchValue)) {
413
+ score *= 2
414
+ }
415
+ const frecencyScore = objResults.obj.path ? frecency.getFrecency(objResults.obj.path) : 0
416
+ return score * (1 + frecencyScore)
417
+ },
418
+ })
419
+
420
+ return result.map((arr) => arr.obj)
421
+ })
422
+
423
+ createEffect(() => {
424
+ filter()
425
+ setStore("selected", 0)
426
+ })
427
+
428
+ function move(direction: -1 | 1) {
429
+ if (!store.visible) return
430
+ if (!options().length) return
431
+ let next = store.selected + direction
432
+ if (next < 0) next = options().length - 1
433
+ if (next >= options().length) next = 0
434
+ moveTo(next)
435
+ }
436
+
437
+ function moveTo(next: number) {
438
+ setStore("selected", next)
439
+ if (!scroll) return
440
+ const viewportHeight = Math.min(height(), options().length)
441
+ const scrollBottom = scroll.scrollTop + viewportHeight
442
+ if (next < scroll.scrollTop) {
443
+ scroll.scrollBy(next - scroll.scrollTop)
444
+ } else if (next + 1 > scrollBottom) {
445
+ scroll.scrollBy(next + 1 - scrollBottom)
446
+ }
447
+ }
448
+
449
+ function select() {
450
+ const selected = options()[store.selected]
451
+ if (!selected) return
452
+ hide()
453
+ selected.onSelect?.()
454
+ }
455
+
456
+ function expandDirectory() {
457
+ const selected = options()[store.selected]
458
+ if (!selected) return
459
+
460
+ const input = props.input()
461
+ const currentCursorOffset = input.cursorOffset
462
+
463
+ const displayText = selected.display.trimEnd()
464
+ const path = displayText.startsWith("@") ? displayText.slice(1) : displayText
465
+
466
+ input.cursorOffset = store.index
467
+ const startCursor = input.logicalCursor
468
+ input.cursorOffset = currentCursorOffset
469
+ const endCursor = input.logicalCursor
470
+
471
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
472
+ input.insertText("@" + path)
473
+
474
+ setStore("selected", 0)
475
+ }
476
+
477
+ function show(mode: "@" | "/") {
478
+ command.keybinds(false)
479
+ setStore({
480
+ visible: mode,
481
+ index: props.input().cursorOffset,
482
+ })
483
+ }
484
+
485
+ function hide() {
486
+ const text = props.input().plainText
487
+ if (store.visible === "/" && !text.endsWith(" ") && text.startsWith("/")) {
488
+ const cursor = props.input().logicalCursor
489
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
490
+ // Sync the prompt store immediately since onContentChange is async
491
+ props.setPrompt((draft) => {
492
+ draft.input = props.input().plainText
493
+ })
494
+ }
495
+ command.keybinds(true)
496
+ setStore("visible", false)
497
+ }
498
+
499
+ onMount(() => {
500
+ props.ref({
501
+ get visible() {
502
+ return store.visible
503
+ },
504
+ onInput(value) {
505
+ if (store.visible) {
506
+ if (
507
+ // Typed text before the trigger
508
+ props.input().cursorOffset <= store.index ||
509
+ // There is a space between the trigger and the cursor
510
+ props.input().getTextRange(store.index, props.input().cursorOffset).match(/\s/) ||
511
+ // "/<command>" is not the sole content
512
+ (store.visible === "/" && value.match(/^\S+\s+\S+\s*$/))
513
+ ) {
514
+ hide()
515
+ }
516
+ return
517
+ }
518
+
519
+ // Check if autocomplete should reopen (e.g., after backspace deleted a space)
520
+ const offset = props.input().cursorOffset
521
+ if (offset === 0) return
522
+
523
+ // Check for "/" at position 0 - reopen slash commands
524
+ if (value.startsWith("/") && !value.slice(0, offset).match(/\s/)) {
525
+ show("/")
526
+ setStore("index", 0)
527
+ return
528
+ }
529
+
530
+ // Check for "@" trigger - find the nearest "@" before cursor with no whitespace between
531
+ const text = value.slice(0, offset)
532
+ const idx = text.lastIndexOf("@")
533
+ if (idx === -1) return
534
+
535
+ const between = text.slice(idx)
536
+ const before = idx === 0 ? undefined : value[idx - 1]
537
+ if ((before === undefined || /\s/.test(before)) && !between.match(/\s/)) {
538
+ show("@")
539
+ setStore("index", idx)
540
+ }
541
+ },
542
+ onKeyDown(e: KeyEvent) {
543
+ if (store.visible) {
544
+ const name = e.name?.toLowerCase()
545
+ const ctrlOnly = e.ctrl && !e.meta && !e.shift
546
+ const isNavUp = name === "up" || (ctrlOnly && name === "p")
547
+ const isNavDown = name === "down" || (ctrlOnly && name === "n")
548
+
549
+ if (isNavUp) {
550
+ setStore("input", "keyboard")
551
+ move(-1)
552
+ e.preventDefault()
553
+ return
554
+ }
555
+ if (isNavDown) {
556
+ setStore("input", "keyboard")
557
+ move(1)
558
+ e.preventDefault()
559
+ return
560
+ }
561
+ if (name === "escape") {
562
+ hide()
563
+ e.preventDefault()
564
+ return
565
+ }
566
+ if (name === "return") {
567
+ select()
568
+ e.preventDefault()
569
+ return
570
+ }
571
+ if (name === "tab") {
572
+ const selected = options()[store.selected]
573
+ if (selected?.isDirectory) {
574
+ expandDirectory()
575
+ } else {
576
+ select()
577
+ }
578
+ e.preventDefault()
579
+ return
580
+ }
581
+ }
582
+ if (!store.visible) {
583
+ if (e.name === "@") {
584
+ const cursorOffset = props.input().cursorOffset
585
+ const charBeforeCursor =
586
+ cursorOffset === 0 ? undefined : props.input().getTextRange(cursorOffset - 1, cursorOffset)
587
+ const canTrigger = charBeforeCursor === undefined || charBeforeCursor === "" || /\s/.test(charBeforeCursor)
588
+ if (canTrigger) show("@")
589
+ }
590
+
591
+ if (e.name === "/") {
592
+ if (props.input().cursorOffset === 0) show("/")
593
+ }
594
+ }
595
+ },
596
+ })
597
+ })
598
+
599
+ const height = createMemo(() => {
600
+ const count = options().length || 1
601
+ if (!store.visible) return Math.min(10, count)
602
+ positionTick()
603
+ return Math.min(10, count, Math.max(1, props.anchor().y))
604
+ })
605
+
606
+ let scroll: ScrollBoxRenderable
607
+
608
+ return (
609
+ <box
610
+ visible={store.visible !== false}
611
+ position="absolute"
612
+ top={position().y - height()}
613
+ left={position().x}
614
+ width={position().width}
615
+ zIndex={100}
616
+ {...SplitBorder}
617
+ borderColor={theme.border}
618
+ >
619
+ <scrollbox
620
+ ref={(r: ScrollBoxRenderable) => (scroll = r)}
621
+ backgroundColor={theme.backgroundMenu}
622
+ height={height()}
623
+ scrollbarOptions={{ visible: false }}
624
+ >
625
+ <Index
626
+ each={options()}
627
+ fallback={
628
+ <box paddingLeft={1} paddingRight={1}>
629
+ <text fg={theme.textMuted}>No matching items</text>
630
+ </box>
631
+ }
632
+ >
633
+ {(option, index) => (
634
+ <box
635
+ paddingLeft={1}
636
+ paddingRight={1}
637
+ backgroundColor={index === store.selected ? theme.primary : undefined}
638
+ flexDirection="row"
639
+ onMouseMove={() => {
640
+ setStore("input", "mouse")
641
+ }}
642
+ onMouseOver={() => {
643
+ if (store.input !== "mouse") return
644
+ moveTo(index)
645
+ }}
646
+ onMouseDown={() => {
647
+ setStore("input", "mouse")
648
+ moveTo(index)
649
+ }}
650
+ onMouseUp={() => select()}
651
+ >
652
+ <text fg={index === store.selected ? selectedForeground(theme) : theme.text} flexShrink={0}>
653
+ {option().display}
654
+ </text>
655
+ <Show when={option().description}>
656
+ <text fg={index === store.selected ? selectedForeground(theme) : theme.textMuted} wrapMode="none">
657
+ {option().description}
658
+ </text>
659
+ </Show>
660
+ </box>
661
+ )}
662
+ </Index>
663
+ </scrollbox>
664
+ </box>
665
+ )
666
+ }