shortcutxl 0.2.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 (423) hide show
  1. package/README.md +59 -0
  2. package/agent-docs/README.md +397 -0
  3. package/agent-docs/docs/compaction.md +390 -0
  4. package/agent-docs/docs/custom-provider.md +580 -0
  5. package/agent-docs/docs/development.md +69 -0
  6. package/agent-docs/docs/extensions.md +1971 -0
  7. package/agent-docs/docs/json.md +79 -0
  8. package/agent-docs/docs/keybindings.md +174 -0
  9. package/agent-docs/docs/models.md +293 -0
  10. package/agent-docs/docs/packages.md +209 -0
  11. package/agent-docs/docs/prompt-templates.md +67 -0
  12. package/agent-docs/docs/providers.md +186 -0
  13. package/agent-docs/docs/rpc.md +1317 -0
  14. package/agent-docs/docs/sdk.md +962 -0
  15. package/agent-docs/docs/session.md +412 -0
  16. package/agent-docs/docs/settings.md +223 -0
  17. package/agent-docs/docs/shell-aliases.md +13 -0
  18. package/agent-docs/docs/skills.md +231 -0
  19. package/agent-docs/docs/terminal-setup.md +70 -0
  20. package/agent-docs/docs/termux.md +127 -0
  21. package/agent-docs/docs/themes.md +295 -0
  22. package/agent-docs/docs/tree.md +219 -0
  23. package/agent-docs/docs/tui.md +887 -0
  24. package/agent-docs/docs/windows.md +17 -0
  25. package/agent-docs/examples/README.md +25 -0
  26. package/agent-docs/examples/extensions/README.md +205 -0
  27. package/agent-docs/examples/extensions/antigravity-image-gen.ts +447 -0
  28. package/agent-docs/examples/extensions/auto-commit-on-exit.ts +49 -0
  29. package/agent-docs/examples/extensions/bash-spawn-hook.ts +30 -0
  30. package/agent-docs/examples/extensions/bookmark.ts +50 -0
  31. package/agent-docs/examples/extensions/built-in-tool-renderer.ts +256 -0
  32. package/agent-docs/examples/extensions/claude-rules.ts +86 -0
  33. package/agent-docs/examples/extensions/commands.ts +75 -0
  34. package/agent-docs/examples/extensions/confirm-destructive.ts +59 -0
  35. package/agent-docs/examples/extensions/custom-compaction.ts +126 -0
  36. package/agent-docs/examples/extensions/custom-footer.ts +63 -0
  37. package/agent-docs/examples/extensions/custom-header.ts +73 -0
  38. package/agent-docs/examples/extensions/custom-provider-anthropic/index.ts +660 -0
  39. package/agent-docs/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  40. package/agent-docs/examples/extensions/custom-provider-anthropic/package.json +19 -0
  41. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/index.ts +362 -0
  42. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  43. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/test.ts +88 -0
  44. package/agent-docs/examples/extensions/custom-provider-qwen-cli/index.ts +349 -0
  45. package/agent-docs/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
  46. package/agent-docs/examples/extensions/dirty-repo-guard.ts +56 -0
  47. package/agent-docs/examples/extensions/doom-overlay/README.md +46 -0
  48. package/agent-docs/examples/extensions/doom-overlay/doom/build.sh +152 -0
  49. package/agent-docs/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  50. package/agent-docs/examples/extensions/doom-overlay/doom-component.ts +133 -0
  51. package/agent-docs/examples/extensions/doom-overlay/doom-engine.ts +186 -0
  52. package/agent-docs/examples/extensions/doom-overlay/doom-keys.ts +108 -0
  53. package/agent-docs/examples/extensions/doom-overlay/index.ts +74 -0
  54. package/agent-docs/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  55. package/agent-docs/examples/extensions/dynamic-resources/SKILL.md +8 -0
  56. package/agent-docs/examples/extensions/dynamic-resources/dynamic.json +79 -0
  57. package/agent-docs/examples/extensions/dynamic-resources/dynamic.md +5 -0
  58. package/agent-docs/examples/extensions/dynamic-resources/index.ts +15 -0
  59. package/agent-docs/examples/extensions/dynamic-tools.ts +77 -0
  60. package/agent-docs/examples/extensions/event-bus.ts +43 -0
  61. package/agent-docs/examples/extensions/file-trigger.ts +41 -0
  62. package/agent-docs/examples/extensions/git-checkpoint.ts +53 -0
  63. package/agent-docs/examples/extensions/handoff.ts +155 -0
  64. package/agent-docs/examples/extensions/hello.ts +25 -0
  65. package/agent-docs/examples/extensions/inline-bash.ts +94 -0
  66. package/agent-docs/examples/extensions/input-transform.ts +43 -0
  67. package/agent-docs/examples/extensions/interactive-shell.ts +209 -0
  68. package/agent-docs/examples/extensions/mac-system-theme.ts +47 -0
  69. package/agent-docs/examples/extensions/message-renderer.ts +59 -0
  70. package/agent-docs/examples/extensions/minimal-mode.ts +430 -0
  71. package/agent-docs/examples/extensions/modal-editor.ts +90 -0
  72. package/agent-docs/examples/extensions/model-status.ts +31 -0
  73. package/agent-docs/examples/extensions/notify.ts +55 -0
  74. package/agent-docs/examples/extensions/overlay-qa-tests.ts +936 -0
  75. package/agent-docs/examples/extensions/overlay-test.ts +159 -0
  76. package/agent-docs/examples/extensions/permission-gate.ts +37 -0
  77. package/agent-docs/examples/extensions/pirate.ts +47 -0
  78. package/agent-docs/examples/extensions/plan-mode/README.md +65 -0
  79. package/agent-docs/examples/extensions/plan-mode/index.ts +363 -0
  80. package/agent-docs/examples/extensions/plan-mode/utils.ts +173 -0
  81. package/agent-docs/examples/extensions/preset.ts +418 -0
  82. package/agent-docs/examples/extensions/protected-paths.ts +30 -0
  83. package/agent-docs/examples/extensions/qna.ts +122 -0
  84. package/agent-docs/examples/extensions/question.ts +278 -0
  85. package/agent-docs/examples/extensions/questionnaire.ts +440 -0
  86. package/agent-docs/examples/extensions/rainbow-editor.ts +90 -0
  87. package/agent-docs/examples/extensions/reload-runtime.ts +37 -0
  88. package/agent-docs/examples/extensions/rpc-demo.ts +124 -0
  89. package/agent-docs/examples/extensions/sandbox/index.ts +324 -0
  90. package/agent-docs/examples/extensions/sandbox/package-lock.json +92 -0
  91. package/agent-docs/examples/extensions/sandbox/package.json +19 -0
  92. package/agent-docs/examples/extensions/send-user-message.ts +97 -0
  93. package/agent-docs/examples/extensions/session-name.ts +27 -0
  94. package/agent-docs/examples/extensions/shutdown-command.ts +69 -0
  95. package/agent-docs/examples/extensions/snake.ts +343 -0
  96. package/agent-docs/examples/extensions/space-invaders.ts +566 -0
  97. package/agent-docs/examples/extensions/ssh.ts +233 -0
  98. package/agent-docs/examples/extensions/status-line.ts +40 -0
  99. package/agent-docs/examples/extensions/subagent/README.md +172 -0
  100. package/agent-docs/examples/extensions/subagent/agents/planner.md +37 -0
  101. package/agent-docs/examples/extensions/subagent/agents/reviewer.md +35 -0
  102. package/agent-docs/examples/extensions/subagent/agents/scout.md +50 -0
  103. package/agent-docs/examples/extensions/subagent/agents/worker.md +24 -0
  104. package/agent-docs/examples/extensions/subagent/agents.ts +130 -0
  105. package/agent-docs/examples/extensions/subagent/index.ts +1068 -0
  106. package/agent-docs/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  107. package/agent-docs/examples/extensions/subagent/prompts/implement.md +10 -0
  108. package/agent-docs/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  109. package/agent-docs/examples/extensions/summarize.ts +206 -0
  110. package/agent-docs/examples/extensions/system-prompt-header.ts +17 -0
  111. package/agent-docs/examples/extensions/timed-confirm.ts +72 -0
  112. package/agent-docs/examples/extensions/titlebar-spinner.ts +58 -0
  113. package/agent-docs/examples/extensions/todo.ts +314 -0
  114. package/agent-docs/examples/extensions/tool-override.ts +146 -0
  115. package/agent-docs/examples/extensions/tools.ts +145 -0
  116. package/agent-docs/examples/extensions/trigger-compact.ts +40 -0
  117. package/agent-docs/examples/extensions/truncated-tool.ts +194 -0
  118. package/agent-docs/examples/extensions/widget-placement.ts +17 -0
  119. package/agent-docs/examples/extensions/with-deps/index.ts +37 -0
  120. package/agent-docs/examples/extensions/with-deps/package-lock.json +31 -0
  121. package/agent-docs/examples/extensions/with-deps/package.json +22 -0
  122. package/agent-docs/examples/rpc-extension-ui.ts +654 -0
  123. package/agent-docs/examples/sdk/01-minimal.ts +22 -0
  124. package/agent-docs/examples/sdk/02-custom-model.ts +48 -0
  125. package/agent-docs/examples/sdk/03-custom-prompt.ts +55 -0
  126. package/agent-docs/examples/sdk/04-skills.ts +53 -0
  127. package/agent-docs/examples/sdk/05-tools.ts +56 -0
  128. package/agent-docs/examples/sdk/06-extensions.ts +88 -0
  129. package/agent-docs/examples/sdk/07-context-files.ts +40 -0
  130. package/agent-docs/examples/sdk/08-prompt-templates.ts +47 -0
  131. package/agent-docs/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  132. package/agent-docs/examples/sdk/10-settings.ts +54 -0
  133. package/agent-docs/examples/sdk/11-sessions.ts +48 -0
  134. package/agent-docs/examples/sdk/12-full-control.ts +82 -0
  135. package/agent-docs/examples/sdk/README.md +144 -0
  136. package/agent-docs/xll-skill.md +61 -0
  137. package/agent-docs/xll-spec.md +110 -0
  138. package/dist/cli/args.js +290 -0
  139. package/dist/cli/config-selector.js +31 -0
  140. package/dist/cli/file-processor.js +79 -0
  141. package/dist/cli/list-models.js +92 -0
  142. package/dist/cli/package-commands.js +210 -0
  143. package/dist/cli/report-settings-errors.js +11 -0
  144. package/dist/cli/session-picker.js +34 -0
  145. package/dist/cli.js +19 -0
  146. package/dist/config.js +288 -0
  147. package/dist/core/abort.js +15 -0
  148. package/dist/core/agent-loop.js +352 -0
  149. package/dist/core/agent-session.js +2019 -0
  150. package/dist/core/agent.js +410 -0
  151. package/dist/core/auth-storage.js +456 -0
  152. package/dist/core/bash-executor.js +222 -0
  153. package/dist/core/compaction/branch-summarization.js +242 -0
  154. package/dist/core/compaction/compaction.js +610 -0
  155. package/dist/core/compaction/index.js +7 -0
  156. package/dist/core/compaction/utils.js +139 -0
  157. package/dist/core/defaults.js +6 -0
  158. package/dist/core/diagnostics.js +2 -0
  159. package/dist/core/event-bus.js +25 -0
  160. package/dist/core/exec.js +71 -0
  161. package/dist/core/export-html/ansi-to-html.js +256 -0
  162. package/dist/core/export-html/index.js +238 -0
  163. package/dist/core/export-html/session-view-model.js +342 -0
  164. package/dist/core/export-html/template.css +1110 -0
  165. package/dist/core/export-html/template.html +76 -0
  166. package/dist/core/export-html/template.js +1990 -0
  167. package/dist/core/export-html/tool-renderer.js +63 -0
  168. package/dist/core/export-html/vendor/highlight.min.js +7725 -0
  169. package/dist/core/export-html/vendor/marked.min.js +1803 -0
  170. package/dist/core/extensions/index.js +9 -0
  171. package/dist/core/extensions/loader.js +422 -0
  172. package/dist/core/extensions/runner.js +651 -0
  173. package/dist/core/extensions/types.js +35 -0
  174. package/dist/core/extensions/wrapper.js +102 -0
  175. package/dist/core/footer-data-provider.js +162 -0
  176. package/dist/core/index.js +9 -0
  177. package/dist/core/keybindings.js +153 -0
  178. package/dist/core/messages.js +133 -0
  179. package/dist/core/model-registry.js +539 -0
  180. package/dist/core/model-resolver.js +370 -0
  181. package/dist/core/package-manager.js +1485 -0
  182. package/dist/core/prompt-templates.js +253 -0
  183. package/dist/core/resolve-config-value.js +59 -0
  184. package/dist/core/resource-loader.js +700 -0
  185. package/dist/core/sdk.js +197 -0
  186. package/dist/core/session-bash.js +99 -0
  187. package/dist/core/session-compaction.js +165 -0
  188. package/dist/core/session-manager.js +1153 -0
  189. package/dist/core/session-models.js +99 -0
  190. package/dist/core/session-retry.js +155 -0
  191. package/dist/core/settings-manager.js +572 -0
  192. package/dist/core/skills.js +382 -0
  193. package/dist/core/slash-commands.js +31 -0
  194. package/dist/core/system-prompt.js +161 -0
  195. package/dist/core/theme.js +770 -0
  196. package/dist/core/timings.js +26 -0
  197. package/dist/core/tools/bash.js +258 -0
  198. package/dist/core/tools/edit-diff.js +245 -0
  199. package/dist/core/tools/edit.js +148 -0
  200. package/dist/core/tools/find.js +208 -0
  201. package/dist/core/tools/grep.js +246 -0
  202. package/dist/core/tools/index.js +67 -0
  203. package/dist/core/tools/ls.js +123 -0
  204. package/dist/core/tools/path-utils.js +81 -0
  205. package/dist/core/tools/read.js +160 -0
  206. package/dist/core/tools/truncate.js +70 -0
  207. package/dist/core/tools/write.js +82 -0
  208. package/dist/custom/agents/action.js +13 -0
  209. package/dist/custom/agents/document-reader.js +70 -0
  210. package/dist/custom/agents/general.js +26 -0
  211. package/dist/custom/agents/index.js +49 -0
  212. package/dist/custom/agents/installation.js +13 -0
  213. package/dist/custom/agents/types.js +7 -0
  214. package/dist/custom/auth/refresh-timer.js +33 -0
  215. package/dist/custom/auth/shortcut-oauth.js +145 -0
  216. package/dist/custom/constants.js +21 -0
  217. package/dist/custom/context/workbook-summary.js +73 -0
  218. package/dist/custom/credits/shortcut-credits.js +29 -0
  219. package/dist/custom/cron/cron-daemon-entry.js +18 -0
  220. package/dist/custom/cron/daemon-ipc.js +131 -0
  221. package/dist/custom/cron/daemon.js +224 -0
  222. package/dist/custom/cron/jobs.js +226 -0
  223. package/dist/custom/cron/run-log.js +51 -0
  224. package/dist/custom/cron/schedule.js +72 -0
  225. package/dist/custom/cron/status-line.js +98 -0
  226. package/dist/custom/cron/store.js +87 -0
  227. package/dist/custom/cron/types.js +8 -0
  228. package/dist/custom/dev/index.js +59 -0
  229. package/dist/custom/dev/trace-export.js +58 -0
  230. package/dist/custom/ensure-excel.js +63 -0
  231. package/dist/custom/excel-config.js +36 -0
  232. package/dist/custom/preflight.js +422 -0
  233. package/dist/custom/prompts/action.js +100 -0
  234. package/dist/custom/prompts/api.js +66 -0
  235. package/dist/custom/prompts/installation.js +124 -0
  236. package/dist/custom/prompts/shared.js +138 -0
  237. package/dist/custom/providers/llm-usage.js +42 -0
  238. package/dist/custom/providers/message-converter.js +74 -0
  239. package/dist/custom/providers/provider-ids.js +9 -0
  240. package/dist/custom/providers/register-openai-codex-provider.js +27 -0
  241. package/dist/custom/providers/register-shortcut-provider.js +52 -0
  242. package/dist/custom/providers/shortcut-invoke.js +117 -0
  243. package/dist/custom/providers/shortcut-stream.js +252 -0
  244. package/dist/custom/providers/sse-protocol.js +38 -0
  245. package/dist/custom/sync-xll.js +130 -0
  246. package/dist/custom/tools/cron.js +413 -0
  247. package/dist/custom/tools/excel-exec.js +167 -0
  248. package/dist/custom/tools/excel-range.js +50 -0
  249. package/dist/custom/tools/llm-analysis.js +265 -0
  250. package/dist/custom/tools/render-helpers.js +38 -0
  251. package/dist/custom/tools/switch-mode.js +94 -0
  252. package/dist/custom/tools/task/agents.js +6 -0
  253. package/dist/custom/tools/task/index.js +8 -0
  254. package/dist/custom/tools/task/render.js +348 -0
  255. package/dist/custom/tools/task/subprocess.js +320 -0
  256. package/dist/custom/tools/task/task.js +205 -0
  257. package/dist/custom/tools/todo-list.js +195 -0
  258. package/dist/custom/tracing/session-upload.js +93 -0
  259. package/dist/index.js +45 -0
  260. package/dist/main.js +613 -0
  261. package/dist/migrations.js +265 -0
  262. package/dist/modes/index.js +8 -0
  263. package/dist/modes/interactive/components/armin.js +337 -0
  264. package/dist/modes/interactive/components/assistant-message.js +94 -0
  265. package/dist/modes/interactive/components/bash-execution.js +171 -0
  266. package/dist/modes/interactive/components/bordered-loader.js +51 -0
  267. package/dist/modes/interactive/components/branch-summary-message.js +45 -0
  268. package/dist/modes/interactive/components/compaction-summary-message.js +46 -0
  269. package/dist/modes/interactive/components/config-selector.js +488 -0
  270. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  271. package/dist/modes/interactive/components/custom-editor.js +93 -0
  272. package/dist/modes/interactive/components/custom-message.js +81 -0
  273. package/dist/modes/interactive/components/daxnuts.js +140 -0
  274. package/dist/modes/interactive/components/diff.js +133 -0
  275. package/dist/modes/interactive/components/dynamic-border.js +21 -0
  276. package/dist/modes/interactive/components/extension-editor.js +105 -0
  277. package/dist/modes/interactive/components/extension-input.js +61 -0
  278. package/dist/modes/interactive/components/extension-selector.js +78 -0
  279. package/dist/modes/interactive/components/footer.js +309 -0
  280. package/dist/modes/interactive/components/index.js +33 -0
  281. package/dist/modes/interactive/components/keybinding-hints.js +61 -0
  282. package/dist/modes/interactive/components/layout.js +64 -0
  283. package/dist/modes/interactive/components/login-dialog.js +148 -0
  284. package/dist/modes/interactive/components/model-selector.js +237 -0
  285. package/dist/modes/interactive/components/oauth-selector.js +111 -0
  286. package/dist/modes/interactive/components/session-selector-search.js +157 -0
  287. package/dist/modes/interactive/components/session-selector.js +860 -0
  288. package/dist/modes/interactive/components/settings-selector.js +123 -0
  289. package/dist/modes/interactive/components/show-images-selector.js +35 -0
  290. package/dist/modes/interactive/components/skill-invocation-message.js +48 -0
  291. package/dist/modes/interactive/components/theme-selector.js +47 -0
  292. package/dist/modes/interactive/components/thinking-selector.js +47 -0
  293. package/dist/modes/interactive/components/tool-execution.js +789 -0
  294. package/dist/modes/interactive/components/tool-group.js +106 -0
  295. package/dist/modes/interactive/components/tree-selector.js +962 -0
  296. package/dist/modes/interactive/components/user-message-selector.js +115 -0
  297. package/dist/modes/interactive/components/user-message.js +48 -0
  298. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  299. package/dist/modes/interactive/file-attachments.js +135 -0
  300. package/dist/modes/interactive/interactive-mode.js +3775 -0
  301. package/dist/modes/interactive/theme/dark.json +85 -0
  302. package/dist/modes/interactive/theme/light.json +85 -0
  303. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  304. package/dist/modes/interactive/theme/theme.js +177 -0
  305. package/dist/modes/print-mode.js +101 -0
  306. package/dist/modes/rpc/rpc-client.js +387 -0
  307. package/dist/modes/rpc/rpc-mode.js +509 -0
  308. package/dist/modes/rpc/rpc-types.js +8 -0
  309. package/dist/subagent-entry.js +145 -0
  310. package/dist/tool-names.js +34 -0
  311. package/dist/tui/autocomplete.js +596 -0
  312. package/dist/tui/components/box.js +104 -0
  313. package/dist/tui/components/cancellable-loader.js +35 -0
  314. package/dist/tui/components/editor.js +1679 -0
  315. package/dist/tui/components/image.js +69 -0
  316. package/dist/tui/components/input.js +433 -0
  317. package/dist/tui/components/loader.js +49 -0
  318. package/dist/tui/components/markdown.js +629 -0
  319. package/dist/tui/components/select-list.js +152 -0
  320. package/dist/tui/components/settings-list.js +185 -0
  321. package/dist/tui/components/spacer.js +23 -0
  322. package/dist/tui/components/text.js +89 -0
  323. package/dist/tui/components/truncated-text.js +51 -0
  324. package/dist/tui/editor-component.js +2 -0
  325. package/dist/tui/fuzzy.js +107 -0
  326. package/dist/tui/get-east-asian-width/index.js +32 -0
  327. package/dist/tui/get-east-asian-width/lookup.js +404 -0
  328. package/dist/tui/index.js +32 -0
  329. package/dist/tui/keybindings.js +114 -0
  330. package/dist/tui/keys.js +959 -0
  331. package/dist/tui/kill-ring.js +44 -0
  332. package/dist/tui/stdin-buffer.js +317 -0
  333. package/dist/tui/terminal-image.js +288 -0
  334. package/dist/tui/terminal.js +249 -0
  335. package/dist/tui/tui/autocomplete.js +596 -0
  336. package/dist/tui/tui/components/box.js +106 -0
  337. package/dist/tui/tui/components/cancellable-loader.js +35 -0
  338. package/dist/tui/tui/components/editor.js +1679 -0
  339. package/dist/tui/tui/components/image.js +69 -0
  340. package/dist/tui/tui/components/input.js +433 -0
  341. package/dist/tui/tui/components/loader.js +49 -0
  342. package/dist/tui/tui/components/markdown.js +629 -0
  343. package/dist/tui/tui/components/select-list.js +152 -0
  344. package/dist/tui/tui/components/settings-list.js +185 -0
  345. package/dist/tui/tui/components/spacer.js +23 -0
  346. package/dist/tui/tui/components/text.js +91 -0
  347. package/dist/tui/tui/components/truncated-text.js +51 -0
  348. package/dist/tui/tui/editor-component.js +2 -0
  349. package/dist/tui/tui/fuzzy.js +107 -0
  350. package/dist/tui/tui/get-east-asian-width/index.js +32 -0
  351. package/dist/tui/tui/get-east-asian-width/lookup.js +404 -0
  352. package/dist/tui/tui/index.js +32 -0
  353. package/dist/tui/tui/keybindings.js +114 -0
  354. package/dist/tui/tui/keys.js +959 -0
  355. package/dist/tui/tui/kill-ring.js +44 -0
  356. package/dist/tui/tui/stdin-buffer.js +317 -0
  357. package/dist/tui/tui/terminal-image.js +288 -0
  358. package/dist/tui/tui/terminal.js +249 -0
  359. package/dist/tui/tui/tui.js +955 -0
  360. package/dist/tui/tui/undo-stack.js +25 -0
  361. package/dist/tui/tui/utils.js +800 -0
  362. package/dist/tui/tui.js +955 -0
  363. package/dist/tui/undo-stack.js +25 -0
  364. package/dist/tui/utils.js +800 -0
  365. package/dist/utils/changelog.js +87 -0
  366. package/dist/utils/clipboard-image.js +164 -0
  367. package/dist/utils/clipboard-native.js +14 -0
  368. package/dist/utils/clipboard.js +67 -0
  369. package/dist/utils/frontmatter.js +26 -0
  370. package/dist/utils/git.js +166 -0
  371. package/dist/utils/image-convert.js +35 -0
  372. package/dist/utils/image-resize.js +183 -0
  373. package/dist/utils/mime.js +26 -0
  374. package/dist/utils/photon.js +121 -0
  375. package/dist/utils/shell.js +217 -0
  376. package/dist/utils/sleep.js +17 -0
  377. package/dist/utils/tools-manager.js +259 -0
  378. package/package.json +78 -0
  379. package/skills/excel-com-api/SKILL.md +74 -0
  380. package/skills/excel-com-api/excel-type-library.py +27767 -0
  381. package/skills/excel-com-api/office-type-library.py +10867 -0
  382. package/skills/integrations/SKILL.md +138 -0
  383. package/skills/integrations/alphasense.md +457 -0
  384. package/skills/integrations/bloomberg.md +803 -0
  385. package/skills/integrations/calcbench.md +315 -0
  386. package/skills/integrations/capiq.md +848 -0
  387. package/skills/integrations/dynamics-365-finance.md +354 -0
  388. package/skills/integrations/earnings_transcripts.md +387 -0
  389. package/skills/integrations/factset.md +758 -0
  390. package/skills/integrations/ice-fixed-income.md +344 -0
  391. package/skills/integrations/moodys-analytics.md +313 -0
  392. package/skills/integrations/morningstar.md +433 -0
  393. package/skills/integrations/nasdaq-data-link.md +249 -0
  394. package/skills/integrations/pitchbook.md +413 -0
  395. package/skills/integrations/preqin.md +422 -0
  396. package/skills/integrations/quickbooks.md +289 -0
  397. package/skills/integrations/quickfs.md +314 -0
  398. package/skills/integrations/refinitiv.md +473 -0
  399. package/skills/integrations/sage-intacct.md +401 -0
  400. package/skills/integrations/visible-alpha.md +320 -0
  401. package/skills/integrations/xero.md +393 -0
  402. package/skills/integrations/ycharts.md +306 -0
  403. package/skills/pdf-creation/SKILL.md +93 -0
  404. package/skills/pdf-extraction/SKILL.md +32 -0
  405. package/skills/powerpoint-creation/SKILL.md +110 -0
  406. package/skills/sec-edgar/SKILL.md +127 -0
  407. package/skills/sec-edgar/sec_to_pdf.py +109 -0
  408. package/xll/ShortcutXL.xll +0 -0
  409. package/xll/modules/debug_render.py +272 -0
  410. package/xll/modules/gameboy.py +241 -0
  411. package/xll/modules/pong.py +188 -0
  412. package/xll/modules/shortcut_xl/__init__.py +18 -0
  413. package/xll/modules/shortcut_xl/_categorize.py +200 -0
  414. package/xll/modules/shortcut_xl/_com.py +108 -0
  415. package/xll/modules/shortcut_xl/_format.py +252 -0
  416. package/xll/modules/shortcut_xl/_log.py +12 -0
  417. package/xll/modules/shortcut_xl/_managed.py +116 -0
  418. package/xll/modules/shortcut_xl/_registry.py +44 -0
  419. package/xll/modules/shortcut_xl/_threading.py +161 -0
  420. package/xll/modules/shortcut_xl/_tracking.py +283 -0
  421. package/xll/modules/stocks.py +100 -0
  422. package/xll/python3.dll +0 -0
  423. package/xll/python312.dll +0 -0
