mastracode 0.0.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 (336) 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 +4 -0
  6. package/package.json +113 -0
  7. package/parsers-config.ts +239 -0
  8. package/script/build.ts +167 -0
  9. package/script/postinstall.mjs +122 -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 +1051 -0
  15. package/src/acp/session.ts +101 -0
  16. package/src/acp/types.ts +22 -0
  17. package/src/agent/agent.ts +398 -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 +10 -0
  22. package/src/agent/prompt/title.txt +36 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +114 -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 +88 -0
  30. package/src/cli/cmd/agent.ts +256 -0
  31. package/src/cli/cmd/auth.ts +391 -0
  32. package/src/cli/cmd/cmd.ts +7 -0
  33. package/src/cli/cmd/debug/config.ts +15 -0
  34. package/src/cli/cmd/debug/file.ts +91 -0
  35. package/src/cli/cmd/debug/index.ts +43 -0
  36. package/src/cli/cmd/debug/lsp.ts +48 -0
  37. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  38. package/src/cli/cmd/debug/scrap.ts +15 -0
  39. package/src/cli/cmd/debug/skill.ts +15 -0
  40. package/src/cli/cmd/debug/snapshot.ts +48 -0
  41. package/src/cli/cmd/export.ts +88 -0
  42. package/src/cli/cmd/generate.ts +38 -0
  43. package/src/cli/cmd/github.ts +1408 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/mcp.ts +654 -0
  46. package/src/cli/cmd/models.ts +77 -0
  47. package/src/cli/cmd/pr.ts +112 -0
  48. package/src/cli/cmd/run.ts +368 -0
  49. package/src/cli/cmd/serve.ts +31 -0
  50. package/src/cli/cmd/session.ts +106 -0
  51. package/src/cli/cmd/stats.ts +298 -0
  52. package/src/cli/cmd/tui/app.tsx +686 -0
  53. package/src/cli/cmd/tui/attach.ts +30 -0
  54. package/src/cli/cmd/tui/component/border.tsx +21 -0
  55. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  56. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  57. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  58. package/src/cli/cmd/tui/component/dialog-model.tsx +230 -0
  59. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  60. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  61. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  62. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  64. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  65. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  66. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  67. package/src/cli/cmd/tui/component/logo.tsx +27 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1117 -0
  71. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  72. package/src/cli/cmd/tui/component/tips.ts +103 -0
  73. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  74. package/src/cli/cmd/tui/context/args.tsx +14 -0
  75. package/src/cli/cmd/tui/context/directory.ts +13 -0
  76. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  77. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  78. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  79. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  80. package/src/cli/cmd/tui/context/local.tsx +339 -0
  81. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  82. package/src/cli/cmd/tui/context/route.tsx +46 -0
  83. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  84. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  85. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  86. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  87. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  88. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  89. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  90. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  91. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  92. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  93. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  94. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  95. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  97. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  98. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  99. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  100. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  101. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  102. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  103. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  104. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  105. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  106. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  107. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  108. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  109. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  111. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  112. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  113. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  114. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  115. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  116. package/src/cli/cmd/tui/event.ts +40 -0
  117. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  118. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  119. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  120. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  121. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  122. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  123. package/src/cli/cmd/tui/routes/session/header.tsx +141 -0
  124. package/src/cli/cmd/tui/routes/session/index.tsx +1885 -0
  125. package/src/cli/cmd/tui/routes/session/sidebar.tsx +322 -0
  126. package/src/cli/cmd/tui/spawn.ts +60 -0
  127. package/src/cli/cmd/tui/thread.ts +120 -0
  128. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  129. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  130. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  131. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  132. package/src/cli/cmd/tui/ui/dialog-select.tsx +332 -0
  133. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  134. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  135. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  136. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  137. package/src/cli/cmd/tui/util/editor.ts +32 -0
  138. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  139. package/src/cli/cmd/tui/worker.ts +63 -0
  140. package/src/cli/cmd/uninstall.ts +344 -0
  141. package/src/cli/cmd/upgrade.ts +67 -0
  142. package/src/cli/cmd/web.ts +84 -0
  143. package/src/cli/error.ts +56 -0
  144. package/src/cli/ui.ts +84 -0
  145. package/src/cli/upgrade.ts +25 -0
  146. package/src/command/index.ts +80 -0
  147. package/src/command/template/initialize.txt +10 -0
  148. package/src/command/template/review.txt +97 -0
  149. package/src/config/config.ts +997 -0
  150. package/src/config/markdown.ts +41 -0
  151. package/src/env/index.ts +26 -0
  152. package/src/file/ignore.ts +83 -0
  153. package/src/file/index.ts +328 -0
  154. package/src/file/ripgrep.ts +393 -0
  155. package/src/file/time.ts +64 -0
  156. package/src/file/watcher.ts +103 -0
  157. package/src/flag/flag.ts +46 -0
  158. package/src/format/formatter.ts +315 -0
  159. package/src/format/index.ts +137 -0
  160. package/src/global/index.ts +52 -0
  161. package/src/id/id.ts +73 -0
  162. package/src/ide/index.ts +76 -0
  163. package/src/index.ts +158 -0
  164. package/src/installation/index.ts +196 -0
  165. package/src/lsp/client.ts +229 -0
  166. package/src/lsp/index.ts +485 -0
  167. package/src/lsp/language.ts +116 -0
  168. package/src/lsp/server.ts +1895 -0
  169. package/src/mcp/auth.ts +135 -0
  170. package/src/mcp/index.ts +654 -0
  171. package/src/mcp/oauth-callback.ts +200 -0
  172. package/src/mcp/oauth-provider.ts +154 -0
  173. package/src/patch/index.ts +622 -0
  174. package/src/permission/index.ts +199 -0
  175. package/src/plugin/index.ts +91 -0
  176. package/src/project/bootstrap.ts +31 -0
  177. package/src/project/instance.ts +78 -0
  178. package/src/project/project.ts +221 -0
  179. package/src/project/state.ts +65 -0
  180. package/src/project/vcs.ts +76 -0
  181. package/src/provider/auth.ts +143 -0
  182. package/src/provider/models-macro.ts +11 -0
  183. package/src/provider/models.ts +106 -0
  184. package/src/provider/provider.ts +1056 -0
  185. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  186. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  187. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  190. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  191. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  192. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  193. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  200. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  201. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  202. package/src/provider/transform.ts +455 -0
  203. package/src/pty/index.ts +231 -0
  204. package/src/server/error.ts +36 -0
  205. package/src/server/project.ts +79 -0
  206. package/src/server/server.ts +2642 -0
  207. package/src/server/tui.ts +71 -0
  208. package/src/session/compaction.ts +223 -0
  209. package/src/session/index.ts +458 -0
  210. package/src/session/llm-mastra.ts +412 -0
  211. package/src/session/llm-shared.ts +172 -0
  212. package/src/session/llm.ts +439 -0
  213. package/src/session/message-v2.ts +675 -0
  214. package/src/session/message.ts +189 -0
  215. package/src/session/processor.ts +171 -0
  216. package/src/session/prompt/anthropic-20250930.txt +166 -0
  217. package/src/session/prompt/anthropic.txt +105 -0
  218. package/src/session/prompt/anthropic_spoof.txt +1 -0
  219. package/src/session/prompt/beast.txt +147 -0
  220. package/src/session/prompt/build-switch.txt +5 -0
  221. package/src/session/prompt/codex.txt +318 -0
  222. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  223. package/src/session/prompt/gemini.txt +155 -0
  224. package/src/session/prompt/max-steps.txt +16 -0
  225. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  226. package/src/session/prompt/plan.txt +26 -0
  227. package/src/session/prompt/polaris.txt +107 -0
  228. package/src/session/prompt/qwen.txt +109 -0
  229. package/src/session/prompt.ts +1454 -0
  230. package/src/session/retry.ts +86 -0
  231. package/src/session/revert.ts +108 -0
  232. package/src/session/status.ts +76 -0
  233. package/src/session/summary.ts +194 -0
  234. package/src/session/system.ts +120 -0
  235. package/src/session/todo.ts +37 -0
  236. package/src/share/share-next.ts +194 -0
  237. package/src/share/share.ts +87 -0
  238. package/src/shell/shell.ts +67 -0
  239. package/src/skill/index.ts +1 -0
  240. package/src/skill/skill.ts +83 -0
  241. package/src/snapshot/index.ts +197 -0
  242. package/src/storage/storage.ts +226 -0
  243. package/src/tool/bash.ts +306 -0
  244. package/src/tool/bash.txt +158 -0
  245. package/src/tool/batch.ts +175 -0
  246. package/src/tool/batch.txt +24 -0
  247. package/src/tool/codesearch.ts +138 -0
  248. package/src/tool/codesearch.txt +12 -0
  249. package/src/tool/edit.ts +675 -0
  250. package/src/tool/edit.txt +10 -0
  251. package/src/tool/glob.ts +65 -0
  252. package/src/tool/glob.txt +6 -0
  253. package/src/tool/grep.ts +121 -0
  254. package/src/tool/grep.txt +8 -0
  255. package/src/tool/invalid.ts +17 -0
  256. package/src/tool/ls.ts +110 -0
  257. package/src/tool/ls.txt +1 -0
  258. package/src/tool/lsp-diagnostics.ts +26 -0
  259. package/src/tool/lsp-diagnostics.txt +1 -0
  260. package/src/tool/lsp-hover.ts +31 -0
  261. package/src/tool/lsp-hover.txt +1 -0
  262. package/src/tool/lsp.ts +87 -0
  263. package/src/tool/lsp.txt +19 -0
  264. package/src/tool/multiedit.ts +46 -0
  265. package/src/tool/multiedit.txt +41 -0
  266. package/src/tool/patch.ts +233 -0
  267. package/src/tool/patch.txt +1 -0
  268. package/src/tool/read.ts +219 -0
  269. package/src/tool/read.txt +12 -0
  270. package/src/tool/registry.ts +162 -0
  271. package/src/tool/skill.ts +100 -0
  272. package/src/tool/task.ts +136 -0
  273. package/src/tool/task.txt +60 -0
  274. package/src/tool/todo.ts +39 -0
  275. package/src/tool/todoread.txt +14 -0
  276. package/src/tool/todowrite.txt +167 -0
  277. package/src/tool/tool.ts +71 -0
  278. package/src/tool/webfetch.ts +187 -0
  279. package/src/tool/webfetch.txt +13 -0
  280. package/src/tool/websearch.ts +150 -0
  281. package/src/tool/websearch.txt +11 -0
  282. package/src/tool/write.ts +110 -0
  283. package/src/tool/write.txt +8 -0
  284. package/src/util/archive.ts +16 -0
  285. package/src/util/color.ts +19 -0
  286. package/src/util/context.ts +25 -0
  287. package/src/util/defer.ts +12 -0
  288. package/src/util/eventloop.ts +20 -0
  289. package/src/util/filesystem.ts +83 -0
  290. package/src/util/fn.ts +11 -0
  291. package/src/util/iife.ts +3 -0
  292. package/src/util/keybind.ts +102 -0
  293. package/src/util/lazy.ts +11 -0
  294. package/src/util/locale.ts +81 -0
  295. package/src/util/lock.ts +98 -0
  296. package/src/util/log.ts +180 -0
  297. package/src/util/queue.ts +32 -0
  298. package/src/util/rpc.ts +42 -0
  299. package/src/util/scrap.ts +10 -0
  300. package/src/util/signal.ts +12 -0
  301. package/src/util/timeout.ts +14 -0
  302. package/src/util/token.ts +7 -0
  303. package/src/util/wildcard.ts +54 -0
  304. package/sst-env.d.ts +9 -0
  305. package/test/agent/agent.test.ts +146 -0
  306. package/test/bun.test.ts +53 -0
  307. package/test/cli/github-remote.test.ts +80 -0
  308. package/test/config/agent-color.test.ts +66 -0
  309. package/test/config/config.test.ts +535 -0
  310. package/test/config/markdown.test.ts +89 -0
  311. package/test/file/ignore.test.ts +10 -0
  312. package/test/fixture/fixture.ts +34 -0
  313. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  314. package/test/ide/ide.test.ts +82 -0
  315. package/test/keybind.test.ts +421 -0
  316. package/test/lsp/client.test.ts +95 -0
  317. package/test/mcp/headers.test.ts +153 -0
  318. package/test/patch/patch.test.ts +348 -0
  319. package/test/preload.ts +57 -0
  320. package/test/project/project.test.ts +72 -0
  321. package/test/provider/provider.test.ts +1809 -0
  322. package/test/provider/transform.test.ts +411 -0
  323. package/test/session/retry.test.ts +61 -0
  324. package/test/session/session.test.ts +71 -0
  325. package/test/skill/skill.test.ts +131 -0
  326. package/test/snapshot/snapshot.test.ts +939 -0
  327. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  328. package/test/tool/bash.test.ts +434 -0
  329. package/test/tool/grep.test.ts +108 -0
  330. package/test/tool/patch.test.ts +259 -0
  331. package/test/tool/read.test.ts +42 -0
  332. package/test/util/iife.test.ts +36 -0
  333. package/test/util/lazy.test.ts +50 -0
  334. package/test/util/timeout.test.ts +21 -0
  335. package/test/util/wildcard.test.ts +55 -0
  336. package/tsconfig.json +16 -0
