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