open-pi-coding-agent 0.57.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (604) hide show
  1. package/CHANGELOG.md +3045 -0
  2. package/README.md +571 -0
  3. package/dist/cli/args.d.ts +48 -0
  4. package/dist/cli/args.d.ts.map +1 -0
  5. package/dist/cli/args.js +299 -0
  6. package/dist/cli/args.js.map +1 -0
  7. package/dist/cli/config-selector.d.ts +14 -0
  8. package/dist/cli/config-selector.d.ts.map +1 -0
  9. package/dist/cli/config-selector.js +31 -0
  10. package/dist/cli/config-selector.js.map +1 -0
  11. package/dist/cli/file-processor.d.ts +15 -0
  12. package/dist/cli/file-processor.d.ts.map +1 -0
  13. package/dist/cli/file-processor.js +79 -0
  14. package/dist/cli/file-processor.js.map +1 -0
  15. package/dist/cli/list-models.d.ts +9 -0
  16. package/dist/cli/list-models.d.ts.map +1 -0
  17. package/dist/cli/list-models.js +92 -0
  18. package/dist/cli/list-models.js.map +1 -0
  19. package/dist/cli/session-picker.d.ts +9 -0
  20. package/dist/cli/session-picker.d.ts.map +1 -0
  21. package/dist/cli/session-picker.js +34 -0
  22. package/dist/cli/session-picker.js.map +1 -0
  23. package/dist/cli.d.ts +3 -0
  24. package/dist/cli.d.ts.map +1 -0
  25. package/dist/cli.js +16 -0
  26. package/dist/cli.js.map +1 -0
  27. package/dist/config.d.ts +68 -0
  28. package/dist/config.d.ts.map +1 -0
  29. package/dist/config.js +203 -0
  30. package/dist/config.js.map +1 -0
  31. package/dist/core/agent-session.d.ts +582 -0
  32. package/dist/core/agent-session.d.ts.map +1 -0
  33. package/dist/core/agent-session.js +2494 -0
  34. package/dist/core/agent-session.js.map +1 -0
  35. package/dist/core/auth-storage.d.ts +130 -0
  36. package/dist/core/auth-storage.d.ts.map +1 -0
  37. package/dist/core/auth-storage.js +419 -0
  38. package/dist/core/auth-storage.js.map +1 -0
  39. package/dist/core/bash-executor.d.ts +47 -0
  40. package/dist/core/bash-executor.d.ts.map +1 -0
  41. package/dist/core/bash-executor.js +212 -0
  42. package/dist/core/bash-executor.js.map +1 -0
  43. package/dist/core/compaction/branch-summarization.d.ts +86 -0
  44. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  45. package/dist/core/compaction/branch-summarization.js +242 -0
  46. package/dist/core/compaction/branch-summarization.js.map +1 -0
  47. package/dist/core/compaction/compaction.d.ts +121 -0
  48. package/dist/core/compaction/compaction.d.ts.map +1 -0
  49. package/dist/core/compaction/compaction.js +610 -0
  50. package/dist/core/compaction/compaction.js.map +1 -0
  51. package/dist/core/compaction/index.d.ts +7 -0
  52. package/dist/core/compaction/index.d.ts.map +1 -0
  53. package/dist/core/compaction/index.js +7 -0
  54. package/dist/core/compaction/index.js.map +1 -0
  55. package/dist/core/compaction/utils.d.ts +38 -0
  56. package/dist/core/compaction/utils.d.ts.map +1 -0
  57. package/dist/core/compaction/utils.js +153 -0
  58. package/dist/core/compaction/utils.js.map +1 -0
  59. package/dist/core/defaults.d.ts +3 -0
  60. package/dist/core/defaults.d.ts.map +1 -0
  61. package/dist/core/defaults.js +2 -0
  62. package/dist/core/defaults.js.map +1 -0
  63. package/dist/core/diagnostics.d.ts +15 -0
  64. package/dist/core/diagnostics.d.ts.map +1 -0
  65. package/dist/core/diagnostics.js +2 -0
  66. package/dist/core/diagnostics.js.map +1 -0
  67. package/dist/core/event-bus.d.ts +9 -0
  68. package/dist/core/event-bus.d.ts.map +1 -0
  69. package/dist/core/event-bus.js +25 -0
  70. package/dist/core/event-bus.js.map +1 -0
  71. package/dist/core/exec.d.ts +29 -0
  72. package/dist/core/exec.d.ts.map +1 -0
  73. package/dist/core/exec.js +71 -0
  74. package/dist/core/exec.js.map +1 -0
  75. package/dist/core/export-html/ansi-to-html.d.ts +22 -0
  76. package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
  77. package/dist/core/export-html/ansi-to-html.js +249 -0
  78. package/dist/core/export-html/ansi-to-html.js.map +1 -0
  79. package/dist/core/export-html/index.d.ts +37 -0
  80. package/dist/core/export-html/index.d.ts.map +1 -0
  81. package/dist/core/export-html/index.js +223 -0
  82. package/dist/core/export-html/index.js.map +1 -0
  83. package/dist/core/export-html/template.css +971 -0
  84. package/dist/core/export-html/template.html +54 -0
  85. package/dist/core/export-html/template.js +1583 -0
  86. package/dist/core/export-html/tool-renderer.d.ts +38 -0
  87. package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
  88. package/dist/core/export-html/tool-renderer.js +70 -0
  89. package/dist/core/export-html/tool-renderer.js.map +1 -0
  90. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  91. package/dist/core/export-html/vendor/marked.min.js +6 -0
  92. package/dist/core/extensions/index.d.ts +11 -0
  93. package/dist/core/extensions/index.d.ts.map +1 -0
  94. package/dist/core/extensions/index.js +9 -0
  95. package/dist/core/extensions/index.js.map +1 -0
  96. package/dist/core/extensions/loader.d.ts +25 -0
  97. package/dist/core/extensions/loader.d.ts.map +1 -0
  98. package/dist/core/extensions/loader.js +426 -0
  99. package/dist/core/extensions/loader.js.map +1 -0
  100. package/dist/core/extensions/runner.d.ts +147 -0
  101. package/dist/core/extensions/runner.d.ts.map +1 -0
  102. package/dist/core/extensions/runner.js +678 -0
  103. package/dist/core/extensions/runner.js.map +1 -0
  104. package/dist/core/extensions/types.d.ts +1036 -0
  105. package/dist/core/extensions/types.d.ts.map +1 -0
  106. package/dist/core/extensions/types.js +35 -0
  107. package/dist/core/extensions/types.js.map +1 -0
  108. package/dist/core/extensions/wrapper.d.ts +27 -0
  109. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  110. package/dist/core/extensions/wrapper.js +102 -0
  111. package/dist/core/extensions/wrapper.js.map +1 -0
  112. package/dist/core/footer-data-provider.d.ts +32 -0
  113. package/dist/core/footer-data-provider.d.ts.map +1 -0
  114. package/dist/core/footer-data-provider.js +134 -0
  115. package/dist/core/footer-data-provider.js.map +1 -0
  116. package/dist/core/index.d.ts +9 -0
  117. package/dist/core/index.d.ts.map +1 -0
  118. package/dist/core/index.js +9 -0
  119. package/dist/core/index.js.map +1 -0
  120. package/dist/core/keybindings.d.ts +55 -0
  121. package/dist/core/keybindings.d.ts.map +1 -0
  122. package/dist/core/keybindings.js +153 -0
  123. package/dist/core/keybindings.js.map +1 -0
  124. package/dist/core/messages.d.ts +77 -0
  125. package/dist/core/messages.d.ts.map +1 -0
  126. package/dist/core/messages.js +123 -0
  127. package/dist/core/messages.js.map +1 -0
  128. package/dist/core/model-registry.d.ts +113 -0
  129. package/dist/core/model-registry.d.ts.map +1 -0
  130. package/dist/core/model-registry.js +537 -0
  131. package/dist/core/model-registry.js.map +1 -0
  132. package/dist/core/model-resolver.d.ts +104 -0
  133. package/dist/core/model-resolver.d.ts.map +1 -0
  134. package/dist/core/model-resolver.js +462 -0
  135. package/dist/core/model-resolver.js.map +1 -0
  136. package/dist/core/package-manager.d.ts +151 -0
  137. package/dist/core/package-manager.d.ts.map +1 -0
  138. package/dist/core/package-manager.js +1447 -0
  139. package/dist/core/package-manager.js.map +1 -0
  140. package/dist/core/prompt-templates.d.ts +50 -0
  141. package/dist/core/prompt-templates.d.ts.map +1 -0
  142. package/dist/core/prompt-templates.js +251 -0
  143. package/dist/core/prompt-templates.js.map +1 -0
  144. package/dist/core/resolve-config-value.d.ts +17 -0
  145. package/dist/core/resolve-config-value.d.ts.map +1 -0
  146. package/dist/core/resolve-config-value.js +59 -0
  147. package/dist/core/resolve-config-value.js.map +1 -0
  148. package/dist/core/resource-loader.d.ts +184 -0
  149. package/dist/core/resource-loader.d.ts.map +1 -0
  150. package/dist/core/resource-loader.js +670 -0
  151. package/dist/core/resource-loader.js.map +1 -0
  152. package/dist/core/sdk.d.ts +90 -0
  153. package/dist/core/sdk.d.ts.map +1 -0
  154. package/dist/core/sdk.js +242 -0
  155. package/dist/core/sdk.js.map +1 -0
  156. package/dist/core/session-manager.d.ts +323 -0
  157. package/dist/core/session-manager.d.ts.map +1 -0
  158. package/dist/core/session-manager.js +1098 -0
  159. package/dist/core/session-manager.js.map +1 -0
  160. package/dist/core/settings-manager.d.ts +232 -0
  161. package/dist/core/settings-manager.d.ts.map +1 -0
  162. package/dist/core/settings-manager.js +691 -0
  163. package/dist/core/settings-manager.js.map +1 -0
  164. package/dist/core/skills.d.ts +58 -0
  165. package/dist/core/skills.d.ts.map +1 -0
  166. package/dist/core/skills.js +364 -0
  167. package/dist/core/skills.js.map +1 -0
  168. package/dist/core/slash-commands.d.ts +15 -0
  169. package/dist/core/slash-commands.d.ts.map +1 -0
  170. package/dist/core/slash-commands.js +22 -0
  171. package/dist/core/slash-commands.js.map +1 -0
  172. package/dist/core/system-prompt.d.ts +28 -0
  173. package/dist/core/system-prompt.d.ts.map +1 -0
  174. package/dist/core/system-prompt.js +159 -0
  175. package/dist/core/system-prompt.js.map +1 -0
  176. package/dist/core/timings.d.ts +7 -0
  177. package/dist/core/timings.d.ts.map +1 -0
  178. package/dist/core/timings.js +25 -0
  179. package/dist/core/timings.js.map +1 -0
  180. package/dist/core/tools/bash.d.ts +55 -0
  181. package/dist/core/tools/bash.d.ts.map +1 -0
  182. package/dist/core/tools/bash.js +242 -0
  183. package/dist/core/tools/bash.js.map +1 -0
  184. package/dist/core/tools/edit-diff.d.ts +63 -0
  185. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  186. package/dist/core/tools/edit-diff.js +243 -0
  187. package/dist/core/tools/edit-diff.js.map +1 -0
  188. package/dist/core/tools/edit.d.ts +39 -0
  189. package/dist/core/tools/edit.d.ts.map +1 -0
  190. package/dist/core/tools/edit.js +146 -0
  191. package/dist/core/tools/edit.js.map +1 -0
  192. package/dist/core/tools/find.d.ts +39 -0
  193. package/dist/core/tools/find.d.ts.map +1 -0
  194. package/dist/core/tools/find.js +206 -0
  195. package/dist/core/tools/find.js.map +1 -0
  196. package/dist/core/tools/grep.d.ts +45 -0
  197. package/dist/core/tools/grep.d.ts.map +1 -0
  198. package/dist/core/tools/grep.js +239 -0
  199. package/dist/core/tools/grep.js.map +1 -0
  200. package/dist/core/tools/index.d.ts +73 -0
  201. package/dist/core/tools/index.d.ts.map +1 -0
  202. package/dist/core/tools/index.js +61 -0
  203. package/dist/core/tools/index.js.map +1 -0
  204. package/dist/core/tools/ls.d.ts +40 -0
  205. package/dist/core/tools/ls.d.ts.map +1 -0
  206. package/dist/core/tools/ls.js +118 -0
  207. package/dist/core/tools/ls.js.map +1 -0
  208. package/dist/core/tools/path-utils.d.ts +8 -0
  209. package/dist/core/tools/path-utils.d.ts.map +1 -0
  210. package/dist/core/tools/path-utils.js +81 -0
  211. package/dist/core/tools/path-utils.js.map +1 -0
  212. package/dist/core/tools/read.d.ts +39 -0
  213. package/dist/core/tools/read.d.ts.map +1 -0
  214. package/dist/core/tools/read.js +166 -0
  215. package/dist/core/tools/read.js.map +1 -0
  216. package/dist/core/tools/truncate.d.ts +70 -0
  217. package/dist/core/tools/truncate.d.ts.map +1 -0
  218. package/dist/core/tools/truncate.js +205 -0
  219. package/dist/core/tools/truncate.js.map +1 -0
  220. package/dist/core/tools/write.d.ts +29 -0
  221. package/dist/core/tools/write.d.ts.map +1 -0
  222. package/dist/core/tools/write.js +78 -0
  223. package/dist/core/tools/write.js.map +1 -0
  224. package/dist/index.d.ts +27 -0
  225. package/dist/index.d.ts.map +1 -0
  226. package/dist/index.js +42 -0
  227. package/dist/index.js.map +1 -0
  228. package/dist/main.d.ts +8 -0
  229. package/dist/main.d.ts.map +1 -0
  230. package/dist/main.js +691 -0
  231. package/dist/main.js.map +1 -0
  232. package/dist/migrations.d.ts +33 -0
  233. package/dist/migrations.d.ts.map +1 -0
  234. package/dist/migrations.js +261 -0
  235. package/dist/migrations.js.map +1 -0
  236. package/dist/modes/index.d.ts +9 -0
  237. package/dist/modes/index.d.ts.map +1 -0
  238. package/dist/modes/index.js +8 -0
  239. package/dist/modes/index.js.map +1 -0
  240. package/dist/modes/interactive/components/armin.d.ts +34 -0
  241. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  242. package/dist/modes/interactive/components/armin.js +333 -0
  243. package/dist/modes/interactive/components/armin.js.map +1 -0
  244. package/dist/modes/interactive/components/assistant-message.d.ts +16 -0
  245. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  246. package/dist/modes/interactive/components/assistant-message.js +96 -0
  247. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  248. package/dist/modes/interactive/components/bash-execution.d.ts +35 -0
  249. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  250. package/dist/modes/interactive/components/bash-execution.js +162 -0
  251. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  252. package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
  253. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  254. package/dist/modes/interactive/components/bordered-loader.js +51 -0
  255. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  256. package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
  257. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  258. package/dist/modes/interactive/components/branch-summary-message.js +44 -0
  259. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  260. package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
  261. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  262. package/dist/modes/interactive/components/compaction-summary-message.js +45 -0
  263. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  264. package/dist/modes/interactive/components/config-selector.d.ts +71 -0
  265. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  266. package/dist/modes/interactive/components/config-selector.js +479 -0
  267. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  268. package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
  269. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  270. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  271. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  272. package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
  273. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  274. package/dist/modes/interactive/components/custom-editor.js +70 -0
  275. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  276. package/dist/modes/interactive/components/custom-message.d.ts +20 -0
  277. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  278. package/dist/modes/interactive/components/custom-message.js +79 -0
  279. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  280. package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
  281. package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
  282. package/dist/modes/interactive/components/daxnuts.js +140 -0
  283. package/dist/modes/interactive/components/daxnuts.js.map +1 -0
  284. package/dist/modes/interactive/components/diff.d.ts +12 -0
  285. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  286. package/dist/modes/interactive/components/diff.js +133 -0
  287. package/dist/modes/interactive/components/diff.js.map +1 -0
  288. package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
  289. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  290. package/dist/modes/interactive/components/dynamic-border.js +21 -0
  291. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  292. package/dist/modes/interactive/components/extension-editor.d.ts +20 -0
  293. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  294. package/dist/modes/interactive/components/extension-editor.js +111 -0
  295. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  296. package/dist/modes/interactive/components/extension-input.d.ts +23 -0
  297. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  298. package/dist/modes/interactive/components/extension-input.js +61 -0
  299. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  300. package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
  301. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  302. package/dist/modes/interactive/components/extension-selector.js +78 -0
  303. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  304. package/dist/modes/interactive/components/footer.d.ts +26 -0
  305. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  306. package/dist/modes/interactive/components/footer.js +198 -0
  307. package/dist/modes/interactive/components/footer.js.map +1 -0
  308. package/dist/modes/interactive/components/index.d.ts +32 -0
  309. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  310. package/dist/modes/interactive/components/index.js +33 -0
  311. package/dist/modes/interactive/components/index.js.map +1 -0
  312. package/dist/modes/interactive/components/keybinding-hints.d.ts +41 -0
  313. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
  314. package/dist/modes/interactive/components/keybinding-hints.js +61 -0
  315. package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
  316. package/dist/modes/interactive/components/login-dialog.d.ts +42 -0
  317. package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
  318. package/dist/modes/interactive/components/login-dialog.js +145 -0
  319. package/dist/modes/interactive/components/login-dialog.js.map +1 -0
  320. package/dist/modes/interactive/components/model-selector.d.ts +47 -0
  321. package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
  322. package/dist/modes/interactive/components/model-selector.js +271 -0
  323. package/dist/modes/interactive/components/model-selector.js.map +1 -0
  324. package/dist/modes/interactive/components/oauth-selector.d.ts +19 -0
  325. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
  326. package/dist/modes/interactive/components/oauth-selector.js +97 -0
  327. package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
  328. package/dist/modes/interactive/components/scoped-models-selector.d.ts +49 -0
  329. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
  330. package/dist/modes/interactive/components/scoped-models-selector.js +275 -0
  331. package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
  332. package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
  333. package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
  334. package/dist/modes/interactive/components/session-selector-search.js +155 -0
  335. package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
  336. package/dist/modes/interactive/components/session-selector.d.ts +95 -0
  337. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  338. package/dist/modes/interactive/components/session-selector.js +851 -0
  339. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  340. package/dist/modes/interactive/components/settings-selector.d.ts +58 -0
  341. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  342. package/dist/modes/interactive/components/settings-selector.js +297 -0
  343. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  344. package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
  345. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  346. package/dist/modes/interactive/components/show-images-selector.js +35 -0
  347. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  348. package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
  349. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  350. package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
  351. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  352. package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
  353. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  354. package/dist/modes/interactive/components/theme-selector.js +46 -0
  355. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  356. package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
  357. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  358. package/dist/modes/interactive/components/thinking-selector.js +47 -0
  359. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  360. package/dist/modes/interactive/components/tool-execution.d.ts +77 -0
  361. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  362. package/dist/modes/interactive/components/tool-execution.js +787 -0
  363. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  364. package/dist/modes/interactive/components/tree-selector.d.ts +87 -0
  365. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  366. package/dist/modes/interactive/components/tree-selector.js +1040 -0
  367. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  368. package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
  369. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  370. package/dist/modes/interactive/components/user-message-selector.js +113 -0
  371. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  372. package/dist/modes/interactive/components/user-message.d.ts +9 -0
  373. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  374. package/dist/modes/interactive/components/user-message.js +27 -0
  375. package/dist/modes/interactive/components/user-message.js.map +1 -0
  376. package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
  377. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  378. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  379. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  380. package/dist/modes/interactive/interactive-mode.d.ts +316 -0
  381. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  382. package/dist/modes/interactive/interactive-mode.js +3779 -0
  383. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  384. package/dist/modes/interactive/theme/dark.json +85 -0
  385. package/dist/modes/interactive/theme/light.json +84 -0
  386. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  387. package/dist/modes/interactive/theme/theme.d.ts +78 -0
  388. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  389. package/dist/modes/interactive/theme/theme.js +949 -0
  390. package/dist/modes/interactive/theme/theme.js.map +1 -0
  391. package/dist/modes/print-mode.d.ts +28 -0
  392. package/dist/modes/print-mode.d.ts.map +1 -0
  393. package/dist/modes/print-mode.js +101 -0
  394. package/dist/modes/print-mode.js.map +1 -0
  395. package/dist/modes/rpc/jsonl.d.ts +17 -0
  396. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  397. package/dist/modes/rpc/jsonl.js +49 -0
  398. package/dist/modes/rpc/jsonl.js.map +1 -0
  399. package/dist/modes/rpc/rpc-client.d.ts +217 -0
  400. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  401. package/dist/modes/rpc/rpc-client.js +401 -0
  402. package/dist/modes/rpc/rpc-client.js.map +1 -0
  403. package/dist/modes/rpc/rpc-mode.d.ts +20 -0
  404. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  405. package/dist/modes/rpc/rpc-mode.js +509 -0
  406. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  407. package/dist/modes/rpc/rpc-types.d.ts +409 -0
  408. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  409. package/dist/modes/rpc/rpc-types.js +8 -0
  410. package/dist/modes/rpc/rpc-types.js.map +1 -0
  411. package/dist/utils/changelog.d.ts +21 -0
  412. package/dist/utils/changelog.d.ts.map +1 -0
  413. package/dist/utils/changelog.js +87 -0
  414. package/dist/utils/changelog.js.map +1 -0
  415. package/dist/utils/clipboard-image.d.ts +11 -0
  416. package/dist/utils/clipboard-image.d.ts.map +1 -0
  417. package/dist/utils/clipboard-image.js +162 -0
  418. package/dist/utils/clipboard-image.js.map +1 -0
  419. package/dist/utils/clipboard-native.d.ts +7 -0
  420. package/dist/utils/clipboard-native.d.ts.map +1 -0
  421. package/dist/utils/clipboard-native.js +14 -0
  422. package/dist/utils/clipboard-native.js.map +1 -0
  423. package/dist/utils/clipboard.d.ts +2 -0
  424. package/dist/utils/clipboard.d.ts.map +1 -0
  425. package/dist/utils/clipboard.js +67 -0
  426. package/dist/utils/clipboard.js.map +1 -0
  427. package/dist/utils/frontmatter.d.ts +8 -0
  428. package/dist/utils/frontmatter.d.ts.map +1 -0
  429. package/dist/utils/frontmatter.js +26 -0
  430. package/dist/utils/frontmatter.js.map +1 -0
  431. package/dist/utils/git.d.ts +26 -0
  432. package/dist/utils/git.d.ts.map +1 -0
  433. package/dist/utils/git.js +163 -0
  434. package/dist/utils/git.js.map +1 -0
  435. package/dist/utils/image-convert.d.ts +9 -0
  436. package/dist/utils/image-convert.d.ts.map +1 -0
  437. package/dist/utils/image-convert.js +35 -0
  438. package/dist/utils/image-convert.js.map +1 -0
  439. package/dist/utils/image-resize.d.ts +36 -0
  440. package/dist/utils/image-resize.d.ts.map +1 -0
  441. package/dist/utils/image-resize.js +181 -0
  442. package/dist/utils/image-resize.js.map +1 -0
  443. package/dist/utils/mime.d.ts +2 -0
  444. package/dist/utils/mime.d.ts.map +1 -0
  445. package/dist/utils/mime.js +26 -0
  446. package/dist/utils/mime.js.map +1 -0
  447. package/dist/utils/photon.d.ts +21 -0
  448. package/dist/utils/photon.d.ts.map +1 -0
  449. package/dist/utils/photon.js +121 -0
  450. package/dist/utils/photon.js.map +1 -0
  451. package/dist/utils/shell.d.ts +26 -0
  452. package/dist/utils/shell.d.ts.map +1 -0
  453. package/dist/utils/shell.js +186 -0
  454. package/dist/utils/shell.js.map +1 -0
  455. package/dist/utils/sleep.d.ts +5 -0
  456. package/dist/utils/sleep.d.ts.map +1 -0
  457. package/dist/utils/sleep.js +17 -0
  458. package/dist/utils/sleep.js.map +1 -0
  459. package/dist/utils/tools-manager.d.ts +3 -0
  460. package/dist/utils/tools-manager.d.ts.map +1 -0
  461. package/dist/utils/tools-manager.js +251 -0
  462. package/dist/utils/tools-manager.js.map +1 -0
  463. package/docs/compaction.md +392 -0
  464. package/docs/custom-provider.md +592 -0
  465. package/docs/development.md +69 -0
  466. package/docs/extensions.md +2023 -0
  467. package/docs/images/doom-extension.png +0 -0
  468. package/docs/images/exy.png +0 -0
  469. package/docs/images/interactive-mode.png +0 -0
  470. package/docs/images/tree-view.png +0 -0
  471. package/docs/json.md +79 -0
  472. package/docs/keybindings.md +182 -0
  473. package/docs/models.md +297 -0
  474. package/docs/packages.md +209 -0
  475. package/docs/prompt-templates.md +67 -0
  476. package/docs/providers.md +188 -0
  477. package/docs/rpc.md +1354 -0
  478. package/docs/sdk.md +968 -0
  479. package/docs/session.md +412 -0
  480. package/docs/settings.md +225 -0
  481. package/docs/shell-aliases.md +13 -0
  482. package/docs/skills.md +231 -0
  483. package/docs/terminal-setup.md +95 -0
  484. package/docs/termux.md +127 -0
  485. package/docs/themes.md +295 -0
  486. package/docs/tmux.md +61 -0
  487. package/docs/tree.md +228 -0
  488. package/docs/tui.md +887 -0
  489. package/docs/windows.md +17 -0
  490. package/examples/README.md +25 -0
  491. package/examples/extensions/README.md +205 -0
  492. package/examples/extensions/antigravity-image-gen.ts +415 -0
  493. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  494. package/examples/extensions/bash-spawn-hook.ts +30 -0
  495. package/examples/extensions/bookmark.ts +50 -0
  496. package/examples/extensions/built-in-tool-renderer.ts +246 -0
  497. package/examples/extensions/claude-rules.ts +86 -0
  498. package/examples/extensions/commands.ts +72 -0
  499. package/examples/extensions/confirm-destructive.ts +59 -0
  500. package/examples/extensions/custom-compaction.ts +114 -0
  501. package/examples/extensions/custom-footer.ts +64 -0
  502. package/examples/extensions/custom-header.ts +73 -0
  503. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  504. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  505. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  506. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  507. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  508. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  509. package/examples/extensions/custom-provider-qwen-cli/index.ts +345 -0
  510. package/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
  511. package/examples/extensions/dirty-repo-guard.ts +56 -0
  512. package/examples/extensions/doom-overlay/README.md +46 -0
  513. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  514. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  515. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  516. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  517. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  518. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  519. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  520. package/examples/extensions/doom-overlay/index.ts +74 -0
  521. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  522. package/examples/extensions/dynamic-resources/SKILL.md +8 -0
  523. package/examples/extensions/dynamic-resources/dynamic.json +79 -0
  524. package/examples/extensions/dynamic-resources/dynamic.md +5 -0
  525. package/examples/extensions/dynamic-resources/index.ts +15 -0
  526. package/examples/extensions/dynamic-tools.ts +74 -0
  527. package/examples/extensions/event-bus.ts +43 -0
  528. package/examples/extensions/file-trigger.ts +41 -0
  529. package/examples/extensions/git-checkpoint.ts +53 -0
  530. package/examples/extensions/handoff.ts +150 -0
  531. package/examples/extensions/hello.ts +25 -0
  532. package/examples/extensions/inline-bash.ts +94 -0
  533. package/examples/extensions/input-transform.ts +43 -0
  534. package/examples/extensions/interactive-shell.ts +196 -0
  535. package/examples/extensions/mac-system-theme.ts +47 -0
  536. package/examples/extensions/message-renderer.ts +59 -0
  537. package/examples/extensions/minimal-mode.ts +426 -0
  538. package/examples/extensions/modal-editor.ts +85 -0
  539. package/examples/extensions/model-status.ts +31 -0
  540. package/examples/extensions/notify.ts +55 -0
  541. package/examples/extensions/overlay-qa-tests.ts +1348 -0
  542. package/examples/extensions/overlay-test.ts +150 -0
  543. package/examples/extensions/permission-gate.ts +34 -0
  544. package/examples/extensions/pirate.ts +47 -0
  545. package/examples/extensions/plan-mode/README.md +65 -0
  546. package/examples/extensions/plan-mode/index.ts +340 -0
  547. package/examples/extensions/plan-mode/utils.ts +168 -0
  548. package/examples/extensions/preset.ts +398 -0
  549. package/examples/extensions/protected-paths.ts +30 -0
  550. package/examples/extensions/provider-payload.ts +14 -0
  551. package/examples/extensions/qna.ts +119 -0
  552. package/examples/extensions/question.ts +264 -0
  553. package/examples/extensions/questionnaire.ts +427 -0
  554. package/examples/extensions/rainbow-editor.ts +88 -0
  555. package/examples/extensions/reload-runtime.ts +37 -0
  556. package/examples/extensions/rpc-demo.ts +124 -0
  557. package/examples/extensions/sandbox/index.ts +318 -0
  558. package/examples/extensions/sandbox/package-lock.json +92 -0
  559. package/examples/extensions/sandbox/package.json +19 -0
  560. package/examples/extensions/send-user-message.ts +97 -0
  561. package/examples/extensions/session-name.ts +27 -0
  562. package/examples/extensions/shutdown-command.ts +63 -0
  563. package/examples/extensions/snake.ts +343 -0
  564. package/examples/extensions/space-invaders.ts +560 -0
  565. package/examples/extensions/ssh.ts +220 -0
  566. package/examples/extensions/status-line.ts +40 -0
  567. package/examples/extensions/subagent/README.md +172 -0
  568. package/examples/extensions/subagent/agents/planner.md +37 -0
  569. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  570. package/examples/extensions/subagent/agents/scout.md +50 -0
  571. package/examples/extensions/subagent/agents/worker.md +24 -0
  572. package/examples/extensions/subagent/agents.ts +126 -0
  573. package/examples/extensions/subagent/index.ts +964 -0
  574. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  575. package/examples/extensions/subagent/prompts/implement.md +10 -0
  576. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  577. package/examples/extensions/summarize.ts +195 -0
  578. package/examples/extensions/system-prompt-header.ts +17 -0
  579. package/examples/extensions/timed-confirm.ts +70 -0
  580. package/examples/extensions/titlebar-spinner.ts +58 -0
  581. package/examples/extensions/todo.ts +299 -0
  582. package/examples/extensions/tool-override.ts +143 -0
  583. package/examples/extensions/tools.ts +146 -0
  584. package/examples/extensions/trigger-compact.ts +40 -0
  585. package/examples/extensions/truncated-tool.ts +192 -0
  586. package/examples/extensions/widget-placement.ts +17 -0
  587. package/examples/extensions/with-deps/index.ts +32 -0
  588. package/examples/extensions/with-deps/package-lock.json +31 -0
  589. package/examples/extensions/with-deps/package.json +22 -0
  590. package/examples/rpc-extension-ui.ts +632 -0
  591. package/examples/sdk/01-minimal.ts +22 -0
  592. package/examples/sdk/02-custom-model.ts +49 -0
  593. package/examples/sdk/03-custom-prompt.ts +55 -0
  594. package/examples/sdk/04-skills.ts +46 -0
  595. package/examples/sdk/05-tools.ts +56 -0
  596. package/examples/sdk/06-extensions.ts +88 -0
  597. package/examples/sdk/07-context-files.ts +40 -0
  598. package/examples/sdk/08-prompt-templates.ts +47 -0
  599. package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  600. package/examples/sdk/10-settings.ts +51 -0
  601. package/examples/sdk/11-sessions.ts +48 -0
  602. package/examples/sdk/12-full-control.ts +82 -0
  603. package/examples/sdk/README.md +144 -0
  604. package/package.json +99 -0