@@ -0,0 +1,85 @@
1
+ import { createMemo, createSignal, For } from "solid-js"
2
+ import { useTheme } from "@tui/context/theme"
3
+ import { useKeybind } from "@tui/context/keybind"
4
+ import { TIPS } from "./tips"
5
+ import { EmptyBorder } from "./border"
6
+
7
+ type TipPart = { text: string; highlight: boolean }
8
+
9
+ function parseTip(tip: string): TipPart[] {
10
+ const parts: TipPart[] = []
11
+ const regex = /\{highlight\}(.*?)\{\/highlight\}/g
12
+ let lastIndex = 0
13
+ let match
14
+
15
+ while ((match = regex.exec(tip)) !== null) {
16
+ if (match.index > lastIndex) {
17
+ parts.push({ text: tip.slice(lastIndex, match.index), highlight: false })
18
+ }
19
+ parts.push({ text: match[1], highlight: true })
20
+ lastIndex = regex.lastIndex
21
+ }
22
+
23
+ if (lastIndex < tip.length) {
24
+ parts.push({ text: tip.slice(lastIndex), highlight: false })
25
+ }
26
+
27
+ return parts
28
+ }
29
+
30
+ const [tipIndex, setTipIndex] = createSignal(Math.floor(Math.random() * TIPS.length))
31
+
32
+ export function randomizeTip() {
33
+ setTipIndex(Math.floor(Math.random() * TIPS.length))
34
+ }
35
+
36
+ const BOX_WIDTH = 42
37
+ const TITLE = " 🅘 Did you know? "
38
+
39
+ export function DidYouKnow() {
40
+ const { theme } = useTheme()
41
+ const keybind = useKeybind()
42
+
43
+ const tipParts = createMemo(() => parseTip(TIPS[tipIndex()]))
44
+
45
+ const dashes = createMemo(() => {
46
+ // ╭─ + title + ─...─ + ╮ = BOX_WIDTH
47
+ // 1 + 1 + title.length + dashes + 1 = BOX_WIDTH
48
+ return Math.max(0, BOX_WIDTH - 2 - TITLE.length - 1)
49
+ })
50
+
51
+ return (
52
+ <box position="absolute" bottom={3} right={2} width={BOX_WIDTH}>
53
+ <text>
54
+ <span style={{ fg: theme.border }}>╭─</span>
55
+ <span style={{ fg: theme.text }}>{TITLE}</span>
56
+ <span style={{ fg: theme.border }}>{"─".repeat(dashes())}╮</span>
57
+ </text>
58
+ <box
59
+ border={["left", "right", "bottom"]}
60
+ borderColor={theme.border}
61
+ customBorderChars={{
62
+ ...EmptyBorder,
63
+ bottomLeft: "╰",
64
+ bottomRight: "╯",
65
+ horizontal: "─",
66
+ vertical: "│",
67
+ }}
68
+ >
69
+ <box paddingLeft={2} paddingRight={2} paddingTop={1} paddingBottom={1}>
70
+ <text>
71
+ <For each={tipParts()}>
72
+ {(part) => <span style={{ fg: part.highlight ? theme.text : theme.textMuted }}>{part.text}</span>}
73
+ </For>
74
+ </text>
75
+ </box>
76
+ </box>
77
+ <box flexDirection="row" justifyContent="flex-end">
78
+ <text>
79
+ <span style={{ fg: theme.text }}>{keybind.print("tips_toggle")}</span>
80
+ <span style={{ fg: theme.textMuted }}> hide tips</span>
81
+ </text>
82
+ </box>
83
+ </box>
84
+ )
85
+ }
@@ -0,0 +1,27 @@
1
+ import { TextAttributes } from "@opentui/core"
2
+ import { For } from "solid-js"
3
+ import { useTheme } from "@tui/context/theme"
4
+
5
+ const LOGO_LEFT = [` `, `█▀▀█ █▀▀█ █▀▀█ █▀▀▄`, `█░░█ █░░█ █▀▀▀ █░░█`, `▀▀▀▀ █▀▀▀ ▀▀▀▀ ▀ ▀`]
6
+
7
+ const LOGO_RIGHT = [` ▄ `, `█▀▀▀ █▀▀█ █▀▀█ █▀▀█`, `█░░░ █░░█ █░░█ █▀▀▀`, `▀▀▀▀ ▀▀▀▀ ▀▀▀▀ ▀▀▀▀`]
8
+
9
+ export function Logo() {
10
+ const { theme } = useTheme()
11
+ return (
12
+ <box>
13
+ <For each={LOGO_LEFT}>
14
+ {(line, index) => (
15
+ <box flexDirection="row" gap={1}>
16
+ <text fg={theme.textMuted} selectable={false}>
17
+ {line}
18
+ </text>
19
+ <text fg={theme.text} attributes={TextAttributes.BOLD} selectable={false}>
20
+ {LOGO_RIGHT[index()]}
21
+ </text>
22
+ </box>
23
+ )}
24
+ </For>
25
+ </box>
26
+ )
27
+ }
@@ -0,0 +1,574 @@
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, For, 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
+
15
+ export type AutocompleteRef = {
16
+ onInput: (value: string) => void
17
+ onKeyDown: (e: KeyEvent) => void
18
+ visible: false | "@" | "/"
19
+ }
20
+
21
+ export type AutocompleteOption = {
22
+ display: string
23
+ aliases?: string[]
24
+ disabled?: boolean
25
+ description?: string
26
+ onSelect?: () => void
27
+ }
28
+
29
+ export function Autocomplete(props: {
30
+ value: string
31
+ sessionID?: string
32
+ setPrompt: (input: (prompt: PromptInfo) => void) => void
33
+ setExtmark: (partIndex: number, extmarkId: number) => void
34
+ anchor: () => BoxRenderable
35
+ input: () => TextareaRenderable
36
+ ref: (ref: AutocompleteRef) => void
37
+ fileStyleId: number
38
+ agentStyleId: number
39
+ promptPartTypeId: () => number
40
+ }) {
41
+ const sdk = useSDK()
42
+ const sync = useSync()
43
+ const command = useCommandDialog()
44
+ const { theme } = useTheme()
45
+ const dimensions = useTerminalDimensions()
46
+
47
+ const [store, setStore] = createStore({
48
+ index: 0,
49
+ selected: 0,
50
+ visible: false as AutocompleteRef["visible"],
51
+ })
52
+
53
+ const [positionTick, setPositionTick] = createSignal(0)
54
+
55
+ createEffect(() => {
56
+ if (store.visible) {
57
+ let lastPos = { x: 0, y: 0, width: 0 }
58
+ const interval = setInterval(() => {
59
+ const anchor = props.anchor()
60
+ if (anchor.x !== lastPos.x || anchor.y !== lastPos.y || anchor.width !== lastPos.width) {
61
+ lastPos = { x: anchor.x, y: anchor.y, width: anchor.width }
62
+ setPositionTick((t) => t + 1)
63
+ }
64
+ }, 50)
65
+
66
+ onCleanup(() => clearInterval(interval))
67
+ }
68
+ })
69
+
70
+ const position = createMemo(() => {
71
+ if (!store.visible) return { x: 0, y: 0, width: 0 }
72
+ const dims = dimensions()
73
+ positionTick()
74
+ const anchor = props.anchor()
75
+ const parent = anchor.parent
76
+ const parentX = parent?.x ?? 0
77
+ const parentY = parent?.y ?? 0
78
+
79
+ return {
80
+ x: anchor.x - parentX,
81
+ y: anchor.y - parentY,
82
+ width: anchor.width,
83
+ }
84
+ })
85
+
86
+ const filter = createMemo(() => {
87
+ if (!store.visible) return
88
+ // Track props.value to make memo reactive to text changes
89
+ props.value // <- there surely is a better way to do this, like making .input() reactive
90
+
91
+ return props.input().getTextRange(store.index + 1, props.input().cursorOffset)
92
+ })
93
+
94
+ function insertPart(text: string, part: PromptInfo["parts"][number]) {
95
+ const input = props.input()
96
+ const currentCursorOffset = input.cursorOffset
97
+
98
+ const charAfterCursor = props.value.at(currentCursorOffset)
99
+ const needsSpace = charAfterCursor !== " "
100
+ const append = "@" + text + (needsSpace ? " " : "")
101
+
102
+ input.cursorOffset = store.index
103
+ const startCursor = input.logicalCursor
104
+ input.cursorOffset = currentCursorOffset
105
+ const endCursor = input.logicalCursor
106
+
107
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
108
+ input.insertText(append)
109
+
110
+ const virtualText = "@" + text
111
+ const extmarkStart = store.index
112
+ const extmarkEnd = extmarkStart + Bun.stringWidth(virtualText)
113
+
114
+ const styleId = part.type === "file" ? props.fileStyleId : part.type === "agent" ? props.agentStyleId : undefined
115
+
116
+ const extmarkId = input.extmarks.create({
117
+ start: extmarkStart,
118
+ end: extmarkEnd,
119
+ virtual: true,
120
+ styleId,
121
+ typeId: props.promptPartTypeId(),
122
+ })
123
+
124
+ props.setPrompt((draft) => {
125
+ if (part.type === "file" && part.source?.text) {
126
+ part.source.text.start = extmarkStart
127
+ part.source.text.end = extmarkEnd
128
+ part.source.text.value = virtualText
129
+ } else if (part.type === "agent" && part.source) {
130
+ part.source.start = extmarkStart
131
+ part.source.end = extmarkEnd
132
+ part.source.value = virtualText
133
+ }
134
+ const partIndex = draft.parts.length
135
+ draft.parts.push(part)
136
+ props.setExtmark(partIndex, extmarkId)
137
+ })
138
+ }
139
+
140
+ const [files] = createResource(
141
+ () => filter(),
142
+ async (query) => {
143
+ if (!store.visible || store.visible === "/") return []
144
+
145
+ // Get files from SDK
146
+ const result = await sdk.client.find.files({
147
+ query: query ?? "",
148
+ })
149
+
150
+ const options: AutocompleteOption[] = []
151
+
152
+ // Add file options
153
+ if (!result.error && result.data) {
154
+ const width = props.anchor().width - 4
155
+ options.push(
156
+ ...result.data.map(
157
+ (item): AutocompleteOption => ({
158
+ display: Locale.truncateMiddle(item, width),
159
+ onSelect: () => {
160
+ insertPart(item, {
161
+ type: "file",
162
+ mime: "text/plain",
163
+ filename: item,
164
+ url: `file://${process.cwd()}/${item}`,
165
+ source: {
166
+ type: "file",
167
+ text: {
168
+ start: 0,
169
+ end: 0,
170
+ value: "",
171
+ },
172
+ path: item,
173
+ },
174
+ })
175
+ },
176
+ }),
177
+ ),
178
+ )
179
+ }
180
+
181
+ return options
182
+ },
183
+ {
184
+ initialValue: [],
185
+ },
186
+ )
187
+
188
+ const agents = createMemo(() => {
189
+ const agents = sync.data.agent
190
+ return agents
191
+ .filter((agent) => !agent.hidden && agent.mode !== "primary")
192
+ .map(
193
+ (agent): AutocompleteOption => ({
194
+ display: "@" + agent.name,
195
+ onSelect: () => {
196
+ insertPart(agent.name, {
197
+ type: "agent",
198
+ name: agent.name,
199
+ source: {
200
+ start: 0,
201
+ end: 0,
202
+ value: "",
203
+ },
204
+ })
205
+ },
206
+ }),
207
+ )
208
+ })
209
+
210
+ const session = createMemo(() => (props.sessionID ? sync.session.get(props.sessionID) : undefined))
211
+ const commands = createMemo((): AutocompleteOption[] => {
212
+ const results: AutocompleteOption[] = []
213
+ const s = session()
214
+ for (const command of sync.data.command) {
215
+ results.push({
216
+ display: "/" + command.name,
217
+ description: command.description,
218
+ onSelect: () => {
219
+ const newText = "/" + command.name + " "
220
+ const cursor = props.input().logicalCursor
221
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
222
+ props.input().insertText(newText)
223
+ props.input().cursorOffset = Bun.stringWidth(newText)
224
+ },
225
+ })
226
+ }
227
+ if (s) {
228
+ results.push(
229
+ {
230
+ display: "/undo",
231
+ description: "undo the last message",
232
+ onSelect: () => {
233
+ command.trigger("session.undo")
234
+ },
235
+ },
236
+ {
237
+ display: "/redo",
238
+ description: "redo the last message",
239
+ onSelect: () => command.trigger("session.redo"),
240
+ },
241
+ {
242
+ display: "/compact",
243
+ aliases: ["/summarize"],
244
+ description: "compact the session",
245
+ onSelect: () => command.trigger("session.compact"),
246
+ },
247
+ {
248
+ display: "/unshare",
249
+ disabled: !s.share,
250
+ description: "unshare a session",
251
+ onSelect: () => command.trigger("session.unshare"),
252
+ },
253
+ {
254
+ display: "/rename",
255
+ description: "rename session",
256
+ onSelect: () => command.trigger("session.rename"),
257
+ },
258
+ {
259
+ display: "/copy",
260
+ description: "copy session transcript to clipboard",
261
+ onSelect: () => command.trigger("session.copy"),
262
+ },
263
+ {
264
+ display: "/export",
265
+ description: "export session transcript to file",
266
+ onSelect: () => command.trigger("session.export"),
267
+ },
268
+ {
269
+ display: "/timeline",
270
+ description: "jump to message",
271
+ onSelect: () => command.trigger("session.timeline"),
272
+ },
273
+ {
274
+ display: "/fork",
275
+ description: "fork from message",
276
+ onSelect: () => command.trigger("session.fork"),
277
+ },
278
+ {
279
+ display: "/thinking",
280
+ description: "toggle thinking visibility",
281
+ onSelect: () => command.trigger("session.toggle.thinking"),
282
+ },
283
+ )
284
+ if (sync.data.config.share !== "disabled") {
285
+ results.push({
286
+ display: "/share",
287
+ disabled: !!s.share?.url,
288
+ description: "share a session",
289
+ onSelect: () => command.trigger("session.share"),
290
+ })
291
+ }
292
+ }
293
+
294
+ results.push(
295
+ {
296
+ display: "/new",
297
+ aliases: ["/clear"],
298
+ description: "create a new session",
299
+ onSelect: () => command.trigger("session.new"),
300
+ },
301
+ {
302
+ display: "/models",
303
+ description: "list models",
304
+ onSelect: () => command.trigger("model.list"),
305
+ },
306
+ {
307
+ display: "/agents",
308
+ description: "list agents",
309
+ onSelect: () => command.trigger("agent.list"),
310
+ },
311
+ {
312
+ display: "/session",
313
+ aliases: ["/resume", "/continue"],
314
+ description: "list sessions",
315
+ onSelect: () => command.trigger("session.list"),
316
+ },
317
+ {
318
+ display: "/status",
319
+ description: "show status",
320
+ onSelect: () => command.trigger("opencode.status"),
321
+ },
322
+ {
323
+ display: "/mcp",
324
+ description: "toggle MCPs",
325
+ onSelect: () => command.trigger("mcp.list"),
326
+ },
327
+ {
328
+ display: "/theme",
329
+ description: "toggle theme",
330
+ onSelect: () => command.trigger("theme.switch"),
331
+ },
332
+ {
333
+ display: "/editor",
334
+ description: "open editor",
335
+ onSelect: () => command.trigger("prompt.editor", "prompt"),
336
+ },
337
+ {
338
+ display: "/connect",
339
+ description: "connect to a provider",
340
+ onSelect: () => command.trigger("provider.connect"),
341
+ },
342
+ {
343
+ display: "/help",
344
+ description: "show help",
345
+ onSelect: () => command.trigger("help.show"),
346
+ },
347
+ {
348
+ display: "/commands",
349
+ description: "show all commands",
350
+ onSelect: () => command.show(),
351
+ },
352
+ {
353
+ display: "/exit",
354
+ aliases: ["/quit", "/q"],
355
+ description: "exit the app",
356
+ onSelect: () => command.trigger("app.exit"),
357
+ },
358
+ )
359
+ const max = firstBy(results, [(x) => x.display.length, "desc"])?.display.length
360
+ if (!max) return results
361
+ return results.map((item) => ({
362
+ ...item,
363
+ display: item.display.padEnd(max + 2),
364
+ }))
365
+ })
366
+
367
+ const options = createMemo((prev: AutocompleteOption[] | undefined) => {
368
+ const filesValue = files()
369
+ const agentsValue = agents()
370
+ const commandsValue = commands()
371
+
372
+ const mixed: AutocompleteOption[] = (
373
+ store.visible === "@" ? [...agentsValue, ...(filesValue || [])] : [...commandsValue]
374
+ ).filter((x) => x.disabled !== true)
375
+
376
+ const currentFilter = filter()
377
+
378
+ if (!currentFilter) {
379
+ return mixed
380
+ }
381
+
382
+ if (files.loading && prev && prev.length > 0) {
383
+ return prev
384
+ }
385
+
386
+ const result = fuzzysort.go(currentFilter, mixed, {
387
+ keys: [(obj) => obj.display.trimEnd(), "description", (obj) => obj.aliases?.join(" ") ?? ""],
388
+ limit: 10,
389
+ scoreFn: (objResults) => {
390
+ const displayResult = objResults[0]
391
+ if (displayResult && displayResult.target.startsWith(store.visible + currentFilter)) {
392
+ return objResults.score * 2
393
+ }
394
+ return objResults.score
395
+ },
396
+ })
397
+
398
+ return result.map((arr) => arr.obj)
399
+ })
400
+
401
+ createEffect(() => {
402
+ filter()
403
+ setStore("selected", 0)
404
+ })
405
+
406
+ function move(direction: -1 | 1) {
407
+ if (!store.visible) return
408
+ if (!options().length) return
409
+ let next = store.selected + direction
410
+ if (next < 0) next = options().length - 1
411
+ if (next >= options().length) next = 0
412
+ moveTo(next)
413
+ }
414
+
415
+ function moveTo(next: number) {
416
+ setStore("selected", next)
417
+ if (!scroll) return
418
+ const viewportHeight = Math.min(height(), options().length)
419
+ const scrollBottom = scroll.scrollTop + viewportHeight
420
+ if (next < scroll.scrollTop) {
421
+ scroll.scrollBy(next - scroll.scrollTop)
422
+ } else if (next + 1 > scrollBottom) {
423
+ scroll.scrollBy(next + 1 - scrollBottom)
424
+ }
425
+ }
426
+
427
+ function select() {
428
+ const selected = options()[store.selected]
429
+ if (!selected) return
430
+ hide()
431
+ selected.onSelect?.()
432
+ }
433
+
434
+ function show(mode: "@" | "/") {
435
+ command.keybinds(false)
436
+ setStore({
437
+ visible: mode,
438
+ index: props.input().cursorOffset,
439
+ })
440
+ }
441
+
442
+ function hide() {
443
+ const text = props.input().plainText
444
+ if (store.visible === "/" && !text.endsWith(" ") && text.startsWith("/")) {
445
+ const cursor = props.input().logicalCursor
446
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
447
+ // Sync the prompt store immediately since onContentChange is async
448
+ props.setPrompt((draft) => {
449
+ draft.input = props.input().plainText
450
+ })
451
+ }
452
+ command.keybinds(true)
453
+ setStore("visible", false)
454
+ }
455
+
456
+ onMount(() => {
457
+ props.ref({
458
+ get visible() {
459
+ return store.visible
460
+ },
461
+ onInput(value) {
462
+ if (store.visible) {
463
+ if (
464
+ // Typed text before the trigger
465
+ props.input().cursorOffset <= store.index ||
466
+ // There is a space between the trigger and the cursor
467
+ props.input().getTextRange(store.index, props.input().cursorOffset).match(/\s/) ||
468
+ // "/<command>" is not the sole content
469
+ (store.visible === "/" && value.match(/^\S+\s+\S+\s*$/))
470
+ ) {
471
+ hide()
472
+ return
473
+ }
474
+ }
475
+ },
476
+ onKeyDown(e: KeyEvent) {
477
+ if (store.visible) {
478
+ const name = e.name?.toLowerCase()
479
+ const ctrlOnly = e.ctrl && !e.meta && !e.shift
480
+ const isNavUp = name === "up" || (ctrlOnly && name === "p")
481
+ const isNavDown = name === "down" || (ctrlOnly && name === "n")
482
+
483
+ if (isNavUp) {
484
+ move(-1)
485
+ e.preventDefault()
486
+ return
487
+ }
488
+ if (isNavDown) {
489
+ move(1)
490
+ e.preventDefault()
491
+ return
492
+ }
493
+ if (name === "escape") {
494
+ hide()
495
+ e.preventDefault()
496
+ return
497
+ }
498
+ if (name === "return" || name === "tab") {
499
+ select()
500
+ e.preventDefault()
501
+ return
502
+ }
503
+ }
504
+ if (!store.visible) {
505
+ if (e.name === "@") {
506
+ const cursorOffset = props.input().cursorOffset
507
+ const charBeforeCursor =
508
+ cursorOffset === 0 ? undefined : props.input().getTextRange(cursorOffset - 1, cursorOffset)
509
+ const canTrigger = charBeforeCursor === undefined || charBeforeCursor === "" || /\s/.test(charBeforeCursor)
510
+ if (canTrigger) show("@")
511
+ }
512
+
513
+ if (e.name === "/") {
514
+ if (props.input().cursorOffset === 0) show("/")
515
+ }
516
+ }
517
+ },
518
+ })
519
+ })
520
+
521
+ const height = createMemo(() => {
522
+ if (options().length) return Math.min(10, options().length)
523
+ return 1
524
+ })
525
+
526
+ let scroll: ScrollBoxRenderable
527
+
528
+ return (
529
+ <box
530
+ visible={store.visible !== false}
531
+ position="absolute"
532
+ top={position().y - height()}
533
+ left={position().x}
534
+ width={position().width}
535
+ zIndex={100}
536
+ {...SplitBorder}
537
+ borderColor={theme.border}
538
+ >
539
+ <scrollbox
540
+ ref={(r: ScrollBoxRenderable) => (scroll = r)}
541
+ backgroundColor={theme.backgroundMenu}
542
+ height={height()}
543
+ scrollbarOptions={{ visible: false }}
544
+ >
545
+ <For
546
+ each={options()}
547
+ fallback={
548
+ <box paddingLeft={1} paddingRight={1}>
549
+ <text fg={theme.textMuted}>No matching items</text>
550
+ </box>
551
+ }
552
+ >
553
+ {(option, index) => (
554
+ <box
555
+ paddingLeft={1}
556
+ paddingRight={1}
557
+ backgroundColor={index() === store.selected ? theme.primary : undefined}
558
+ flexDirection="row"
559
+ >
560
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.text} flexShrink={0}>
561
+ {option.display}
562
+ </text>
563
+ <Show when={option.description}>
564
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.textMuted} wrapMode="none">
565
+ {option.description}
566
+ </text>
567
+ </Show>
568
+ </box>
569
+ )}
570
+ </For>
571
+ </scrollbox>
572
+ </box>
573
+ )
574
+ }