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,430 @@
1
+ /**
2
+ * Minimal Mode Example - Demonstrates a "minimal" tool display mode
3
+ *
4
+ * This extension overrides built-in tools to provide custom rendering:
5
+ * - Collapsed mode: Only shows the tool call (command/path), no output
6
+ * - Expanded mode: Shows full output like the built-in renderers
7
+ *
8
+ * This demonstrates how a "minimal mode" could work, where ctrl+o cycles through:
9
+ * - Standard: Shows truncated output (current default)
10
+ * - Expanded: Shows full output (current expanded)
11
+ * - Minimal: Shows only tool call, no output (this extension's collapsed mode)
12
+ *
13
+ * Usage:
14
+ * shortcut -e ./minimal-mode.ts
15
+ *
16
+ * Then use ctrl+o to toggle between minimal (collapsed) and full (expanded) views.
17
+ */
18
+
19
+ import { homedir } from 'os';
20
+ import type { ExtensionAPI } from 'shortcutxl';
21
+ import {
22
+ createBashTool,
23
+ createEditTool,
24
+ createFindTool,
25
+ createGrepTool,
26
+ createLsTool,
27
+ createReadTool,
28
+ createWriteTool,
29
+ Text
30
+ } from 'shortcutxl';
31
+
32
+ /**
33
+ * Shorten a path by replacing home directory with ~
34
+ */
35
+ function shortenPath(path: string): string {
36
+ const home = homedir();
37
+ if (path.startsWith(home)) {
38
+ return `~${path.slice(home.length)}`;
39
+ }
40
+ return path;
41
+ }
42
+
43
+ // Cache for built-in tools by cwd
44
+ const toolCache = new Map<string, ReturnType<typeof createBuiltInTools>>();
45
+
46
+ function createBuiltInTools(cwd: string) {
47
+ return {
48
+ read: createReadTool(cwd),
49
+ bash: createBashTool(cwd),
50
+ edit: createEditTool(cwd),
51
+ write: createWriteTool(cwd),
52
+ find: createFindTool(cwd),
53
+ grep: createGrepTool(cwd),
54
+ ls: createLsTool(cwd)
55
+ };
56
+ }
57
+
58
+ function getBuiltInTools(cwd: string) {
59
+ let tools = toolCache.get(cwd);
60
+ if (!tools) {
61
+ tools = createBuiltInTools(cwd);
62
+ toolCache.set(cwd, tools);
63
+ }
64
+ return tools;
65
+ }
66
+
67
+ export default function (shortcut: ExtensionAPI) {
68
+ // =========================================================================
69
+ // Read Tool
70
+ // =========================================================================
71
+ shortcut.registerTool({
72
+ name: 'read',
73
+ label: 'read',
74
+ description:
75
+ 'Read the contents of a file. Supports text files and images (jpg, png, gif, webp). Images are sent as attachments. For text files, output is truncated to 2000 lines or 50KB (whichever is hit first). Use offset/limit for large files.',
76
+ parameters: getBuiltInTools(process.cwd()).read.parameters,
77
+
78
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
79
+ const tools = getBuiltInTools(ctx.cwd);
80
+ return tools.read.execute(toolCallId, params, signal, onUpdate);
81
+ },
82
+
83
+ renderCall(args, theme) {
84
+ const path = shortenPath(args.path || '');
85
+ let pathDisplay = path ? theme.fg('accent', path) : theme.fg('toolOutput', '...');
86
+
87
+ // Show line range if specified
88
+ if (args.offset !== undefined || args.limit !== undefined) {
89
+ const startLine = args.offset ?? 1;
90
+ const endLine = args.limit !== undefined ? startLine + args.limit - 1 : '';
91
+ pathDisplay += theme.fg('warning', `:${startLine}${endLine ? `-${endLine}` : ''}`);
92
+ }
93
+
94
+ return new Text(`${theme.fg('toolTitle', theme.bold('read'))} ${pathDisplay}`, 0, 0);
95
+ },
96
+
97
+ renderResult(result, { expanded }, theme) {
98
+ // Minimal mode: show nothing in collapsed state
99
+ if (!expanded) {
100
+ return new Text('', 0, 0);
101
+ }
102
+
103
+ // Expanded mode: show full output
104
+ const textContent = result.content.find((c) => c.type === 'text');
105
+ if (!textContent || textContent.type !== 'text') {
106
+ return new Text('', 0, 0);
107
+ }
108
+
109
+ const lines = textContent.text.split('\n');
110
+ const output = lines.map((line) => theme.fg('toolOutput', line)).join('\n');
111
+ return new Text(`\n${output}`, 0, 0);
112
+ }
113
+ });
114
+
115
+ // =========================================================================
116
+ // Bash Tool
117
+ // =========================================================================
118
+ shortcut.registerTool({
119
+ name: 'bash',
120
+ label: 'bash',
121
+ description:
122
+ 'Execute a bash command in the current working directory. Returns stdout and stderr. Output is truncated to last 2000 lines or 50KB (whichever is hit first).',
123
+ parameters: getBuiltInTools(process.cwd()).bash.parameters,
124
+
125
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
126
+ const tools = getBuiltInTools(ctx.cwd);
127
+ return tools.bash.execute(toolCallId, params, signal, onUpdate);
128
+ },
129
+
130
+ renderCall(args, theme) {
131
+ const command = args.command || '...';
132
+ const timeout = args.timeout as number | undefined;
133
+ const timeoutSuffix = timeout ? theme.fg('muted', ` (timeout ${timeout}s)`) : '';
134
+
135
+ return new Text(theme.fg('toolTitle', theme.bold(`$ ${command}`)) + timeoutSuffix, 0, 0);
136
+ },
137
+
138
+ renderResult(result, { expanded }, theme) {
139
+ // Minimal mode: show nothing in collapsed state
140
+ if (!expanded) {
141
+ return new Text('', 0, 0);
142
+ }
143
+
144
+ // Expanded mode: show full output
145
+ const textContent = result.content.find((c) => c.type === 'text');
146
+ if (!textContent || textContent.type !== 'text') {
147
+ return new Text('', 0, 0);
148
+ }
149
+
150
+ const output = textContent.text
151
+ .trim()
152
+ .split('\n')
153
+ .map((line) => theme.fg('toolOutput', line))
154
+ .join('\n');
155
+
156
+ if (!output) {
157
+ return new Text('', 0, 0);
158
+ }
159
+
160
+ return new Text(`\n${output}`, 0, 0);
161
+ }
162
+ });
163
+
164
+ // =========================================================================
165
+ // Write Tool
166
+ // =========================================================================
167
+ shortcut.registerTool({
168
+ name: 'write',
169
+ label: 'write',
170
+ description:
171
+ "Write content to a file. Creates the file if it doesn't exist, overwrites if it does. Automatically creates parent directories.",
172
+ parameters: getBuiltInTools(process.cwd()).write.parameters,
173
+
174
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
175
+ const tools = getBuiltInTools(ctx.cwd);
176
+ return tools.write.execute(toolCallId, params, signal, onUpdate);
177
+ },
178
+
179
+ renderCall(args, theme) {
180
+ const path = shortenPath(args.path || '');
181
+ const pathDisplay = path ? theme.fg('accent', path) : theme.fg('toolOutput', '...');
182
+ const lineCount = args.content ? args.content.split('\n').length : 0;
183
+ const lineInfo = lineCount > 0 ? theme.fg('muted', ` (${lineCount} lines)`) : '';
184
+
185
+ return new Text(
186
+ `${theme.fg('toolTitle', theme.bold('write'))} ${pathDisplay}${lineInfo}`,
187
+ 0,
188
+ 0
189
+ );
190
+ },
191
+
192
+ renderResult(result, { expanded }, theme) {
193
+ // Minimal mode: show nothing (file was written)
194
+ if (!expanded) {
195
+ return new Text('', 0, 0);
196
+ }
197
+
198
+ // Expanded mode: show error if any
199
+ if (result.content.some((c) => c.type === 'text' && c.text)) {
200
+ const textContent = result.content.find((c) => c.type === 'text');
201
+ if (textContent?.type === 'text' && textContent.text) {
202
+ return new Text(`\n${theme.fg('error', textContent.text)}`, 0, 0);
203
+ }
204
+ }
205
+
206
+ return new Text('', 0, 0);
207
+ }
208
+ });
209
+
210
+ // =========================================================================
211
+ // Edit Tool
212
+ // =========================================================================
213
+ shortcut.registerTool({
214
+ name: 'edit',
215
+ label: 'edit',
216
+ description:
217
+ 'Edit a file by replacing exact text. The oldText must match exactly (including whitespace). Use this for precise, surgical edits.',
218
+ parameters: getBuiltInTools(process.cwd()).edit.parameters,
219
+
220
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
221
+ const tools = getBuiltInTools(ctx.cwd);
222
+ return tools.edit.execute(toolCallId, params, signal, onUpdate);
223
+ },
224
+
225
+ renderCall(args, theme) {
226
+ const path = shortenPath(args.path || '');
227
+ const pathDisplay = path ? theme.fg('accent', path) : theme.fg('toolOutput', '...');
228
+
229
+ return new Text(`${theme.fg('toolTitle', theme.bold('edit'))} ${pathDisplay}`, 0, 0);
230
+ },
231
+
232
+ renderResult(result, { expanded }, theme) {
233
+ // Minimal mode: show nothing in collapsed state
234
+ if (!expanded) {
235
+ return new Text('', 0, 0);
236
+ }
237
+
238
+ // Expanded mode: show diff or error
239
+ const textContent = result.content.find((c) => c.type === 'text');
240
+ if (!textContent || textContent.type !== 'text') {
241
+ return new Text('', 0, 0);
242
+ }
243
+
244
+ // For errors, show the error message
245
+ const text = textContent.text;
246
+ if (text.includes('Error') || text.includes('error')) {
247
+ return new Text(`\n${theme.fg('error', text)}`, 0, 0);
248
+ }
249
+
250
+ // Otherwise show the text (would be nice to show actual diff here)
251
+ return new Text(`\n${theme.fg('toolOutput', text)}`, 0, 0);
252
+ }
253
+ });
254
+
255
+ // =========================================================================
256
+ // Find Tool
257
+ // =========================================================================
258
+ shortcut.registerTool({
259
+ name: 'find',
260
+ label: 'find',
261
+ description:
262
+ 'Find files by name pattern (glob). Searches recursively from the specified path. Output limited to 200 results.',
263
+ parameters: getBuiltInTools(process.cwd()).find.parameters,
264
+
265
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
266
+ const tools = getBuiltInTools(ctx.cwd);
267
+ return tools.find.execute(toolCallId, params, signal, onUpdate);
268
+ },
269
+
270
+ renderCall(args, theme) {
271
+ const pattern = args.pattern || '';
272
+ const path = shortenPath(args.path || '.');
273
+ const limit = args.limit;
274
+
275
+ let text = `${theme.fg('toolTitle', theme.bold('find'))} ${theme.fg('accent', pattern)}`;
276
+ text += theme.fg('toolOutput', ` in ${path}`);
277
+ if (limit !== undefined) {
278
+ text += theme.fg('toolOutput', ` (limit ${limit})`);
279
+ }
280
+
281
+ return new Text(text, 0, 0);
282
+ },
283
+
284
+ renderResult(result, { expanded }, theme) {
285
+ if (!expanded) {
286
+ // Minimal: just show count
287
+ const textContent = result.content.find((c) => c.type === 'text');
288
+ if (textContent?.type === 'text') {
289
+ const count = textContent.text.trim().split('\n').filter(Boolean).length;
290
+ if (count > 0) {
291
+ return new Text(theme.fg('muted', ` → ${count} files`), 0, 0);
292
+ }
293
+ }
294
+ return new Text('', 0, 0);
295
+ }
296
+
297
+ // Expanded: show full results
298
+ const textContent = result.content.find((c) => c.type === 'text');
299
+ if (!textContent || textContent.type !== 'text') {
300
+ return new Text('', 0, 0);
301
+ }
302
+
303
+ const output = textContent.text
304
+ .trim()
305
+ .split('\n')
306
+ .map((line) => theme.fg('toolOutput', line))
307
+ .join('\n');
308
+
309
+ return new Text(`\n${output}`, 0, 0);
310
+ }
311
+ });
312
+
313
+ // =========================================================================
314
+ // Grep Tool
315
+ // =========================================================================
316
+ shortcut.registerTool({
317
+ name: 'grep',
318
+ label: 'grep',
319
+ description:
320
+ 'Search file contents by regex pattern. Uses ripgrep for fast searching. Output limited to 200 matches.',
321
+ parameters: getBuiltInTools(process.cwd()).grep.parameters,
322
+
323
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
324
+ const tools = getBuiltInTools(ctx.cwd);
325
+ return tools.grep.execute(toolCallId, params, signal, onUpdate);
326
+ },
327
+
328
+ renderCall(args, theme) {
329
+ const pattern = args.pattern || '';
330
+ const path = shortenPath(args.path || '.');
331
+ const glob = args.glob;
332
+ const limit = args.limit;
333
+
334
+ let text = `${theme.fg('toolTitle', theme.bold('grep'))} ${theme.fg('accent', `/${pattern}/`)}`;
335
+ text += theme.fg('toolOutput', ` in ${path}`);
336
+ if (glob) {
337
+ text += theme.fg('toolOutput', ` (${glob})`);
338
+ }
339
+ if (limit !== undefined) {
340
+ text += theme.fg('toolOutput', ` limit ${limit}`);
341
+ }
342
+
343
+ return new Text(text, 0, 0);
344
+ },
345
+
346
+ renderResult(result, { expanded }, theme) {
347
+ if (!expanded) {
348
+ // Minimal: just show match count
349
+ const textContent = result.content.find((c) => c.type === 'text');
350
+ if (textContent?.type === 'text') {
351
+ const count = textContent.text.trim().split('\n').filter(Boolean).length;
352
+ if (count > 0) {
353
+ return new Text(theme.fg('muted', ` → ${count} matches`), 0, 0);
354
+ }
355
+ }
356
+ return new Text('', 0, 0);
357
+ }
358
+
359
+ // Expanded: show full results
360
+ const textContent = result.content.find((c) => c.type === 'text');
361
+ if (!textContent || textContent.type !== 'text') {
362
+ return new Text('', 0, 0);
363
+ }
364
+
365
+ const output = textContent.text
366
+ .trim()
367
+ .split('\n')
368
+ .map((line) => theme.fg('toolOutput', line))
369
+ .join('\n');
370
+
371
+ return new Text(`\n${output}`, 0, 0);
372
+ }
373
+ });
374
+
375
+ // =========================================================================
376
+ // Ls Tool
377
+ // =========================================================================
378
+ shortcut.registerTool({
379
+ name: 'ls',
380
+ label: 'ls',
381
+ description:
382
+ 'List directory contents with file sizes. Shows files and directories with their sizes. Output limited to 500 entries.',
383
+ parameters: getBuiltInTools(process.cwd()).ls.parameters,
384
+
385
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
386
+ const tools = getBuiltInTools(ctx.cwd);
387
+ return tools.ls.execute(toolCallId, params, signal, onUpdate);
388
+ },
389
+
390
+ renderCall(args, theme) {
391
+ const path = shortenPath(args.path || '.');
392
+ const limit = args.limit;
393
+
394
+ let text = `${theme.fg('toolTitle', theme.bold('ls'))} ${theme.fg('accent', path)}`;
395
+ if (limit !== undefined) {
396
+ text += theme.fg('toolOutput', ` (limit ${limit})`);
397
+ }
398
+
399
+ return new Text(text, 0, 0);
400
+ },
401
+
402
+ renderResult(result, { expanded }, theme) {
403
+ if (!expanded) {
404
+ // Minimal: just show entry count
405
+ const textContent = result.content.find((c) => c.type === 'text');
406
+ if (textContent?.type === 'text') {
407
+ const count = textContent.text.trim().split('\n').filter(Boolean).length;
408
+ if (count > 0) {
409
+ return new Text(theme.fg('muted', ` → ${count} entries`), 0, 0);
410
+ }
411
+ }
412
+ return new Text('', 0, 0);
413
+ }
414
+
415
+ // Expanded: show full listing
416
+ const textContent = result.content.find((c) => c.type === 'text');
417
+ if (!textContent || textContent.type !== 'text') {
418
+ return new Text('', 0, 0);
419
+ }
420
+
421
+ const output = textContent.text
422
+ .trim()
423
+ .split('\n')
424
+ .map((line) => theme.fg('toolOutput', line))
425
+ .join('\n');
426
+
427
+ return new Text(`\n${output}`, 0, 0);
428
+ }
429
+ });
430
+ }
@@ -0,0 +1,90 @@
1
+ /**
2
+ * Modal Editor - vim-like modal editing example
3
+ *
4
+ * Usage: shortcut --extension ./examples/extensions/modal-editor.ts
5
+ *
6
+ * - Escape: insert → normal mode (in normal mode, aborts agent)
7
+ * - i: normal → insert mode
8
+ * - hjkl: navigation in normal mode
9
+ * - ctrl+c, ctrl+d, etc. work in both modes
10
+ */
11
+
12
+ import {
13
+ CustomEditor,
14
+ matchesKey,
15
+ truncateToWidth,
16
+ visibleWidth,
17
+ type ExtensionAPI
18
+ } from 'shortcutxl';
19
+
20
+ // Normal mode key mappings: key -> escape sequence (or null for mode switch)
21
+ const NORMAL_KEYS: Record<string, string | null> = {
22
+ h: '\x1b[D', // left
23
+ j: '\x1b[B', // down
24
+ k: '\x1b[A', // up
25
+ l: '\x1b[C', // right
26
+ '0': '\x01', // line start
27
+ $: '\x05', // line end
28
+ x: '\x1b[3~', // delete char
29
+ i: null, // insert mode
30
+ a: null // append (insert + right)
31
+ };
32
+
33
+ class ModalEditor extends CustomEditor {
34
+ private mode: 'normal' | 'insert' = 'insert';
35
+
36
+ handleInput(data: string): void {
37
+ // Escape toggles to normal mode, or passes through for app handling
38
+ if (matchesKey(data, 'escape')) {
39
+ if (this.mode === 'insert') {
40
+ this.mode = 'normal';
41
+ } else {
42
+ super.handleInput(data); // abort agent, etc.
43
+ }
44
+ return;
45
+ }
46
+
47
+ // Insert mode: pass everything through
48
+ if (this.mode === 'insert') {
49
+ super.handleInput(data);
50
+ return;
51
+ }
52
+
53
+ // Normal mode: check mapped keys
54
+ if (data in NORMAL_KEYS) {
55
+ const seq = NORMAL_KEYS[data];
56
+ if (data === 'i') {
57
+ this.mode = 'insert';
58
+ } else if (data === 'a') {
59
+ this.mode = 'insert';
60
+ super.handleInput('\x1b[C'); // move right first
61
+ } else if (seq) {
62
+ super.handleInput(seq);
63
+ }
64
+ return;
65
+ }
66
+
67
+ // Pass control sequences (ctrl+c, etc.) to super, ignore printable chars
68
+ if (data.length === 1 && data.charCodeAt(0) >= 32) return;
69
+ super.handleInput(data);
70
+ }
71
+
72
+ render(width: number): string[] {
73
+ const lines = super.render(width);
74
+ if (lines.length === 0) return lines;
75
+
76
+ // Add mode indicator to bottom border
77
+ const label = this.mode === 'normal' ? ' NORMAL ' : ' INSERT ';
78
+ const last = lines.length - 1;
79
+ if (visibleWidth(lines[last]!) >= label.length) {
80
+ lines[last] = truncateToWidth(lines[last]!, width - label.length, '') + label;
81
+ }
82
+ return lines;
83
+ }
84
+ }
85
+
86
+ export default function (shortcut: ExtensionAPI) {
87
+ shortcut.on('session_start', (_event, ctx) => {
88
+ ctx.ui.setEditorComponent((tui, theme, kb) => new ModalEditor(tui, theme, kb));
89
+ });
90
+ }
@@ -0,0 +1,31 @@
1
+ /**
2
+ * Model status extension - shows model changes in the status bar.
3
+ *
4
+ * Demonstrates the `model_select` hook which fires when the model changes
5
+ * via /model command, Ctrl+P cycling, or session restore.
6
+ *
7
+ * Usage: shortcut -e ./model-status.ts
8
+ */
9
+
10
+ import type { ExtensionAPI } from 'shortcutxl';
11
+
12
+ export default function (shortcut: ExtensionAPI) {
13
+ shortcut.on('model_select', async (event, ctx) => {
14
+ const { model, previousModel, source } = event;
15
+
16
+ // Format model identifiers
17
+ const next = `${model.provider}/${model.id}`;
18
+ const prev = previousModel ? `${previousModel.provider}/${previousModel.id}` : 'none';
19
+
20
+ // Show notification on change
21
+ if (source !== 'restore') {
22
+ ctx.ui.notify(`Model: ${next}`, 'info');
23
+ }
24
+
25
+ // Update status bar with current model
26
+ ctx.ui.setStatus('model', `🤖 ${model.id}`);
27
+
28
+ // Log change details (visible in debug output)
29
+ console.log(`[model_select] ${prev} → ${next} (${source})`);
30
+ });
31
+ }
@@ -0,0 +1,55 @@
1
+ /**
2
+ * Pi Notify Extension
3
+ *
4
+ * Sends a native terminal notification when Pi agent is done and waiting for input.
5
+ * Supports multiple terminal protocols:
6
+ * - OSC 777: Ghostty, iTerm2, WezTerm, rxvt-unicode
7
+ * - OSC 99: Kitty
8
+ * - Windows toast: Windows Terminal (WSL)
9
+ */
10
+
11
+ import type { ExtensionAPI } from 'shortcutxl';
12
+
13
+ function windowsToastScript(title: string, body: string): string {
14
+ const type = 'Windows.UI.Notifications';
15
+ const mgr = `[${type}.ToastNotificationManager, ${type}, ContentType = WindowsRuntime]`;
16
+ const template = `[${type}.ToastTemplateType]::ToastText01`;
17
+ const toast = `[${type}.ToastNotification]::new($xml)`;
18
+ return [
19
+ `${mgr} > $null`,
20
+ `$xml = [${type}.ToastNotificationManager]::GetTemplateContent(${template})`,
21
+ `$xml.GetElementsByTagName('text')[0].AppendChild($xml.CreateTextNode('${body}')) > $null`,
22
+ `[${type}.ToastNotificationManager]::CreateToastNotifier('${title}').Show(${toast})`
23
+ ].join('; ');
24
+ }
25
+
26
+ function notifyOSC777(title: string, body: string): void {
27
+ process.stdout.write(`\x1b]777;notify;${title};${body}\x07`);
28
+ }
29
+
30
+ function notifyOSC99(title: string, body: string): void {
31
+ // Kitty OSC 99: i=notification id, d=0 means not done yet, p=body for second part
32
+ process.stdout.write(`\x1b]99;i=1:d=0;${title}\x1b\\`);
33
+ process.stdout.write(`\x1b]99;i=1:p=body;${body}\x1b\\`);
34
+ }
35
+
36
+ function notifyWindows(title: string, body: string): void {
37
+ const { execFile } = require('child_process');
38
+ execFile('powershell.exe', ['-NoProfile', '-Command', windowsToastScript(title, body)]);
39
+ }
40
+
41
+ function notify(title: string, body: string): void {
42
+ if (process.env.WT_SESSION) {
43
+ notifyWindows(title, body);
44
+ } else if (process.env.KITTY_WINDOW_ID) {
45
+ notifyOSC99(title, body);
46
+ } else {
47
+ notifyOSC777(title, body);
48
+ }
49
+ }
50
+
51
+ export default function (shortcut: ExtensionAPI) {
52
+ shortcut.on('agent_end', async () => {
53
+ notify('Pi', 'Ready for input');
54
+ });
55
+ }