@stonerzju/opencode 1.2.16-offline.1

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