@nghyane/arcane 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (738) hide show
  1. package/CHANGELOG.md +3 -0
  2. package/README.md +12 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +109 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/custom-tools/todo/index.ts +206 -0
  7. package/examples/extensions/README.md +143 -0
  8. package/examples/extensions/api-demo.ts +89 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +32 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +550 -0
  13. package/examples/extensions/reload-runtime.ts +37 -0
  14. package/examples/extensions/todo.ts +296 -0
  15. package/examples/extensions/tools.ts +144 -0
  16. package/examples/extensions/with-deps/index.ts +35 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +16 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +116 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +150 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +119 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +43 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +80 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +36 -0
  39. package/examples/sdk/08-slash-commands.ts +41 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/README.md +150 -0
  43. package/package.json +464 -0
  44. package/scripts/format-prompts.ts +184 -0
  45. package/scripts/generate-docs-index.ts +40 -0
  46. package/scripts/generate-template.ts +32 -0
  47. package/src/bun-imports.d.ts +22 -0
  48. package/src/capability/context-file.ts +39 -0
  49. package/src/capability/extension-module.ts +33 -0
  50. package/src/capability/extension.ts +47 -0
  51. package/src/capability/fs.ts +89 -0
  52. package/src/capability/hook.ts +39 -0
  53. package/src/capability/index.ts +432 -0
  54. package/src/capability/instruction.ts +36 -0
  55. package/src/capability/mcp.ts +60 -0
  56. package/src/capability/prompt.ts +34 -0
  57. package/src/capability/rule.ts +223 -0
  58. package/src/capability/settings.ts +34 -0
  59. package/src/capability/skill.ts +48 -0
  60. package/src/capability/slash-command.ts +39 -0
  61. package/src/capability/ssh.ts +41 -0
  62. package/src/capability/system-prompt.ts +34 -0
  63. package/src/capability/tool.ts +37 -0
  64. package/src/capability/types.ts +156 -0
  65. package/src/cli/args.ts +259 -0
  66. package/src/cli/config-cli.ts +357 -0
  67. package/src/cli/file-processor.ts +124 -0
  68. package/src/cli/grep-cli.ts +152 -0
  69. package/src/cli/jupyter-cli.ts +106 -0
  70. package/src/cli/list-models.ts +103 -0
  71. package/src/cli/plugin-cli.ts +661 -0
  72. package/src/cli/session-picker.ts +42 -0
  73. package/src/cli/setup-cli.ts +376 -0
  74. package/src/cli/shell-cli.ts +174 -0
  75. package/src/cli/ssh-cli.ts +179 -0
  76. package/src/cli/stats-cli.ts +197 -0
  77. package/src/cli/update-cli.ts +286 -0
  78. package/src/cli/web-search-cli.ts +143 -0
  79. package/src/cli.ts +65 -0
  80. package/src/commands/commit.ts +36 -0
  81. package/src/commands/config.ts +51 -0
  82. package/src/commands/grep.ts +41 -0
  83. package/src/commands/jupyter.ts +32 -0
  84. package/src/commands/launch.ts +139 -0
  85. package/src/commands/plugin.ts +70 -0
  86. package/src/commands/setup.ts +42 -0
  87. package/src/commands/shell.ts +29 -0
  88. package/src/commands/ssh.ts +60 -0
  89. package/src/commands/stats.ts +29 -0
  90. package/src/commands/update.ts +21 -0
  91. package/src/commands/web-search.ts +42 -0
  92. package/src/commit/agentic/agent.ts +311 -0
  93. package/src/commit/agentic/fallback.ts +96 -0
  94. package/src/commit/agentic/index.ts +359 -0
  95. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  96. package/src/commit/agentic/prompts/session-user.md +25 -0
  97. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  98. package/src/commit/agentic/prompts/system.md +38 -0
  99. package/src/commit/agentic/state.ts +69 -0
  100. package/src/commit/agentic/tools/analyze-file.ts +118 -0
  101. package/src/commit/agentic/tools/git-file-diff.ts +194 -0
  102. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  103. package/src/commit/agentic/tools/git-overview.ts +84 -0
  104. package/src/commit/agentic/tools/index.ts +56 -0
  105. package/src/commit/agentic/tools/propose-changelog.ts +128 -0
  106. package/src/commit/agentic/tools/propose-commit.ts +154 -0
  107. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  108. package/src/commit/agentic/tools/split-commit.ts +280 -0
  109. package/src/commit/agentic/topo-sort.ts +44 -0
  110. package/src/commit/agentic/trivial.ts +51 -0
  111. package/src/commit/agentic/validation.ts +200 -0
  112. package/src/commit/analysis/conventional.ts +165 -0
  113. package/src/commit/analysis/index.ts +4 -0
  114. package/src/commit/analysis/scope.ts +242 -0
  115. package/src/commit/analysis/summary.ts +112 -0
  116. package/src/commit/analysis/validation.ts +66 -0
  117. package/src/commit/changelog/detect.ts +37 -0
  118. package/src/commit/changelog/generate.ts +110 -0
  119. package/src/commit/changelog/index.ts +234 -0
  120. package/src/commit/changelog/parse.ts +44 -0
  121. package/src/commit/cli.ts +93 -0
  122. package/src/commit/git/diff.ts +148 -0
  123. package/src/commit/git/errors.ts +9 -0
  124. package/src/commit/git/index.ts +211 -0
  125. package/src/commit/git/operations.ts +54 -0
  126. package/src/commit/index.ts +5 -0
  127. package/src/commit/map-reduce/index.ts +64 -0
  128. package/src/commit/map-reduce/map-phase.ts +178 -0
  129. package/src/commit/map-reduce/reduce-phase.ts +145 -0
  130. package/src/commit/map-reduce/utils.ts +9 -0
  131. package/src/commit/message.ts +11 -0
  132. package/src/commit/model-selection.ts +69 -0
  133. package/src/commit/pipeline.ts +243 -0
  134. package/src/commit/prompts/analysis-system.md +148 -0
  135. package/src/commit/prompts/analysis-user.md +38 -0
  136. package/src/commit/prompts/changelog-system.md +50 -0
  137. package/src/commit/prompts/changelog-user.md +18 -0
  138. package/src/commit/prompts/file-observer-system.md +24 -0
  139. package/src/commit/prompts/file-observer-user.md +8 -0
  140. package/src/commit/prompts/reduce-system.md +50 -0
  141. package/src/commit/prompts/reduce-user.md +17 -0
  142. package/src/commit/prompts/summary-retry.md +3 -0
  143. package/src/commit/prompts/summary-system.md +38 -0
  144. package/src/commit/prompts/summary-user.md +13 -0
  145. package/src/commit/prompts/types-description.md +2 -0
  146. package/src/commit/types.ts +109 -0
  147. package/src/commit/utils/exclusions.ts +42 -0
  148. package/src/config/file-lock.ts +121 -0
  149. package/src/config/keybindings.ts +280 -0
  150. package/src/config/model-registry.ts +1140 -0
  151. package/src/config/model-resolver.ts +812 -0
  152. package/src/config/prompt-templates.ts +526 -0
  153. package/src/config/resolve-config-value.ts +92 -0
  154. package/src/config/settings-schema.ts +1236 -0
  155. package/src/config/settings.ts +706 -0
  156. package/src/config.ts +414 -0
  157. package/src/cursor.ts +239 -0
  158. package/src/debug/index.ts +431 -0
  159. package/src/debug/log-formatting.ts +60 -0
  160. package/src/debug/log-viewer.ts +903 -0
  161. package/src/debug/profiler.ts +158 -0
  162. package/src/debug/report-bundle.ts +366 -0
  163. package/src/debug/system-info.ts +112 -0
  164. package/src/discovery/agents-md.ts +68 -0
  165. package/src/discovery/agents.ts +199 -0
  166. package/src/discovery/builtin.ts +815 -0
  167. package/src/discovery/claude-plugins.ts +205 -0
  168. package/src/discovery/claude.ts +506 -0
  169. package/src/discovery/cline.ts +83 -0
  170. package/src/discovery/codex.ts +532 -0
  171. package/src/discovery/cursor.ts +218 -0
  172. package/src/discovery/gemini.ts +395 -0
  173. package/src/discovery/github.ts +117 -0
  174. package/src/discovery/helpers.ts +698 -0
  175. package/src/discovery/index.ts +89 -0
  176. package/src/discovery/mcp-json.ts +156 -0
  177. package/src/discovery/opencode.ts +394 -0
  178. package/src/discovery/ssh.ts +160 -0
  179. package/src/discovery/vscode.ts +103 -0
  180. package/src/discovery/windsurf.ts +145 -0
  181. package/src/exa/company.ts +57 -0
  182. package/src/exa/index.ts +62 -0
  183. package/src/exa/linkedin.ts +57 -0
  184. package/src/exa/mcp-client.ts +289 -0
  185. package/src/exa/render.ts +244 -0
  186. package/src/exa/researcher.ts +89 -0
  187. package/src/exa/search.ts +330 -0
  188. package/src/exa/types.ts +166 -0
  189. package/src/exa/websets.ts +247 -0
  190. package/src/exec/bash-executor.ts +184 -0
  191. package/src/exec/exec.ts +53 -0
  192. package/src/export/custom-share.ts +65 -0
  193. package/src/export/html/index.ts +162 -0
  194. package/src/export/html/template.css +889 -0
  195. package/src/export/html/template.generated.ts +2 -0
  196. package/src/export/html/template.html +45 -0
  197. package/src/export/html/template.js +1329 -0
  198. package/src/export/html/template.macro.ts +24 -0
  199. package/src/export/html/vendor/highlight.min.js +1213 -0
  200. package/src/export/html/vendor/marked.min.js +6 -0
  201. package/src/export/ttsr.ts +434 -0
  202. package/src/extensibility/custom-commands/bundled/review/index.ts +433 -0
  203. package/src/extensibility/custom-commands/index.ts +15 -0
  204. package/src/extensibility/custom-commands/loader.ts +231 -0
  205. package/src/extensibility/custom-commands/types.ts +111 -0
  206. package/src/extensibility/custom-tools/index.ts +22 -0
  207. package/src/extensibility/custom-tools/loader.ts +235 -0
  208. package/src/extensibility/custom-tools/types.ts +226 -0
  209. package/src/extensibility/custom-tools/wrapper.ts +45 -0
  210. package/src/extensibility/extensions/index.ts +136 -0
  211. package/src/extensibility/extensions/loader.ts +520 -0
  212. package/src/extensibility/extensions/runner.ts +774 -0
  213. package/src/extensibility/extensions/types.ts +1293 -0
  214. package/src/extensibility/extensions/wrapper.ts +188 -0
  215. package/src/extensibility/hooks/index.ts +16 -0
  216. package/src/extensibility/hooks/loader.ts +273 -0
  217. package/src/extensibility/hooks/runner.ts +441 -0
  218. package/src/extensibility/hooks/tool-wrapper.ts +106 -0
  219. package/src/extensibility/hooks/types.ts +817 -0
  220. package/src/extensibility/plugins/doctor.ts +65 -0
  221. package/src/extensibility/plugins/git-url.ts +281 -0
  222. package/src/extensibility/plugins/index.ts +33 -0
  223. package/src/extensibility/plugins/installer.ts +192 -0
  224. package/src/extensibility/plugins/loader.ts +338 -0
  225. package/src/extensibility/plugins/manager.ts +716 -0
  226. package/src/extensibility/plugins/parser.ts +105 -0
  227. package/src/extensibility/plugins/types.ts +190 -0
  228. package/src/extensibility/skills.ts +385 -0
  229. package/src/extensibility/slash-commands.ts +287 -0
  230. package/src/extensibility/tool-proxy.ts +25 -0
  231. package/src/index.ts +275 -0
  232. package/src/internal-urls/agent-protocol.ts +136 -0
  233. package/src/internal-urls/artifact-protocol.ts +97 -0
  234. package/src/internal-urls/docs-index.generated.ts +54 -0
  235. package/src/internal-urls/docs-protocol.ts +84 -0
  236. package/src/internal-urls/index.ts +31 -0
  237. package/src/internal-urls/json-query.ts +126 -0
  238. package/src/internal-urls/memory-protocol.ts +133 -0
  239. package/src/internal-urls/router.ts +70 -0
  240. package/src/internal-urls/rule-protocol.ts +55 -0
  241. package/src/internal-urls/skill-protocol.ts +111 -0
  242. package/src/internal-urls/types.ts +52 -0
  243. package/src/ipy/executor.ts +556 -0
  244. package/src/ipy/gateway-coordinator.ts +426 -0
  245. package/src/ipy/kernel.ts +892 -0
  246. package/src/ipy/modules.ts +109 -0
  247. package/src/ipy/prelude.py +831 -0
  248. package/src/ipy/prelude.ts +3 -0
  249. package/src/ipy/runtime.ts +222 -0
  250. package/src/lsp/client.ts +867 -0
  251. package/src/lsp/clients/biome-client.ts +202 -0
  252. package/src/lsp/clients/index.ts +50 -0
  253. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  254. package/src/lsp/clients/swiftlint-client.ts +120 -0
  255. package/src/lsp/config.ts +397 -0
  256. package/src/lsp/defaults.json +464 -0
  257. package/src/lsp/edits.ts +109 -0
  258. package/src/lsp/index.ts +1268 -0
  259. package/src/lsp/lspmux.ts +250 -0
  260. package/src/lsp/render.ts +689 -0
  261. package/src/lsp/types.ts +414 -0
  262. package/src/lsp/utils.ts +549 -0
  263. package/src/main.ts +773 -0
  264. package/src/mcp/client.ts +239 -0
  265. package/src/mcp/config-writer.ts +215 -0
  266. package/src/mcp/config.ts +363 -0
  267. package/src/mcp/index.ts +55 -0
  268. package/src/mcp/json-rpc.ts +84 -0
  269. package/src/mcp/loader.ts +124 -0
  270. package/src/mcp/manager.ts +490 -0
  271. package/src/mcp/oauth-discovery.ts +274 -0
  272. package/src/mcp/oauth-flow.ts +229 -0
  273. package/src/mcp/render.ts +123 -0
  274. package/src/mcp/tool-bridge.ts +372 -0
  275. package/src/mcp/tool-cache.ts +121 -0
  276. package/src/mcp/transports/http.ts +332 -0
  277. package/src/mcp/transports/index.ts +6 -0
  278. package/src/mcp/transports/stdio.ts +281 -0
  279. package/src/mcp/types.ts +248 -0
  280. package/src/memories/index.ts +1099 -0
  281. package/src/memories/storage.ts +563 -0
  282. package/src/modes/components/agent-dashboard.ts +1130 -0
  283. package/src/modes/components/assistant-message.ts +144 -0
  284. package/src/modes/components/bash-execution.ts +218 -0
  285. package/src/modes/components/bordered-loader.ts +41 -0
  286. package/src/modes/components/branch-summary-message.ts +45 -0
  287. package/src/modes/components/codemode-group.ts +369 -0
  288. package/src/modes/components/compaction-summary-message.ts +51 -0
  289. package/src/modes/components/countdown-timer.ts +46 -0
  290. package/src/modes/components/custom-editor.ts +181 -0
  291. package/src/modes/components/custom-message.ts +91 -0
  292. package/src/modes/components/diff.ts +186 -0
  293. package/src/modes/components/dynamic-border.ts +25 -0
  294. package/src/modes/components/extensions/extension-dashboard.ts +325 -0
  295. package/src/modes/components/extensions/extension-list.ts +484 -0
  296. package/src/modes/components/extensions/index.ts +9 -0
  297. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  298. package/src/modes/components/extensions/state-manager.ts +586 -0
  299. package/src/modes/components/extensions/types.ts +191 -0
  300. package/src/modes/components/footer.ts +315 -0
  301. package/src/modes/components/history-search.ts +157 -0
  302. package/src/modes/components/hook-editor.ts +101 -0
  303. package/src/modes/components/hook-input.ts +72 -0
  304. package/src/modes/components/hook-message.ts +100 -0
  305. package/src/modes/components/hook-selector.ts +155 -0
  306. package/src/modes/components/index.ts +41 -0
  307. package/src/modes/components/keybinding-hints.ts +65 -0
  308. package/src/modes/components/login-dialog.ts +164 -0
  309. package/src/modes/components/mcp-add-wizard.ts +1295 -0
  310. package/src/modes/components/model-selector.ts +625 -0
  311. package/src/modes/components/oauth-selector.ts +210 -0
  312. package/src/modes/components/plugin-settings.ts +477 -0
  313. package/src/modes/components/python-execution.ts +196 -0
  314. package/src/modes/components/queue-mode-selector.ts +56 -0
  315. package/src/modes/components/read-tool-group.ts +119 -0
  316. package/src/modes/components/session-selector.ts +242 -0
  317. package/src/modes/components/settings-defs.ts +340 -0
  318. package/src/modes/components/settings-selector.ts +529 -0
  319. package/src/modes/components/show-images-selector.ts +45 -0
  320. package/src/modes/components/skill-message.ts +90 -0
  321. package/src/modes/components/status-line/index.ts +4 -0
  322. package/src/modes/components/status-line/presets.ts +94 -0
  323. package/src/modes/components/status-line/segments.ts +352 -0
  324. package/src/modes/components/status-line/separators.ts +55 -0
  325. package/src/modes/components/status-line/types.ts +75 -0
  326. package/src/modes/components/status-line-segment-editor.ts +354 -0
  327. package/src/modes/components/status-line.ts +421 -0
  328. package/src/modes/components/theme-selector.ts +63 -0
  329. package/src/modes/components/thinking-selector.ts +64 -0
  330. package/src/modes/components/todo-display.ts +115 -0
  331. package/src/modes/components/todo-reminder.ts +40 -0
  332. package/src/modes/components/tool-execution.ts +703 -0
  333. package/src/modes/components/tree-selector.ts +904 -0
  334. package/src/modes/components/ttsr-notification.ts +80 -0
  335. package/src/modes/components/user-message-selector.ts +146 -0
  336. package/src/modes/components/user-message.ts +22 -0
  337. package/src/modes/components/visual-truncate.ts +63 -0
  338. package/src/modes/components/welcome.ts +247 -0
  339. package/src/modes/controllers/command-controller.ts +1120 -0
  340. package/src/modes/controllers/event-controller.ts +479 -0
  341. package/src/modes/controllers/extension-ui-controller.ts +778 -0
  342. package/src/modes/controllers/input-controller.ts +671 -0
  343. package/src/modes/controllers/mcp-command-controller.ts +1315 -0
  344. package/src/modes/controllers/selector-controller.ts +712 -0
  345. package/src/modes/controllers/ssh-command-controller.ts +452 -0
  346. package/src/modes/index.ts +15 -0
  347. package/src/modes/interactive-mode.ts +1027 -0
  348. package/src/modes/print-mode.ts +191 -0
  349. package/src/modes/rpc/rpc-client.ts +583 -0
  350. package/src/modes/rpc/rpc-mode.ts +700 -0
  351. package/src/modes/rpc/rpc-types.ts +236 -0
  352. package/src/modes/theme/dark.json +95 -0
  353. package/src/modes/theme/defaults/alabaster.json +93 -0
  354. package/src/modes/theme/defaults/amethyst.json +96 -0
  355. package/src/modes/theme/defaults/anthracite.json +93 -0
  356. package/src/modes/theme/defaults/basalt.json +91 -0
  357. package/src/modes/theme/defaults/birch.json +95 -0
  358. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  359. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  360. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  361. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  362. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  363. package/src/modes/theme/defaults/dark-copper.json +95 -0
  364. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  365. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  366. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  367. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  368. package/src/modes/theme/defaults/dark-ember.json +95 -0
  369. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  370. package/src/modes/theme/defaults/dark-forest.json +96 -0
  371. package/src/modes/theme/defaults/dark-github.json +105 -0
  372. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  373. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  374. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  375. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  376. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  377. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  378. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  379. package/src/modes/theme/defaults/dark-nord.json +97 -0
  380. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  381. package/src/modes/theme/defaults/dark-one.json +100 -0
  382. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  383. package/src/modes/theme/defaults/dark-reef.json +91 -0
  384. package/src/modes/theme/defaults/dark-retro.json +92 -0
  385. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  386. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  387. package/src/modes/theme/defaults/dark-slate.json +95 -0
  388. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  389. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  390. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  391. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  392. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  393. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  394. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  395. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  396. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  397. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  398. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  399. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  400. package/src/modes/theme/defaults/graphite.json +92 -0
  401. package/src/modes/theme/defaults/index.ts +195 -0
  402. package/src/modes/theme/defaults/light-arctic.json +107 -0
  403. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  404. package/src/modes/theme/defaults/light-canyon.json +91 -0
  405. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  406. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  407. package/src/modes/theme/defaults/light-coral.json +95 -0
  408. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  409. package/src/modes/theme/defaults/light-dawn.json +90 -0
  410. package/src/modes/theme/defaults/light-dunes.json +91 -0
  411. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  412. package/src/modes/theme/defaults/light-forest.json +100 -0
  413. package/src/modes/theme/defaults/light-frost.json +95 -0
  414. package/src/modes/theme/defaults/light-github.json +115 -0
  415. package/src/modes/theme/defaults/light-glacier.json +91 -0
  416. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  417. package/src/modes/theme/defaults/light-haze.json +90 -0
  418. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  419. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  420. package/src/modes/theme/defaults/light-lavender.json +95 -0
  421. package/src/modes/theme/defaults/light-meadow.json +91 -0
  422. package/src/modes/theme/defaults/light-mint.json +95 -0
  423. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  424. package/src/modes/theme/defaults/light-ocean.json +99 -0
  425. package/src/modes/theme/defaults/light-one.json +99 -0
  426. package/src/modes/theme/defaults/light-opal.json +91 -0
  427. package/src/modes/theme/defaults/light-orchard.json +91 -0
  428. package/src/modes/theme/defaults/light-paper.json +95 -0
  429. package/src/modes/theme/defaults/light-prism.json +90 -0
  430. package/src/modes/theme/defaults/light-retro.json +98 -0
  431. package/src/modes/theme/defaults/light-sand.json +95 -0
  432. package/src/modes/theme/defaults/light-savanna.json +91 -0
  433. package/src/modes/theme/defaults/light-solarized.json +102 -0
  434. package/src/modes/theme/defaults/light-soleil.json +90 -0
  435. package/src/modes/theme/defaults/light-sunset.json +99 -0
  436. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  437. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  438. package/src/modes/theme/defaults/light-wetland.json +91 -0
  439. package/src/modes/theme/defaults/light-zenith.json +89 -0
  440. package/src/modes/theme/defaults/limestone.json +94 -0
  441. package/src/modes/theme/defaults/mahogany.json +97 -0
  442. package/src/modes/theme/defaults/marble.json +93 -0
  443. package/src/modes/theme/defaults/obsidian.json +91 -0
  444. package/src/modes/theme/defaults/onyx.json +91 -0
  445. package/src/modes/theme/defaults/pearl.json +93 -0
  446. package/src/modes/theme/defaults/porcelain.json +91 -0
  447. package/src/modes/theme/defaults/quartz.json +96 -0
  448. package/src/modes/theme/defaults/sandstone.json +95 -0
  449. package/src/modes/theme/defaults/titanium.json +90 -0
  450. package/src/modes/theme/light.json +93 -0
  451. package/src/modes/theme/mermaid-cache.ts +111 -0
  452. package/src/modes/theme/theme-schema.json +429 -0
  453. package/src/modes/theme/theme.ts +2333 -0
  454. package/src/modes/types.ts +216 -0
  455. package/src/modes/utils/ui-helpers.ts +529 -0
  456. package/src/patch/applicator.ts +1482 -0
  457. package/src/patch/diff.ts +425 -0
  458. package/src/patch/fuzzy.ts +784 -0
  459. package/src/patch/hashline.ts +972 -0
  460. package/src/patch/index.ts +964 -0
  461. package/src/patch/normalize.ts +397 -0
  462. package/src/patch/normative.ts +72 -0
  463. package/src/patch/parser.ts +532 -0
  464. package/src/patch/shared.ts +400 -0
  465. package/src/patch/types.ts +292 -0
  466. package/src/priority.json +35 -0
  467. package/src/prompts/agents/explore.md +48 -0
  468. package/src/prompts/agents/frontmatter.md +9 -0
  469. package/src/prompts/agents/init.md +36 -0
  470. package/src/prompts/agents/librarian.md +53 -0
  471. package/src/prompts/agents/oracle.md +51 -0
  472. package/src/prompts/agents/reviewer.md +70 -0
  473. package/src/prompts/agents/task.md +14 -0
  474. package/src/prompts/compaction/branch-summary-context.md +5 -0
  475. package/src/prompts/compaction/branch-summary-preamble.md +2 -0
  476. package/src/prompts/compaction/branch-summary.md +30 -0
  477. package/src/prompts/compaction/compaction-short-summary.md +9 -0
  478. package/src/prompts/compaction/compaction-summary-context.md +5 -0
  479. package/src/prompts/compaction/compaction-summary.md +38 -0
  480. package/src/prompts/compaction/compaction-turn-prefix.md +17 -0
  481. package/src/prompts/compaction/compaction-update-summary.md +45 -0
  482. package/src/prompts/memories/consolidation.md +30 -0
  483. package/src/prompts/memories/read_path.md +11 -0
  484. package/src/prompts/memories/stage_one_input.md +6 -0
  485. package/src/prompts/memories/stage_one_system.md +21 -0
  486. package/src/prompts/review-request.md +64 -0
  487. package/src/prompts/system/agent-creation-architect.md +65 -0
  488. package/src/prompts/system/agent-creation-user.md +6 -0
  489. package/src/prompts/system/custom-system-prompt.md +68 -0
  490. package/src/prompts/system/file-operations.md +10 -0
  491. package/src/prompts/system/subagent-submit-reminder.md +11 -0
  492. package/src/prompts/system/subagent-system-prompt.md +31 -0
  493. package/src/prompts/system/subagent-user-prompt.md +8 -0
  494. package/src/prompts/system/summarization-system.md +3 -0
  495. package/src/prompts/system/system-prompt.md +300 -0
  496. package/src/prompts/system/title-system.md +2 -0
  497. package/src/prompts/system/ttsr-interrupt.md +7 -0
  498. package/src/prompts/system/web-search.md +28 -0
  499. package/src/prompts/tools/ask.md +44 -0
  500. package/src/prompts/tools/bash.md +24 -0
  501. package/src/prompts/tools/browser.md +33 -0
  502. package/src/prompts/tools/calculator.md +12 -0
  503. package/src/prompts/tools/explore.md +29 -0
  504. package/src/prompts/tools/fetch.md +16 -0
  505. package/src/prompts/tools/find.md +18 -0
  506. package/src/prompts/tools/gemini-image.md +23 -0
  507. package/src/prompts/tools/grep.md +28 -0
  508. package/src/prompts/tools/hashline.md +232 -0
  509. package/src/prompts/tools/librarian.md +24 -0
  510. package/src/prompts/tools/lsp.md +28 -0
  511. package/src/prompts/tools/oracle.md +26 -0
  512. package/src/prompts/tools/patch.md +74 -0
  513. package/src/prompts/tools/python.md +66 -0
  514. package/src/prompts/tools/read.md +36 -0
  515. package/src/prompts/tools/replace.md +38 -0
  516. package/src/prompts/tools/reviewer.md +41 -0
  517. package/src/prompts/tools/ssh.md +51 -0
  518. package/src/prompts/tools/task-summary.md +28 -0
  519. package/src/prompts/tools/task.md +275 -0
  520. package/src/prompts/tools/todo-write.md +65 -0
  521. package/src/prompts/tools/undo-edit.md +7 -0
  522. package/src/prompts/tools/web-search.md +19 -0
  523. package/src/prompts/tools/write.md +18 -0
  524. package/src/sdk.ts +1287 -0
  525. package/src/secrets/index.ts +116 -0
  526. package/src/secrets/obfuscator.ts +269 -0
  527. package/src/secrets/regex.ts +21 -0
  528. package/src/session/agent-session.ts +4669 -0
  529. package/src/session/agent-storage.ts +621 -0
  530. package/src/session/artifacts.ts +132 -0
  531. package/src/session/auth-storage.ts +1433 -0
  532. package/src/session/blob-store.ts +103 -0
  533. package/src/session/compaction/branch-summarization.ts +315 -0
  534. package/src/session/compaction/compaction.ts +864 -0
  535. package/src/session/compaction/index.ts +7 -0
  536. package/src/session/compaction/pruning.ts +91 -0
  537. package/src/session/compaction/utils.ts +171 -0
  538. package/src/session/history-storage.ts +170 -0
  539. package/src/session/messages.ts +317 -0
  540. package/src/session/session-manager.ts +2276 -0
  541. package/src/session/session-storage.ts +342 -0
  542. package/src/session/streaming-output.ts +565 -0
  543. package/src/slash-commands/builtin-registry.ts +439 -0
  544. package/src/ssh/config-writer.ts +183 -0
  545. package/src/ssh/connection-manager.ts +444 -0
  546. package/src/ssh/ssh-executor.ts +127 -0
  547. package/src/ssh/sshfs-mount.ts +135 -0
  548. package/src/stt/downloader.ts +71 -0
  549. package/src/stt/index.ts +3 -0
  550. package/src/stt/recorder.ts +351 -0
  551. package/src/stt/setup.ts +52 -0
  552. package/src/stt/stt-controller.ts +160 -0
  553. package/src/stt/transcribe.py +70 -0
  554. package/src/stt/transcriber.ts +91 -0
  555. package/src/system-prompt.ts +685 -0
  556. package/src/task/agents.ts +155 -0
  557. package/src/task/batch.ts +102 -0
  558. package/src/task/commands.ts +134 -0
  559. package/src/task/discovery.ts +126 -0
  560. package/src/task/executor.ts +908 -0
  561. package/src/task/index.ts +223 -0
  562. package/src/task/output-manager.ts +107 -0
  563. package/src/task/parallel.ts +84 -0
  564. package/src/task/render.ts +326 -0
  565. package/src/task/subprocess-tool-registry.ts +88 -0
  566. package/src/task/template.ts +32 -0
  567. package/src/task/types.ts +144 -0
  568. package/src/tools/ask.ts +523 -0
  569. package/src/tools/bash-interactive.ts +419 -0
  570. package/src/tools/bash-interceptor.ts +105 -0
  571. package/src/tools/bash-normalize.ts +107 -0
  572. package/src/tools/bash-skill-urls.ts +177 -0
  573. package/src/tools/bash.ts +347 -0
  574. package/src/tools/browser.ts +1374 -0
  575. package/src/tools/calculator.ts +537 -0
  576. package/src/tools/context.ts +39 -0
  577. package/src/tools/explore.ts +23 -0
  578. package/src/tools/fetch.ts +1091 -0
  579. package/src/tools/find.ts +540 -0
  580. package/src/tools/fs-cache-invalidation.ts +28 -0
  581. package/src/tools/gemini-image.ts +907 -0
  582. package/src/tools/grep.ts +489 -0
  583. package/src/tools/index.ts +337 -0
  584. package/src/tools/json-tree.ts +231 -0
  585. package/src/tools/jtd-to-json-schema.ts +247 -0
  586. package/src/tools/jtd-to-typescript.ts +198 -0
  587. package/src/tools/librarian.ts +33 -0
  588. package/src/tools/list-limit.ts +40 -0
  589. package/src/tools/notebook.ts +287 -0
  590. package/src/tools/oracle.ts +40 -0
  591. package/src/tools/output-meta.ts +459 -0
  592. package/src/tools/output-utils.ts +63 -0
  593. package/src/tools/path-utils.ts +116 -0
  594. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  595. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  596. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  597. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  598. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  599. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  600. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  601. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  602. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  603. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  604. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  605. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  606. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  607. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  608. package/src/tools/python.ts +1118 -0
  609. package/src/tools/read.ts +1193 -0
  610. package/src/tools/render-utils.ts +680 -0
  611. package/src/tools/renderers.ts +60 -0
  612. package/src/tools/reviewer-tool.ts +41 -0
  613. package/src/tools/ssh.ts +326 -0
  614. package/src/tools/subagent-tool.ts +169 -0
  615. package/src/tools/submit-result.ts +152 -0
  616. package/src/tools/todo-write.ts +255 -0
  617. package/src/tools/tool-errors.ts +92 -0
  618. package/src/tools/tool-result.ts +86 -0
  619. package/src/tools/undo-edit.ts +145 -0
  620. package/src/tools/undo-history.ts +22 -0
  621. package/src/tools/write.ts +274 -0
  622. package/src/tui/code-cell.ts +108 -0
  623. package/src/tui/file-list.ts +47 -0
  624. package/src/tui/index.ts +11 -0
  625. package/src/tui/output-block.ts +144 -0
  626. package/src/tui/status-line.ts +39 -0
  627. package/src/tui/tree-list.ts +53 -0
  628. package/src/tui/types.ts +16 -0
  629. package/src/tui/utils.ts +116 -0
  630. package/src/utils/changelog.ts +98 -0
  631. package/src/utils/event-bus.ts +33 -0
  632. package/src/utils/external-editor.ts +59 -0
  633. package/src/utils/file-display-mode.ts +36 -0
  634. package/src/utils/file-mentions.ts +384 -0
  635. package/src/utils/frontmatter.ts +101 -0
  636. package/src/utils/fuzzy.ts +108 -0
  637. package/src/utils/ignore-files.ts +119 -0
  638. package/src/utils/image-convert.ts +27 -0
  639. package/src/utils/image-resize.ts +236 -0
  640. package/src/utils/mime.ts +30 -0
  641. package/src/utils/open.ts +20 -0
  642. package/src/utils/shell-snapshot.ts +199 -0
  643. package/src/utils/timings.ts +26 -0
  644. package/src/utils/title-generator.ts +167 -0
  645. package/src/utils/tools-manager.ts +362 -0
  646. package/src/web/scrapers/artifacthub.ts +215 -0
  647. package/src/web/scrapers/arxiv.ts +88 -0
  648. package/src/web/scrapers/aur.ts +175 -0
  649. package/src/web/scrapers/biorxiv.ts +141 -0
  650. package/src/web/scrapers/bluesky.ts +284 -0
  651. package/src/web/scrapers/brew.ts +177 -0
  652. package/src/web/scrapers/cheatsh.ts +78 -0
  653. package/src/web/scrapers/chocolatey.ts +158 -0
  654. package/src/web/scrapers/choosealicense.ts +110 -0
  655. package/src/web/scrapers/cisa-kev.ts +100 -0
  656. package/src/web/scrapers/clojars.ts +180 -0
  657. package/src/web/scrapers/coingecko.ts +184 -0
  658. package/src/web/scrapers/crates-io.ts +128 -0
  659. package/src/web/scrapers/crossref.ts +149 -0
  660. package/src/web/scrapers/devto.ts +177 -0
  661. package/src/web/scrapers/discogs.ts +307 -0
  662. package/src/web/scrapers/discourse.ts +221 -0
  663. package/src/web/scrapers/dockerhub.ts +160 -0
  664. package/src/web/scrapers/fdroid.ts +158 -0
  665. package/src/web/scrapers/firefox-addons.ts +214 -0
  666. package/src/web/scrapers/flathub.ts +239 -0
  667. package/src/web/scrapers/github-gist.ts +68 -0
  668. package/src/web/scrapers/github.ts +490 -0
  669. package/src/web/scrapers/gitlab.ts +456 -0
  670. package/src/web/scrapers/go-pkg.ts +275 -0
  671. package/src/web/scrapers/hackage.ts +94 -0
  672. package/src/web/scrapers/hackernews.ts +208 -0
  673. package/src/web/scrapers/hex.ts +121 -0
  674. package/src/web/scrapers/huggingface.ts +385 -0
  675. package/src/web/scrapers/iacr.ts +86 -0
  676. package/src/web/scrapers/index.ts +249 -0
  677. package/src/web/scrapers/jetbrains-marketplace.ts +169 -0
  678. package/src/web/scrapers/lemmy.ts +220 -0
  679. package/src/web/scrapers/lobsters.ts +186 -0
  680. package/src/web/scrapers/mastodon.ts +310 -0
  681. package/src/web/scrapers/maven.ts +152 -0
  682. package/src/web/scrapers/mdn.ts +172 -0
  683. package/src/web/scrapers/metacpan.ts +253 -0
  684. package/src/web/scrapers/musicbrainz.ts +272 -0
  685. package/src/web/scrapers/npm.ts +114 -0
  686. package/src/web/scrapers/nuget.ts +205 -0
  687. package/src/web/scrapers/nvd.ts +243 -0
  688. package/src/web/scrapers/ollama.ts +265 -0
  689. package/src/web/scrapers/open-vsx.ts +119 -0
  690. package/src/web/scrapers/opencorporates.ts +275 -0
  691. package/src/web/scrapers/openlibrary.ts +319 -0
  692. package/src/web/scrapers/orcid.ts +298 -0
  693. package/src/web/scrapers/osv.ts +192 -0
  694. package/src/web/scrapers/packagist.ts +174 -0
  695. package/src/web/scrapers/pub-dev.ts +185 -0
  696. package/src/web/scrapers/pubmed.ts +177 -0
  697. package/src/web/scrapers/pypi.ts +129 -0
  698. package/src/web/scrapers/rawg.ts +124 -0
  699. package/src/web/scrapers/readthedocs.ts +125 -0
  700. package/src/web/scrapers/reddit.ts +104 -0
  701. package/src/web/scrapers/repology.ts +262 -0
  702. package/src/web/scrapers/rfc.ts +209 -0
  703. package/src/web/scrapers/rubygems.ts +117 -0
  704. package/src/web/scrapers/searchcode.ts +217 -0
  705. package/src/web/scrapers/sec-edgar.ts +274 -0
  706. package/src/web/scrapers/semantic-scholar.ts +190 -0
  707. package/src/web/scrapers/snapcraft.ts +200 -0
  708. package/src/web/scrapers/sourcegraph.ts +373 -0
  709. package/src/web/scrapers/spdx.ts +121 -0
  710. package/src/web/scrapers/spotify.ts +217 -0
  711. package/src/web/scrapers/stackoverflow.ts +124 -0
  712. package/src/web/scrapers/terraform.ts +304 -0
  713. package/src/web/scrapers/tldr.ts +51 -0
  714. package/src/web/scrapers/twitter.ts +97 -0
  715. package/src/web/scrapers/types.ts +200 -0
  716. package/src/web/scrapers/utils.ts +142 -0
  717. package/src/web/scrapers/vimeo.ts +152 -0
  718. package/src/web/scrapers/vscode-marketplace.ts +195 -0
  719. package/src/web/scrapers/w3c.ts +163 -0
  720. package/src/web/scrapers/wikidata.ts +357 -0
  721. package/src/web/scrapers/wikipedia.ts +95 -0
  722. package/src/web/scrapers/youtube.ts +312 -0
  723. package/src/web/search/auth.ts +178 -0
  724. package/src/web/search/index.ts +598 -0
  725. package/src/web/search/provider.ts +77 -0
  726. package/src/web/search/providers/anthropic.ts +284 -0
  727. package/src/web/search/providers/base.ts +22 -0
  728. package/src/web/search/providers/brave.ts +165 -0
  729. package/src/web/search/providers/codex.ts +377 -0
  730. package/src/web/search/providers/exa.ts +158 -0
  731. package/src/web/search/providers/gemini.ts +437 -0
  732. package/src/web/search/providers/jina.ts +99 -0
  733. package/src/web/search/providers/kimi.ts +196 -0
  734. package/src/web/search/providers/perplexity.ts +546 -0
  735. package/src/web/search/providers/synthetic.ts +136 -0
  736. package/src/web/search/providers/zai.ts +352 -0
  737. package/src/web/search/render.ts +299 -0
  738. package/src/web/search/types.ts +437 -0