@@ -0,0 +1,1348 @@
1
+ /**
2
+ * Overlay QA Tests - comprehensive overlay positioning and edge case tests
3
+ *
4
+ * Usage: pi --extension ./examples/extensions/overlay-qa-tests.ts
5
+ *
6
+ * Commands:
7
+ * /overlay-animation - Real-time animation demo (~30 FPS, proves DOOM-like rendering works)
8
+ * /overlay-anchors - Cycle through all 9 anchor positions
9
+ * /overlay-margins - Test margin and offset options
10
+ * /overlay-stack - Test stacked overlays
11
+ * /overlay-overflow - Test width overflow with streaming process output
12
+ * /overlay-edge - Test overlay positioned at terminal edge
13
+ * /overlay-percent - Test percentage-based positioning
14
+ * /overlay-maxheight - Test maxHeight truncation
15
+ * /overlay-sidepanel - Responsive sidepanel (hides when terminal < 100 cols)
16
+ * /overlay-toggle - Toggle visibility demo (demonstrates OverlayHandle.setHidden)
17
+ * /overlay-passive - Non-capturing overlay demo (passive info panel alongside active overlay)
18
+ * /overlay-focus - Focus cycling and rendering order with non-capturing overlays
19
+ * /overlay-streaming - Multiple input panels with simulated streaming (Tab to cycle focus)
20
+ */
21
+
22
+ import type { ExtensionAPI, ExtensionCommandContext, Theme } from "open-pi-coding-agent";
23
+ import type { Component, OverlayAnchor, OverlayHandle, OverlayOptions, TUI } from "open-pi-tui";
24
+ import { matchesKey, truncateToWidth, visibleWidth } from "open-pi-tui";
25
+ import { spawn } from "child_process";
26
+
27
+ // Global handle for toggle demo (in real code, use a more elegant pattern)
28
+ let globalToggleHandle: OverlayHandle | null = null;
29
+
30
+ export default function (pi: ExtensionAPI) {
31
+ // Animation demo - proves overlays can handle real-time updates (like pi-doom would need)
32
+ pi.registerCommand("overlay-animation", {
33
+ description: "Test real-time animation in overlay (~30 FPS)",
34
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
35
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new AnimationDemoComponent(tui, theme, done), {
36
+ overlay: true,
37
+ overlayOptions: { anchor: "center", width: 50, maxHeight: 20 },
38
+ });
39
+ },
40
+ });
41
+
42
+ // Test all 9 anchor positions
43
+ pi.registerCommand("overlay-anchors", {
44
+ description: "Cycle through all anchor positions",
45
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
46
+ const anchors: OverlayAnchor[] = [
47
+ "top-left",
48
+ "top-center",
49
+ "top-right",
50
+ "left-center",
51
+ "center",
52
+ "right-center",
53
+ "bottom-left",
54
+ "bottom-center",
55
+ "bottom-right",
56
+ ];
57
+
58
+ let index = 0;
59
+ while (true) {
60
+ const result = await ctx.ui.custom<"next" | "confirm" | "cancel">(
61
+ (_tui, theme, _kb, done) => new AnchorTestComponent(theme, anchors[index]!, done),
62
+ {
63
+ overlay: true,
64
+ overlayOptions: { anchor: anchors[index], width: 40 },
65
+ },
66
+ );
67
+
68
+ if (result === "next") {
69
+ index = (index + 1) % anchors.length;
70
+ continue;
71
+ }
72
+ if (result === "confirm") {
73
+ ctx.ui.notify(`Selected: ${anchors[index]}`, "info");
74
+ }
75
+ break;
76
+ }
77
+ },
78
+ });
79
+
80
+ // Test margins and offsets
81
+ pi.registerCommand("overlay-margins", {
82
+ description: "Test margin and offset options",
83
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
84
+ const configs: { name: string; options: OverlayOptions }[] = [
85
+ { name: "No margin (top-left)", options: { anchor: "top-left", width: 35 } },
86
+ { name: "Margin: 3 all sides", options: { anchor: "top-left", width: 35, margin: 3 } },
87
+ {
88
+ name: "Margin: top=5, left=10",
89
+ options: { anchor: "top-left", width: 35, margin: { top: 5, left: 10 } },
90
+ },
91
+ { name: "Center + offset (10, -3)", options: { anchor: "center", width: 35, offsetX: 10, offsetY: -3 } },
92
+ { name: "Bottom-right, margin: 2", options: { anchor: "bottom-right", width: 35, margin: 2 } },
93
+ ];
94
+
95
+ let index = 0;
96
+ while (true) {
97
+ const result = await ctx.ui.custom<"next" | "close">(
98
+ (_tui, theme, _kb, done) => new MarginTestComponent(theme, configs[index]!, done),
99
+ {
100
+ overlay: true,
101
+ overlayOptions: configs[index]!.options,
102
+ },
103
+ );
104
+
105
+ if (result === "next") {
106
+ index = (index + 1) % configs.length;
107
+ continue;
108
+ }
109
+ break;
110
+ }
111
+ },
112
+ });
113
+
114
+ // Test stacked overlays
115
+ pi.registerCommand("overlay-stack", {
116
+ description: "Test stacked overlays",
117
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
118
+ // Three large overlays that overlap in the center area
119
+ // Each offset slightly so you can see the stacking
120
+
121
+ ctx.ui.notify("Showing overlay 1 (back)...", "info");
122
+ const p1 = ctx.ui.custom<string>(
123
+ (_tui, theme, _kb, done) => new StackOverlayComponent(theme, 1, "back (red border)", done),
124
+ {
125
+ overlay: true,
126
+ overlayOptions: { anchor: "center", width: 50, offsetX: -8, offsetY: -4, maxHeight: 15 },
127
+ },
128
+ );
129
+
130
+ await sleep(400);
131
+
132
+ ctx.ui.notify("Showing overlay 2 (middle)...", "info");
133
+ const p2 = ctx.ui.custom<string>(
134
+ (_tui, theme, _kb, done) => new StackOverlayComponent(theme, 2, "middle (green border)", done),
135
+ {
136
+ overlay: true,
137
+ overlayOptions: { anchor: "center", width: 50, offsetX: 0, offsetY: 0, maxHeight: 15 },
138
+ },
139
+ );
140
+
141
+ await sleep(400);
142
+
143
+ ctx.ui.notify("Showing overlay 3 (front)...", "info");
144
+ const p3 = ctx.ui.custom<string>(
145
+ (_tui, theme, _kb, done) => new StackOverlayComponent(theme, 3, "front (blue border)", done),
146
+ {
147
+ overlay: true,
148
+ overlayOptions: { anchor: "center", width: 50, offsetX: 8, offsetY: 4, maxHeight: 15 },
149
+ },
150
+ );
151
+
152
+ // Wait for all to close
153
+ const results = await Promise.all([p1, p2, p3]);
154
+ ctx.ui.notify(`Closed in order: ${results.join(", ")}`, "info");
155
+ },
156
+ });
157
+
158
+ // Test width overflow scenarios (original crash case) - streams real process output
159
+ pi.registerCommand("overlay-overflow", {
160
+ description: "Test width overflow with streaming process output",
161
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
162
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new StreamingOverflowComponent(tui, theme, done), {
163
+ overlay: true,
164
+ overlayOptions: { anchor: "center", width: 90, maxHeight: 20 },
165
+ });
166
+ },
167
+ });
168
+
169
+ // Test overlay at terminal edge
170
+ pi.registerCommand("overlay-edge", {
171
+ description: "Test overlay positioned at terminal edge",
172
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
173
+ await ctx.ui.custom<void>((_tui, theme, _kb, done) => new EdgeTestComponent(theme, done), {
174
+ overlay: true,
175
+ overlayOptions: { anchor: "right-center", width: 40, margin: { right: 0 } },
176
+ });
177
+ },
178
+ });
179
+
180
+ // Test percentage-based positioning
181
+ pi.registerCommand("overlay-percent", {
182
+ description: "Test percentage-based positioning",
183
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
184
+ const configs = [
185
+ { name: "rowPercent: 0 (top)", row: 0, col: 50 },
186
+ { name: "rowPercent: 50 (middle)", row: 50, col: 50 },
187
+ { name: "rowPercent: 100 (bottom)", row: 100, col: 50 },
188
+ { name: "colPercent: 0 (left)", row: 50, col: 0 },
189
+ { name: "colPercent: 100 (right)", row: 50, col: 100 },
190
+ ];
191
+
192
+ let index = 0;
193
+ while (true) {
194
+ const config = configs[index]!;
195
+ const result = await ctx.ui.custom<"next" | "close">(
196
+ (_tui, theme, _kb, done) => new PercentTestComponent(theme, config, done),
197
+ {
198
+ overlay: true,
199
+ overlayOptions: {
200
+ width: 30,
201
+ row: `${config.row}%`,
202
+ col: `${config.col}%`,
203
+ },
204
+ },
205
+ );
206
+
207
+ if (result === "next") {
208
+ index = (index + 1) % configs.length;
209
+ continue;
210
+ }
211
+ break;
212
+ }
213
+ },
214
+ });
215
+
216
+ // Test maxHeight
217
+ pi.registerCommand("overlay-maxheight", {
218
+ description: "Test maxHeight truncation",
219
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
220
+ await ctx.ui.custom<void>((_tui, theme, _kb, done) => new MaxHeightTestComponent(theme, done), {
221
+ overlay: true,
222
+ overlayOptions: { anchor: "center", width: 50, maxHeight: 10 },
223
+ });
224
+ },
225
+ });
226
+
227
+ // Test responsive sidepanel - only shows when terminal is wide enough
228
+ pi.registerCommand("overlay-sidepanel", {
229
+ description: "Test responsive sidepanel (hides when terminal < 100 cols)",
230
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
231
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new SidepanelComponent(tui, theme, done), {
232
+ overlay: true,
233
+ overlayOptions: {
234
+ anchor: "right-center",
235
+ width: "25%",
236
+ minWidth: 30,
237
+ margin: { right: 1 },
238
+ // Only show when terminal is wide enough
239
+ visible: (termWidth) => termWidth >= 100,
240
+ },
241
+ });
242
+ },
243
+ });
244
+
245
+ // Test toggle overlay - demonstrates OverlayHandle.setHidden() via onHandle callback
246
+ pi.registerCommand("overlay-toggle", {
247
+ description: "Test overlay toggle (press 't' to toggle visibility)",
248
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
249
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new ToggleDemoComponent(tui, theme, done), {
250
+ overlay: true,
251
+ overlayOptions: { anchor: "center", width: 50 },
252
+ // onHandle callback provides access to the OverlayHandle for visibility control
253
+ onHandle: (handle) => {
254
+ // Store handle globally so component can access it
255
+ // (In real code, you'd use a more elegant pattern like a store or event emitter)
256
+ globalToggleHandle = handle;
257
+ },
258
+ });
259
+ globalToggleHandle = null;
260
+ },
261
+ });
262
+
263
+ // Non-capturing overlay demo - passive info panel that doesn't steal focus
264
+ pi.registerCommand("overlay-passive", {
265
+ description: "Test non-capturing overlay (passive info panel alongside active overlay)",
266
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
267
+ ctx.ui.setEditorText("");
268
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new PassiveDemoController(tui, theme, done), {
269
+ overlay: true,
270
+ overlayOptions: { anchor: "center", width: 48 },
271
+ });
272
+ },
273
+ });
274
+
275
+ // Focus cycling demo - demonstrates focus(), unfocus(), isFocused() and rendering order
276
+ pi.registerCommand("overlay-focus", {
277
+ description: "Test focus cycling and rendering order with non-capturing overlays",
278
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
279
+ ctx.ui.setEditorText("");
280
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new FocusDemoController(tui, theme, done), {
281
+ overlay: true,
282
+ overlayOptions: { anchor: "bottom-center", width: 55, margin: { bottom: 1 } },
283
+ });
284
+ },
285
+ });
286
+
287
+ // Test multiple input panels with simulated streaming
288
+ pi.registerCommand("overlay-streaming", {
289
+ description: "Multiple input panels with simulated streaming (Tab to cycle focus)",
290
+ handler: async (_args: string, ctx: ExtensionCommandContext) => {
291
+ ctx.ui.setEditorText("");
292
+ await ctx.ui.custom<void>((tui, theme, _kb, done) => new StreamingInputController(tui, theme, done), {
293
+ overlay: true,
294
+ overlayOptions: { anchor: "bottom-center", width: 60, margin: { bottom: 1 } },
295
+ });
296
+ },
297
+ });
298
+ }
299
+
300
+ function sleep(ms: number): Promise<void> {
301
+ return new Promise((resolve) => setTimeout(resolve, ms));
302
+ }
303
+
304
+ // Base overlay component with common rendering
305
+ abstract class BaseOverlay {
306
+ constructor(protected theme: Theme) {}
307
+
308
+ protected box(lines: string[], width: number, title?: string): string[] {
309
+ const th = this.theme;
310
+ const innerW = Math.max(1, width - 2);
311
+ const result: string[] = [];
312
+
313
+ const titleStr = title ? truncateToWidth(` ${title} `, innerW) : "";
314
+ const titleW = visibleWidth(titleStr);
315
+ const topLine = "─".repeat(Math.floor((innerW - titleW) / 2));
316
+ const topLine2 = "─".repeat(Math.max(0, innerW - titleW - topLine.length));
317
+ result.push(th.fg("border", `╭${topLine}`) + th.fg("accent", titleStr) + th.fg("border", `${topLine2}╮`));
318
+
319
+ for (const line of lines) {
320
+ result.push(th.fg("border", "│") + truncateToWidth(line, innerW, "...", true) + th.fg("border", "│"));
321
+ }
322
+
323
+ result.push(th.fg("border", `╰${"─".repeat(innerW)}╯`));
324
+ return result;
325
+ }
326
+
327
+ invalidate(): void {}
328
+ dispose(): void {}
329
+ }
330
+
331
+ // Anchor position test
332
+ class AnchorTestComponent extends BaseOverlay {
333
+ constructor(
334
+ theme: Theme,
335
+ private anchor: OverlayAnchor,
336
+ private done: (result: "next" | "confirm" | "cancel") => void,
337
+ ) {
338
+ super(theme);
339
+ }
340
+
341
+ handleInput(data: string): void {
342
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
343
+ this.done("cancel");
344
+ } else if (matchesKey(data, "return")) {
345
+ this.done("confirm");
346
+ } else if (matchesKey(data, "space") || matchesKey(data, "right")) {
347
+ this.done("next");
348
+ }
349
+ }
350
+
351
+ render(width: number): string[] {
352
+ const th = this.theme;
353
+ return this.box(
354
+ [
355
+ "",
356
+ ` Current: ${th.fg("accent", this.anchor)}`,
357
+ "",
358
+ ` ${th.fg("dim", "Space/→ = next anchor")}`,
359
+ ` ${th.fg("dim", "Enter = confirm")}`,
360
+ ` ${th.fg("dim", "Esc = cancel")}`,
361
+ "",
362
+ ],
363
+ width,
364
+ "Anchor Test",
365
+ );
366
+ }
367
+ }
368
+
369
+ // Margin/offset test
370
+ class MarginTestComponent extends BaseOverlay {
371
+ constructor(
372
+ theme: Theme,
373
+ private config: { name: string; options: OverlayOptions },
374
+ private done: (result: "next" | "close") => void,
375
+ ) {
376
+ super(theme);
377
+ }
378
+
379
+ handleInput(data: string): void {
380
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
381
+ this.done("close");
382
+ } else if (matchesKey(data, "space") || matchesKey(data, "right")) {
383
+ this.done("next");
384
+ }
385
+ }
386
+
387
+ render(width: number): string[] {
388
+ const th = this.theme;
389
+ return this.box(
390
+ [
391
+ "",
392
+ ` ${th.fg("accent", this.config.name)}`,
393
+ "",
394
+ ` ${th.fg("dim", "Space/→ = next config")}`,
395
+ ` ${th.fg("dim", "Esc = close")}`,
396
+ "",
397
+ ],
398
+ width,
399
+ "Margin Test",
400
+ );
401
+ }
402
+ }
403
+
404
+ // Stacked overlay test
405
+ class StackOverlayComponent extends BaseOverlay {
406
+ constructor(
407
+ theme: Theme,
408
+ private num: number,
409
+ private position: string,
410
+ private done: (result: string) => void,
411
+ ) {
412
+ super(theme);
413
+ }
414
+
415
+ handleInput(data: string): void {
416
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c") || matchesKey(data, "return")) {
417
+ this.done(`Overlay ${this.num}`);
418
+ }
419
+ }
420
+
421
+ render(width: number): string[] {
422
+ const th = this.theme;
423
+ // Use different colors for each overlay to show stacking
424
+ const colors = ["error", "success", "accent"] as const;
425
+ const color = colors[(this.num - 1) % colors.length]!;
426
+ const innerW = Math.max(1, width - 2);
427
+ const border = (char: string) => th.fg(color, char);
428
+ const padLine = (s: string) => truncateToWidth(s, innerW, "...", true);
429
+ const lines: string[] = [];
430
+
431
+ lines.push(border(`╭${"─".repeat(innerW)}╮`));
432
+ lines.push(border("│") + padLine(` Overlay ${th.fg("accent", `#${this.num}`)}`) + border("│"));
433
+ lines.push(border("│") + padLine(` Layer: ${th.fg(color, this.position)}`) + border("│"));
434
+ lines.push(border("│") + padLine("") + border("│"));
435
+ // Add extra lines to make it taller
436
+ for (let i = 0; i < 5; i++) {
437
+ lines.push(border("│") + padLine(` ${"░".repeat(innerW - 2)} `) + border("│"));
438
+ }
439
+ lines.push(border("│") + padLine("") + border("│"));
440
+ lines.push(border("│") + padLine(th.fg("dim", " Press Enter/Esc to close")) + border("│"));
441
+ lines.push(border(`╰${"─".repeat(innerW)}╯`));
442
+
443
+ return lines;
444
+ }
445
+ }
446
+
447
+ // Streaming overflow test - spawns real process with colored output (original crash scenario)
448
+ class StreamingOverflowComponent extends BaseOverlay {
449
+ private lines: string[] = [];
450
+ private proc: ReturnType<typeof spawn> | null = null;
451
+ private scrollOffset = 0;
452
+ private maxVisibleLines = 15;
453
+ private finished = false;
454
+ private disposed = false;
455
+
456
+ constructor(
457
+ private tui: TUI,
458
+ theme: Theme,
459
+ private done: () => void,
460
+ ) {
461
+ super(theme);
462
+ this.startProcess();
463
+ }
464
+
465
+ private startProcess(): void {
466
+ // Run a command that produces many lines with ANSI colors
467
+ // Using find with -ls produces file listings, or use ls --color
468
+ this.proc = spawn("bash", [
469
+ "-c",
470
+ `
471
+ echo "Starting streaming overflow test (30+ seconds)..."
472
+ echo "This simulates subagent output with colors, hyperlinks, and long paths"
473
+ echo ""
474
+ for i in $(seq 1 100); do
475
+ # Simulate long file paths with OSC 8 hyperlinks (clickable) - tests width overflow
476
+ DIR="/Users/nicobailon/Documents/development/pi-mono/packages/coding-agent/src/modes/interactive"
477
+ FILE="\${DIR}/components/very-long-component-name-that-exceeds-width-\${i}.ts"
478
+ echo -e "\\033]8;;file://\${FILE}\\007▶ read: \${FILE}\\033]8;;\\007"
479
+
480
+ # Add some colored status messages with long text
481
+ if [ $((i % 5)) -eq 0 ]; then
482
+ echo -e " \\033[32m✓ Successfully processed \${i} files in /Users/nicobailon/Documents/development/pi-mono\\033[0m"
483
+ fi
484
+ if [ $((i % 7)) -eq 0 ]; then
485
+ echo -e " \\033[33m⚠ Warning: potential issue detected at line \${i} in very-long-component-name-that-exceeds-width.ts\\033[0m"
486
+ fi
487
+ if [ $((i % 11)) -eq 0 ]; then
488
+ echo -e " \\033[31m✗ Error: file not found /some/really/long/path/that/definitely/exceeds/the/overlay/width/limit/file-\${i}.ts\\033[0m"
489
+ fi
490
+ sleep 0.3
491
+ done
492
+ echo ""
493
+ echo -e "\\033[32m✓ Complete - 100 files processed in 30 seconds\\033[0m"
494
+ echo "Press Esc to close"
495
+ `,
496
+ ]);
497
+
498
+ this.proc.stdout?.on("data", (data: Buffer) => {
499
+ if (this.disposed) return; // Guard against callbacks after dispose
500
+ const text = data.toString();
501
+ const newLines = text.split("\n");
502
+ for (const line of newLines) {
503
+ if (line) this.lines.push(line);
504
+ }
505
+ // Auto-scroll to bottom
506
+ this.scrollOffset = Math.max(0, this.lines.length - this.maxVisibleLines);
507
+ this.tui.requestRender();
508
+ });
509
+
510
+ this.proc.stderr?.on("data", (data: Buffer) => {
511
+ if (this.disposed) return; // Guard against callbacks after dispose
512
+ this.lines.push(this.theme.fg("error", data.toString().trim()));
513
+ this.tui.requestRender();
514
+ });
515
+
516
+ this.proc.on("close", () => {
517
+ if (this.disposed) return; // Guard against callbacks after dispose
518
+ this.finished = true;
519
+ this.tui.requestRender();
520
+ });
521
+ }
522
+
523
+ handleInput(data: string): void {
524
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
525
+ this.proc?.kill();
526
+ this.done();
527
+ } else if (matchesKey(data, "up")) {
528
+ this.scrollOffset = Math.max(0, this.scrollOffset - 1);
529
+ this.tui.requestRender(); // Trigger re-render after scroll
530
+ } else if (matchesKey(data, "down")) {
531
+ this.scrollOffset = Math.min(Math.max(0, this.lines.length - this.maxVisibleLines), this.scrollOffset + 1);
532
+ this.tui.requestRender(); // Trigger re-render after scroll
533
+ }
534
+ }
535
+
536
+ render(width: number): string[] {
537
+ const th = this.theme;
538
+ const innerW = Math.max(1, width - 2);
539
+ const padLine = (s: string) => truncateToWidth(s, innerW, "...", true);
540
+ const border = (c: string) => th.fg("border", c);
541
+
542
+ const result: string[] = [];
543
+ const title = truncateToWidth(` Streaming Output (${this.lines.length} lines) `, innerW);
544
+ const titlePad = Math.max(0, innerW - visibleWidth(title));
545
+ result.push(border("╭") + th.fg("accent", title) + border(`${"─".repeat(titlePad)}╮`));
546
+
547
+ // Scroll indicators
548
+ const canScrollUp = this.scrollOffset > 0;
549
+ const canScrollDown = this.scrollOffset < this.lines.length - this.maxVisibleLines;
550
+ const scrollInfo = `↑${this.scrollOffset} | ↓${Math.max(0, this.lines.length - this.maxVisibleLines - this.scrollOffset)}`;
551
+
552
+ result.push(
553
+ border("│") + padLine(canScrollUp || canScrollDown ? th.fg("dim", ` ${scrollInfo}`) : "") + border("│"),
554
+ );
555
+
556
+ // Visible lines - truncate long lines to fit within border
557
+ const visibleLines = this.lines.slice(this.scrollOffset, this.scrollOffset + this.maxVisibleLines);
558
+ for (const line of visibleLines) {
559
+ result.push(border("│") + padLine(` ${line}`) + border("│"));
560
+ }
561
+
562
+ // Pad to maxVisibleLines
563
+ for (let i = visibleLines.length; i < this.maxVisibleLines; i++) {
564
+ result.push(border("│") + padLine("") + border("│"));
565
+ }
566
+
567
+ const status = this.finished ? th.fg("success", "✓ Done") : th.fg("warning", "● Running");
568
+ result.push(border("│") + padLine(` ${status} ${th.fg("dim", "| ↑↓ scroll | Esc close")}`) + border("│"));
569
+ result.push(border(`╰${"─".repeat(innerW)}╯`));
570
+
571
+ return result;
572
+ }
573
+
574
+ dispose(): void {
575
+ this.disposed = true;
576
+ this.proc?.kill();
577
+ }
578
+ }
579
+
580
+ // Edge position test
581
+ class EdgeTestComponent extends BaseOverlay {
582
+ constructor(
583
+ theme: Theme,
584
+ private done: () => void,
585
+ ) {
586
+ super(theme);
587
+ }
588
+
589
+ handleInput(data: string): void {
590
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
591
+ this.done();
592
+ }
593
+ }
594
+
595
+ render(width: number): string[] {
596
+ const th = this.theme;
597
+ return this.box(
598
+ [
599
+ "",
600
+ " This overlay is at the",
601
+ " right edge of terminal.",
602
+ "",
603
+ ` ${th.fg("dim", "Verify right border")}`,
604
+ ` ${th.fg("dim", "aligns with edge.")}`,
605
+ "",
606
+ ` ${th.fg("dim", "Press Esc to close")}`,
607
+ "",
608
+ ],
609
+ width,
610
+ "Edge Test",
611
+ );
612
+ }
613
+ }
614
+
615
+ // Percentage positioning test
616
+ class PercentTestComponent extends BaseOverlay {
617
+ constructor(
618
+ theme: Theme,
619
+ private config: { name: string; row: number; col: number },
620
+ private done: (result: "next" | "close") => void,
621
+ ) {
622
+ super(theme);
623
+ }
624
+
625
+ handleInput(data: string): void {
626
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
627
+ this.done("close");
628
+ } else if (matchesKey(data, "space") || matchesKey(data, "right")) {
629
+ this.done("next");
630
+ }
631
+ }
632
+
633
+ render(width: number): string[] {
634
+ const th = this.theme;
635
+ return this.box(
636
+ [
637
+ "",
638
+ ` ${th.fg("accent", this.config.name)}`,
639
+ "",
640
+ ` ${th.fg("dim", "Space/→ = next")}`,
641
+ ` ${th.fg("dim", "Esc = close")}`,
642
+ "",
643
+ ],
644
+ width,
645
+ "Percent Test",
646
+ );
647
+ }
648
+ }
649
+
650
+ // MaxHeight test - renders 20 lines, truncated to 10 by maxHeight
651
+ class MaxHeightTestComponent extends BaseOverlay {
652
+ constructor(
653
+ theme: Theme,
654
+ private done: () => void,
655
+ ) {
656
+ super(theme);
657
+ }
658
+
659
+ handleInput(data: string): void {
660
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
661
+ this.done();
662
+ }
663
+ }
664
+
665
+ render(width: number): string[] {
666
+ const th = this.theme;
667
+ // Intentionally render 21 lines - maxHeight: 10 will truncate to first 10
668
+ // You should see header + lines 1-6, with bottom border cut off
669
+ const contentLines: string[] = [
670
+ th.fg("warning", " ⚠ Rendering 21 lines, maxHeight: 10"),
671
+ th.fg("dim", " Lines 11-21 truncated (no bottom border)"),
672
+ "",
673
+ ];
674
+
675
+ for (let i = 1; i <= 14; i++) {
676
+ contentLines.push(` Line ${i} of 14`);
677
+ }
678
+
679
+ contentLines.push("", th.fg("dim", " Press Esc to close"));
680
+
681
+ return this.box(contentLines, width, "MaxHeight Test");
682
+ }
683
+ }
684
+
685
+ // Responsive sidepanel - demonstrates percentage width and visibility callback
686
+ class SidepanelComponent extends BaseOverlay {
687
+ private items = ["Dashboard", "Messages", "Settings", "Help", "About"];
688
+ private selectedIndex = 0;
689
+
690
+ constructor(
691
+ private tui: TUI,
692
+ theme: Theme,
693
+ private done: () => void,
694
+ ) {
695
+ super(theme);
696
+ }
697
+
698
+ handleInput(data: string): void {
699
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
700
+ this.done();
701
+ } else if (matchesKey(data, "up")) {
702
+ this.selectedIndex = Math.max(0, this.selectedIndex - 1);
703
+ this.tui.requestRender();
704
+ } else if (matchesKey(data, "down")) {
705
+ this.selectedIndex = Math.min(this.items.length - 1, this.selectedIndex + 1);
706
+ this.tui.requestRender();
707
+ } else if (matchesKey(data, "return")) {
708
+ // Could trigger an action here
709
+ this.tui.requestRender();
710
+ }
711
+ }
712
+
713
+ render(width: number): string[] {
714
+ const th = this.theme;
715
+ const innerW = Math.max(1, width - 2);
716
+ const padLine = (s: string) => truncateToWidth(s, innerW, "...", true);
717
+ const border = (c: string) => th.fg("border", c);
718
+ const lines: string[] = [];
719
+
720
+ // Header
721
+ lines.push(border(`╭${"─".repeat(innerW)}╮`));
722
+ lines.push(border("│") + padLine(th.fg("accent", " Responsive Sidepanel")) + border("│"));
723
+ lines.push(border("├") + border("─".repeat(innerW)) + border("┤"));
724
+
725
+ // Menu items
726
+ for (let i = 0; i < this.items.length; i++) {
727
+ const item = this.items[i]!;
728
+ const isSelected = i === this.selectedIndex;
729
+ const prefix = isSelected ? th.fg("accent", "→ ") : " ";
730
+ const text = isSelected ? th.fg("accent", item) : item;
731
+ lines.push(border("│") + padLine(`${prefix}${text}`) + border("│"));
732
+ }
733
+
734
+ // Footer with responsive behavior info
735
+ lines.push(border("├") + border("─".repeat(innerW)) + border("┤"));
736
+ lines.push(border("│") + padLine(th.fg("warning", " ⚠ Resize terminal < 100 cols")) + border("│"));
737
+ lines.push(border("│") + padLine(th.fg("warning", " to see panel auto-hide")) + border("│"));
738
+ lines.push(border("│") + padLine(th.fg("dim", " Uses visible: (w) => w >= 100")) + border("│"));
739
+ lines.push(border("│") + padLine(th.fg("dim", " ↑↓ navigate | Esc close")) + border("│"));
740
+ lines.push(border(`╰${"─".repeat(innerW)}╯`));
741
+
742
+ return lines;
743
+ }
744
+ }
745
+
746
+ // Animation demo - proves overlays can handle real-time updates like pi-doom
747
+ class AnimationDemoComponent extends BaseOverlay {
748
+ private frame = 0;
749
+ private interval: ReturnType<typeof setInterval> | null = null;
750
+ private fps = 0;
751
+ private lastFpsUpdate = Date.now();
752
+ private framesSinceLastFps = 0;
753
+
754
+ constructor(
755
+ private tui: TUI,
756
+ theme: Theme,
757
+ private done: () => void,
758
+ ) {
759
+ super(theme);
760
+ this.startAnimation();
761
+ }
762
+
763
+ private startAnimation(): void {
764
+ // Run at ~30 FPS (same as DOOM target)
765
+ this.interval = setInterval(() => {
766
+ this.frame++;
767
+ this.framesSinceLastFps++;
768
+
769
+ // Update FPS counter every second
770
+ const now = Date.now();
771
+ if (now - this.lastFpsUpdate >= 1000) {
772
+ this.fps = this.framesSinceLastFps;
773
+ this.framesSinceLastFps = 0;
774
+ this.lastFpsUpdate = now;
775
+ }
776
+
777
+ this.tui.requestRender();
778
+ }, 1000 / 30);
779
+ }
780
+
781
+ handleInput(data: string): void {
782
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
783
+ this.dispose();
784
+ this.done();
785
+ }
786
+ }
787
+
788
+ render(width: number): string[] {
789
+ const th = this.theme;
790
+ const innerW = Math.max(1, width - 2);
791
+ const padLine = (s: string) => truncateToWidth(s, innerW, "...", true);
792
+ const border = (c: string) => th.fg("border", c);
793
+
794
+ const lines: string[] = [];
795
+ lines.push(border(`╭${"─".repeat(innerW)}╮`));
796
+ lines.push(border("│") + padLine(th.fg("accent", " Animation Demo (~30 FPS)")) + border("│"));
797
+ lines.push(border("│") + padLine(``) + border("│"));
798
+ lines.push(border("│") + padLine(` Frame: ${th.fg("accent", String(this.frame))}`) + border("│"));
799
+ lines.push(border("│") + padLine(` FPS: ${th.fg("success", String(this.fps))}`) + border("│"));
800
+ lines.push(border("│") + padLine(``) + border("│"));
801
+
802
+ // Animated content - bouncing bar
803
+ const barWidth = Math.max(12, innerW - 4); // Ensure enough space for bar
804
+ const pos = Math.max(0, Math.floor(((Math.sin(this.frame / 10) + 1) * (barWidth - 10)) / 2));
805
+ const bar = " ".repeat(pos) + th.fg("accent", "██████████") + " ".repeat(Math.max(0, barWidth - 10 - pos));
806
+ lines.push(border("│") + padLine(` ${bar}`) + border("│"));
807
+
808
+ // Spinning character
809
+ const spinChars = ["◐", "◓", "◑", "◒"];
810
+ const spin = spinChars[this.frame % spinChars.length];
811
+ lines.push(border("│") + padLine(` Spinner: ${th.fg("warning", spin!)}`) + border("│"));
812
+
813
+ // Color cycling
814
+ const hue = (this.frame * 3) % 360;
815
+ const rgb = hslToRgb(hue / 360, 0.8, 0.5);
816
+ const colorBlock = `\x1b[48;2;${rgb[0]};${rgb[1]};${rgb[2]}m${" ".repeat(10)}\x1b[0m`;
817
+ lines.push(border("│") + padLine(` Color: ${colorBlock}`) + border("│"));
818
+
819
+ lines.push(border("│") + padLine(``) + border("│"));
820
+ lines.push(border("│") + padLine(th.fg("dim", " This proves overlays can handle")) + border("│"));
821
+ lines.push(border("│") + padLine(th.fg("dim", " real-time game-like rendering.")) + border("│"));
822
+ lines.push(border("│") + padLine(th.fg("dim", " (pi-doom uses same approach)")) + border("│"));
823
+ lines.push(border("│") + padLine(``) + border("│"));
824
+ lines.push(border("│") + padLine(th.fg("dim", " Press Esc to close")) + border("│"));
825
+ lines.push(border(`╰${"─".repeat(innerW)}╯`));
826
+
827
+ return lines;
828
+ }
829
+
830
+ dispose(): void {
831
+ if (this.interval) {
832
+ clearInterval(this.interval);
833
+ this.interval = null;
834
+ }
835
+ }
836
+ }
837
+
838
+ // HSL to RGB helper for color cycling animation
839
+ function hslToRgb(h: number, s: number, l: number): [number, number, number] {
840
+ let r: number, g: number, b: number;
841
+ if (s === 0) {
842
+ r = g = b = l;
843
+ } else {
844
+ const hue2rgb = (p: number, q: number, t: number) => {
845
+ if (t < 0) t += 1;
846
+ if (t > 1) t -= 1;
847
+ if (t < 1 / 6) return p + (q - p) * 6 * t;
848
+ if (t < 1 / 2) return q;
849
+ if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
850
+ return p;
851
+ };
852
+ const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
853
+ const p = 2 * l - q;
854
+ r = hue2rgb(p, q, h + 1 / 3);
855
+ g = hue2rgb(p, q, h);
856
+ b = hue2rgb(p, q, h - 1 / 3);
857
+ }
858
+ return [Math.round(r * 255), Math.round(g * 255), Math.round(b * 255)];
859
+ }
860
+
861
+ // Toggle demo - demonstrates OverlayHandle.setHidden() via onHandle callback
862
+ class ToggleDemoComponent extends BaseOverlay {
863
+ private toggleCount = 0;
864
+ private isToggling = false;
865
+
866
+ constructor(
867
+ private tui: TUI,
868
+ theme: Theme,
869
+ private done: () => void,
870
+ ) {
871
+ super(theme);
872
+ }
873
+
874
+ handleInput(data: string): void {
875
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
876
+ this.done();
877
+ } else if (matchesKey(data, "t") && globalToggleHandle && !this.isToggling) {
878
+ // Demonstrate toggle by hiding for 1 second then showing again
879
+ // (In real usage, a global keybinding would control visibility)
880
+ this.isToggling = true;
881
+ this.toggleCount++;
882
+ globalToggleHandle.setHidden(true);
883
+
884
+ // Auto-restore after 1 second to demonstrate the API
885
+ setTimeout(() => {
886
+ if (globalToggleHandle) {
887
+ globalToggleHandle.setHidden(false);
888
+ this.isToggling = false;
889
+ this.tui.requestRender();
890
+ }
891
+ }, 1000);
892
+ }
893
+ }
894
+
895
+ render(width: number): string[] {
896
+ const th = this.theme;
897
+ return this.box(
898
+ [
899
+ "",
900
+ th.fg("accent", " Toggle Demo"),
901
+ "",
902
+ " This overlay demonstrates the",
903
+ " onHandle callback API.",
904
+ "",
905
+ ` Toggle count: ${th.fg("accent", String(this.toggleCount))}`,
906
+ "",
907
+ th.fg("dim", " Press 't' to hide for 1 second"),
908
+ th.fg("dim", " (demonstrates setHidden API)"),
909
+ "",
910
+ th.fg("dim", " In real usage, a global keybinding"),
911
+ th.fg("dim", " would toggle visibility externally."),
912
+ "",
913
+ th.fg("dim", " Press Esc to close"),
914
+ "",
915
+ ],
916
+ width,
917
+ "Toggle Demo",
918
+ );
919
+ }
920
+ }
921
+
922
+ // === Non-capturing passive overlay demo ===
923
+
924
+ class PassiveDemoController extends BaseOverlay {
925
+ focused = false;
926
+ private typed = "";
927
+ private timerComponent: TimerPanel;
928
+ private timerHandle: OverlayHandle | null = null;
929
+ private interval: ReturnType<typeof setInterval> | null = null;
930
+ private inputCount = 0;
931
+ private lastInputDebug = "";
932
+
933
+ constructor(
934
+ private tui: TUI,
935
+ theme: Theme,
936
+ private done: () => void,
937
+ ) {
938
+ super(theme);
939
+ this.timerComponent = new TimerPanel(theme);
940
+ this.timerHandle = this.tui.showOverlay(this.timerComponent, {
941
+ nonCapturing: true,
942
+ anchor: "top-right",
943
+ width: 22,
944
+ margin: { top: 1, right: 2 },
945
+ });
946
+ this.interval = setInterval(() => {
947
+ this.timerComponent.tick();
948
+ this.tui.requestRender();
949
+ }, 1000);
950
+ }
951
+
952
+ handleInput(data: string): void {
953
+ this.inputCount++;
954
+ this.lastInputDebug = `len=${data.length} c0=${data.charCodeAt(0)}`;
955
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
956
+ this.cleanup();
957
+ this.done();
958
+ } else if (matchesKey(data, "backspace")) {
959
+ this.typed = this.typed.slice(0, -1);
960
+ } else if (data.length === 1 && data.charCodeAt(0) >= 32) {
961
+ this.typed += data;
962
+ }
963
+ }
964
+
965
+ render(width: number): string[] {
966
+ const th = this.theme;
967
+ const display = this.typed.length > 0 ? this.typed : th.fg("dim", "(type here)");
968
+ return this.box(
969
+ [
970
+ "",
971
+ ` ${th.fg("dim", `focused=${this.focused} inputs=${this.inputCount}`)}`,
972
+ ` ${th.fg("dim", `last: ${this.lastInputDebug || "none"}`)}`,
973
+ "",
974
+ ` > ${display}`,
975
+ "",
976
+ th.fg("dim", " Type to prove input goes here."),
977
+ th.fg("dim", " Press Esc to close both."),
978
+ "",
979
+ ],
980
+ width,
981
+ "Non-Capturing Demo",
982
+ );
983
+ }
984
+
985
+ private cleanup(): void {
986
+ if (this.interval) {
987
+ clearInterval(this.interval);
988
+ this.interval = null;
989
+ }
990
+ this.timerHandle?.hide();
991
+ this.timerHandle = null;
992
+ }
993
+
994
+ override dispose(): void {
995
+ this.cleanup();
996
+ }
997
+ }
998
+
999
+ class TimerPanel extends BaseOverlay {
1000
+ private seconds = 0;
1001
+
1002
+ tick(): void {
1003
+ this.seconds++;
1004
+ }
1005
+
1006
+ render(width: number): string[] {
1007
+ const th = this.theme;
1008
+ const mins = Math.floor(this.seconds / 60);
1009
+ const secs = this.seconds % 60;
1010
+ const time = `${String(mins).padStart(2, "0")}:${String(secs).padStart(2, "0")}`;
1011
+ return this.box([` ${th.fg("accent", time)}`, th.fg("dim", " nonCapturing: true")], width, "Timer");
1012
+ }
1013
+ }
1014
+
1015
+ // === Focus cycling demo ===
1016
+
1017
+ class FocusDemoController extends BaseOverlay {
1018
+ private panels: FocusPanel[] = [];
1019
+ private handles: OverlayHandle[] = [];
1020
+ private focusIndex = -1;
1021
+
1022
+ constructor(
1023
+ private tui: TUI,
1024
+ theme: Theme,
1025
+ private done: () => void,
1026
+ ) {
1027
+ super(theme);
1028
+ const colors = ["error", "success", "accent"] as const;
1029
+ const labels = ["Alpha", "Beta", "Gamma"];
1030
+
1031
+ for (let i = 0; i < 3; i++) {
1032
+ const panel = new FocusPanel(
1033
+ theme,
1034
+ labels[i]!,
1035
+ colors[i]!,
1036
+ () => this.cycleFocus(),
1037
+ () => this.close(),
1038
+ );
1039
+ const handle = this.tui.showOverlay(panel, {
1040
+ nonCapturing: true,
1041
+ row: 2,
1042
+ col: 5 + i * 6,
1043
+ width: 28,
1044
+ });
1045
+ panel.handle = handle;
1046
+ this.panels.push(panel);
1047
+ this.handles.push(handle);
1048
+ }
1049
+ }
1050
+
1051
+ private cycleFocus(): void {
1052
+ if (this.focusIndex >= 0 && this.focusIndex < this.handles.length) {
1053
+ this.handles[this.focusIndex]!.unfocus();
1054
+ }
1055
+ this.focusIndex++;
1056
+ if (this.focusIndex >= this.handles.length) {
1057
+ this.focusIndex = -1;
1058
+ } else {
1059
+ this.handles[this.focusIndex]!.focus();
1060
+ }
1061
+ this.tui.requestRender();
1062
+ }
1063
+
1064
+ private close(): void {
1065
+ for (const handle of this.handles) handle.hide();
1066
+ this.handles = [];
1067
+ this.panels = [];
1068
+ this.done();
1069
+ }
1070
+
1071
+ handleInput(data: string): void {
1072
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
1073
+ this.close();
1074
+ } else if (matchesKey(data, "tab")) {
1075
+ this.cycleFocus();
1076
+ }
1077
+ }
1078
+
1079
+ render(width: number): string[] {
1080
+ const th = this.theme;
1081
+ const focused = this.focusIndex === -1 ? "Controller" : (this.panels[this.focusIndex]?.label ?? "?");
1082
+ return this.box(
1083
+ [
1084
+ "",
1085
+ ` Current focus: ${th.fg("accent", focused)}`,
1086
+ "",
1087
+ " Three overlapping panels above are",
1088
+ ` all ${th.fg("accent", "nonCapturing")}. Press Tab to`,
1089
+ " cycle focus() between them.",
1090
+ "",
1091
+ " Focused panel renders on top",
1092
+ " (focus-based rendering order).",
1093
+ "",
1094
+ th.fg("dim", " Tab = cycle focus | Esc = close"),
1095
+ "",
1096
+ ],
1097
+ width,
1098
+ "Focus Demo",
1099
+ );
1100
+ }
1101
+
1102
+ override dispose(): void {
1103
+ for (const handle of this.handles) handle.hide();
1104
+ }
1105
+ }
1106
+
1107
+ class FocusPanel extends BaseOverlay {
1108
+ handle: OverlayHandle | null = null;
1109
+ readonly label: string;
1110
+
1111
+ constructor(
1112
+ theme: Theme,
1113
+ label: string,
1114
+ private color: "error" | "success" | "accent",
1115
+ private onTab: () => void,
1116
+ private onClose: () => void,
1117
+ ) {
1118
+ super(theme);
1119
+ this.label = label;
1120
+ }
1121
+
1122
+ handleInput(data: string): void {
1123
+ if (matchesKey(data, "tab")) {
1124
+ this.onTab();
1125
+ } else if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
1126
+ this.onClose();
1127
+ }
1128
+ }
1129
+
1130
+ render(width: number): string[] {
1131
+ const th = this.theme;
1132
+ const focused = this.handle?.isFocused() ?? false;
1133
+ const innerW = Math.max(1, width - 2);
1134
+ const border = (c: string) => th.fg(this.color, c);
1135
+ const padLine = (s: string) => truncateToWidth(s, innerW, "...", true);
1136
+ const lines: string[] = [];
1137
+
1138
+ lines.push(border(`╭${"─".repeat(innerW)}╮`));
1139
+ lines.push(border("│") + padLine(` ${th.fg("accent", this.label)}`) + border("│"));
1140
+ lines.push(border("│") + padLine("") + border("│"));
1141
+ if (focused) {
1142
+ lines.push(border("│") + padLine(th.fg("success", " ● FOCUSED")) + border("│"));
1143
+ lines.push(border("│") + padLine(th.fg("dim", " (receiving input)")) + border("│"));
1144
+ } else {
1145
+ lines.push(border("│") + padLine(th.fg("dim", " ○ unfocused")) + border("│"));
1146
+ lines.push(border("│") + padLine(th.fg("dim", " (passive)")) + border("│"));
1147
+ }
1148
+ lines.push(border("│") + padLine("") + border("│"));
1149
+ lines.push(border(`╰${"─".repeat(innerW)}╯`));
1150
+
1151
+ return lines;
1152
+ }
1153
+ }
1154
+
1155
+ // === Streaming input panel test (/overlay-streaming) ===
1156
+
1157
+ class StreamingInputController extends BaseOverlay {
1158
+ private panels: StreamingInputPanel[] = [];
1159
+ private handles: OverlayHandle[] = [];
1160
+ private focusIndex = -1; // -1 = controller focused, 0-2 = panel focused
1161
+ private streamLines: string[] = [];
1162
+ private streamInterval: ReturnType<typeof setInterval> | null = null;
1163
+ private lineCount = 0;
1164
+
1165
+ constructor(
1166
+ private tui: TUI,
1167
+ theme: Theme,
1168
+ private done: () => void,
1169
+ ) {
1170
+ super(theme);
1171
+
1172
+ // Create 3 input panels as non-capturing overlays
1173
+ const colors = ["error", "success", "accent"] as const;
1174
+ const labels = ["Panel A", "Panel B", "Panel C"];
1175
+
1176
+ for (let i = 0; i < 3; i++) {
1177
+ const panel = new StreamingInputPanel(
1178
+ theme,
1179
+ labels[i]!,
1180
+ colors[i]!,
1181
+ () => this.cycleFocus(),
1182
+ () => this.close(),
1183
+ );
1184
+ const handle = this.tui.showOverlay(panel, {
1185
+ nonCapturing: true,
1186
+ row: 1 + i * 9,
1187
+ col: 2,
1188
+ width: 35,
1189
+ });
1190
+ panel.handle = handle;
1191
+ this.panels.push(panel);
1192
+ this.handles.push(handle);
1193
+ }
1194
+
1195
+ // Start with controller focused (focusIndex = -1)
1196
+
1197
+ // Start simulated streaming
1198
+ this.streamInterval = setInterval(() => {
1199
+ this.lineCount++;
1200
+ const timestamp = new Date().toLocaleTimeString();
1201
+ this.streamLines.push(`[${timestamp}] Streaming line ${this.lineCount}...`);
1202
+ if (this.streamLines.length > 8) {
1203
+ this.streamLines.shift();
1204
+ }
1205
+ this.tui.requestRender();
1206
+ }, 500);
1207
+ }
1208
+
1209
+ private cycleFocus(): void {
1210
+ // Unfocus current panel if any
1211
+ if (this.focusIndex >= 0 && this.focusIndex < this.handles.length) {
1212
+ this.handles[this.focusIndex]!.unfocus();
1213
+ }
1214
+
1215
+ // Cycle: -1 (controller) → 0 → 1 → 2 → -1 ...
1216
+ this.focusIndex++;
1217
+ if (this.focusIndex >= this.handles.length) {
1218
+ this.focusIndex = -1; // Back to controller
1219
+ }
1220
+
1221
+ // Focus new panel if any
1222
+ if (this.focusIndex >= 0) {
1223
+ this.handles[this.focusIndex]!.focus();
1224
+ }
1225
+
1226
+ this.tui.requestRender();
1227
+ }
1228
+
1229
+ private close(): void {
1230
+ if (this.streamInterval) {
1231
+ clearInterval(this.streamInterval);
1232
+ this.streamInterval = null;
1233
+ }
1234
+ for (const handle of this.handles) handle.hide();
1235
+ this.handles = [];
1236
+ this.panels = [];
1237
+ this.done();
1238
+ }
1239
+
1240
+ handleInput(data: string): void {
1241
+ if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
1242
+ this.close();
1243
+ } else if (matchesKey(data, "tab")) {
1244
+ this.cycleFocus();
1245
+ }
1246
+ }
1247
+
1248
+ render(width: number): string[] {
1249
+ const th = this.theme;
1250
+ const focusedLabel =
1251
+ this.focusIndex === -1
1252
+ ? th.fg("success", "Controller (this panel)")
1253
+ : (this.panels[this.focusIndex]?.label ?? "?");
1254
+
1255
+ const lines = [
1256
+ "",
1257
+ ` Current focus: ${th.fg("accent", focusedLabel)}`,
1258
+ "",
1259
+ " Simulated streaming output:",
1260
+ th.fg("dim", " ─".repeat((width - 2) / 2)),
1261
+ ];
1262
+
1263
+ for (const line of this.streamLines) {
1264
+ lines.push(` ${th.fg("dim", line)}`);
1265
+ }
1266
+
1267
+ while (lines.length < 12) {
1268
+ lines.push("");
1269
+ }
1270
+
1271
+ lines.push(th.fg("dim", " ─".repeat((width - 2) / 2)));
1272
+ lines.push("");
1273
+ lines.push(` Three ${th.fg("accent", "nonCapturing")} input panels on the left.`);
1274
+ lines.push(" Tab cycles: Controller → Panel A → B → C → Controller");
1275
+ lines.push(" Type in each panel to test input routing.");
1276
+ lines.push("");
1277
+ lines.push(th.fg("dim", " Tab = cycle focus | Esc = close all"));
1278
+ lines.push("");
1279
+
1280
+ return this.box(lines, width, "Streaming + Input Test");
1281
+ }
1282
+
1283
+ override dispose(): void {
1284
+ this.close();
1285
+ }
1286
+ }
1287
+
1288
+ class StreamingInputPanel implements Component {
1289
+ handle: OverlayHandle | null = null;
1290
+ private typed = "";
1291
+ readonly label: string;
1292
+
1293
+ constructor(
1294
+ private theme: Theme,
1295
+ label: string,
1296
+ private color: "error" | "success" | "accent",
1297
+ private onTab: () => void,
1298
+ private onClose: () => void,
1299
+ ) {
1300
+ this.label = label;
1301
+ }
1302
+
1303
+ handleInput(data: string): void {
1304
+ if (matchesKey(data, "tab")) {
1305
+ this.onTab();
1306
+ } else if (matchesKey(data, "escape") || matchesKey(data, "ctrl+c")) {
1307
+ this.onClose();
1308
+ } else if (matchesKey(data, "backspace")) {
1309
+ this.typed = this.typed.slice(0, -1);
1310
+ } else if (data.length === 1 && data.charCodeAt(0) >= 32) {
1311
+ this.typed += data;
1312
+ }
1313
+ }
1314
+
1315
+ render(width: number): string[] {
1316
+ const th = this.theme;
1317
+ const focused = this.handle?.isFocused() ?? false;
1318
+ const innerW = Math.max(1, width - 2);
1319
+ const border = (c: string) => th.fg(this.color, c);
1320
+ const padLine = (s: string) => {
1321
+ const w = visibleWidth(s);
1322
+ return s + " ".repeat(Math.max(0, innerW - w));
1323
+ };
1324
+
1325
+ const inputDisplay = this.typed.length > 0 ? this.typed : th.fg("dim", "(type here)");
1326
+ const truncatedInput = truncateToWidth(` > ${inputDisplay}`, innerW, "...", true);
1327
+
1328
+ const lines: string[] = [];
1329
+ lines.push(border(`╭${"─".repeat(innerW)}╮`));
1330
+ lines.push(border("│") + padLine(` ${th.fg("accent", this.label)}`) + border("│"));
1331
+ lines.push(border("│") + padLine("") + border("│"));
1332
+ if (focused) {
1333
+ lines.push(border("│") + padLine(th.fg("success", " ● FOCUSED")) + border("│"));
1334
+ lines.push(border("│") + padLine(th.fg("dim", " (receiving input)")) + border("│"));
1335
+ } else {
1336
+ lines.push(border("│") + padLine(th.fg("dim", " ○ unfocused")) + border("│"));
1337
+ lines.push(border("│") + padLine("") + border("│"));
1338
+ }
1339
+ lines.push(border("│") + padLine(truncatedInput) + border("│"));
1340
+ lines.push(border("│") + padLine("") + border("│"));
1341
+ lines.push(border("│") + padLine(th.fg("dim", " Tab | Esc")) + border("│"));
1342
+ lines.push(border(`╰${"─".repeat(innerW)}╯`));
1343
+
1344
+ return lines;
1345
+ }
1346
+
1347
+ invalidate(): void {}
1348
+ }