@@ -0,0 +1,962 @@
1
+ import { Container, getEditorKeybindings, Input, matchesKey, Spacer, Text, TruncatedText, truncateToWidth } from '../../../tui/index.js';
2
+ import { theme } from '../../../core/theme.js';
3
+ import { DynamicBorder } from './dynamic-border.js';
4
+ import { keyHint } from './keybinding-hints.js';
5
+ class TreeList {
6
+ flatNodes = [];
7
+ filteredNodes = [];
8
+ selectedIndex = 0;
9
+ currentLeafId;
10
+ maxVisibleLines;
11
+ filterMode = 'default';
12
+ searchQuery = '';
13
+ toolCallMap = new Map();
14
+ multipleRoots = false;
15
+ activePathIds = new Set();
16
+ lastSelectedId = null;
17
+ onSelect;
18
+ onCancel;
19
+ onLabelEdit;
20
+ constructor(tree, currentLeafId, maxVisibleLines, initialSelectedId) {
21
+ this.currentLeafId = currentLeafId;
22
+ this.maxVisibleLines = maxVisibleLines;
23
+ this.multipleRoots = tree.length > 1;
24
+ this.flatNodes = this.flattenTree(tree);
25
+ this.buildActivePath();
26
+ this.applyFilter();
27
+ // Start with initialSelectedId if provided, otherwise current leaf
28
+ const targetId = initialSelectedId ?? currentLeafId;
29
+ this.selectedIndex = this.findNearestVisibleIndex(targetId);
30
+ this.lastSelectedId = this.filteredNodes[this.selectedIndex]?.node.entry.id ?? null;
31
+ }
32
+ /**
33
+ * Find the index of the nearest visible entry, walking up the parent chain if needed.
34
+ * Returns the index in filteredNodes, or the last index as fallback.
35
+ */
36
+ findNearestVisibleIndex(entryId) {
37
+ if (this.filteredNodes.length === 0)
38
+ return 0;
39
+ // Build a map for parent lookup
40
+ const entryMap = new Map();
41
+ for (const flatNode of this.flatNodes) {
42
+ entryMap.set(flatNode.node.entry.id, flatNode);
43
+ }
44
+ // Build a map of visible entry IDs to their indices in filteredNodes
45
+ const visibleIdToIndex = new Map(this.filteredNodes.map((node, i) => [node.node.entry.id, i]));
46
+ // Walk from entryId up to root, looking for a visible entry
47
+ let currentId = entryId;
48
+ while (currentId !== null) {
49
+ const index = visibleIdToIndex.get(currentId);
50
+ if (index !== undefined)
51
+ return index;
52
+ const node = entryMap.get(currentId);
53
+ if (!node)
54
+ break;
55
+ currentId = node.node.entry.parentId ?? null;
56
+ }
57
+ // Fallback: last visible entry
58
+ return this.filteredNodes.length - 1;
59
+ }
60
+ /** Build the set of entry IDs on the path from root to current leaf */
61
+ buildActivePath() {
62
+ this.activePathIds.clear();
63
+ if (!this.currentLeafId)
64
+ return;
65
+ // Build a map of id -> entry for parent lookup
66
+ const entryMap = new Map();
67
+ for (const flatNode of this.flatNodes) {
68
+ entryMap.set(flatNode.node.entry.id, flatNode);
69
+ }
70
+ // Walk from leaf to root
71
+ let currentId = this.currentLeafId;
72
+ while (currentId) {
73
+ this.activePathIds.add(currentId);
74
+ const node = entryMap.get(currentId);
75
+ if (!node)
76
+ break;
77
+ currentId = node.node.entry.parentId ?? null;
78
+ }
79
+ }
80
+ flattenTree(roots) {
81
+ const result = [];
82
+ this.toolCallMap.clear();
83
+ const stack = [];
84
+ // Determine which subtrees contain the active leaf (to sort current branch first)
85
+ // Use iterative post-order traversal to avoid stack overflow
86
+ const containsActive = new Map();
87
+ const leafId = this.currentLeafId;
88
+ {
89
+ // Build list in pre-order, then process in reverse for post-order effect
90
+ const allNodes = [];
91
+ const preOrderStack = [...roots];
92
+ while (preOrderStack.length > 0) {
93
+ const node = preOrderStack.pop();
94
+ allNodes.push(node);
95
+ // Push children in reverse so they're processed left-to-right
96
+ for (let i = node.children.length - 1; i >= 0; i--) {
97
+ preOrderStack.push(node.children[i]);
98
+ }
99
+ }
100
+ // Process in reverse (post-order): children before parents
101
+ for (let i = allNodes.length - 1; i >= 0; i--) {
102
+ const node = allNodes[i];
103
+ let has = leafId !== null && node.entry.id === leafId;
104
+ for (const child of node.children) {
105
+ if (containsActive.get(child)) {
106
+ has = true;
107
+ }
108
+ }
109
+ containsActive.set(node, has);
110
+ }
111
+ }
112
+ // Add roots in reverse order, prioritizing the one containing the active leaf
113
+ // If multiple roots, treat them as children of a virtual root that branches
114
+ const multipleRoots = roots.length > 1;
115
+ const orderedRoots = [...roots].sort((a, b) => Number(containsActive.get(b)) - Number(containsActive.get(a)));
116
+ for (let i = orderedRoots.length - 1; i >= 0; i--) {
117
+ const isLast = i === orderedRoots.length - 1;
118
+ stack.push([
119
+ orderedRoots[i],
120
+ multipleRoots ? 1 : 0,
121
+ multipleRoots,
122
+ multipleRoots,
123
+ isLast,
124
+ [],
125
+ multipleRoots
126
+ ]);
127
+ }
128
+ while (stack.length > 0) {
129
+ const [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop();
130
+ // Extract tool calls from assistant messages for later lookup
131
+ const entry = node.entry;
132
+ if (entry.type === 'message' && entry.message.role === 'assistant') {
133
+ const content = entry.message.content;
134
+ if (Array.isArray(content)) {
135
+ for (const block of content) {
136
+ if (typeof block === 'object' &&
137
+ block !== null &&
138
+ 'type' in block &&
139
+ block.type === 'toolCall') {
140
+ const tc = block;
141
+ this.toolCallMap.set(tc.id, { name: tc.name, arguments: tc.arguments });
142
+ }
143
+ }
144
+ }
145
+ }
146
+ result.push({ node, indent, showConnector, isLast, gutters, isVirtualRootChild });
147
+ const children = node.children;
148
+ const multipleChildren = children.length > 1;
149
+ // Order children so the branch containing the active leaf comes first
150
+ const orderedChildren = (() => {
151
+ const prioritized = [];
152
+ const rest = [];
153
+ for (const child of children) {
154
+ if (containsActive.get(child)) {
155
+ prioritized.push(child);
156
+ }
157
+ else {
158
+ rest.push(child);
159
+ }
160
+ }
161
+ return [...prioritized, ...rest];
162
+ })();
163
+ // Calculate child indent
164
+ let childIndent;
165
+ if (multipleChildren) {
166
+ // Parent branches: children get +1
167
+ childIndent = indent + 1;
168
+ }
169
+ else if (justBranched && indent > 0) {
170
+ // First generation after a branch: +1 for visual grouping
171
+ childIndent = indent + 1;
172
+ }
173
+ else {
174
+ // Single-child chain: stay flat
175
+ childIndent = indent;
176
+ }
177
+ // Build gutters for children
178
+ // If this node showed a connector, add a gutter entry for descendants
179
+ // Only add gutter if connector is actually displayed (not suppressed for virtual root children)
180
+ const connectorDisplayed = showConnector && !isVirtualRootChild;
181
+ // When connector is displayed, add a gutter entry at the connector's position
182
+ // Connector is at position (displayIndent - 1), so gutter should be there too
183
+ const currentDisplayIndent = this.multipleRoots ? Math.max(0, indent - 1) : indent;
184
+ const connectorPosition = Math.max(0, currentDisplayIndent - 1);
185
+ const childGutters = connectorDisplayed
186
+ ? [...gutters, { position: connectorPosition, show: !isLast }]
187
+ : gutters;
188
+ // Add children in reverse order
189
+ for (let i = orderedChildren.length - 1; i >= 0; i--) {
190
+ const childIsLast = i === orderedChildren.length - 1;
191
+ stack.push([
192
+ orderedChildren[i],
193
+ childIndent,
194
+ multipleChildren,
195
+ multipleChildren,
196
+ childIsLast,
197
+ childGutters,
198
+ false
199
+ ]);
200
+ }
201
+ }
202
+ return result;
203
+ }
204
+ applyFilter() {
205
+ // Update lastSelectedId only when we have a valid selection (non-empty list)
206
+ // This preserves the selection when switching through empty filter results
207
+ if (this.filteredNodes.length > 0) {
208
+ this.lastSelectedId =
209
+ this.filteredNodes[this.selectedIndex]?.node.entry.id ?? this.lastSelectedId;
210
+ }
211
+ const searchTokens = this.searchQuery.toLowerCase().split(/\s+/).filter(Boolean);
212
+ this.filteredNodes = this.flatNodes.filter((flatNode) => {
213
+ const entry = flatNode.node.entry;
214
+ const isCurrentLeaf = entry.id === this.currentLeafId;
215
+ // Skip assistant messages with only tool calls (no text) unless error/aborted
216
+ // Always show current leaf so active position is visible
217
+ if (entry.type === 'message' && entry.message.role === 'assistant' && !isCurrentLeaf) {
218
+ const msg = entry.message;
219
+ const hasText = this.hasTextContent(msg.content);
220
+ const isErrorOrAborted = msg.stopReason && msg.stopReason !== 'stop' && msg.stopReason !== 'toolUse';
221
+ // Only hide if no text AND not an error/aborted message
222
+ if (!hasText && !isErrorOrAborted) {
223
+ return false;
224
+ }
225
+ }
226
+ // Apply filter mode
227
+ let passesFilter = true;
228
+ // Entry types hidden in default view (settings/bookkeeping)
229
+ const isSettingsEntry = entry.type === 'label' ||
230
+ entry.type === 'custom' ||
231
+ entry.type === 'model_change' ||
232
+ entry.type === 'thinking_level_change' ||
233
+ entry.type === 'turn_start' ||
234
+ entry.type === 'turn_end' ||
235
+ entry.type === 'system_prompt' ||
236
+ entry.type === 'session_info';
237
+ switch (this.filterMode) {
238
+ case 'user-only':
239
+ // Just user messages
240
+ passesFilter = entry.type === 'message' && entry.message.role === 'user';
241
+ break;
242
+ case 'no-tools':
243
+ // Default minus tool results
244
+ passesFilter =
245
+ !isSettingsEntry && !(entry.type === 'message' && entry.message.role === 'toolResult');
246
+ break;
247
+ case 'labeled-only':
248
+ // Just labeled entries
249
+ passesFilter = flatNode.node.label !== undefined;
250
+ break;
251
+ case 'all':
252
+ // Show everything
253
+ passesFilter = true;
254
+ break;
255
+ default:
256
+ // Default mode: hide settings/bookkeeping entries
257
+ passesFilter = !isSettingsEntry;
258
+ break;
259
+ }
260
+ if (!passesFilter)
261
+ return false;
262
+ // Apply search filter
263
+ if (searchTokens.length > 0) {
264
+ const nodeText = this.getSearchableText(flatNode.node).toLowerCase();
265
+ return searchTokens.every((token) => nodeText.includes(token));
266
+ }
267
+ return true;
268
+ });
269
+ // Recalculate visual structure (indent, connectors, gutters) based on visible tree
270
+ this.recalculateVisualStructure();
271
+ // Try to preserve cursor on the same node, or find nearest visible ancestor
272
+ if (this.lastSelectedId) {
273
+ this.selectedIndex = this.findNearestVisibleIndex(this.lastSelectedId);
274
+ }
275
+ else if (this.selectedIndex >= this.filteredNodes.length) {
276
+ // Clamp index if out of bounds
277
+ this.selectedIndex = Math.max(0, this.filteredNodes.length - 1);
278
+ }
279
+ // Update lastSelectedId to the actual selection (may have changed due to parent walk)
280
+ if (this.filteredNodes.length > 0) {
281
+ this.lastSelectedId =
282
+ this.filteredNodes[this.selectedIndex]?.node.entry.id ?? this.lastSelectedId;
283
+ }
284
+ }
285
+ /**
286
+ * Recompute indentation/connectors for the filtered view
287
+ *
288
+ * Filtering can hide intermediate entries; descendants attach to the nearest visible ancestor.
289
+ * Keep indentation semantics aligned with flattenTree() so single-child chains don't drift right.
290
+ */
291
+ recalculateVisualStructure() {
292
+ if (this.filteredNodes.length === 0)
293
+ return;
294
+ const visibleIds = new Set(this.filteredNodes.map((n) => n.node.entry.id));
295
+ // Build entry map for efficient parent lookup (using full tree)
296
+ const entryMap = new Map();
297
+ for (const flatNode of this.flatNodes) {
298
+ entryMap.set(flatNode.node.entry.id, flatNode);
299
+ }
300
+ // Find nearest visible ancestor for a node
301
+ const findVisibleAncestor = (nodeId) => {
302
+ let currentId = entryMap.get(nodeId)?.node.entry.parentId ?? null;
303
+ while (currentId !== null) {
304
+ if (visibleIds.has(currentId)) {
305
+ return currentId;
306
+ }
307
+ currentId = entryMap.get(currentId)?.node.entry.parentId ?? null;
308
+ }
309
+ return null;
310
+ };
311
+ // Build visible tree structure:
312
+ // - visibleParent: nodeId → nearest visible ancestor (or null for roots)
313
+ // - visibleChildren: parentId → list of visible children (in filteredNodes order)
314
+ const visibleParent = new Map();
315
+ const visibleChildren = new Map();
316
+ visibleChildren.set(null, []); // root-level nodes
317
+ for (const flatNode of this.filteredNodes) {
318
+ const nodeId = flatNode.node.entry.id;
319
+ const ancestorId = findVisibleAncestor(nodeId);
320
+ visibleParent.set(nodeId, ancestorId);
321
+ if (!visibleChildren.has(ancestorId)) {
322
+ visibleChildren.set(ancestorId, []);
323
+ }
324
+ visibleChildren.get(ancestorId).push(nodeId);
325
+ }
326
+ // Update multipleRoots based on visible roots
327
+ const visibleRootIds = visibleChildren.get(null);
328
+ this.multipleRoots = visibleRootIds.length > 1;
329
+ // Build a map for quick lookup: nodeId → FlatNode
330
+ const filteredNodeMap = new Map();
331
+ for (const flatNode of this.filteredNodes) {
332
+ filteredNodeMap.set(flatNode.node.entry.id, flatNode);
333
+ }
334
+ const stack = [];
335
+ // Add visible roots in reverse order (to process in forward order via stack)
336
+ for (let i = visibleRootIds.length - 1; i >= 0; i--) {
337
+ const isLast = i === visibleRootIds.length - 1;
338
+ stack.push([
339
+ visibleRootIds[i],
340
+ this.multipleRoots ? 1 : 0,
341
+ this.multipleRoots,
342
+ this.multipleRoots,
343
+ isLast,
344
+ [],
345
+ this.multipleRoots
346
+ ]);
347
+ }
348
+ while (stack.length > 0) {
349
+ const [nodeId, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop();
350
+ const flatNode = filteredNodeMap.get(nodeId);
351
+ if (!flatNode)
352
+ continue;
353
+ // Update this node's visual properties
354
+ flatNode.indent = indent;
355
+ flatNode.showConnector = showConnector;
356
+ flatNode.isLast = isLast;
357
+ flatNode.gutters = gutters;
358
+ flatNode.isVirtualRootChild = isVirtualRootChild;
359
+ // Get visible children of this node
360
+ const children = visibleChildren.get(nodeId) || [];
361
+ const multipleChildren = children.length > 1;
362
+ // Child indent follows flattenTree(): branch points (and first generation after a branch) shift +1
363
+ let childIndent;
364
+ if (multipleChildren) {
365
+ childIndent = indent + 1;
366
+ }
367
+ else if (justBranched && indent > 0) {
368
+ childIndent = indent + 1;
369
+ }
370
+ else {
371
+ childIndent = indent;
372
+ }
373
+ // Child gutters follow flattenTree() connector/gutter rules
374
+ const connectorDisplayed = showConnector && !isVirtualRootChild;
375
+ const currentDisplayIndent = this.multipleRoots ? Math.max(0, indent - 1) : indent;
376
+ const connectorPosition = Math.max(0, currentDisplayIndent - 1);
377
+ const childGutters = connectorDisplayed
378
+ ? [...gutters, { position: connectorPosition, show: !isLast }]
379
+ : gutters;
380
+ // Add children in reverse order (to process in forward order via stack)
381
+ for (let i = children.length - 1; i >= 0; i--) {
382
+ const childIsLast = i === children.length - 1;
383
+ stack.push([
384
+ children[i],
385
+ childIndent,
386
+ multipleChildren,
387
+ multipleChildren,
388
+ childIsLast,
389
+ childGutters,
390
+ false
391
+ ]);
392
+ }
393
+ }
394
+ }
395
+ /** Get searchable text content from a node */
396
+ getSearchableText(node) {
397
+ const entry = node.entry;
398
+ const parts = [];
399
+ if (node.label) {
400
+ parts.push(node.label);
401
+ }
402
+ switch (entry.type) {
403
+ case 'message': {
404
+ const msg = entry.message;
405
+ parts.push(msg.role);
406
+ if ('content' in msg && msg.content) {
407
+ parts.push(this.extractContent(msg.content));
408
+ }
409
+ if (msg.role === 'bashExecution') {
410
+ const bashMsg = msg;
411
+ if (bashMsg.command)
412
+ parts.push(bashMsg.command);
413
+ }
414
+ break;
415
+ }
416
+ case 'custom_message': {
417
+ parts.push(entry.customType);
418
+ if (typeof entry.content === 'string') {
419
+ parts.push(entry.content);
420
+ }
421
+ else {
422
+ parts.push(this.extractContent(entry.content));
423
+ }
424
+ break;
425
+ }
426
+ case 'compaction':
427
+ parts.push('compaction');
428
+ break;
429
+ case 'branch_summary':
430
+ parts.push('branch summary', entry.summary);
431
+ break;
432
+ case 'model_change':
433
+ parts.push('model', entry.modelId);
434
+ break;
435
+ case 'thinking_level_change':
436
+ parts.push('thinking', entry.thinkingLevel);
437
+ break;
438
+ case 'custom':
439
+ parts.push('custom', entry.customType);
440
+ break;
441
+ case 'label':
442
+ parts.push('label', entry.label ?? '');
443
+ break;
444
+ }
445
+ return parts.join(' ');
446
+ }
447
+ invalidate() { }
448
+ getSearchQuery() {
449
+ return this.searchQuery;
450
+ }
451
+ getSelectedNode() {
452
+ return this.filteredNodes[this.selectedIndex]?.node;
453
+ }
454
+ updateNodeLabel(entryId, label) {
455
+ for (const flatNode of this.flatNodes) {
456
+ if (flatNode.node.entry.id === entryId) {
457
+ flatNode.node.label = label;
458
+ break;
459
+ }
460
+ }
461
+ }
462
+ getFilterLabel() {
463
+ switch (this.filterMode) {
464
+ case 'no-tools':
465
+ return ' [no-tools]';
466
+ case 'user-only':
467
+ return ' [user]';
468
+ case 'labeled-only':
469
+ return ' [labeled]';
470
+ case 'all':
471
+ return ' [all]';
472
+ default:
473
+ return '';
474
+ }
475
+ }
476
+ render(width) {
477
+ const lines = [];
478
+ if (this.filteredNodes.length === 0) {
479
+ lines.push(truncateToWidth(theme.fg('muted', ' No entries found'), width));
480
+ lines.push(truncateToWidth(theme.fg('muted', ` (0/0)${this.getFilterLabel()}`), width));
481
+ return lines;
482
+ }
483
+ const startIndex = Math.max(0, Math.min(this.selectedIndex - Math.floor(this.maxVisibleLines / 2), this.filteredNodes.length - this.maxVisibleLines));
484
+ const endIndex = Math.min(startIndex + this.maxVisibleLines, this.filteredNodes.length);
485
+ for (let i = startIndex; i < endIndex; i++) {
486
+ const flatNode = this.filteredNodes[i];
487
+ const entry = flatNode.node.entry;
488
+ const isSelected = i === this.selectedIndex;
489
+ // Build line: cursor + prefix + path marker + label + content
490
+ const cursor = isSelected ? theme.fg('accent', '› ') : ' ';
491
+ // If multiple roots, shift display (roots at 0, not 1)
492
+ const displayIndent = this.multipleRoots ? Math.max(0, flatNode.indent - 1) : flatNode.indent;
493
+ // Build prefix with gutters at their correct positions
494
+ // Each gutter has a position (displayIndent where its connector was shown)
495
+ const connector = flatNode.showConnector && !flatNode.isVirtualRootChild
496
+ ? flatNode.isLast
497
+ ? '└─ '
498
+ : '├─ '
499
+ : '';
500
+ const connectorPosition = connector ? displayIndent - 1 : -1;
501
+ // Build prefix char by char, placing gutters and connector at their positions
502
+ const totalChars = displayIndent * 3;
503
+ const prefixChars = [];
504
+ for (let i = 0; i < totalChars; i++) {
505
+ const level = Math.floor(i / 3);
506
+ const posInLevel = i % 3;
507
+ // Check if there's a gutter at this level
508
+ const gutter = flatNode.gutters.find((g) => g.position === level);
509
+ if (gutter) {
510
+ if (posInLevel === 0) {
511
+ prefixChars.push(gutter.show ? '│' : ' ');
512
+ }
513
+ else {
514
+ prefixChars.push(' ');
515
+ }
516
+ }
517
+ else if (connector && level === connectorPosition) {
518
+ // Connector at this level
519
+ if (posInLevel === 0) {
520
+ prefixChars.push(flatNode.isLast ? '└' : '├');
521
+ }
522
+ else if (posInLevel === 1) {
523
+ prefixChars.push('─');
524
+ }
525
+ else {
526
+ prefixChars.push(' ');
527
+ }
528
+ }
529
+ else {
530
+ prefixChars.push(' ');
531
+ }
532
+ }
533
+ const prefix = prefixChars.join('');
534
+ // Active path marker - shown right before the entry text
535
+ const isOnActivePath = this.activePathIds.has(entry.id);
536
+ const pathMarker = isOnActivePath ? theme.fg('accent', '• ') : '';
537
+ const label = flatNode.node.label ? theme.fg('warning', `[${flatNode.node.label}] `) : '';
538
+ const content = this.getEntryDisplayText(flatNode.node, isSelected);
539
+ let line = cursor + theme.fg('dim', prefix) + pathMarker + label + content;
540
+ if (isSelected) {
541
+ line = theme.bg('selectedBg', line);
542
+ }
543
+ lines.push(truncateToWidth(line, width));
544
+ }
545
+ lines.push(truncateToWidth(theme.fg('muted', ` (${this.selectedIndex + 1}/${this.filteredNodes.length})${this.getFilterLabel()}`), width));
546
+ return lines;
547
+ }
548
+ getEntryDisplayText(node, isSelected) {
549
+ const entry = node.entry;
550
+ let result;
551
+ const normalize = (s) => s.replace(/[\n\t]/g, ' ').trim();
552
+ switch (entry.type) {
553
+ case 'message': {
554
+ const msg = entry.message;
555
+ const role = msg.role;
556
+ if (role === 'user') {
557
+ const msgWithContent = msg;
558
+ const content = normalize(this.extractContent(msgWithContent.content));
559
+ result = theme.fg('accent', 'user: ') + content;
560
+ }
561
+ else if (role === 'assistant') {
562
+ const msgWithContent = msg;
563
+ const textContent = normalize(this.extractContent(msgWithContent.content));
564
+ if (textContent) {
565
+ result = theme.fg('success', 'assistant: ') + textContent;
566
+ }
567
+ else if (msgWithContent.stopReason === 'aborted') {
568
+ result = theme.fg('success', 'assistant: ') + theme.fg('muted', '(aborted)');
569
+ }
570
+ else if (msgWithContent.errorMessage) {
571
+ const errMsg = normalize(msgWithContent.errorMessage).slice(0, 80);
572
+ result = theme.fg('success', 'assistant: ') + theme.fg('error', errMsg);
573
+ }
574
+ else {
575
+ result = theme.fg('success', 'assistant: ') + theme.fg('muted', '(no content)');
576
+ }
577
+ }
578
+ else if (role === 'toolResult') {
579
+ const toolMsg = msg;
580
+ const toolCall = toolMsg.toolCallId
581
+ ? this.toolCallMap.get(toolMsg.toolCallId)
582
+ : undefined;
583
+ if (toolCall) {
584
+ result = theme.fg('muted', this.formatToolCall(toolCall.name, toolCall.arguments));
585
+ }
586
+ else {
587
+ result = theme.fg('muted', `[${toolMsg.toolName ?? 'tool'}]`);
588
+ }
589
+ }
590
+ else if (role === 'bashExecution') {
591
+ const bashMsg = msg;
592
+ result = theme.fg('dim', `[bash]: ${normalize(bashMsg.command ?? '')}`);
593
+ }
594
+ else {
595
+ result = theme.fg('dim', `[${role}]`);
596
+ }
597
+ break;
598
+ }
599
+ case 'custom_message': {
600
+ const content = typeof entry.content === 'string'
601
+ ? entry.content
602
+ : entry.content
603
+ .filter((c) => c.type === 'text')
604
+ .map((c) => c.text)
605
+ .join('');
606
+ result = theme.fg('customMessageLabel', `[${entry.customType}]: `) + normalize(content);
607
+ break;
608
+ }
609
+ case 'compaction': {
610
+ const tokens = Math.round(entry.tokensBefore / 1000);
611
+ result = theme.fg('borderAccent', `[compaction: ${tokens}k tokens]`);
612
+ break;
613
+ }
614
+ case 'branch_summary':
615
+ result = theme.fg('warning', `[branch summary]: `) + normalize(entry.summary);
616
+ break;
617
+ case 'model_change':
618
+ result = theme.fg('dim', `[model: ${entry.modelId}]`);
619
+ break;
620
+ case 'thinking_level_change':
621
+ result = theme.fg('dim', `[thinking: ${entry.thinkingLevel}]`);
622
+ break;
623
+ case 'custom':
624
+ result = theme.fg('dim', `[custom: ${entry.customType}]`);
625
+ break;
626
+ case 'label':
627
+ result = theme.fg('dim', `[label: ${entry.label ?? '(cleared)'}]`);
628
+ break;
629
+ default:
630
+ result = '';
631
+ }
632
+ return isSelected ? theme.bold(result) : result;
633
+ }
634
+ extractContent(content) {
635
+ const maxLen = 200;
636
+ if (typeof content === 'string')
637
+ return content.slice(0, maxLen);
638
+ if (Array.isArray(content)) {
639
+ let result = '';
640
+ for (const c of content) {
641
+ if (typeof c === 'object' && c !== null && 'type' in c && c.type === 'text') {
642
+ result += c.text;
643
+ if (result.length >= maxLen)
644
+ return result.slice(0, maxLen);
645
+ }
646
+ }
647
+ return result;
648
+ }
649
+ return '';
650
+ }
651
+ hasTextContent(content) {
652
+ if (typeof content === 'string')
653
+ return content.trim().length > 0;
654
+ if (Array.isArray(content)) {
655
+ for (const c of content) {
656
+ if (typeof c === 'object' && c !== null && 'type' in c && c.type === 'text') {
657
+ const text = c.text;
658
+ if (text && text.trim().length > 0)
659
+ return true;
660
+ }
661
+ }
662
+ }
663
+ return false;
664
+ }
665
+ formatToolCall(name, args) {
666
+ const shortenPath = (p) => {
667
+ const home = process.env.HOME || process.env.USERPROFILE || '';
668
+ if (home && p.startsWith(home))
669
+ return `~${p.slice(home.length)}`;
670
+ return p;
671
+ };
672
+ switch (name) {
673
+ case 'read': {
674
+ const path = shortenPath(String(args.path || args.file_path || ''));
675
+ const offset = args.offset;
676
+ const limit = args.limit;
677
+ let display = path;
678
+ if (offset !== undefined || limit !== undefined) {
679
+ const start = offset ?? 1;
680
+ const end = limit !== undefined ? start + limit - 1 : '';
681
+ display += `:${start}${end ? `-${end}` : ''}`;
682
+ }
683
+ return `[read: ${display}]`;
684
+ }
685
+ case 'write': {
686
+ const path = shortenPath(String(args.path || args.file_path || ''));
687
+ return `[write: ${path}]`;
688
+ }
689
+ case 'edit': {
690
+ const path = shortenPath(String(args.path || args.file_path || ''));
691
+ return `[edit: ${path}]`;
692
+ }
693
+ case 'bash': {
694
+ const rawCmd = String(args.command || '');
695
+ const cmd = rawCmd
696
+ .replace(/[\n\t]/g, ' ')
697
+ .trim()
698
+ .slice(0, 50);
699
+ return `[bash: ${cmd}${rawCmd.length > 50 ? '...' : ''}]`;
700
+ }
701
+ case 'grep': {
702
+ const pattern = String(args.pattern || '');
703
+ const path = shortenPath(String(args.path || '.'));
704
+ return `[grep: /${pattern}/ in ${path}]`;
705
+ }
706
+ case 'find': {
707
+ const pattern = String(args.pattern || '');
708
+ const path = shortenPath(String(args.path || '.'));
709
+ return `[find: ${pattern} in ${path}]`;
710
+ }
711
+ case 'ls': {
712
+ const path = shortenPath(String(args.path || '.'));
713
+ return `[ls: ${path}]`;
714
+ }
715
+ default: {
716
+ // Custom tool - show name and truncated JSON args
717
+ const argsStr = JSON.stringify(args).slice(0, 40);
718
+ return `[${name}: ${argsStr}${JSON.stringify(args).length > 40 ? '...' : ''}]`;
719
+ }
720
+ }
721
+ }
722
+ handleInput(keyData) {
723
+ const kb = getEditorKeybindings();
724
+ if (kb.matches(keyData, 'selectUp')) {
725
+ this.selectedIndex =
726
+ this.selectedIndex === 0 ? this.filteredNodes.length - 1 : this.selectedIndex - 1;
727
+ }
728
+ else if (kb.matches(keyData, 'selectDown')) {
729
+ this.selectedIndex =
730
+ this.selectedIndex === this.filteredNodes.length - 1 ? 0 : this.selectedIndex + 1;
731
+ }
732
+ else if (kb.matches(keyData, 'cursorLeft')) {
733
+ // Page up
734
+ this.selectedIndex = Math.max(0, this.selectedIndex - this.maxVisibleLines);
735
+ }
736
+ else if (kb.matches(keyData, 'cursorRight')) {
737
+ // Page down
738
+ this.selectedIndex = Math.min(this.filteredNodes.length - 1, this.selectedIndex + this.maxVisibleLines);
739
+ }
740
+ else if (kb.matches(keyData, 'selectConfirm')) {
741
+ const selected = this.filteredNodes[this.selectedIndex];
742
+ if (selected && this.onSelect) {
743
+ this.onSelect(selected.node.entry.id);
744
+ }
745
+ }
746
+ else if (kb.matches(keyData, 'selectCancel')) {
747
+ if (this.searchQuery) {
748
+ this.searchQuery = '';
749
+ this.applyFilter();
750
+ }
751
+ else {
752
+ this.onCancel?.();
753
+ }
754
+ }
755
+ else if (matchesKey(keyData, 'ctrl+d')) {
756
+ // Direct filter: default
757
+ this.filterMode = 'default';
758
+ this.applyFilter();
759
+ }
760
+ else if (matchesKey(keyData, 'ctrl+t')) {
761
+ // Toggle filter: no-tools ↔ default
762
+ this.filterMode = this.filterMode === 'no-tools' ? 'default' : 'no-tools';
763
+ this.applyFilter();
764
+ }
765
+ else if (matchesKey(keyData, 'ctrl+u')) {
766
+ // Toggle filter: user-only ↔ default
767
+ this.filterMode = this.filterMode === 'user-only' ? 'default' : 'user-only';
768
+ this.applyFilter();
769
+ }
770
+ else if (matchesKey(keyData, 'ctrl+l')) {
771
+ // Toggle filter: labeled-only ↔ default
772
+ this.filterMode = this.filterMode === 'labeled-only' ? 'default' : 'labeled-only';
773
+ this.applyFilter();
774
+ }
775
+ else if (matchesKey(keyData, 'ctrl+a')) {
776
+ // Toggle filter: all ↔ default
777
+ this.filterMode = this.filterMode === 'all' ? 'default' : 'all';
778
+ this.applyFilter();
779
+ }
780
+ else if (matchesKey(keyData, 'shift+ctrl+o')) {
781
+ // Cycle filter backwards
782
+ const modes = ['default', 'no-tools', 'user-only', 'labeled-only', 'all'];
783
+ const currentIndex = modes.indexOf(this.filterMode);
784
+ this.filterMode = modes[(currentIndex - 1 + modes.length) % modes.length];
785
+ this.applyFilter();
786
+ }
787
+ else if (matchesKey(keyData, 'ctrl+o')) {
788
+ // Cycle filter forwards: default → no-tools → user-only → labeled-only → all → default
789
+ const modes = ['default', 'no-tools', 'user-only', 'labeled-only', 'all'];
790
+ const currentIndex = modes.indexOf(this.filterMode);
791
+ this.filterMode = modes[(currentIndex + 1) % modes.length];
792
+ this.applyFilter();
793
+ }
794
+ else if (kb.matches(keyData, 'deleteCharBackward')) {
795
+ if (this.searchQuery.length > 0) {
796
+ this.searchQuery = this.searchQuery.slice(0, -1);
797
+ this.applyFilter();
798
+ }
799
+ }
800
+ else if (matchesKey(keyData, 'shift+l')) {
801
+ const selected = this.filteredNodes[this.selectedIndex];
802
+ if (selected && this.onLabelEdit) {
803
+ this.onLabelEdit(selected.node.entry.id, selected.node.label);
804
+ }
805
+ }
806
+ else {
807
+ const hasControlChars = [...keyData].some((ch) => {
808
+ const code = ch.charCodeAt(0);
809
+ return code < 32 || code === 0x7f || (code >= 0x80 && code <= 0x9f);
810
+ });
811
+ if (!hasControlChars && keyData.length > 0) {
812
+ this.searchQuery += keyData;
813
+ this.applyFilter();
814
+ }
815
+ }
816
+ }
817
+ }
818
+ /** Component that displays the current search query */
819
+ class SearchLine {
820
+ treeList;
821
+ constructor(treeList) {
822
+ this.treeList = treeList;
823
+ }
824
+ invalidate() { }
825
+ render(width) {
826
+ const query = this.treeList.getSearchQuery();
827
+ if (query) {
828
+ return [
829
+ truncateToWidth(` ${theme.fg('muted', 'Type to search:')} ${theme.fg('accent', query)}`, width)
830
+ ];
831
+ }
832
+ return [truncateToWidth(` ${theme.fg('muted', 'Type to search:')}`, width)];
833
+ }
834
+ handleInput(_keyData) { }
835
+ }
836
+ /** Label input component shown when editing a label */
837
+ class LabelInput {
838
+ input;
839
+ entryId;
840
+ onSubmit;
841
+ onCancel;
842
+ // Focusable implementation - propagate to input for IME cursor positioning
843
+ _focused = false;
844
+ get focused() {
845
+ return this._focused;
846
+ }
847
+ set focused(value) {
848
+ this._focused = value;
849
+ this.input.focused = value;
850
+ }
851
+ constructor(entryId, currentLabel) {
852
+ this.entryId = entryId;
853
+ this.input = new Input();
854
+ if (currentLabel) {
855
+ this.input.setValue(currentLabel);
856
+ }
857
+ }
858
+ invalidate() { }
859
+ render(width) {
860
+ const lines = [];
861
+ const indent = ' ';
862
+ const availableWidth = width - indent.length;
863
+ lines.push(truncateToWidth(`${indent}${theme.fg('muted', 'Label (empty to remove):')}`, width));
864
+ lines.push(...this.input.render(availableWidth).map((line) => truncateToWidth(`${indent}${line}`, width)));
865
+ lines.push(truncateToWidth(`${indent}${keyHint('selectConfirm', 'save')} ${keyHint('selectCancel', 'cancel')}`, width));
866
+ return lines;
867
+ }
868
+ handleInput(keyData) {
869
+ const kb = getEditorKeybindings();
870
+ if (kb.matches(keyData, 'selectConfirm')) {
871
+ const value = this.input.getValue().trim();
872
+ this.onSubmit?.(this.entryId, value || undefined);
873
+ }
874
+ else if (kb.matches(keyData, 'selectCancel')) {
875
+ this.onCancel?.();
876
+ }
877
+ else {
878
+ this.input.handleInput(keyData);
879
+ }
880
+ }
881
+ }
882
+ /**
883
+ * Component that renders a session tree selector for navigation
884
+ */
885
+ export class TreeSelectorComponent extends Container {
886
+ treeList;
887
+ labelInput = null;
888
+ labelInputContainer;
889
+ treeContainer;
890
+ onLabelChangeCallback;
891
+ // Focusable implementation - propagate to labelInput when active for IME cursor positioning
892
+ _focused = false;
893
+ get focused() {
894
+ return this._focused;
895
+ }
896
+ set focused(value) {
897
+ this._focused = value;
898
+ // Propagate to labelInput when it's active
899
+ if (this.labelInput) {
900
+ this.labelInput.focused = value;
901
+ }
902
+ }
903
+ constructor(tree, currentLeafId, terminalHeight, onSelect, onCancel, onLabelChange, initialSelectedId) {
904
+ super();
905
+ this.onLabelChangeCallback = onLabelChange;
906
+ const maxVisibleLines = Math.max(5, Math.floor(terminalHeight / 2));
907
+ this.treeList = new TreeList(tree, currentLeafId, maxVisibleLines, initialSelectedId);
908
+ this.treeList.onSelect = onSelect;
909
+ this.treeList.onCancel = onCancel;
910
+ this.treeList.onLabelEdit = (entryId, currentLabel) => this.showLabelInput(entryId, currentLabel);
911
+ this.treeContainer = new Container();
912
+ this.treeContainer.addChild(this.treeList);
913
+ this.labelInputContainer = new Container();
914
+ this.addChild(new Spacer(1));
915
+ this.addChild(new DynamicBorder());
916
+ this.addChild(new Text(theme.bold(' Session Tree'), 1, 0));
917
+ this.addChild(new TruncatedText(theme.fg('muted', ' ↑/↓: move. ←/→: page. Shift+L: label. ') +
918
+ theme.fg('muted', '^D/^T/^U/^L/^A: filters (^O/⇧^O cycle)'), 0, 0));
919
+ this.addChild(new SearchLine(this.treeList));
920
+ this.addChild(new DynamicBorder());
921
+ this.addChild(new Spacer(1));
922
+ this.addChild(this.treeContainer);
923
+ this.addChild(this.labelInputContainer);
924
+ this.addChild(new Spacer(1));
925
+ this.addChild(new DynamicBorder());
926
+ if (tree.length === 0) {
927
+ setTimeout(() => onCancel(), 100);
928
+ }
929
+ }
930
+ showLabelInput(entryId, currentLabel) {
931
+ this.labelInput = new LabelInput(entryId, currentLabel);
932
+ this.labelInput.onSubmit = (id, label) => {
933
+ this.treeList.updateNodeLabel(id, label);
934
+ this.onLabelChangeCallback?.(id, label);
935
+ this.hideLabelInput();
936
+ };
937
+ this.labelInput.onCancel = () => this.hideLabelInput();
938
+ // Propagate current focused state to the new labelInput
939
+ this.labelInput.focused = this._focused;
940
+ this.treeContainer.clear();
941
+ this.labelInputContainer.clear();
942
+ this.labelInputContainer.addChild(this.labelInput);
943
+ }
944
+ hideLabelInput() {
945
+ this.labelInput = null;
946
+ this.labelInputContainer.clear();
947
+ this.treeContainer.clear();
948
+ this.treeContainer.addChild(this.treeList);
949
+ }
950
+ handleInput(keyData) {
951
+ if (this.labelInput) {
952
+ this.labelInput.handleInput(keyData);
953
+ }
954
+ else {
955
+ this.treeList.handleInput(keyData);
956
+ }
957
+ }
958
+ getTreeList() {
959
+ return this.treeList;
960
+ }
961
+ }
962
+ //# sourceMappingURL=tree-selector.js.map