bincode-cli 1.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 (300) hide show
  1. package/AGENTS.md +27 -0
  2. package/README.md +15 -0
  3. package/bin/bincode +98 -0
  4. package/bunfig.toml +4 -0
  5. package/package.json +124 -0
  6. package/parsers-config.ts +239 -0
  7. package/script/build.ts +167 -0
  8. package/script/postinstall.mjs +206 -0
  9. package/script/publish.ts +99 -0
  10. package/script/schema.ts +47 -0
  11. package/src/acp/README.md +164 -0
  12. package/src/acp/agent.ts +1051 -0
  13. package/src/acp/session.ts +101 -0
  14. package/src/acp/types.ts +22 -0
  15. package/src/agent/agent.ts +398 -0
  16. package/src/agent/generate.txt +75 -0
  17. package/src/agent/prompt/compaction.txt +12 -0
  18. package/src/agent/prompt/explore.txt +18 -0
  19. package/src/agent/prompt/summary.txt +10 -0
  20. package/src/agent/prompt/title.txt +36 -0
  21. package/src/auth/bineric-login.ts +506 -0
  22. package/src/auth/index.ts +70 -0
  23. package/src/bun/index.ts +114 -0
  24. package/src/bus/bus-event.ts +43 -0
  25. package/src/bus/global.ts +10 -0
  26. package/src/bus/index.ts +105 -0
  27. package/src/cli/auth-check.ts +61 -0
  28. package/src/cli/bootstrap.ts +21 -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 +436 -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 +1399 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/login.ts +112 -0
  46. package/src/cli/cmd/logout.ts +38 -0
  47. package/src/cli/cmd/mcp.ts +654 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +368 -0
  51. package/src/cli/cmd/serve.ts +31 -0
  52. package/src/cli/cmd/session.ts +106 -0
  53. package/src/cli/cmd/stats.ts +298 -0
  54. package/src/cli/cmd/tui/app.tsx +669 -0
  55. package/src/cli/cmd/tui/attach.ts +30 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +123 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -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 +32 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +560 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1052 -0
  71. package/src/cli/cmd/tui/context/args.tsx +14 -0
  72. package/src/cli/cmd/tui/context/directory.ts +13 -0
  73. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  74. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  75. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  76. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  77. package/src/cli/cmd/tui/context/local.tsx +339 -0
  78. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  79. package/src/cli/cmd/tui/context/route.tsx +46 -0
  80. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  81. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  82. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  83. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  84. package/src/cli/cmd/tui/context/theme/bincode.json +245 -0
  85. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  86. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  87. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  88. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  89. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  90. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  91. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  92. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  93. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  95. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  96. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  97. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  98. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  99. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  100. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  101. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  102. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  103. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  104. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  106. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  107. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  108. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  109. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  110. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  111. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  112. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  113. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  114. package/src/cli/cmd/tui/event.ts +40 -0
  115. package/src/cli/cmd/tui/routes/home.tsx +105 -0
  116. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  117. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  118. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  119. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  120. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  121. package/src/cli/cmd/tui/routes/session/header.tsx +141 -0
  122. package/src/cli/cmd/tui/routes/session/index.tsx +1888 -0
  123. package/src/cli/cmd/tui/routes/session/sidebar.tsx +321 -0
  124. package/src/cli/cmd/tui/spawn.ts +60 -0
  125. package/src/cli/cmd/tui/thread.ts +120 -0
  126. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  127. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  128. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  129. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  130. package/src/cli/cmd/tui/ui/dialog-select.tsx +330 -0
  131. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  132. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  133. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  134. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  135. package/src/cli/cmd/tui/util/editor.ts +32 -0
  136. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  137. package/src/cli/cmd/tui/worker.ts +63 -0
  138. package/src/cli/cmd/uninstall.ts +344 -0
  139. package/src/cli/cmd/upgrade.ts +67 -0
  140. package/src/cli/cmd/web.ts +84 -0
  141. package/src/cli/error.ts +55 -0
  142. package/src/cli/ui.ts +84 -0
  143. package/src/cli/upgrade.ts +25 -0
  144. package/src/command/index.ts +80 -0
  145. package/src/command/template/initialize.txt +10 -0
  146. package/src/command/template/review.txt +97 -0
  147. package/src/config/config.ts +995 -0
  148. package/src/config/markdown.ts +41 -0
  149. package/src/env/index.ts +26 -0
  150. package/src/file/ignore.ts +83 -0
  151. package/src/file/index.ts +328 -0
  152. package/src/file/ripgrep.ts +393 -0
  153. package/src/file/time.ts +64 -0
  154. package/src/file/watcher.ts +103 -0
  155. package/src/flag/flag.ts +46 -0
  156. package/src/format/formatter.ts +315 -0
  157. package/src/format/index.ts +137 -0
  158. package/src/global/index.ts +52 -0
  159. package/src/id/id.ts +73 -0
  160. package/src/ide/index.ts +76 -0
  161. package/src/index.ts +217 -0
  162. package/src/installation/index.ts +196 -0
  163. package/src/lsp/client.ts +229 -0
  164. package/src/lsp/index.ts +485 -0
  165. package/src/lsp/language.ts +116 -0
  166. package/src/lsp/server.ts +1895 -0
  167. package/src/mcp/auth.ts +135 -0
  168. package/src/mcp/index.ts +654 -0
  169. package/src/mcp/oauth-callback.ts +200 -0
  170. package/src/mcp/oauth-provider.ts +154 -0
  171. package/src/patch/index.ts +622 -0
  172. package/src/permission/index.ts +199 -0
  173. package/src/plugin/index.ts +101 -0
  174. package/src/project/bootstrap.ts +31 -0
  175. package/src/project/instance.ts +78 -0
  176. package/src/project/project.ts +221 -0
  177. package/src/project/state.ts +65 -0
  178. package/src/project/vcs.ts +76 -0
  179. package/src/provider/auth.ts +143 -0
  180. package/src/provider/models-macro.ts +11 -0
  181. package/src/provider/models.ts +106 -0
  182. package/src/provider/provider.ts +1071 -0
  183. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  184. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  185. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +101 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  190. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  191. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  192. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  193. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  200. package/src/provider/transform.ts +455 -0
  201. package/src/pty/index.ts +231 -0
  202. package/src/server/error.ts +36 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2642 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +223 -0
  207. package/src/session/index.ts +458 -0
  208. package/src/session/llm.ts +201 -0
  209. package/src/session/message-v2.ts +659 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +409 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +104 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/polaris.txt +106 -0
  224. package/src/session/prompt/qwen.txt +109 -0
  225. package/src/session/prompt.ts +1446 -0
  226. package/src/session/retry.ts +86 -0
  227. package/src/session/revert.ts +108 -0
  228. package/src/session/status.ts +76 -0
  229. package/src/session/summary.ts +194 -0
  230. package/src/session/system.ts +120 -0
  231. package/src/session/todo.ts +37 -0
  232. package/src/share/share-next.ts +194 -0
  233. package/src/share/share.ts +87 -0
  234. package/src/shell/shell.ts +67 -0
  235. package/src/skill/index.ts +1 -0
  236. package/src/skill/skill.ts +83 -0
  237. package/src/snapshot/index.ts +197 -0
  238. package/src/storage/storage.ts +226 -0
  239. package/src/tool/bash.ts +306 -0
  240. package/src/tool/bash.txt +158 -0
  241. package/src/tool/batch.ts +175 -0
  242. package/src/tool/batch.txt +24 -0
  243. package/src/tool/codesearch.ts +138 -0
  244. package/src/tool/codesearch.txt +12 -0
  245. package/src/tool/edit.ts +675 -0
  246. package/src/tool/edit.txt +10 -0
  247. package/src/tool/glob.ts +65 -0
  248. package/src/tool/glob.txt +6 -0
  249. package/src/tool/grep.ts +121 -0
  250. package/src/tool/grep.txt +8 -0
  251. package/src/tool/invalid.ts +17 -0
  252. package/src/tool/ls.ts +110 -0
  253. package/src/tool/ls.txt +1 -0
  254. package/src/tool/lsp-diagnostics.ts +26 -0
  255. package/src/tool/lsp-diagnostics.txt +1 -0
  256. package/src/tool/lsp-hover.ts +31 -0
  257. package/src/tool/lsp-hover.txt +1 -0
  258. package/src/tool/lsp.ts +87 -0
  259. package/src/tool/lsp.txt +19 -0
  260. package/src/tool/multiedit.ts +46 -0
  261. package/src/tool/multiedit.txt +41 -0
  262. package/src/tool/patch.ts +233 -0
  263. package/src/tool/patch.txt +1 -0
  264. package/src/tool/read.ts +219 -0
  265. package/src/tool/read.txt +12 -0
  266. package/src/tool/registry.ts +162 -0
  267. package/src/tool/skill.ts +100 -0
  268. package/src/tool/task.ts +136 -0
  269. package/src/tool/task.txt +60 -0
  270. package/src/tool/todo.ts +39 -0
  271. package/src/tool/todoread.txt +14 -0
  272. package/src/tool/todowrite.txt +167 -0
  273. package/src/tool/tool.ts +71 -0
  274. package/src/tool/webfetch.ts +187 -0
  275. package/src/tool/webfetch.txt +13 -0
  276. package/src/tool/websearch.ts +150 -0
  277. package/src/tool/websearch.txt +11 -0
  278. package/src/tool/write.ts +110 -0
  279. package/src/tool/write.txt +8 -0
  280. package/src/util/archive.ts +16 -0
  281. package/src/util/color.ts +19 -0
  282. package/src/util/context.ts +25 -0
  283. package/src/util/defer.ts +12 -0
  284. package/src/util/eventloop.ts +20 -0
  285. package/src/util/filesystem.ts +83 -0
  286. package/src/util/fn.ts +11 -0
  287. package/src/util/iife.ts +3 -0
  288. package/src/util/keybind.ts +102 -0
  289. package/src/util/lazy.ts +11 -0
  290. package/src/util/locale.ts +81 -0
  291. package/src/util/lock.ts +98 -0
  292. package/src/util/log.ts +180 -0
  293. package/src/util/queue.ts +32 -0
  294. package/src/util/rpc.ts +42 -0
  295. package/src/util/scrap.ts +10 -0
  296. package/src/util/signal.ts +12 -0
  297. package/src/util/timeout.ts +14 -0
  298. package/src/util/token.ts +7 -0
  299. package/src/util/wildcard.ts +54 -0
  300. package/tsconfig.json +16 -0