@@ -0,0 +1,904 @@
1
+ import {
2
+ type Component,
3
+ Container,
4
+ Input,
5
+ matchesKey,
6
+ Spacer,
7
+ Text,
8
+ TruncatedText,
9
+ truncateToWidth,
10
+ } from "@nghyane/arcane-tui";
11
+ import { theme } from "../../modes/theme/theme";
12
+ import type { SessionTreeNode } from "../../session/session-manager";
13
+ import { shortenPath } from "../../tools/render-utils";
14
+ import { DynamicBorder } from "./dynamic-border";
15
+
16
+ /** Gutter info: position (displayIndent where connector was) and whether to show │ */
17
+ interface GutterInfo {
18
+ position: number; // displayIndent level where the connector was shown
19
+ show: boolean; // true = show │, false = show spaces
20
+ }
21
+
22
+ /** Flattened tree node for navigation */
23
+ interface FlatNode {
24
+ node: SessionTreeNode;
25
+ /** Indentation level (each level = 3 chars) */
26
+ indent: number;
27
+ /** Whether to show connector (├─ or └─) - true if parent has multiple children */
28
+ showConnector: boolean;
29
+ /** If showConnector, true = last sibling (└─), false = not last (├─) */
30
+ isLast: boolean;
31
+ /** Gutter info for each ancestor branch point */
32
+ gutters: GutterInfo[];
33
+ /** True if this node is a root under a virtual branching root (multiple roots) */
34
+ isVirtualRootChild: boolean;
35
+ }
36
+
37
+ /** Filter mode for tree display */
38
+ type FilterMode = "default" | "no-tools" | "user-only" | "labeled-only" | "all";
39
+
40
+ /**
41
+ * Tree list component with selection and ASCII art visualization
42
+ */
43
+ /** Tool call info for lookup */
44
+ interface ToolCallInfo {
45
+ name: string;
46
+ arguments: Record<string, unknown>;
47
+ }
48
+
49
+ class TreeList implements Component {
50
+ #flatNodes: FlatNode[] = [];
51
+ #filteredNodes: FlatNode[] = [];
52
+ #selectedIndex = 0;
53
+ #filterMode: FilterMode = "default";
54
+ #searchQuery = "";
55
+ #toolCallMap: Map<string, ToolCallInfo> = new Map();
56
+ #multipleRoots = false;
57
+ #activePathIds: Set<string> = new Set();
58
+ #lastSelectedId: string | null = null;
59
+
60
+ onSelect?: (entryId: string) => void;
61
+ onCancel?: () => void;
62
+ onLabelEdit?: (entryId: string, currentLabel: string | undefined) => void;
63
+
64
+ constructor(
65
+ tree: SessionTreeNode[],
66
+ private readonly currentLeafId: string | null,
67
+ private readonly maxVisibleLines: number,
68
+ initialSelectedId?: string,
69
+ ) {
70
+ this.#multipleRoots = tree.length > 1;
71
+ this.#flatNodes = this.#flattenTree(tree);
72
+ this.#buildActivePath();
73
+ this.#applyFilter();
74
+
75
+ // Start with initialSelectedId if provided, otherwise current leaf
76
+ const targetId = initialSelectedId ?? currentLeafId;
77
+ this.#selectedIndex = this.#findNearestVisibleIndex(targetId);
78
+ this.#lastSelectedId = this.#filteredNodes[this.#selectedIndex]?.node.entry.id ?? null;
79
+ }
80
+
81
+ /** Build the set of entry IDs on the path from root to current leaf */
82
+ #buildActivePath(): void {
83
+ this.#activePathIds.clear();
84
+ if (!this.currentLeafId) return;
85
+
86
+ // Build a map of id -> entry for parent lookup
87
+ const entryMap = new Map<string, FlatNode>();
88
+ for (const flatNode of this.#flatNodes) {
89
+ entryMap.set(flatNode.node.entry.id, flatNode);
90
+ }
91
+
92
+ // Walk from leaf to root
93
+ let currentId: string | null = this.currentLeafId;
94
+ while (currentId) {
95
+ this.#activePathIds.add(currentId);
96
+ const node = entryMap.get(currentId);
97
+ if (!node) break;
98
+ currentId = node.node.entry.parentId ?? null;
99
+ }
100
+ }
101
+
102
+ /**
103
+ * Find the index of the nearest visible entry, walking up the parent chain if needed.
104
+ * Returns the index in filteredNodes, or the last index as fallback.
105
+ */
106
+ #findNearestVisibleIndex(entryId: string | null): number {
107
+ if (this.#filteredNodes.length === 0) return 0;
108
+
109
+ // Build a map for parent lookup
110
+ const entryMap = new Map<string, FlatNode>();
111
+ for (const flatNode of this.#flatNodes) {
112
+ entryMap.set(flatNode.node.entry.id, flatNode);
113
+ }
114
+
115
+ // Build a map of visible entry IDs to their indices in filteredNodes
116
+ const visibleIdToIndex = new Map<string, number>(this.#filteredNodes.map((node, i) => [node.node.entry.id, i]));
117
+
118
+ // Walk from entryId up to root, looking for a visible entry
119
+ let currentId = entryId;
120
+ while (currentId !== null) {
121
+ const index = visibleIdToIndex.get(currentId);
122
+ if (index !== undefined) return index;
123
+ const node = entryMap.get(currentId);
124
+ if (!node) break;
125
+ currentId = node.node.entry.parentId ?? null;
126
+ }
127
+
128
+ // Fallback: last visible entry
129
+ return this.#filteredNodes.length - 1;
130
+ }
131
+
132
+ #flattenTree(roots: SessionTreeNode[]): FlatNode[] {
133
+ const result: FlatNode[] = [];
134
+ this.#toolCallMap.clear();
135
+
136
+ // Indentation rules:
137
+ // - At indent 0: stay at 0 unless parent has >1 children (then +1)
138
+ // - At indent 1: children always go to indent 2 (visual grouping of subtree)
139
+ // - At indent 2+: stay flat for single-child chains, +1 only if parent branches
140
+
141
+ // Stack items: [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild]
142
+ type StackItem = [SessionTreeNode, number, boolean, boolean, boolean, GutterInfo[], boolean];
143
+ const stack: StackItem[] = [];
144
+
145
+ // Determine which subtrees contain the active leaf (to sort current branch first)
146
+ // Use iterative post-order traversal to avoid stack overflow
147
+ const containsActive = new Map<SessionTreeNode, boolean>();
148
+ const leafId = this.currentLeafId;
149
+ {
150
+ // Build list in pre-order, then process in reverse for post-order effect
151
+ const allNodes: SessionTreeNode[] = [];
152
+ const preOrderStack: SessionTreeNode[] = [...roots];
153
+ while (preOrderStack.length > 0) {
154
+ const node = preOrderStack.pop()!;
155
+ allNodes.push(node);
156
+ // Push children in reverse so they're processed left-to-right
157
+ for (let i = node.children.length - 1; i >= 0; i--) {
158
+ preOrderStack.push(node.children[i]);
159
+ }
160
+ }
161
+ // Process in reverse (post-order): children before parents
162
+ for (let i = allNodes.length - 1; i >= 0; i--) {
163
+ const node = allNodes[i];
164
+ let has = leafId !== null && node.entry.id === leafId;
165
+ for (const child of node.children) {
166
+ if (containsActive.get(child)) {
167
+ has = true;
168
+ }
169
+ }
170
+ containsActive.set(node, has);
171
+ }
172
+ }
173
+
174
+ // Add roots in reverse order, prioritizing the one containing the active leaf
175
+ // If multiple roots, treat them as children of a virtual root that branches
176
+ const multipleRoots = roots.length > 1;
177
+ const orderedRoots = [...roots].sort((a, b) => Number(containsActive.get(b)) - Number(containsActive.get(a)));
178
+ for (let i = orderedRoots.length - 1; i >= 0; i--) {
179
+ const isLast = i === orderedRoots.length - 1;
180
+ stack.push([orderedRoots[i], multipleRoots ? 1 : 0, multipleRoots, multipleRoots, isLast, [], multipleRoots]);
181
+ }
182
+
183
+ while (stack.length > 0) {
184
+ const [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop()!;
185
+
186
+ // Extract tool calls from assistant messages for later lookup
187
+ const entry = node.entry;
188
+ if (entry.type === "message" && entry.message.role === "assistant") {
189
+ const content = (entry.message as { content?: unknown }).content;
190
+ if (Array.isArray(content)) {
191
+ for (const block of content) {
192
+ if (typeof block === "object" && block !== null && "type" in block && block.type === "toolCall") {
193
+ const tc = block as { id: string; name: string; arguments: Record<string, unknown> };
194
+ this.#toolCallMap.set(tc.id, { name: tc.name, arguments: tc.arguments });
195
+ }
196
+ }
197
+ }
198
+ }
199
+
200
+ result.push({ node, indent, showConnector, isLast, gutters, isVirtualRootChild });
201
+
202
+ const children = node.children;
203
+ const multipleChildren = children.length > 1;
204
+
205
+ // Order children so the branch containing the active leaf comes first
206
+ const orderedChildren = (() => {
207
+ const prioritized: SessionTreeNode[] = [];
208
+ const rest: SessionTreeNode[] = [];
209
+ for (const child of children) {
210
+ if (containsActive.get(child)) {
211
+ prioritized.push(child);
212
+ } else {
213
+ rest.push(child);
214
+ }
215
+ }
216
+ return [...prioritized, ...rest];
217
+ })();
218
+
219
+ // Calculate child indent
220
+ let childIndent: number;
221
+ if (multipleChildren) {
222
+ // Parent branches: children get +1
223
+ childIndent = indent + 1;
224
+ } else if (justBranched && indent > 0) {
225
+ // First generation after a branch: +1 for visual grouping
226
+ childIndent = indent + 1;
227
+ } else {
228
+ // Single-child chain: stay flat
229
+ childIndent = indent;
230
+ }
231
+
232
+ // Build gutters for children
233
+ // If this node showed a connector, add a gutter entry for descendants
234
+ // Only add gutter if connector is actually displayed (not suppressed for virtual root children)
235
+ const connectorDisplayed = showConnector && !isVirtualRootChild;
236
+ // When connector is displayed, add a gutter entry at the connector's position
237
+ // Connector is at position (displayIndent - 1), so gutter should be there too
238
+ const currentDisplayIndent = this.#multipleRoots ? Math.max(0, indent - 1) : indent;
239
+ const connectorPosition = Math.max(0, currentDisplayIndent - 1);
240
+ const childGutters: GutterInfo[] = connectorDisplayed
241
+ ? [...gutters, { position: connectorPosition, show: !isLast }]
242
+ : gutters;
243
+
244
+ // Add children in reverse order
245
+ for (let i = orderedChildren.length - 1; i >= 0; i--) {
246
+ const childIsLast = i === orderedChildren.length - 1;
247
+ stack.push([
248
+ orderedChildren[i],
249
+ childIndent,
250
+ multipleChildren,
251
+ multipleChildren,
252
+ childIsLast,
253
+ childGutters,
254
+ false,
255
+ ]);
256
+ }
257
+ }
258
+
259
+ return result;
260
+ }
261
+
262
+ #applyFilter(): void {
263
+ // Update lastSelectedId only when we have a valid selection (non-empty list)
264
+ // This preserves the selection when switching through empty filter results
265
+ if (this.#filteredNodes.length > 0) {
266
+ this.#lastSelectedId = this.#filteredNodes[this.#selectedIndex]?.node.entry.id ?? this.#lastSelectedId;
267
+ }
268
+
269
+ const searchTokens = this.#searchQuery.toLowerCase().split(/\s+/).filter(Boolean);
270
+
271
+ this.#filteredNodes = this.#flatNodes.filter(flatNode => {
272
+ const entry = flatNode.node.entry;
273
+ const isCurrentLeaf = entry.id === this.currentLeafId;
274
+
275
+ // Skip assistant messages with only tool calls (no text) unless error/aborted
276
+ // Always show current leaf so active position is visible
277
+ if (entry.type === "message" && entry.message.role === "assistant" && !isCurrentLeaf) {
278
+ const msg = entry.message as { stopReason?: string; content?: unknown };
279
+ const hasText = this.#hasTextContent(msg.content);
280
+ const isErrorOrAborted = msg.stopReason && msg.stopReason !== "stop" && msg.stopReason !== "toolUse";
281
+ // Only hide if no text AND not an error/aborted message
282
+ if (!hasText && !isErrorOrAborted) {
283
+ return false;
284
+ }
285
+ }
286
+
287
+ // Apply filter mode
288
+ let passesFilter = true;
289
+ // Entry types hidden in default view (settings/bookkeeping)
290
+ const isSettingsEntry =
291
+ entry.type === "label" ||
292
+ entry.type === "custom" ||
293
+ entry.type === "model_change" ||
294
+ entry.type === "thinking_level_change";
295
+
296
+ switch (this.#filterMode) {
297
+ case "user-only":
298
+ // Just user messages
299
+ passesFilter = entry.type === "message" && entry.message.role === "user";
300
+ break;
301
+ case "no-tools":
302
+ // Default minus tool results
303
+ passesFilter = !isSettingsEntry && !(entry.type === "message" && entry.message.role === "toolResult");
304
+ break;
305
+ case "labeled-only":
306
+ // Just labeled entries
307
+ passesFilter = flatNode.node.label !== undefined;
308
+ break;
309
+ case "all":
310
+ // Show everything
311
+ passesFilter = true;
312
+ break;
313
+ default:
314
+ // Default mode: hide settings/bookkeeping entries
315
+ passesFilter = !isSettingsEntry;
316
+ break;
317
+ }
318
+
319
+ if (!passesFilter) return false;
320
+
321
+ // Apply search filter
322
+ if (searchTokens.length > 0) {
323
+ const nodeText = this.#getSearchableText(flatNode.node).toLowerCase();
324
+ return searchTokens.every(token => nodeText.includes(token));
325
+ }
326
+
327
+ return true;
328
+ });
329
+
330
+ // Try to preserve cursor on the same node, or find nearest visible ancestor
331
+ if (this.#lastSelectedId) {
332
+ this.#selectedIndex = this.#findNearestVisibleIndex(this.#lastSelectedId);
333
+ } else if (this.#selectedIndex >= this.#filteredNodes.length) {
334
+ // Clamp index if out of bounds
335
+ this.#selectedIndex = Math.max(0, this.#filteredNodes.length - 1);
336
+ }
337
+
338
+ // Update lastSelectedId to the actual selection (may have changed due to parent walk)
339
+ if (this.#filteredNodes.length > 0) {
340
+ this.#lastSelectedId = this.#filteredNodes[this.#selectedIndex]?.node.entry.id ?? this.#lastSelectedId;
341
+ }
342
+ }
343
+
344
+ /** Get searchable text content from a node */
345
+ #getSearchableText(node: SessionTreeNode): string {
346
+ const entry = node.entry;
347
+ const parts: string[] = [];
348
+
349
+ if (node.label) {
350
+ parts.push(node.label);
351
+ }
352
+
353
+ switch (entry.type) {
354
+ case "message": {
355
+ const msg = entry.message;
356
+ parts.push(msg.role);
357
+ if ("content" in msg && msg.content) {
358
+ parts.push(this.#extractContent(msg.content));
359
+ }
360
+ if (msg.role === "bashExecution") {
361
+ const bashMsg = msg as { command?: string };
362
+ if (bashMsg.command) parts.push(bashMsg.command);
363
+ }
364
+ break;
365
+ }
366
+ case "custom_message": {
367
+ parts.push(entry.customType);
368
+ if (typeof entry.content === "string") {
369
+ parts.push(entry.content);
370
+ } else {
371
+ parts.push(this.#extractContent(entry.content));
372
+ }
373
+ break;
374
+ }
375
+ case "compaction":
376
+ parts.push("compaction");
377
+ break;
378
+ case "branch_summary":
379
+ parts.push("branch summary", entry.summary);
380
+ break;
381
+ case "model_change":
382
+ parts.push("model", entry.model);
383
+ break;
384
+ case "thinking_level_change":
385
+ parts.push("thinking", entry.thinkingLevel);
386
+ break;
387
+ case "custom":
388
+ parts.push("custom", entry.customType);
389
+ break;
390
+ case "label":
391
+ parts.push("label", entry.label ?? "");
392
+ break;
393
+ }
394
+
395
+ return parts.join(" ");
396
+ }
397
+
398
+ invalidate(): void {}
399
+
400
+ getSearchQuery(): string {
401
+ return this.#searchQuery;
402
+ }
403
+
404
+ getSelectedNode(): SessionTreeNode | undefined {
405
+ return this.#filteredNodes[this.#selectedIndex]?.node;
406
+ }
407
+
408
+ updateNodeLabel(entryId: string, label: string | undefined): void {
409
+ for (const flatNode of this.#flatNodes) {
410
+ if (flatNode.node.entry.id === entryId) {
411
+ flatNode.node.label = label;
412
+ break;
413
+ }
414
+ }
415
+ }
416
+
417
+ #getFilterLabel(): string {
418
+ switch (this.#filterMode) {
419
+ case "no-tools":
420
+ return " [no-tools]";
421
+ case "user-only":
422
+ return " [user]";
423
+ case "labeled-only":
424
+ return " [labeled]";
425
+ case "all":
426
+ return " [all]";
427
+ default:
428
+ return "";
429
+ }
430
+ }
431
+
432
+ render(width: number): string[] {
433
+ const lines: string[] = [];
434
+
435
+ if (this.#filteredNodes.length === 0) {
436
+ lines.push(truncateToWidth(theme.fg("muted", " No entries found"), width));
437
+ lines.push(truncateToWidth(theme.fg("muted", ` (0/0)${this.#getFilterLabel()}`), width));
438
+ return lines;
439
+ }
440
+
441
+ const startIndex = Math.max(
442
+ 0,
443
+ Math.min(
444
+ this.#selectedIndex - Math.floor(this.maxVisibleLines / 2),
445
+ this.#filteredNodes.length - this.maxVisibleLines,
446
+ ),
447
+ );
448
+ const endIndex = Math.min(startIndex + this.maxVisibleLines, this.#filteredNodes.length);
449
+
450
+ for (let i = startIndex; i < endIndex; i++) {
451
+ const flatNode = this.#filteredNodes[i];
452
+ const entry = flatNode.node.entry;
453
+ const isSelected = i === this.#selectedIndex;
454
+
455
+ // Build line: cursor + prefix + path marker + label + content
456
+ const cursor = isSelected ? theme.fg("accent", "› ") : " ";
457
+
458
+ // If multiple roots, shift display (roots at 0, not 1)
459
+ const displayIndent = this.#multipleRoots ? Math.max(0, flatNode.indent - 1) : flatNode.indent;
460
+
461
+ // Build prefix with gutters at their correct positions
462
+ // Each gutter has a position (displayIndent where its connector was shown)
463
+ const hasConnector = flatNode.showConnector && !flatNode.isVirtualRootChild;
464
+ const connectorSymbol = hasConnector ? (flatNode.isLast ? theme.tree.last : theme.tree.branch) : "";
465
+ const connectorChars = hasConnector ? Array.from(connectorSymbol) : [];
466
+ const connectorPosition = hasConnector ? displayIndent - 1 : -1;
467
+
468
+ // Build prefix char by char, placing gutters and connector at their positions
469
+ const totalChars = displayIndent * 3;
470
+ const prefixChars: string[] = [];
471
+ for (let i = 0; i < totalChars; i++) {
472
+ const level = Math.floor(i / 3);
473
+ const posInLevel = i % 3;
474
+
475
+ // Check if there's a gutter at this level
476
+ const gutter = flatNode.gutters.find(g => g.position === level);
477
+ if (gutter) {
478
+ if (posInLevel === 0) {
479
+ prefixChars.push(gutter.show ? theme.tree.vertical : " ");
480
+ } else {
481
+ prefixChars.push(" ");
482
+ }
483
+ } else if (hasConnector && level === connectorPosition) {
484
+ // Connector at this level
485
+ if (posInLevel === 0) {
486
+ prefixChars.push(connectorChars[0] ?? " ");
487
+ } else if (posInLevel === 1) {
488
+ prefixChars.push(connectorChars[1] ?? theme.tree.horizontal);
489
+ } else {
490
+ prefixChars.push(connectorChars[2] ?? " ");
491
+ }
492
+ } else {
493
+ prefixChars.push(" ");
494
+ }
495
+ }
496
+ const prefix = prefixChars.join("");
497
+
498
+ // Active path marker - shown right before the entry text
499
+ const isOnActivePath = this.#activePathIds.has(entry.id);
500
+ const pathMarker = isOnActivePath ? theme.fg("accent", `${theme.md.bullet} `) : "";
501
+
502
+ const label = flatNode.node.label ? theme.fg("warning", `[${flatNode.node.label}] `) : "";
503
+ const content = this.#getEntryDisplayText(flatNode.node, isSelected);
504
+
505
+ let line = cursor + theme.fg("dim", prefix) + pathMarker + label + content;
506
+ if (isSelected) {
507
+ line = theme.bg("selectedBg", line);
508
+ }
509
+ lines.push(truncateToWidth(line, width));
510
+ }
511
+
512
+ lines.push(
513
+ truncateToWidth(
514
+ theme.fg("muted", ` (${this.#selectedIndex + 1}/${this.#filteredNodes.length})${this.#getFilterLabel()}`),
515
+ width,
516
+ ),
517
+ );
518
+
519
+ return lines;
520
+ }
521
+
522
+ #getEntryDisplayText(node: SessionTreeNode, isSelected: boolean): string {
523
+ const entry = node.entry;
524
+ let result: string;
525
+
526
+ const normalize = (s: string) => s.replace(/[\n\t]/g, " ").trim();
527
+
528
+ switch (entry.type) {
529
+ case "message": {
530
+ const msg = entry.message;
531
+ const role = msg.role;
532
+ if (role === "user") {
533
+ const msgWithContent = msg as { content?: unknown };
534
+ const content = normalize(this.#extractContent(msgWithContent.content));
535
+ result = theme.fg("accent", "user: ") + content;
536
+ } else if (role === "assistant") {
537
+ const msgWithContent = msg as { content?: unknown; stopReason?: string; errorMessage?: string };
538
+ const textContent = normalize(this.#extractContent(msgWithContent.content));
539
+ if (textContent) {
540
+ result = theme.fg("success", "assistant: ") + textContent;
541
+ } else if (msgWithContent.stopReason === "aborted") {
542
+ result = theme.fg("success", "assistant: ") + theme.fg("muted", "(aborted)");
543
+ } else if (msgWithContent.errorMessage) {
544
+ const errMsg = normalize(msgWithContent.errorMessage).slice(0, 80);
545
+ result = theme.fg("success", "assistant: ") + theme.fg("error", errMsg);
546
+ } else {
547
+ result = theme.fg("success", "assistant: ") + theme.fg("muted", "(no content)");
548
+ }
549
+ } else if (role === "toolResult") {
550
+ const toolMsg = msg as { toolCallId?: string; toolName?: string };
551
+ const toolCall = toolMsg.toolCallId ? this.#toolCallMap.get(toolMsg.toolCallId) : undefined;
552
+ if (toolCall) {
553
+ result = theme.fg("muted", this.#formatToolCall(toolCall.name, toolCall.arguments));
554
+ } else {
555
+ result = theme.fg("muted", `[${toolMsg.toolName ?? "tool"}]`);
556
+ }
557
+ } else if (role === "bashExecution") {
558
+ const bashMsg = msg as { command?: string };
559
+ result = theme.fg("dim", `[bash]: ${normalize(bashMsg.command ?? "")}`);
560
+ } else {
561
+ result = theme.fg("dim", `[${role}]`);
562
+ }
563
+ break;
564
+ }
565
+ case "custom_message": {
566
+ const content =
567
+ typeof entry.content === "string"
568
+ ? entry.content
569
+ : entry.content
570
+ .filter((c): c is { type: "text"; text: string } => c.type === "text")
571
+ .map(c => c.text)
572
+ .join("");
573
+ result = theme.fg("customMessageLabel", `[${entry.customType}]: `) + normalize(content);
574
+ break;
575
+ }
576
+ case "compaction": {
577
+ const tokens = Math.round(entry.tokensBefore / 1000);
578
+ result = theme.fg("borderAccent", `[compaction: ${tokens}k tokens]`);
579
+ break;
580
+ }
581
+ case "branch_summary":
582
+ result = theme.fg("warning", `[branch summary]: `) + normalize(entry.summary);
583
+ break;
584
+ case "model_change":
585
+ result = theme.fg("dim", `[model: ${entry.model}]`);
586
+ break;
587
+ case "thinking_level_change":
588
+ result = theme.fg("dim", `[thinking: ${entry.thinkingLevel}]`);
589
+ break;
590
+ case "custom":
591
+ result = theme.fg("dim", `[custom: ${entry.customType}]`);
592
+ break;
593
+ case "label":
594
+ result = theme.fg("dim", `[label: ${entry.label ?? "(cleared)"}]`);
595
+ break;
596
+ default:
597
+ result = "";
598
+ }
599
+
600
+ return isSelected ? theme.bold(result) : result;
601
+ }
602
+
603
+ #extractContent(content: unknown): string {
604
+ const maxLen = 200;
605
+ if (typeof content === "string") return content.slice(0, maxLen);
606
+ if (Array.isArray(content)) {
607
+ let result = "";
608
+ for (const c of content) {
609
+ if (typeof c === "object" && c !== null && "type" in c && c.type === "text") {
610
+ result += (c as { text: string }).text;
611
+ if (result.length >= maxLen) return result.slice(0, maxLen);
612
+ }
613
+ }
614
+ return result;
615
+ }
616
+ return "";
617
+ }
618
+
619
+ #hasTextContent(content: unknown): boolean {
620
+ if (typeof content === "string") return content.trim().length > 0;
621
+ if (Array.isArray(content)) {
622
+ for (const c of content) {
623
+ if (typeof c === "object" && c !== null && "type" in c && c.type === "text") {
624
+ const text = (c as { text?: string }).text;
625
+ if (text && text.trim().length > 0) return true;
626
+ }
627
+ }
628
+ }
629
+ return false;
630
+ }
631
+
632
+ #formatToolCall(name: string, args: Record<string, unknown>): string {
633
+ switch (name) {
634
+ case "read": {
635
+ const path = shortenPath(String(args.path || args.file_path || ""));
636
+ const offset = args.offset as number | undefined;
637
+ const limit = args.limit as number | undefined;
638
+ let display = path;
639
+ if (offset !== undefined || limit !== undefined) {
640
+ const start = offset ?? 1;
641
+ const end = limit !== undefined ? start + limit - 1 : "";
642
+ display += `:${start}${end ? `-${end}` : ""}`;
643
+ }
644
+ return `[read: ${display}]`;
645
+ }
646
+ case "write": {
647
+ const path = shortenPath(String(args.path || args.file_path || ""));
648
+ return `[write: ${path}]`;
649
+ }
650
+ case "edit": {
651
+ const path = shortenPath(String(args.path || args.file_path || ""));
652
+ return `[edit: ${path}]`;
653
+ }
654
+ case "bash": {
655
+ const rawCmd = String(args.command || "");
656
+ const cmd = rawCmd
657
+ .replace(/[\n\t]/g, " ")
658
+ .trim()
659
+ .slice(0, 50);
660
+ return `[bash: ${cmd}${rawCmd.length > 50 ? "..." : ""}]`;
661
+ }
662
+ case "grep": {
663
+ const pattern = String(args.pattern || "");
664
+ const path = shortenPath(String(args.path || "."));
665
+ return `[grep: /${pattern}/ in ${path}]`;
666
+ }
667
+ case "find": {
668
+ const pattern = String(args.pattern || "");
669
+ const path = shortenPath(String(args.path || "."));
670
+ return `[find: ${pattern} in ${path}]`;
671
+ }
672
+ case "ls": {
673
+ const path = shortenPath(String(args.path || "."));
674
+ return `[ls: ${path}]`;
675
+ }
676
+ default: {
677
+ // Custom tool - show name and truncated JSON args
678
+ const argsStr = JSON.stringify(args).slice(0, 40);
679
+ return `[${name}: ${argsStr}${JSON.stringify(args).length > 40 ? "..." : ""}]`;
680
+ }
681
+ }
682
+ }
683
+
684
+ handleInput(keyData: string): void {
685
+ if (matchesKey(keyData, "up")) {
686
+ this.#selectedIndex = this.#selectedIndex === 0 ? this.#filteredNodes.length - 1 : this.#selectedIndex - 1;
687
+ } else if (matchesKey(keyData, "down")) {
688
+ this.#selectedIndex = this.#selectedIndex === this.#filteredNodes.length - 1 ? 0 : this.#selectedIndex + 1;
689
+ } else if (matchesKey(keyData, "left")) {
690
+ // Page up
691
+ this.#selectedIndex = Math.max(0, this.#selectedIndex - this.maxVisibleLines);
692
+ } else if (matchesKey(keyData, "right")) {
693
+ // Page down
694
+ this.#selectedIndex = Math.min(this.#filteredNodes.length - 1, this.#selectedIndex + this.maxVisibleLines);
695
+ } else if (matchesKey(keyData, "enter") || matchesKey(keyData, "return") || keyData === "\n") {
696
+ const selected = this.#filteredNodes[this.#selectedIndex];
697
+ if (selected && this.onSelect) {
698
+ this.onSelect(selected.node.entry.id);
699
+ }
700
+ } else if (matchesKey(keyData, "escape") || matchesKey(keyData, "esc")) {
701
+ if (this.#searchQuery) {
702
+ this.#searchQuery = "";
703
+ this.#applyFilter();
704
+ } else {
705
+ this.onCancel?.();
706
+ }
707
+ } else if (matchesKey(keyData, "ctrl+c")) {
708
+ this.onCancel?.();
709
+ } else if (matchesKey(keyData, "shift+ctrl+o") || matchesKey(keyData, "ctrl+shift+o")) {
710
+ // Cycle filter backwards
711
+ const modes: FilterMode[] = ["default", "no-tools", "user-only", "labeled-only", "all"];
712
+ const currentIndex = modes.indexOf(this.#filterMode);
713
+ this.#filterMode = modes[(currentIndex - 1 + modes.length) % modes.length];
714
+ this.#applyFilter();
715
+ } else if (matchesKey(keyData, "ctrl+o")) {
716
+ // Cycle filter forwards: default → no-tools → user-only → labeled-only → all → default
717
+ const modes: FilterMode[] = ["default", "no-tools", "user-only", "labeled-only", "all"];
718
+ const currentIndex = modes.indexOf(this.#filterMode);
719
+ this.#filterMode = modes[(currentIndex + 1) % modes.length];
720
+ this.#applyFilter();
721
+ } else if (matchesKey(keyData, "alt+d")) {
722
+ this.#filterMode = "default";
723
+ this.#applyFilter();
724
+ } else if (matchesKey(keyData, "alt+t")) {
725
+ this.#filterMode = "no-tools";
726
+ this.#applyFilter();
727
+ } else if (matchesKey(keyData, "alt+u")) {
728
+ this.#filterMode = "user-only";
729
+ this.#applyFilter();
730
+ } else if (matchesKey(keyData, "alt+l")) {
731
+ this.#filterMode = "labeled-only";
732
+ this.#applyFilter();
733
+ } else if (matchesKey(keyData, "alt+a")) {
734
+ this.#filterMode = "all";
735
+ this.#applyFilter();
736
+ } else if (matchesKey(keyData, "backspace")) {
737
+ if (this.#searchQuery.length > 0) {
738
+ this.#searchQuery = this.#searchQuery.slice(0, -1);
739
+ this.#applyFilter();
740
+ }
741
+ } else if (matchesKey(keyData, "shift+l") && !this.#searchQuery) {
742
+ const selected = this.#filteredNodes[this.#selectedIndex];
743
+ if (selected && this.onLabelEdit) {
744
+ this.onLabelEdit(selected.node.entry.id, selected.node.label);
745
+ }
746
+ } else {
747
+ const hasControlChars = [...keyData].some(ch => {
748
+ const code = ch.charCodeAt(0);
749
+ return code < 32 || code === 0x7f || (code >= 0x80 && code <= 0x9f);
750
+ });
751
+ if (!hasControlChars && keyData.length > 0) {
752
+ this.#searchQuery += keyData;
753
+ this.#applyFilter();
754
+ }
755
+ }
756
+ }
757
+ }
758
+
759
+ /** Component that displays the current search query */
760
+ class SearchLine implements Component {
761
+ constructor(private treeList: TreeList) {}
762
+
763
+ invalidate(): void {}
764
+
765
+ render(width: number): string[] {
766
+ const query = this.treeList.getSearchQuery();
767
+ if (query) {
768
+ return [truncateToWidth(` ${theme.fg("muted", "Search:")} ${theme.fg("accent", query)}`, width)];
769
+ }
770
+ return [truncateToWidth(` ${theme.fg("muted", "Search:")}`, width)];
771
+ }
772
+
773
+ handleInput(_keyData: string): void {}
774
+ }
775
+
776
+ /** Label input component shown when editing a label */
777
+ class LabelInput implements Component {
778
+ #input: Input;
779
+ onSubmit?: (entryId: string, label: string | undefined) => void;
780
+ onCancel?: () => void;
781
+
782
+ constructor(
783
+ private readonly entryId: string,
784
+ currentLabel: string | undefined,
785
+ ) {
786
+ this.#input = new Input();
787
+ if (currentLabel) {
788
+ this.#input.setValue(currentLabel);
789
+ }
790
+ }
791
+
792
+ invalidate(): void {}
793
+
794
+ render(width: number): string[] {
795
+ const lines: string[] = [];
796
+ const indent = " ";
797
+ const availableWidth = width - indent.length;
798
+ lines.push(truncateToWidth(`${indent}${theme.fg("muted", "Label (empty to remove):")}`, width));
799
+ lines.push(...this.#input.render(availableWidth).map(line => truncateToWidth(`${indent}${line}`, width)));
800
+ lines.push(truncateToWidth(`${indent}${theme.fg("dim", "enter: save esc: cancel")}`, width));
801
+ return lines;
802
+ }
803
+
804
+ handleInput(keyData: string): void {
805
+ if (matchesKey(keyData, "enter") || matchesKey(keyData, "return") || keyData === "\n") {
806
+ const value = this.#input.getValue().trim();
807
+ this.onSubmit?.(this.entryId, value || undefined);
808
+ } else if (matchesKey(keyData, "escape") || matchesKey(keyData, "esc")) {
809
+ this.onCancel?.();
810
+ } else {
811
+ this.#input.handleInput(keyData);
812
+ }
813
+ }
814
+ }
815
+
816
+ /**
817
+ * Component that renders a session tree selector for navigation
818
+ */
819
+ export class TreeSelectorComponent extends Container {
820
+ #treeList: TreeList;
821
+ #labelInput: LabelInput | null = null;
822
+ #labelInputContainer: Container;
823
+ #treeContainer: Container;
824
+
825
+ constructor(
826
+ tree: SessionTreeNode[],
827
+ currentLeafId: string | null,
828
+ terminalHeight: number,
829
+ onSelect: (entryId: string) => void,
830
+ onCancel: () => void,
831
+ private readonly onLabelChangeCallback?: (entryId: string, label: string | undefined) => void,
832
+ ) {
833
+ super();
834
+ const maxVisibleLines = Math.max(5, Math.floor(terminalHeight / 2));
835
+
836
+ this.#treeList = new TreeList(tree, currentLeafId, maxVisibleLines);
837
+ this.#treeList.onSelect = onSelect;
838
+ this.#treeList.onCancel = onCancel;
839
+ this.#treeList.onLabelEdit = (entryId, currentLabel) => this.#showLabelInput(entryId, currentLabel);
840
+
841
+ this.#treeContainer = new Container();
842
+ this.#treeContainer.addChild(this.#treeList);
843
+
844
+ this.#labelInputContainer = new Container();
845
+
846
+ this.addChild(new Spacer(1));
847
+ this.addChild(new DynamicBorder());
848
+ this.addChild(new Text(theme.bold(" Session Tree"), 1, 0));
849
+ this.addChild(
850
+ new TruncatedText(
851
+ theme.fg(
852
+ "muted",
853
+ "Up/Down: move. Left/Right: page. Shift+L: label. Ctrl+O/Shift+Ctrl+O: filter. Alt+D/T/U/L/A: filter. Type to search",
854
+ ),
855
+ 0,
856
+ 0,
857
+ ),
858
+ );
859
+ this.addChild(new SearchLine(this.#treeList));
860
+ this.addChild(new DynamicBorder());
861
+ this.addChild(new Spacer(1));
862
+ this.addChild(this.#treeContainer);
863
+ this.addChild(this.#labelInputContainer);
864
+ this.addChild(new Spacer(1));
865
+ this.addChild(new DynamicBorder());
866
+
867
+ if (tree.length === 0) {
868
+ setTimeout(() => onCancel(), 100);
869
+ }
870
+ }
871
+
872
+ #showLabelInput(entryId: string, currentLabel: string | undefined): void {
873
+ this.#labelInput = new LabelInput(entryId, currentLabel);
874
+ this.#labelInput.onSubmit = (id, label) => {
875
+ this.#treeList.updateNodeLabel(id, label);
876
+ this.onLabelChangeCallback?.(id, label);
877
+ this.#hideLabelInput();
878
+ };
879
+ this.#labelInput.onCancel = () => this.#hideLabelInput();
880
+
881
+ this.#treeContainer.clear();
882
+ this.#labelInputContainer.clear();
883
+ this.#labelInputContainer.addChild(this.#labelInput);
884
+ }
885
+
886
+ #hideLabelInput(): void {
887
+ this.#labelInput = null;
888
+ this.#labelInputContainer.clear();
889
+ this.#treeContainer.clear();
890
+ this.#treeContainer.addChild(this.#treeList);
891
+ }
892
+
893
+ handleInput(keyData: string): void {
894
+ if (this.#labelInput) {
895
+ this.#labelInput.handleInput(keyData);
896
+ } else {
897
+ this.#treeList.handleInput(keyData);
898
+ }
899
+ }
900
+
901
+ getTreeList(): TreeList {
902
+ return this.#treeList;
903
+ }
904
+ }