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.
- package/CHANGELOG.md +3045 -0
- package/README.md +571 -0
- package/dist/cli/args.d.ts +48 -0
- package/dist/cli/args.d.ts.map +1 -0
- package/dist/cli/args.js +299 -0
- package/dist/cli/args.js.map +1 -0
- package/dist/cli/config-selector.d.ts +14 -0
- package/dist/cli/config-selector.d.ts.map +1 -0
- package/dist/cli/config-selector.js +31 -0
- package/dist/cli/config-selector.js.map +1 -0
- package/dist/cli/file-processor.d.ts +15 -0
- package/dist/cli/file-processor.d.ts.map +1 -0
- package/dist/cli/file-processor.js +79 -0
- package/dist/cli/file-processor.js.map +1 -0
- package/dist/cli/list-models.d.ts +9 -0
- package/dist/cli/list-models.d.ts.map +1 -0
- package/dist/cli/list-models.js +92 -0
- package/dist/cli/list-models.js.map +1 -0
- package/dist/cli/session-picker.d.ts +9 -0
- package/dist/cli/session-picker.d.ts.map +1 -0
- package/dist/cli/session-picker.js +34 -0
- package/dist/cli/session-picker.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +16 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +68 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +203 -0
- package/dist/config.js.map +1 -0
- package/dist/core/agent-session.d.ts +582 -0
- package/dist/core/agent-session.d.ts.map +1 -0
- package/dist/core/agent-session.js +2494 -0
- package/dist/core/agent-session.js.map +1 -0
- package/dist/core/auth-storage.d.ts +130 -0
- package/dist/core/auth-storage.d.ts.map +1 -0
- package/dist/core/auth-storage.js +419 -0
- package/dist/core/auth-storage.js.map +1 -0
- package/dist/core/bash-executor.d.ts +47 -0
- package/dist/core/bash-executor.d.ts.map +1 -0
- package/dist/core/bash-executor.js +212 -0
- package/dist/core/bash-executor.js.map +1 -0
- package/dist/core/compaction/branch-summarization.d.ts +86 -0
- package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
- package/dist/core/compaction/branch-summarization.js +242 -0
- package/dist/core/compaction/branch-summarization.js.map +1 -0
- package/dist/core/compaction/compaction.d.ts +121 -0
- package/dist/core/compaction/compaction.d.ts.map +1 -0
- package/dist/core/compaction/compaction.js +610 -0
- package/dist/core/compaction/compaction.js.map +1 -0
- package/dist/core/compaction/index.d.ts +7 -0
- package/dist/core/compaction/index.d.ts.map +1 -0
- package/dist/core/compaction/index.js +7 -0
- package/dist/core/compaction/index.js.map +1 -0
- package/dist/core/compaction/utils.d.ts +38 -0
- package/dist/core/compaction/utils.d.ts.map +1 -0
- package/dist/core/compaction/utils.js +153 -0
- package/dist/core/compaction/utils.js.map +1 -0
- package/dist/core/defaults.d.ts +3 -0
- package/dist/core/defaults.d.ts.map +1 -0
- package/dist/core/defaults.js +2 -0
- package/dist/core/defaults.js.map +1 -0
- package/dist/core/diagnostics.d.ts +15 -0
- package/dist/core/diagnostics.d.ts.map +1 -0
- package/dist/core/diagnostics.js +2 -0
- package/dist/core/diagnostics.js.map +1 -0
- package/dist/core/event-bus.d.ts +9 -0
- package/dist/core/event-bus.d.ts.map +1 -0
- package/dist/core/event-bus.js +25 -0
- package/dist/core/event-bus.js.map +1 -0
- package/dist/core/exec.d.ts +29 -0
- package/dist/core/exec.d.ts.map +1 -0
- package/dist/core/exec.js +71 -0
- package/dist/core/exec.js.map +1 -0
- package/dist/core/export-html/ansi-to-html.d.ts +22 -0
- package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
- package/dist/core/export-html/ansi-to-html.js +249 -0
- package/dist/core/export-html/ansi-to-html.js.map +1 -0
- package/dist/core/export-html/index.d.ts +37 -0
- package/dist/core/export-html/index.d.ts.map +1 -0
- package/dist/core/export-html/index.js +223 -0
- package/dist/core/export-html/index.js.map +1 -0
- package/dist/core/export-html/template.css +971 -0
- package/dist/core/export-html/template.html +54 -0
- package/dist/core/export-html/template.js +1583 -0
- package/dist/core/export-html/tool-renderer.d.ts +38 -0
- package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
- package/dist/core/export-html/tool-renderer.js +70 -0
- package/dist/core/export-html/tool-renderer.js.map +1 -0
- package/dist/core/export-html/vendor/highlight.min.js +1213 -0
- package/dist/core/export-html/vendor/marked.min.js +6 -0
- package/dist/core/extensions/index.d.ts +11 -0
- package/dist/core/extensions/index.d.ts.map +1 -0
- package/dist/core/extensions/index.js +9 -0
- package/dist/core/extensions/index.js.map +1 -0
- package/dist/core/extensions/loader.d.ts +25 -0
- package/dist/core/extensions/loader.d.ts.map +1 -0
- package/dist/core/extensions/loader.js +426 -0
- package/dist/core/extensions/loader.js.map +1 -0
- package/dist/core/extensions/runner.d.ts +147 -0
- package/dist/core/extensions/runner.d.ts.map +1 -0
- package/dist/core/extensions/runner.js +678 -0
- package/dist/core/extensions/runner.js.map +1 -0
- package/dist/core/extensions/types.d.ts +1036 -0
- package/dist/core/extensions/types.d.ts.map +1 -0
- package/dist/core/extensions/types.js +35 -0
- package/dist/core/extensions/types.js.map +1 -0
- package/dist/core/extensions/wrapper.d.ts +27 -0
- package/dist/core/extensions/wrapper.d.ts.map +1 -0
- package/dist/core/extensions/wrapper.js +102 -0
- package/dist/core/extensions/wrapper.js.map +1 -0
- package/dist/core/footer-data-provider.d.ts +32 -0
- package/dist/core/footer-data-provider.d.ts.map +1 -0
- package/dist/core/footer-data-provider.js +134 -0
- package/dist/core/footer-data-provider.js.map +1 -0
- package/dist/core/index.d.ts +9 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +9 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/keybindings.d.ts +55 -0
- package/dist/core/keybindings.d.ts.map +1 -0
- package/dist/core/keybindings.js +153 -0
- package/dist/core/keybindings.js.map +1 -0
- package/dist/core/messages.d.ts +77 -0
- package/dist/core/messages.d.ts.map +1 -0
- package/dist/core/messages.js +123 -0
- package/dist/core/messages.js.map +1 -0
- package/dist/core/model-registry.d.ts +113 -0
- package/dist/core/model-registry.d.ts.map +1 -0
- package/dist/core/model-registry.js +537 -0
- package/dist/core/model-registry.js.map +1 -0
- package/dist/core/model-resolver.d.ts +104 -0
- package/dist/core/model-resolver.d.ts.map +1 -0
- package/dist/core/model-resolver.js +462 -0
- package/dist/core/model-resolver.js.map +1 -0
- package/dist/core/package-manager.d.ts +151 -0
- package/dist/core/package-manager.d.ts.map +1 -0
- package/dist/core/package-manager.js +1447 -0
- package/dist/core/package-manager.js.map +1 -0
- package/dist/core/prompt-templates.d.ts +50 -0
- package/dist/core/prompt-templates.d.ts.map +1 -0
- package/dist/core/prompt-templates.js +251 -0
- package/dist/core/prompt-templates.js.map +1 -0
- package/dist/core/resolve-config-value.d.ts +17 -0
- package/dist/core/resolve-config-value.d.ts.map +1 -0
- package/dist/core/resolve-config-value.js +59 -0
- package/dist/core/resolve-config-value.js.map +1 -0
- package/dist/core/resource-loader.d.ts +184 -0
- package/dist/core/resource-loader.d.ts.map +1 -0
- package/dist/core/resource-loader.js +670 -0
- package/dist/core/resource-loader.js.map +1 -0
- package/dist/core/sdk.d.ts +90 -0
- package/dist/core/sdk.d.ts.map +1 -0
- package/dist/core/sdk.js +242 -0
- package/dist/core/sdk.js.map +1 -0
- package/dist/core/session-manager.d.ts +323 -0
- package/dist/core/session-manager.d.ts.map +1 -0
- package/dist/core/session-manager.js +1098 -0
- package/dist/core/session-manager.js.map +1 -0
- package/dist/core/settings-manager.d.ts +232 -0
- package/dist/core/settings-manager.d.ts.map +1 -0
- package/dist/core/settings-manager.js +691 -0
- package/dist/core/settings-manager.js.map +1 -0
- package/dist/core/skills.d.ts +58 -0
- package/dist/core/skills.d.ts.map +1 -0
- package/dist/core/skills.js +364 -0
- package/dist/core/skills.js.map +1 -0
- package/dist/core/slash-commands.d.ts +15 -0
- package/dist/core/slash-commands.d.ts.map +1 -0
- package/dist/core/slash-commands.js +22 -0
- package/dist/core/slash-commands.js.map +1 -0
- package/dist/core/system-prompt.d.ts +28 -0
- package/dist/core/system-prompt.d.ts.map +1 -0
- package/dist/core/system-prompt.js +159 -0
- package/dist/core/system-prompt.js.map +1 -0
- package/dist/core/timings.d.ts +7 -0
- package/dist/core/timings.d.ts.map +1 -0
- package/dist/core/timings.js +25 -0
- package/dist/core/timings.js.map +1 -0
- package/dist/core/tools/bash.d.ts +55 -0
- package/dist/core/tools/bash.d.ts.map +1 -0
- package/dist/core/tools/bash.js +242 -0
- package/dist/core/tools/bash.js.map +1 -0
- package/dist/core/tools/edit-diff.d.ts +63 -0
- package/dist/core/tools/edit-diff.d.ts.map +1 -0
- package/dist/core/tools/edit-diff.js +243 -0
- package/dist/core/tools/edit-diff.js.map +1 -0
- package/dist/core/tools/edit.d.ts +39 -0
- package/dist/core/tools/edit.d.ts.map +1 -0
- package/dist/core/tools/edit.js +146 -0
- package/dist/core/tools/edit.js.map +1 -0
- package/dist/core/tools/find.d.ts +39 -0
- package/dist/core/tools/find.d.ts.map +1 -0
- package/dist/core/tools/find.js +206 -0
- package/dist/core/tools/find.js.map +1 -0
- package/dist/core/tools/grep.d.ts +45 -0
- package/dist/core/tools/grep.d.ts.map +1 -0
- package/dist/core/tools/grep.js +239 -0
- package/dist/core/tools/grep.js.map +1 -0
- package/dist/core/tools/index.d.ts +73 -0
- package/dist/core/tools/index.d.ts.map +1 -0
- package/dist/core/tools/index.js +61 -0
- package/dist/core/tools/index.js.map +1 -0
- package/dist/core/tools/ls.d.ts +40 -0
- package/dist/core/tools/ls.d.ts.map +1 -0
- package/dist/core/tools/ls.js +118 -0
- package/dist/core/tools/ls.js.map +1 -0
- package/dist/core/tools/path-utils.d.ts +8 -0
- package/dist/core/tools/path-utils.d.ts.map +1 -0
- package/dist/core/tools/path-utils.js +81 -0
- package/dist/core/tools/path-utils.js.map +1 -0
- package/dist/core/tools/read.d.ts +39 -0
- package/dist/core/tools/read.d.ts.map +1 -0
- package/dist/core/tools/read.js +166 -0
- package/dist/core/tools/read.js.map +1 -0
- package/dist/core/tools/truncate.d.ts +70 -0
- package/dist/core/tools/truncate.d.ts.map +1 -0
- package/dist/core/tools/truncate.js +205 -0
- package/dist/core/tools/truncate.js.map +1 -0
- package/dist/core/tools/write.d.ts +29 -0
- package/dist/core/tools/write.d.ts.map +1 -0
- package/dist/core/tools/write.js +78 -0
- package/dist/core/tools/write.js.map +1 -0
- package/dist/index.d.ts +27 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +42 -0
- package/dist/index.js.map +1 -0
- package/dist/main.d.ts +8 -0
- package/dist/main.d.ts.map +1 -0
- package/dist/main.js +691 -0
- package/dist/main.js.map +1 -0
- package/dist/migrations.d.ts +33 -0
- package/dist/migrations.d.ts.map +1 -0
- package/dist/migrations.js +261 -0
- package/dist/migrations.js.map +1 -0
- package/dist/modes/index.d.ts +9 -0
- package/dist/modes/index.d.ts.map +1 -0
- package/dist/modes/index.js +8 -0
- package/dist/modes/index.js.map +1 -0
- package/dist/modes/interactive/components/armin.d.ts +34 -0
- package/dist/modes/interactive/components/armin.d.ts.map +1 -0
- package/dist/modes/interactive/components/armin.js +333 -0
- package/dist/modes/interactive/components/armin.js.map +1 -0
- package/dist/modes/interactive/components/assistant-message.d.ts +16 -0
- package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/assistant-message.js +96 -0
- package/dist/modes/interactive/components/assistant-message.js.map +1 -0
- package/dist/modes/interactive/components/bash-execution.d.ts +35 -0
- package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/bash-execution.js +162 -0
- package/dist/modes/interactive/components/bash-execution.js.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.js +51 -0
- package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.js +44 -0
- package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.js +45 -0
- package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/config-selector.d.ts +71 -0
- package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/config-selector.js +479 -0
- package/dist/modes/interactive/components/config-selector.js.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.js +33 -0
- package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
- package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
- package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-editor.js +70 -0
- package/dist/modes/interactive/components/custom-editor.js.map +1 -0
- package/dist/modes/interactive/components/custom-message.d.ts +20 -0
- package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-message.js +79 -0
- package/dist/modes/interactive/components/custom-message.js.map +1 -0
- package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
- package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
- package/dist/modes/interactive/components/daxnuts.js +140 -0
- package/dist/modes/interactive/components/daxnuts.js.map +1 -0
- package/dist/modes/interactive/components/diff.d.ts +12 -0
- package/dist/modes/interactive/components/diff.d.ts.map +1 -0
- package/dist/modes/interactive/components/diff.js +133 -0
- package/dist/modes/interactive/components/diff.js.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.js +21 -0
- package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
- package/dist/modes/interactive/components/extension-editor.d.ts +20 -0
- package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-editor.js +111 -0
- package/dist/modes/interactive/components/extension-editor.js.map +1 -0
- package/dist/modes/interactive/components/extension-input.d.ts +23 -0
- package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-input.js +61 -0
- package/dist/modes/interactive/components/extension-input.js.map +1 -0
- package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
- package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-selector.js +78 -0
- package/dist/modes/interactive/components/extension-selector.js.map +1 -0
- package/dist/modes/interactive/components/footer.d.ts +26 -0
- package/dist/modes/interactive/components/footer.d.ts.map +1 -0
- package/dist/modes/interactive/components/footer.js +198 -0
- package/dist/modes/interactive/components/footer.js.map +1 -0
- package/dist/modes/interactive/components/index.d.ts +32 -0
- package/dist/modes/interactive/components/index.d.ts.map +1 -0
- package/dist/modes/interactive/components/index.js +33 -0
- package/dist/modes/interactive/components/index.js.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts +41 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.js +61 -0
- package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
- package/dist/modes/interactive/components/login-dialog.d.ts +42 -0
- package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
- package/dist/modes/interactive/components/login-dialog.js +145 -0
- package/dist/modes/interactive/components/login-dialog.js.map +1 -0
- package/dist/modes/interactive/components/model-selector.d.ts +47 -0
- package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/model-selector.js +271 -0
- package/dist/modes/interactive/components/model-selector.js.map +1 -0
- package/dist/modes/interactive/components/oauth-selector.d.ts +19 -0
- package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/oauth-selector.js +97 -0
- package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
- package/dist/modes/interactive/components/scoped-models-selector.d.ts +49 -0
- package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/scoped-models-selector.js +275 -0
- package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.js +155 -0
- package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
- package/dist/modes/interactive/components/session-selector.d.ts +95 -0
- package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector.js +851 -0
- package/dist/modes/interactive/components/session-selector.js.map +1 -0
- package/dist/modes/interactive/components/settings-selector.d.ts +58 -0
- package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/settings-selector.js +297 -0
- package/dist/modes/interactive/components/settings-selector.js.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.js +35 -0
- package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
- package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
- package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
- package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/theme-selector.js +46 -0
- package/dist/modes/interactive/components/theme-selector.js.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.js +47 -0
- package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
- package/dist/modes/interactive/components/tool-execution.d.ts +77 -0
- package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/tool-execution.js +787 -0
- package/dist/modes/interactive/components/tool-execution.js.map +1 -0
- package/dist/modes/interactive/components/tree-selector.d.ts +87 -0
- package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/tree-selector.js +1040 -0
- package/dist/modes/interactive/components/tree-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.js +113 -0
- package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message.d.ts +9 -0
- package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message.js +27 -0
- package/dist/modes/interactive/components/user-message.js.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.js +33 -0
- package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
- package/dist/modes/interactive/interactive-mode.d.ts +316 -0
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
- package/dist/modes/interactive/interactive-mode.js +3779 -0
- package/dist/modes/interactive/interactive-mode.js.map +1 -0
- package/dist/modes/interactive/theme/dark.json +85 -0
- package/dist/modes/interactive/theme/light.json +84 -0
- package/dist/modes/interactive/theme/theme-schema.json +335 -0
- package/dist/modes/interactive/theme/theme.d.ts +78 -0
- package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/dist/modes/interactive/theme/theme.js +949 -0
- package/dist/modes/interactive/theme/theme.js.map +1 -0
- package/dist/modes/print-mode.d.ts +28 -0
- package/dist/modes/print-mode.d.ts.map +1 -0
- package/dist/modes/print-mode.js +101 -0
- package/dist/modes/print-mode.js.map +1 -0
- package/dist/modes/rpc/jsonl.d.ts +17 -0
- package/dist/modes/rpc/jsonl.d.ts.map +1 -0
- package/dist/modes/rpc/jsonl.js +49 -0
- package/dist/modes/rpc/jsonl.js.map +1 -0
- package/dist/modes/rpc/rpc-client.d.ts +217 -0
- package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-client.js +401 -0
- package/dist/modes/rpc/rpc-client.js.map +1 -0
- package/dist/modes/rpc/rpc-mode.d.ts +20 -0
- package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-mode.js +509 -0
- package/dist/modes/rpc/rpc-mode.js.map +1 -0
- package/dist/modes/rpc/rpc-types.d.ts +409 -0
- package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-types.js +8 -0
- package/dist/modes/rpc/rpc-types.js.map +1 -0
- package/dist/utils/changelog.d.ts +21 -0
- package/dist/utils/changelog.d.ts.map +1 -0
- package/dist/utils/changelog.js +87 -0
- package/dist/utils/changelog.js.map +1 -0
- package/dist/utils/clipboard-image.d.ts +11 -0
- package/dist/utils/clipboard-image.d.ts.map +1 -0
- package/dist/utils/clipboard-image.js +162 -0
- package/dist/utils/clipboard-image.js.map +1 -0
- package/dist/utils/clipboard-native.d.ts +7 -0
- package/dist/utils/clipboard-native.d.ts.map +1 -0
- package/dist/utils/clipboard-native.js +14 -0
- package/dist/utils/clipboard-native.js.map +1 -0
- package/dist/utils/clipboard.d.ts +2 -0
- package/dist/utils/clipboard.d.ts.map +1 -0
- package/dist/utils/clipboard.js +67 -0
- package/dist/utils/clipboard.js.map +1 -0
- package/dist/utils/frontmatter.d.ts +8 -0
- package/dist/utils/frontmatter.d.ts.map +1 -0
- package/dist/utils/frontmatter.js +26 -0
- package/dist/utils/frontmatter.js.map +1 -0
- package/dist/utils/git.d.ts +26 -0
- package/dist/utils/git.d.ts.map +1 -0
- package/dist/utils/git.js +163 -0
- package/dist/utils/git.js.map +1 -0
- package/dist/utils/image-convert.d.ts +9 -0
- package/dist/utils/image-convert.d.ts.map +1 -0
- package/dist/utils/image-convert.js +35 -0
- package/dist/utils/image-convert.js.map +1 -0
- package/dist/utils/image-resize.d.ts +36 -0
- package/dist/utils/image-resize.d.ts.map +1 -0
- package/dist/utils/image-resize.js +181 -0
- package/dist/utils/image-resize.js.map +1 -0
- package/dist/utils/mime.d.ts +2 -0
- package/dist/utils/mime.d.ts.map +1 -0
- package/dist/utils/mime.js +26 -0
- package/dist/utils/mime.js.map +1 -0
- package/dist/utils/photon.d.ts +21 -0
- package/dist/utils/photon.d.ts.map +1 -0
- package/dist/utils/photon.js +121 -0
- package/dist/utils/photon.js.map +1 -0
- package/dist/utils/shell.d.ts +26 -0
- package/dist/utils/shell.d.ts.map +1 -0
- package/dist/utils/shell.js +186 -0
- package/dist/utils/shell.js.map +1 -0
- package/dist/utils/sleep.d.ts +5 -0
- package/dist/utils/sleep.d.ts.map +1 -0
- package/dist/utils/sleep.js +17 -0
- package/dist/utils/sleep.js.map +1 -0
- package/dist/utils/tools-manager.d.ts +3 -0
- package/dist/utils/tools-manager.d.ts.map +1 -0
- package/dist/utils/tools-manager.js +251 -0
- package/dist/utils/tools-manager.js.map +1 -0
- package/docs/compaction.md +392 -0
- package/docs/custom-provider.md +592 -0
- package/docs/development.md +69 -0
- package/docs/extensions.md +2023 -0
- package/docs/images/doom-extension.png +0 -0
- package/docs/images/exy.png +0 -0
- package/docs/images/interactive-mode.png +0 -0
- package/docs/images/tree-view.png +0 -0
- package/docs/json.md +79 -0
- package/docs/keybindings.md +182 -0
- package/docs/models.md +297 -0
- package/docs/packages.md +209 -0
- package/docs/prompt-templates.md +67 -0
- package/docs/providers.md +188 -0
- package/docs/rpc.md +1354 -0
- package/docs/sdk.md +968 -0
- package/docs/session.md +412 -0
- package/docs/settings.md +225 -0
- package/docs/shell-aliases.md +13 -0
- package/docs/skills.md +231 -0
- package/docs/terminal-setup.md +95 -0
- package/docs/termux.md +127 -0
- package/docs/themes.md +295 -0
- package/docs/tmux.md +61 -0
- package/docs/tree.md +228 -0
- package/docs/tui.md +887 -0
- package/docs/windows.md +17 -0
- package/examples/README.md +25 -0
- package/examples/extensions/README.md +205 -0
- package/examples/extensions/antigravity-image-gen.ts +415 -0
- package/examples/extensions/auto-commit-on-exit.ts +49 -0
- package/examples/extensions/bash-spawn-hook.ts +30 -0
- package/examples/extensions/bookmark.ts +50 -0
- package/examples/extensions/built-in-tool-renderer.ts +246 -0
- package/examples/extensions/claude-rules.ts +86 -0
- package/examples/extensions/commands.ts +72 -0
- package/examples/extensions/confirm-destructive.ts +59 -0
- package/examples/extensions/custom-compaction.ts +114 -0
- package/examples/extensions/custom-footer.ts +64 -0
- package/examples/extensions/custom-header.ts +73 -0
- package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
- package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
- package/examples/extensions/custom-provider-anthropic/package.json +19 -0
- package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
- package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
- package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
- package/examples/extensions/custom-provider-qwen-cli/index.ts +345 -0
- package/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
- package/examples/extensions/dirty-repo-guard.ts +56 -0
- package/examples/extensions/doom-overlay/README.md +46 -0
- package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
- package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
- package/examples/extensions/doom-overlay/doom/build.sh +152 -0
- package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
- package/examples/extensions/doom-overlay/doom-component.ts +132 -0
- package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
- package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
- package/examples/extensions/doom-overlay/index.ts +74 -0
- package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
- package/examples/extensions/dynamic-resources/SKILL.md +8 -0
- package/examples/extensions/dynamic-resources/dynamic.json +79 -0
- package/examples/extensions/dynamic-resources/dynamic.md +5 -0
- package/examples/extensions/dynamic-resources/index.ts +15 -0
- package/examples/extensions/dynamic-tools.ts +74 -0
- package/examples/extensions/event-bus.ts +43 -0
- package/examples/extensions/file-trigger.ts +41 -0
- package/examples/extensions/git-checkpoint.ts +53 -0
- package/examples/extensions/handoff.ts +150 -0
- package/examples/extensions/hello.ts +25 -0
- package/examples/extensions/inline-bash.ts +94 -0
- package/examples/extensions/input-transform.ts +43 -0
- package/examples/extensions/interactive-shell.ts +196 -0
- package/examples/extensions/mac-system-theme.ts +47 -0
- package/examples/extensions/message-renderer.ts +59 -0
- package/examples/extensions/minimal-mode.ts +426 -0
- package/examples/extensions/modal-editor.ts +85 -0
- package/examples/extensions/model-status.ts +31 -0
- package/examples/extensions/notify.ts +55 -0
- package/examples/extensions/overlay-qa-tests.ts +1348 -0
- package/examples/extensions/overlay-test.ts +150 -0
- package/examples/extensions/permission-gate.ts +34 -0
- package/examples/extensions/pirate.ts +47 -0
- package/examples/extensions/plan-mode/README.md +65 -0
- package/examples/extensions/plan-mode/index.ts +340 -0
- package/examples/extensions/plan-mode/utils.ts +168 -0
- package/examples/extensions/preset.ts +398 -0
- package/examples/extensions/protected-paths.ts +30 -0
- package/examples/extensions/provider-payload.ts +14 -0
- package/examples/extensions/qna.ts +119 -0
- package/examples/extensions/question.ts +264 -0
- package/examples/extensions/questionnaire.ts +427 -0
- package/examples/extensions/rainbow-editor.ts +88 -0
- package/examples/extensions/reload-runtime.ts +37 -0
- package/examples/extensions/rpc-demo.ts +124 -0
- package/examples/extensions/sandbox/index.ts +318 -0
- package/examples/extensions/sandbox/package-lock.json +92 -0
- package/examples/extensions/sandbox/package.json +19 -0
- package/examples/extensions/send-user-message.ts +97 -0
- package/examples/extensions/session-name.ts +27 -0
- package/examples/extensions/shutdown-command.ts +63 -0
- package/examples/extensions/snake.ts +343 -0
- package/examples/extensions/space-invaders.ts +560 -0
- package/examples/extensions/ssh.ts +220 -0
- package/examples/extensions/status-line.ts +40 -0
- package/examples/extensions/subagent/README.md +172 -0
- package/examples/extensions/subagent/agents/planner.md +37 -0
- package/examples/extensions/subagent/agents/reviewer.md +35 -0
- package/examples/extensions/subagent/agents/scout.md +50 -0
- package/examples/extensions/subagent/agents/worker.md +24 -0
- package/examples/extensions/subagent/agents.ts +126 -0
- package/examples/extensions/subagent/index.ts +964 -0
- package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
- package/examples/extensions/subagent/prompts/implement.md +10 -0
- package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
- package/examples/extensions/summarize.ts +195 -0
- package/examples/extensions/system-prompt-header.ts +17 -0
- package/examples/extensions/timed-confirm.ts +70 -0
- package/examples/extensions/titlebar-spinner.ts +58 -0
- package/examples/extensions/todo.ts +299 -0
- package/examples/extensions/tool-override.ts +143 -0
- package/examples/extensions/tools.ts +146 -0
- package/examples/extensions/trigger-compact.ts +40 -0
- package/examples/extensions/truncated-tool.ts +192 -0
- package/examples/extensions/widget-placement.ts +17 -0
- package/examples/extensions/with-deps/index.ts +32 -0
- package/examples/extensions/with-deps/package-lock.json +31 -0
- package/examples/extensions/with-deps/package.json +22 -0
- package/examples/rpc-extension-ui.ts +632 -0
- package/examples/sdk/01-minimal.ts +22 -0
- package/examples/sdk/02-custom-model.ts +49 -0
- package/examples/sdk/03-custom-prompt.ts +55 -0
- package/examples/sdk/04-skills.ts +46 -0
- package/examples/sdk/05-tools.ts +56 -0
- package/examples/sdk/06-extensions.ts +88 -0
- package/examples/sdk/07-context-files.ts +40 -0
- package/examples/sdk/08-prompt-templates.ts +47 -0
- package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
- package/examples/sdk/10-settings.ts +51 -0
- package/examples/sdk/11-sessions.ts +48 -0
- package/examples/sdk/12-full-control.ts +82 -0
- package/examples/sdk/README.md +144 -0
- 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
|
+
}
|