@@ -0,0 +1,560 @@
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("bincode.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(() => {
368
+ const mixed: AutocompleteOption[] = (
369
+ store.visible === "@" ? [...agents(), ...(files() || [])] : [...commands()]
370
+ ).filter((x) => x.disabled !== true)
371
+ const currentFilter = filter()
372
+ if (!currentFilter) return mixed
373
+ const result = fuzzysort.go(currentFilter, mixed, {
374
+ keys: [(obj) => obj.display.trimEnd(), "description", (obj) => obj.aliases?.join(" ") ?? ""],
375
+ limit: 10,
376
+ scoreFn: (objResults) => {
377
+ const displayResult = objResults[0]
378
+ if (displayResult && displayResult.target.startsWith(store.visible + currentFilter)) {
379
+ return objResults.score * 2
380
+ }
381
+ return objResults.score
382
+ },
383
+ })
384
+ return result.map((arr) => arr.obj)
385
+ })
386
+
387
+ createEffect(() => {
388
+ filter()
389
+ setStore("selected", 0)
390
+ })
391
+
392
+ function move(direction: -1 | 1) {
393
+ if (!store.visible) return
394
+ if (!options().length) return
395
+ let next = store.selected + direction
396
+ if (next < 0) next = options().length - 1
397
+ if (next >= options().length) next = 0
398
+ moveTo(next)
399
+ }
400
+
401
+ function moveTo(next: number) {
402
+ setStore("selected", next)
403
+ if (!scroll) return
404
+ const viewportHeight = Math.min(height(), options().length)
405
+ const scrollBottom = scroll.scrollTop + viewportHeight
406
+ if (next < scroll.scrollTop) {
407
+ scroll.scrollBy(next - scroll.scrollTop)
408
+ } else if (next + 1 > scrollBottom) {
409
+ scroll.scrollBy(next + 1 - scrollBottom)
410
+ }
411
+ }
412
+
413
+ function select() {
414
+ const selected = options()[store.selected]
415
+ if (!selected) return
416
+ hide()
417
+ selected.onSelect?.()
418
+ }
419
+
420
+ function show(mode: "@" | "/") {
421
+ command.keybinds(false)
422
+ setStore({
423
+ visible: mode,
424
+ index: props.input().cursorOffset,
425
+ })
426
+ }
427
+
428
+ function hide() {
429
+ const text = props.input().plainText
430
+ if (store.visible === "/" && !text.endsWith(" ") && text.startsWith("/")) {
431
+ const cursor = props.input().logicalCursor
432
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
433
+ // Sync the prompt store immediately since onContentChange is async
434
+ props.setPrompt((draft) => {
435
+ draft.input = props.input().plainText
436
+ })
437
+ }
438
+ command.keybinds(true)
439
+ setStore("visible", false)
440
+ }
441
+
442
+ onMount(() => {
443
+ props.ref({
444
+ get visible() {
445
+ return store.visible
446
+ },
447
+ onInput(value) {
448
+ if (store.visible) {
449
+ if (
450
+ // Typed text before the trigger
451
+ props.input().cursorOffset <= store.index ||
452
+ // There is a space between the trigger and the cursor
453
+ props.input().getTextRange(store.index, props.input().cursorOffset).match(/\s/) ||
454
+ // "/<command>" is not the sole content
455
+ (store.visible === "/" && value.match(/^\S+\s+\S+\s*$/))
456
+ ) {
457
+ hide()
458
+ return
459
+ }
460
+ }
461
+ },
462
+ onKeyDown(e: KeyEvent) {
463
+ if (store.visible) {
464
+ const name = e.name?.toLowerCase()
465
+ const ctrlOnly = e.ctrl && !e.meta && !e.shift
466
+ const isNavUp = name === "up" || (ctrlOnly && name === "p")
467
+ const isNavDown = name === "down" || (ctrlOnly && name === "n")
468
+
469
+ if (isNavUp) {
470
+ move(-1)
471
+ e.preventDefault()
472
+ return
473
+ }
474
+ if (isNavDown) {
475
+ move(1)
476
+ e.preventDefault()
477
+ return
478
+ }
479
+ if (name === "escape") {
480
+ hide()
481
+ e.preventDefault()
482
+ return
483
+ }
484
+ if (name === "return" || name === "tab") {
485
+ select()
486
+ e.preventDefault()
487
+ return
488
+ }
489
+ }
490
+ if (!store.visible) {
491
+ if (e.name === "@") {
492
+ const cursorOffset = props.input().cursorOffset
493
+ const charBeforeCursor =
494
+ cursorOffset === 0 ? undefined : props.input().getTextRange(cursorOffset - 1, cursorOffset)
495
+ const canTrigger = charBeforeCursor === undefined || charBeforeCursor === "" || /\s/.test(charBeforeCursor)
496
+ if (canTrigger) show("@")
497
+ }
498
+
499
+ if (e.name === "/") {
500
+ if (props.input().cursorOffset === 0) show("/")
501
+ }
502
+ }
503
+ },
504
+ })
505
+ })
506
+
507
+ const height = createMemo(() => {
508
+ if (options().length) return Math.min(10, options().length)
509
+ return 1
510
+ })
511
+
512
+ let scroll: ScrollBoxRenderable
513
+
514
+ return (
515
+ <box
516
+ visible={store.visible !== false}
517
+ position="absolute"
518
+ top={position().y - height()}
519
+ left={position().x}
520
+ width={position().width}
521
+ zIndex={100}
522
+ {...SplitBorder}
523
+ borderColor={theme.border}
524
+ >
525
+ <scrollbox
526
+ ref={(r: ScrollBoxRenderable) => (scroll = r)}
527
+ backgroundColor={theme.backgroundMenu}
528
+ height={height()}
529
+ scrollbarOptions={{ visible: false }}
530
+ >
531
+ <For
532
+ each={options()}
533
+ fallback={
534
+ <box paddingLeft={1} paddingRight={1}>
535
+ <text fg={theme.textMuted}>No matching items</text>
536
+ </box>
537
+ }
538
+ >
539
+ {(option, index) => (
540
+ <box
541
+ paddingLeft={1}
542
+ paddingRight={1}
543
+ backgroundColor={index() === store.selected ? theme.primary : undefined}
544
+ flexDirection="row"
545
+ >
546
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.text} flexShrink={0}>
547
+ {option.display}
548
+ </text>
549
+ <Show when={option.description}>
550
+ <text fg={index() === store.selected ? selectedForeground(theme) : theme.textMuted} wrapMode="none">
551
+ {option.description}
552
+ </text>
553
+ </Show>
554
+ </box>
555
+ )}
556
+ </For>
557
+ </scrollbox>
558
+ </box>
559
+ )
560
+ }
@@ -0,0 +1,108 @@
1
+ import path from "path"
2
+ import { Global } from "@/global"
3
+ import { onMount } from "solid-js"
4
+ import { createStore, produce } from "solid-js/store"
5
+ import { clone } from "remeda"
6
+ import { createSimpleContext } from "../../context/helper"
7
+ import { appendFile, writeFile } from "fs/promises"
8
+ import type { AgentPart, FilePart, TextPart } from "@bincode-ai/sdk/v2"
9
+
10
+ export type PromptInfo = {
11
+ input: string
12
+ mode?: "normal" | "shell"
13
+ parts: (
14
+ | Omit<FilePart, "id" | "messageID" | "sessionID">
15
+ | Omit<AgentPart, "id" | "messageID" | "sessionID">
16
+ | (Omit<TextPart, "id" | "messageID" | "sessionID"> & {
17
+ source?: {
18
+ text: {
19
+ start: number
20
+ end: number
21
+ value: string
22
+ }
23
+ }
24
+ })
25
+ )[]
26
+ }
27
+
28
+ const MAX_HISTORY_ENTRIES = 50
29
+
30
+ export const { use: usePromptHistory, provider: PromptHistoryProvider } = createSimpleContext({
31
+ name: "PromptHistory",
32
+ init: () => {
33
+ const historyFile = Bun.file(path.join(Global.Path.state, "prompt-history.jsonl"))
34
+ onMount(async () => {
35
+ const text = await historyFile.text().catch(() => "")
36
+ const lines = text
37
+ .split("\n")
38
+ .filter(Boolean)
39
+ .map((line) => {
40
+ try {
41
+ return JSON.parse(line)
42
+ } catch {
43
+ return null
44
+ }
45
+ })
46
+ .filter((line): line is PromptInfo => line !== null)
47
+ .slice(-MAX_HISTORY_ENTRIES)
48
+
49
+ setStore("history", lines)
50
+
51
+ // Rewrite file with only valid entries to self-heal corruption
52
+ if (lines.length > 0) {
53
+ const content = lines.map((line) => JSON.stringify(line)).join("\n") + "\n"
54
+ writeFile(historyFile.name!, content).catch(() => {})
55
+ }
56
+ })
57
+
58
+ const [store, setStore] = createStore({
59
+ index: 0,
60
+ history: [] as PromptInfo[],
61
+ })
62
+
63
+ return {
64
+ move(direction: 1 | -1, input: string) {
65
+ if (!store.history.length) return undefined
66
+ const current = store.history.at(store.index)
67
+ if (!current) return undefined
68
+ if (current.input !== input && input.length) return
69
+ setStore(
70
+ produce((draft) => {
71
+ const next = store.index + direction
72
+ if (Math.abs(next) > store.history.length) return
73
+ if (next > 0) return
74
+ draft.index = next
75
+ }),
76
+ )
77
+ if (store.index === 0)
78
+ return {
79
+ input: "",
80
+ parts: [],
81
+ }
82
+ return store.history.at(store.index)
83
+ },
84
+ append(item: PromptInfo) {
85
+ const entry = clone(item)
86
+ let trimmed = false
87
+ setStore(
88
+ produce((draft) => {
89
+ draft.history.push(entry)
90
+ if (draft.history.length > MAX_HISTORY_ENTRIES) {
91
+ draft.history = draft.history.slice(-MAX_HISTORY_ENTRIES)
92
+ trimmed = true
93
+ }
94
+ draft.index = 0
95
+ }),
96
+ )
97
+
98
+ if (trimmed) {
99
+ const content = store.history.map((line) => JSON.stringify(line)).join("\n") + "\n"
100
+ writeFile(historyFile.name!, content).catch(() => {})
101
+ return
102
+ }
103
+
104
+ appendFile(historyFile.name!, JSON.stringify(entry) + "\n").catch(() => {})
105
+ },
106
+ }
107
+ },
108
+ })