@fairyhunter13/opentui-core 0.1.113 → 0.1.114
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/dev/keypress-debug-renderer.ts +148 -0
- package/dev/keypress-debug.ts +43 -0
- package/dev/print-env-vars.ts +32 -0
- package/dev/test-tmux-graphics-334.sh +68 -0
- package/dev/thai-debug-test.ts +68 -0
- package/docs/development.md +144 -0
- package/package.json +62 -53
- package/scripts/build.ts +400 -0
- package/scripts/publish.ts +60 -0
- package/src/3d/SpriteResourceManager.ts +286 -0
- package/src/3d/SpriteUtils.ts +70 -0
- package/src/3d/TextureUtils.ts +196 -0
- package/src/3d/ThreeRenderable.ts +197 -0
- package/src/3d/WGPURenderer.ts +294 -0
- package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
- package/src/3d/animation/SpriteAnimator.ts +633 -0
- package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
- package/src/3d/canvas.ts +464 -0
- package/src/3d/index.ts +12 -0
- package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
- package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
- package/src/3d/physics/physics-interface.ts +31 -0
- package/src/3d/shaders/supersampling.wgsl +201 -0
- package/src/3d.ts +3 -0
- package/src/NativeSpanFeed.ts +300 -0
- package/src/Renderable.ts +1704 -0
- package/src/__snapshots__/buffer.test.ts.snap +28 -0
- package/src/animation/Timeline.test.ts +2709 -0
- package/src/animation/Timeline.ts +598 -0
- package/src/ansi.ts +18 -0
- package/src/benchmark/attenuation-benchmark.ts +81 -0
- package/src/benchmark/colormatrix-benchmark.ts +128 -0
- package/src/benchmark/gain-benchmark.ts +80 -0
- package/src/benchmark/latest-all-bench-run.json +707 -0
- package/src/benchmark/latest-async-bench-run.json +336 -0
- package/src/benchmark/latest-default-bench-run.json +657 -0
- package/src/benchmark/latest-large-bench-run.json +707 -0
- package/src/benchmark/latest-quick-bench-run.json +207 -0
- package/src/benchmark/markdown-benchmark.ts +1796 -0
- package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
- package/src/benchmark/native-span-feed-benchmark.md +56 -0
- package/src/benchmark/native-span-feed-benchmark.ts +596 -0
- package/src/benchmark/native-span-feed-compare.ts +280 -0
- package/src/benchmark/renderer-benchmark.ts +754 -0
- package/src/benchmark/text-table-benchmark.ts +948 -0
- package/src/buffer.test.ts +291 -0
- package/src/buffer.ts +554 -0
- package/src/console.test.ts +612 -0
- package/src/console.ts +1254 -0
- package/src/edit-buffer.test.ts +1769 -0
- package/src/edit-buffer.ts +411 -0
- package/src/editor-view.test.ts +1032 -0
- package/src/editor-view.ts +284 -0
- package/src/examples/ascii-font-selection-demo.ts +245 -0
- package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
- package/src/examples/assets/concrete.png +0 -0
- package/src/examples/assets/crate.png +0 -0
- package/src/examples/assets/crate_emissive.png +0 -0
- package/src/examples/assets/forrest_background.png +0 -0
- package/src/examples/assets/hast-example.json +1018 -0
- package/src/examples/assets/heart.png +0 -0
- package/src/examples/assets/main_char_heavy_attack.png +0 -0
- package/src/examples/assets/main_char_idle.png +0 -0
- package/src/examples/assets/main_char_jump_end.png +0 -0
- package/src/examples/assets/main_char_jump_landing.png +0 -0
- package/src/examples/assets/main_char_jump_start.png +0 -0
- package/src/examples/assets/main_char_run_loop.png +0 -0
- package/src/examples/assets/roughness_map.jpg +0 -0
- package/src/examples/build.ts +115 -0
- package/src/examples/code-demo.ts +924 -0
- package/src/examples/console-demo.ts +358 -0
- package/src/examples/core-plugin-slots-demo.ts +759 -0
- package/src/examples/diff-demo.ts +701 -0
- package/src/examples/draggable-three-demo.ts +259 -0
- package/src/examples/editor-demo.ts +322 -0
- package/src/examples/extmarks-demo.ts +196 -0
- package/src/examples/focus-restore-demo.ts +310 -0
- package/src/examples/fonts.ts +245 -0
- package/src/examples/fractal-shader-demo.ts +268 -0
- package/src/examples/framebuffer-demo.ts +674 -0
- package/src/examples/full-unicode-demo.ts +241 -0
- package/src/examples/golden-star-demo.ts +933 -0
- package/src/examples/grayscale-buffer-demo.ts +249 -0
- package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
- package/src/examples/index.ts +926 -0
- package/src/examples/input-demo.ts +377 -0
- package/src/examples/input-select-layout-demo.ts +425 -0
- package/src/examples/install.sh +143 -0
- package/src/examples/keypress-debug-demo.ts +452 -0
- package/src/examples/lib/HexList.ts +122 -0
- package/src/examples/lib/PaletteGrid.ts +125 -0
- package/src/examples/lib/standalone-keys.ts +25 -0
- package/src/examples/lib/tab-controller.ts +243 -0
- package/src/examples/lights-phong-demo.ts +290 -0
- package/src/examples/link-demo.ts +220 -0
- package/src/examples/live-state-demo.ts +480 -0
- package/src/examples/markdown-demo.ts +725 -0
- package/src/examples/mouse-interaction-demo.ts +428 -0
- package/src/examples/nested-zindex-demo.ts +357 -0
- package/src/examples/opacity-example.ts +235 -0
- package/src/examples/opentui-demo.ts +1057 -0
- package/src/examples/physx-planck-2d-demo.ts +623 -0
- package/src/examples/physx-rapier-2d-demo.ts +655 -0
- package/src/examples/relative-positioning-demo.ts +323 -0
- package/src/examples/scroll-example.ts +214 -0
- package/src/examples/scrollbox-mouse-test.ts +112 -0
- package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
- package/src/examples/select-demo.ts +237 -0
- package/src/examples/shader-cube-demo.ts +1015 -0
- package/src/examples/simple-layout-example.ts +591 -0
- package/src/examples/slider-demo.ts +617 -0
- package/src/examples/split-mode-demo.ts +453 -0
- package/src/examples/sprite-animation-demo.ts +443 -0
- package/src/examples/sprite-particle-generator-demo.ts +486 -0
- package/src/examples/static-sprite-demo.ts +193 -0
- package/src/examples/sticky-scroll-example.ts +308 -0
- package/src/examples/styled-text-demo.ts +282 -0
- package/src/examples/tab-select-demo.ts +219 -0
- package/src/examples/terminal-title.ts +29 -0
- package/src/examples/terminal.ts +305 -0
- package/src/examples/text-node-demo.ts +416 -0
- package/src/examples/text-selection-demo.ts +377 -0
- package/src/examples/text-table-demo.ts +503 -0
- package/src/examples/text-truncation-demo.ts +481 -0
- package/src/examples/text-wrap.ts +757 -0
- package/src/examples/texture-loading-demo.ts +259 -0
- package/src/examples/timeline-example.ts +670 -0
- package/src/examples/transparency-demo.ts +400 -0
- package/src/examples/vnode-composition-demo.ts +404 -0
- package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
- package/src/index.ts +24 -0
- package/src/lib/KeyHandler.integration.test.ts +292 -0
- package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
- package/src/lib/KeyHandler.test.ts +662 -0
- package/src/lib/KeyHandler.ts +222 -0
- package/src/lib/RGBA.test.ts +984 -0
- package/src/lib/RGBA.ts +204 -0
- package/src/lib/ascii.font.ts +330 -0
- package/src/lib/border.test.ts +83 -0
- package/src/lib/border.ts +170 -0
- package/src/lib/bunfs.test.ts +27 -0
- package/src/lib/bunfs.ts +18 -0
- package/src/lib/clipboard.test.ts +41 -0
- package/src/lib/clipboard.ts +47 -0
- package/src/lib/clock.ts +35 -0
- package/src/lib/data-paths.test.ts +133 -0
- package/src/lib/data-paths.ts +109 -0
- package/src/lib/debounce.ts +106 -0
- package/src/lib/detect-links.test.ts +98 -0
- package/src/lib/detect-links.ts +56 -0
- package/src/lib/env.test.ts +228 -0
- package/src/lib/env.ts +209 -0
- package/src/lib/extmarks-history.ts +51 -0
- package/src/lib/extmarks-multiwidth.test.ts +322 -0
- package/src/lib/extmarks.test.ts +3457 -0
- package/src/lib/extmarks.ts +843 -0
- package/src/lib/fonts/block.json +405 -0
- package/src/lib/fonts/grid.json +265 -0
- package/src/lib/fonts/huge.json +741 -0
- package/src/lib/fonts/pallet.json +314 -0
- package/src/lib/fonts/shade.json +591 -0
- package/src/lib/fonts/slick.json +321 -0
- package/src/lib/fonts/tiny.json +69 -0
- package/src/lib/hast-styled-text.ts +59 -0
- package/src/lib/index.ts +21 -0
- package/src/lib/keymapping.test.ts +317 -0
- package/src/lib/keymapping.ts +115 -0
- package/src/lib/objects-in-viewport.test.ts +787 -0
- package/src/lib/objects-in-viewport.ts +153 -0
- package/src/lib/output.capture.ts +58 -0
- package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
- package/src/lib/parse.keypress-kitty.test.ts +663 -0
- package/src/lib/parse.keypress-kitty.ts +439 -0
- package/src/lib/parse.keypress.test.ts +1849 -0
- package/src/lib/parse.keypress.ts +397 -0
- package/src/lib/parse.mouse.test.ts +552 -0
- package/src/lib/parse.mouse.ts +232 -0
- package/src/lib/paste.ts +16 -0
- package/src/lib/queue.ts +65 -0
- package/src/lib/renderable.validations.test.ts +87 -0
- package/src/lib/renderable.validations.ts +83 -0
- package/src/lib/scroll-acceleration.ts +98 -0
- package/src/lib/selection.ts +240 -0
- package/src/lib/singleton.ts +28 -0
- package/src/lib/stdin-parser.test.ts +2290 -0
- package/src/lib/stdin-parser.ts +1810 -0
- package/src/lib/styled-text.ts +178 -0
- package/src/lib/terminal-capability-detection.test.ts +202 -0
- package/src/lib/terminal-capability-detection.ts +79 -0
- package/src/lib/terminal-palette.test.ts +878 -0
- package/src/lib/terminal-palette.ts +383 -0
- package/src/lib/tree-sitter/assets/README.md +118 -0
- package/src/lib/tree-sitter/assets/update.ts +334 -0
- package/src/lib/tree-sitter/assets.d.ts +9 -0
- package/src/lib/tree-sitter/cache.test.ts +273 -0
- package/src/lib/tree-sitter/client.test.ts +1165 -0
- package/src/lib/tree-sitter/client.ts +607 -0
- package/src/lib/tree-sitter/default-parsers.ts +86 -0
- package/src/lib/tree-sitter/download-utils.ts +148 -0
- package/src/lib/tree-sitter/index.ts +28 -0
- package/src/lib/tree-sitter/parser.worker.ts +1042 -0
- package/src/lib/tree-sitter/parsers-config.ts +81 -0
- package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
- package/src/lib/tree-sitter/resolve-ft.ts +189 -0
- package/src/lib/tree-sitter/types.ts +82 -0
- package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
- package/src/lib/tree-sitter-styled-text.ts +306 -0
- package/src/lib/validate-dir-name.ts +55 -0
- package/src/lib/yoga.options.test.ts +628 -0
- package/src/lib/yoga.options.ts +346 -0
- package/src/plugins/core-slot.ts +579 -0
- package/src/plugins/registry.ts +402 -0
- package/src/plugins/types.ts +46 -0
- package/src/post/effects.ts +930 -0
- package/src/post/filters.ts +489 -0
- package/src/post/matrices.ts +288 -0
- package/src/renderables/ASCIIFont.ts +219 -0
- package/src/renderables/Box.test.ts +205 -0
- package/src/renderables/Box.ts +326 -0
- package/src/renderables/Code.test.ts +2062 -0
- package/src/renderables/Code.ts +357 -0
- package/src/renderables/Diff.regression.test.ts +226 -0
- package/src/renderables/Diff.test.ts +3101 -0
- package/src/renderables/Diff.ts +1211 -0
- package/src/renderables/EditBufferRenderable.test.ts +288 -0
- package/src/renderables/EditBufferRenderable.ts +1166 -0
- package/src/renderables/FrameBuffer.ts +47 -0
- package/src/renderables/Input.test.ts +1228 -0
- package/src/renderables/Input.ts +247 -0
- package/src/renderables/LineNumberRenderable.ts +724 -0
- package/src/renderables/Markdown.ts +1393 -0
- package/src/renderables/ScrollBar.ts +422 -0
- package/src/renderables/ScrollBox.ts +883 -0
- package/src/renderables/Select.test.ts +1033 -0
- package/src/renderables/Select.ts +524 -0
- package/src/renderables/Slider.test.ts +456 -0
- package/src/renderables/Slider.ts +342 -0
- package/src/renderables/TabSelect.test.ts +197 -0
- package/src/renderables/TabSelect.ts +455 -0
- package/src/renderables/Text.selection-buffer.test.ts +123 -0
- package/src/renderables/Text.test.ts +2660 -0
- package/src/renderables/Text.ts +147 -0
- package/src/renderables/TextBufferRenderable.ts +518 -0
- package/src/renderables/TextNode.test.ts +1058 -0
- package/src/renderables/TextNode.ts +325 -0
- package/src/renderables/TextTable.test.ts +1421 -0
- package/src/renderables/TextTable.ts +1344 -0
- package/src/renderables/Textarea.ts +430 -0
- package/src/renderables/TimeToFirstDraw.ts +89 -0
- package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
- package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
- package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
- package/src/renderables/__tests__/Markdown.test.ts +2518 -0
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
- package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
- package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
- package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
- package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
- package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
- package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
- package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
- package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
- package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
- package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
- package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
- package/src/renderables/composition/README.md +8 -0
- package/src/renderables/composition/VRenderable.ts +32 -0
- package/src/renderables/composition/constructs.ts +127 -0
- package/src/renderables/composition/vnode.ts +289 -0
- package/src/renderables/index.ts +23 -0
- package/src/renderables/markdown-parser.ts +66 -0
- package/src/renderer.ts +2681 -0
- package/src/runtime-plugin-support.ts +39 -0
- package/src/runtime-plugin.ts +615 -0
- package/src/syntax-style.test.ts +841 -0
- package/src/syntax-style.ts +257 -0
- package/src/testing/README.md +210 -0
- package/src/testing/capture-spans.test.ts +194 -0
- package/src/testing/integration.test.ts +276 -0
- package/src/testing/manual-clock.ts +117 -0
- package/src/testing/mock-keys.test.ts +1378 -0
- package/src/testing/mock-keys.ts +457 -0
- package/src/testing/mock-mouse.test.ts +218 -0
- package/src/testing/mock-mouse.ts +247 -0
- package/src/testing/mock-tree-sitter-client.ts +73 -0
- package/src/testing/spy.ts +13 -0
- package/src/testing/test-recorder.test.ts +415 -0
- package/src/testing/test-recorder.ts +145 -0
- package/src/testing/test-renderer.ts +132 -0
- package/src/testing.ts +7 -0
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
- package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
- package/src/tests/allocator-stats.test.ts +38 -0
- package/src/tests/destroy-during-render.test.ts +200 -0
- package/src/tests/destroy-on-exit.fixture.ts +36 -0
- package/src/tests/destroy-on-exit.test.ts +41 -0
- package/src/tests/hover-cursor.test.ts +98 -0
- package/src/tests/native-span-feed-async.test.ts +173 -0
- package/src/tests/native-span-feed-close.test.ts +120 -0
- package/src/tests/native-span-feed-coverage.test.ts +227 -0
- package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
- package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
- package/src/tests/opacity.test.ts +123 -0
- package/src/tests/renderable.snapshot.test.ts +524 -0
- package/src/tests/renderable.test.ts +1281 -0
- package/src/tests/renderer.clock.test.ts +158 -0
- package/src/tests/renderer.console-startup.test.ts +185 -0
- package/src/tests/renderer.control.test.ts +425 -0
- package/src/tests/renderer.core-slot-binding.test.ts +952 -0
- package/src/tests/renderer.cursor.test.ts +26 -0
- package/src/tests/renderer.destroy-during-render.test.ts +147 -0
- package/src/tests/renderer.focus-restore.test.ts +257 -0
- package/src/tests/renderer.focus.test.ts +294 -0
- package/src/tests/renderer.idle.test.ts +219 -0
- package/src/tests/renderer.input.test.ts +2237 -0
- package/src/tests/renderer.kitty-flags.test.ts +195 -0
- package/src/tests/renderer.mouse.test.ts +1274 -0
- package/src/tests/renderer.palette.test.ts +629 -0
- package/src/tests/renderer.selection.test.ts +49 -0
- package/src/tests/renderer.slot-registry.test.ts +684 -0
- package/src/tests/renderer.useMouse.test.ts +47 -0
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
- package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
- package/src/tests/runtime-plugin-support.fixture.ts +11 -0
- package/src/tests/runtime-plugin-support.test.ts +19 -0
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
- package/src/tests/runtime-plugin.fixture.ts +40 -0
- package/src/tests/runtime-plugin.test.ts +354 -0
- package/src/tests/scrollbox-culling-bug.test.ts +114 -0
- package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
- package/src/tests/scrollbox-hitgrid.test.ts +909 -0
- package/src/tests/scrollbox.test.ts +1530 -0
- package/src/tests/wrap-resize-perf.test.ts +276 -0
- package/src/tests/yoga-setters.test.ts +921 -0
- package/src/text-buffer-view.test.ts +705 -0
- package/src/text-buffer-view.ts +189 -0
- package/src/text-buffer.test.ts +347 -0
- package/src/text-buffer.ts +250 -0
- package/src/types.ts +161 -0
- package/src/utils.ts +88 -0
- package/src/zig/ansi.zig +268 -0
- package/src/zig/bench/README.md +50 -0
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
- package/src/zig/bench/edit-buffer_bench.zig +476 -0
- package/src/zig/bench/native-span-feed_bench.zig +100 -0
- package/src/zig/bench/rope-markers_bench.zig +713 -0
- package/src/zig/bench/rope_bench.zig +514 -0
- package/src/zig/bench/styled-text_bench.zig +470 -0
- package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
- package/src/zig/bench/text-buffer-view_bench.zig +459 -0
- package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
- package/src/zig/bench/utf8_bench.zig +799 -0
- package/src/zig/bench-utils.zig +431 -0
- package/src/zig/bench.zig +217 -0
- package/src/zig/buffer-methods.zig +211 -0
- package/src/zig/buffer.zig +2281 -0
- package/src/zig/build.zig +289 -0
- package/src/zig/build.zig.zon +16 -0
- package/src/zig/edit-buffer.zig +825 -0
- package/src/zig/editor-view.zig +802 -0
- package/src/zig/event-bus.zig +13 -0
- package/src/zig/event-emitter.zig +65 -0
- package/src/zig/file-logger.zig +92 -0
- package/src/zig/grapheme.zig +599 -0
- package/src/zig/lib.zig +1854 -0
- package/src/zig/link.zig +333 -0
- package/src/zig/logger.zig +43 -0
- package/src/zig/mem-registry.zig +125 -0
- package/src/zig/native-span-feed-bench-lib.zig +7 -0
- package/src/zig/native-span-feed.zig +708 -0
- package/src/zig/renderer.zig +1393 -0
- package/src/zig/rope.zig +1220 -0
- package/src/zig/syntax-style.zig +161 -0
- package/src/zig/terminal.zig +987 -0
- package/src/zig/test.zig +72 -0
- package/src/zig/tests/README.md +18 -0
- package/src/zig/tests/buffer-methods_test.zig +1109 -0
- package/src/zig/tests/buffer_test.zig +2557 -0
- package/src/zig/tests/edit-buffer-history_test.zig +271 -0
- package/src/zig/tests/edit-buffer_test.zig +1689 -0
- package/src/zig/tests/editor-view_test.zig +3299 -0
- package/src/zig/tests/event-emitter_test.zig +249 -0
- package/src/zig/tests/grapheme_test.zig +1304 -0
- package/src/zig/tests/link_test.zig +190 -0
- package/src/zig/tests/mem-registry_test.zig +473 -0
- package/src/zig/tests/memory_leak_regression_test.zig +159 -0
- package/src/zig/tests/native-span-feed_test.zig +1264 -0
- package/src/zig/tests/renderer_test.zig +1017 -0
- package/src/zig/tests/rope-nested_test.zig +712 -0
- package/src/zig/tests/rope_fuzz_test.zig +238 -0
- package/src/zig/tests/rope_test.zig +2362 -0
- package/src/zig/tests/segment-merge.test.zig +148 -0
- package/src/zig/tests/syntax-style_test.zig +557 -0
- package/src/zig/tests/terminal_test.zig +754 -0
- package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
- package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
- package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
- package/src/zig/tests/text-buffer-segment_test.zig +320 -0
- package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
- package/src/zig/tests/text-buffer-view_test.zig +3649 -0
- package/src/zig/tests/text-buffer_test.zig +2191 -0
- package/src/zig/tests/unicode-width-map.zon +3909 -0
- package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
- package/src/zig/tests/utf8_test.zig +4057 -0
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
- package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
- package/src/zig/tests/word-wrap-editing_test.zig +498 -0
- package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
- package/src/zig/text-buffer-iterators.zig +499 -0
- package/src/zig/text-buffer-segment.zig +404 -0
- package/src/zig/text-buffer-view.zig +1371 -0
- package/src/zig/text-buffer.zig +1180 -0
- package/src/zig/utf8.zig +1948 -0
- package/src/zig/utils.zig +9 -0
- package/src/zig-structs.ts +261 -0
- package/src/zig.ts +3884 -0
- package/tsconfig.build.json +24 -0
- package/tsconfig.json +27 -0
- package/3d/SpriteResourceManager.d.ts +0 -74
- package/3d/SpriteUtils.d.ts +0 -13
- package/3d/TextureUtils.d.ts +0 -24
- package/3d/ThreeRenderable.d.ts +0 -40
- package/3d/WGPURenderer.d.ts +0 -61
- package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
- package/3d/animation/SpriteAnimator.d.ts +0 -124
- package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
- package/3d/canvas.d.ts +0 -44
- package/3d/index.d.ts +0 -12
- package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
- package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
- package/3d/physics/physics-interface.d.ts +0 -27
- package/3d.d.ts +0 -2
- package/3d.js +0 -34041
- package/3d.js.map +0 -155
- package/LICENSE +0 -21
- package/NativeSpanFeed.d.ts +0 -41
- package/Renderable.d.ts +0 -334
- package/animation/Timeline.d.ts +0 -126
- package/ansi.d.ts +0 -13
- package/buffer.d.ts +0 -111
- package/console.d.ts +0 -144
- package/edit-buffer.d.ts +0 -98
- package/editor-view.d.ts +0 -73
- package/index-9vwc3fg6.js +0 -12260
- package/index-9vwc3fg6.js.map +0 -42
- package/index-dcj62y8t.js +0 -20614
- package/index-dcj62y8t.js.map +0 -67
- package/index-f7n39gpy.js +0 -411
- package/index-f7n39gpy.js.map +0 -10
- package/index.d.ts +0 -23
- package/index.js +0 -478
- package/index.js.map +0 -9
- package/lib/KeyHandler.d.ts +0 -61
- package/lib/RGBA.d.ts +0 -25
- package/lib/ascii.font.d.ts +0 -508
- package/lib/border.d.ts +0 -51
- package/lib/bunfs.d.ts +0 -7
- package/lib/clipboard.d.ts +0 -17
- package/lib/clock.d.ts +0 -15
- package/lib/data-paths.d.ts +0 -26
- package/lib/debounce.d.ts +0 -42
- package/lib/detect-links.d.ts +0 -6
- package/lib/env.d.ts +0 -42
- package/lib/extmarks-history.d.ts +0 -17
- package/lib/extmarks.d.ts +0 -89
- package/lib/hast-styled-text.d.ts +0 -17
- package/lib/index.d.ts +0 -21
- package/lib/keymapping.d.ts +0 -25
- package/lib/objects-in-viewport.d.ts +0 -24
- package/lib/output.capture.d.ts +0 -24
- package/lib/parse.keypress-kitty.d.ts +0 -2
- package/lib/parse.keypress.d.ts +0 -26
- package/lib/parse.mouse.d.ts +0 -30
- package/lib/paste.d.ts +0 -7
- package/lib/queue.d.ts +0 -15
- package/lib/renderable.validations.d.ts +0 -12
- package/lib/scroll-acceleration.d.ts +0 -43
- package/lib/selection.d.ts +0 -63
- package/lib/singleton.d.ts +0 -7
- package/lib/stdin-parser.d.ts +0 -87
- package/lib/styled-text.d.ts +0 -63
- package/lib/terminal-capability-detection.d.ts +0 -30
- package/lib/terminal-palette.d.ts +0 -50
- package/lib/tree-sitter/assets/update.d.ts +0 -11
- package/lib/tree-sitter/client.d.ts +0 -47
- package/lib/tree-sitter/default-parsers.d.ts +0 -2
- package/lib/tree-sitter/download-utils.d.ts +0 -21
- package/lib/tree-sitter/index.d.ts +0 -8
- package/lib/tree-sitter/parser.worker.d.ts +0 -1
- package/lib/tree-sitter/parsers-config.d.ts +0 -53
- package/lib/tree-sitter/resolve-ft.d.ts +0 -5
- package/lib/tree-sitter/types.d.ts +0 -82
- package/lib/tree-sitter-styled-text.d.ts +0 -14
- package/lib/validate-dir-name.d.ts +0 -1
- package/lib/yoga.options.d.ts +0 -32
- package/parser.worker.js +0 -899
- package/parser.worker.js.map +0 -12
- package/plugins/core-slot.d.ts +0 -72
- package/plugins/registry.d.ts +0 -42
- package/plugins/types.d.ts +0 -34
- package/post/effects.d.ts +0 -147
- package/post/filters.d.ts +0 -65
- package/post/matrices.d.ts +0 -20
- package/renderables/ASCIIFont.d.ts +0 -52
- package/renderables/Box.d.ts +0 -81
- package/renderables/Code.d.ts +0 -78
- package/renderables/Diff.d.ts +0 -142
- package/renderables/EditBufferRenderable.d.ts +0 -237
- package/renderables/FrameBuffer.d.ts +0 -16
- package/renderables/Input.d.ts +0 -67
- package/renderables/LineNumberRenderable.d.ts +0 -78
- package/renderables/Markdown.d.ts +0 -185
- package/renderables/ScrollBar.d.ts +0 -77
- package/renderables/ScrollBox.d.ts +0 -124
- package/renderables/Select.d.ts +0 -115
- package/renderables/Slider.d.ts +0 -47
- package/renderables/TabSelect.d.ts +0 -96
- package/renderables/Text.d.ts +0 -36
- package/renderables/TextBufferRenderable.d.ts +0 -105
- package/renderables/TextNode.d.ts +0 -91
- package/renderables/TextTable.d.ts +0 -140
- package/renderables/Textarea.d.ts +0 -63
- package/renderables/TimeToFirstDraw.d.ts +0 -24
- package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
- package/renderables/composition/VRenderable.d.ts +0 -16
- package/renderables/composition/constructs.d.ts +0 -35
- package/renderables/composition/vnode.d.ts +0 -46
- package/renderables/index.d.ts +0 -23
- package/renderables/markdown-parser.d.ts +0 -10
- package/renderer.d.ts +0 -419
- package/runtime-plugin-support.d.ts +0 -3
- package/runtime-plugin-support.js +0 -29
- package/runtime-plugin-support.js.map +0 -10
- package/runtime-plugin.d.ts +0 -16
- package/runtime-plugin.js +0 -16
- package/runtime-plugin.js.map +0 -9
- package/syntax-style.d.ts +0 -54
- package/testing/manual-clock.d.ts +0 -17
- package/testing/mock-keys.d.ts +0 -81
- package/testing/mock-mouse.d.ts +0 -38
- package/testing/mock-tree-sitter-client.d.ts +0 -23
- package/testing/spy.d.ts +0 -7
- package/testing/test-recorder.d.ts +0 -61
- package/testing/test-renderer.d.ts +0 -23
- package/testing.d.ts +0 -6
- package/testing.js +0 -697
- package/testing.js.map +0 -15
- package/text-buffer-view.d.ts +0 -42
- package/text-buffer.d.ts +0 -67
- package/types.d.ts +0 -139
- package/utils.d.ts +0 -14
- package/zig-structs.d.ts +0 -155
- package/zig.d.ts +0 -353
- /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -0,0 +1,987 @@
|
|
|
1
|
+
const std = @import("std");
|
|
2
|
+
const builtin = @import("builtin");
|
|
3
|
+
const atomic = std.atomic;
|
|
4
|
+
const assert = std.debug.assert;
|
|
5
|
+
const ansi = @import("ansi.zig");
|
|
6
|
+
const utf8 = @import("utf8.zig");
|
|
7
|
+
const logger = @import("logger.zig");
|
|
8
|
+
|
|
9
|
+
const WidthMethod = utf8.WidthMethod;
|
|
10
|
+
|
|
11
|
+
/// Terminal capability detection and management
|
|
12
|
+
pub const Terminal = @This();
|
|
13
|
+
|
|
14
|
+
pub const Capabilities = struct {
|
|
15
|
+
kitty_keyboard: bool = false,
|
|
16
|
+
kitty_graphics: bool = false,
|
|
17
|
+
rgb: bool = false,
|
|
18
|
+
unicode: WidthMethod = .unicode,
|
|
19
|
+
sgr_pixels: bool = false,
|
|
20
|
+
color_scheme_updates: bool = false,
|
|
21
|
+
explicit_width: bool = false,
|
|
22
|
+
scaled_text: bool = false,
|
|
23
|
+
sixel: bool = false,
|
|
24
|
+
focus_tracking: bool = false,
|
|
25
|
+
sync: bool = false,
|
|
26
|
+
bracketed_paste: bool = false,
|
|
27
|
+
hyperlinks: bool = false,
|
|
28
|
+
osc52: bool = false,
|
|
29
|
+
explicit_cursor_positioning: bool = false,
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
pub const MouseLevel = enum {
|
|
33
|
+
none,
|
|
34
|
+
basic, // click only
|
|
35
|
+
drag, // click + drag
|
|
36
|
+
motion, // all motion
|
|
37
|
+
pixels, // pixel coordinates
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
pub const CursorStyle = enum {
|
|
41
|
+
block,
|
|
42
|
+
line,
|
|
43
|
+
underline,
|
|
44
|
+
default,
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
pub const MousePointerStyle = enum(u8) {
|
|
48
|
+
default = 0,
|
|
49
|
+
pointer = 1,
|
|
50
|
+
text = 2,
|
|
51
|
+
crosshair = 3,
|
|
52
|
+
move = 4,
|
|
53
|
+
not_allowed = 5,
|
|
54
|
+
|
|
55
|
+
pub fn toName(self: MousePointerStyle) []const u8 {
|
|
56
|
+
return if (self == .not_allowed) "not-allowed" else @tagName(self);
|
|
57
|
+
}
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
pub const ClipboardTarget = enum {
|
|
61
|
+
clipboard, // "c"
|
|
62
|
+
primary, // "p"
|
|
63
|
+
secondary, // "s"
|
|
64
|
+
query, // "q"
|
|
65
|
+
|
|
66
|
+
pub fn toChar(self: ClipboardTarget) u8 {
|
|
67
|
+
return switch (self) {
|
|
68
|
+
.clipboard => 'c',
|
|
69
|
+
.primary => 'p',
|
|
70
|
+
.secondary => 's',
|
|
71
|
+
.query => 'q',
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
pub const Options = struct {
|
|
77
|
+
// Kitty keyboard protocol flags (progressive enhancement):
|
|
78
|
+
// See: https://sw.kovidgoyal.net/kitty/keyboard-protocol/#progressive-enhancement
|
|
79
|
+
// Bit 0 (0b1): Disambiguate escape codes (fixes ESC timing, alt+key ambiguity, ctrl+c as event)
|
|
80
|
+
// Bit 1 (0b10): Report event types (press/repeat/release)
|
|
81
|
+
// Bit 2 (0b100): Report alternate keys (e.g., numpad vs regular, shifted, base layout)
|
|
82
|
+
// Bit 3 (0b1000): Report all keys as escape codes
|
|
83
|
+
// Bit 4 (0b10000): Report text associated with key events
|
|
84
|
+
// Default 0b00101 (5) = disambiguate + alternate keys
|
|
85
|
+
// Use 0b00111 (7) to also enable event types for key release detection
|
|
86
|
+
kitty_keyboard_flags: u8 = 0b00101,
|
|
87
|
+
remote: bool = false,
|
|
88
|
+
// Optional override for environment lookups. Caller owns the map.
|
|
89
|
+
env_map: ?*const std.process.EnvMap = null,
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
pub const TerminalInfo = struct {
|
|
93
|
+
name: [64]u8 = [_]u8{0} ** 64,
|
|
94
|
+
name_len: usize = 0,
|
|
95
|
+
version: [32]u8 = [_]u8{0} ** 32,
|
|
96
|
+
version_len: usize = 0,
|
|
97
|
+
from_xtversion: bool = false,
|
|
98
|
+
};
|
|
99
|
+
|
|
100
|
+
caps: Capabilities = .{},
|
|
101
|
+
opts: Options = .{},
|
|
102
|
+
host_env_map: ?std.process.EnvMap = null,
|
|
103
|
+
|
|
104
|
+
in_tmux: bool = false,
|
|
105
|
+
skip_graphics_query: bool = false,
|
|
106
|
+
skip_explicit_width_query: bool = false,
|
|
107
|
+
graphics_query_pending: bool = false,
|
|
108
|
+
capability_queries_pending: bool = false,
|
|
109
|
+
|
|
110
|
+
state: struct {
|
|
111
|
+
alt_screen: bool = false,
|
|
112
|
+
kitty_keyboard: bool = false,
|
|
113
|
+
kitty_keyboard_flags: u8 = 0,
|
|
114
|
+
bracketed_paste: bool = false,
|
|
115
|
+
mouse: bool = false,
|
|
116
|
+
mouse_movement: bool = true,
|
|
117
|
+
pixel_mouse: bool = false,
|
|
118
|
+
color_scheme_updates: bool = false,
|
|
119
|
+
focus_tracking: bool = false,
|
|
120
|
+
modify_other_keys: bool = false,
|
|
121
|
+
mouse_pointer: MousePointerStyle = .default,
|
|
122
|
+
cursor: struct {
|
|
123
|
+
row: u16 = 0,
|
|
124
|
+
col: u16 = 0,
|
|
125
|
+
x: u32 = 1, // 1-based for rendering
|
|
126
|
+
y: u32 = 1, // 1-based for rendering
|
|
127
|
+
visible: bool = true,
|
|
128
|
+
style: CursorStyle = .default,
|
|
129
|
+
blinking: bool = false,
|
|
130
|
+
color: [4]f32 = .{ 1.0, 1.0, 1.0, 1.0 }, // RGBA
|
|
131
|
+
} = .{},
|
|
132
|
+
} = .{},
|
|
133
|
+
|
|
134
|
+
term_info: TerminalInfo = .{},
|
|
135
|
+
|
|
136
|
+
pub fn init(opts: Options) Terminal {
|
|
137
|
+
var term: Terminal = .{
|
|
138
|
+
.opts = opts,
|
|
139
|
+
};
|
|
140
|
+
|
|
141
|
+
term.checkEnvironmentOverrides();
|
|
142
|
+
return term;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
pub fn deinit(self: *Terminal) void {
|
|
146
|
+
if (self.host_env_map) |*env_map| {
|
|
147
|
+
env_map.deinit();
|
|
148
|
+
self.host_env_map = null;
|
|
149
|
+
}
|
|
150
|
+
self.opts.env_map = null;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
pub fn setHostEnvVar(self: *Terminal, allocator: std.mem.Allocator, key: []const u8, value: []const u8) !void {
|
|
154
|
+
if (self.host_env_map == null) {
|
|
155
|
+
self.host_env_map = std.process.EnvMap.init(allocator);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
const env_map = &self.host_env_map.?;
|
|
159
|
+
try env_map.put(key, value);
|
|
160
|
+
self.opts.env_map = env_map;
|
|
161
|
+
self.checkEnvironmentOverrides();
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
pub fn resetState(self: *Terminal, tty: anytype) !void {
|
|
165
|
+
try tty.writeAll(ansi.ANSI.showCursor);
|
|
166
|
+
try tty.writeAll(ansi.ANSI.reset);
|
|
167
|
+
try tty.writeAll(ansi.ANSI.resetMousePointer);
|
|
168
|
+
self.state.mouse_pointer = .default;
|
|
169
|
+
|
|
170
|
+
if (self.state.kitty_keyboard) {
|
|
171
|
+
try self.setKittyKeyboard(tty, false, 0);
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
if (self.state.modify_other_keys) {
|
|
175
|
+
try self.setModifyOtherKeys(tty, false);
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
if (self.state.mouse) {
|
|
179
|
+
try self.setMouseMode(tty, false, self.state.mouse_movement);
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
if (self.state.bracketed_paste) {
|
|
183
|
+
try self.setBracketedPaste(tty, false);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
if (self.state.focus_tracking) {
|
|
187
|
+
try self.setFocusTracking(tty, false);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
if (self.state.alt_screen) {
|
|
191
|
+
try self.exitAltScreen(tty);
|
|
192
|
+
} else {
|
|
193
|
+
switch (builtin.os.tag) {
|
|
194
|
+
.windows => {
|
|
195
|
+
try tty.writeByte('\r');
|
|
196
|
+
var i: u16 = 0;
|
|
197
|
+
while (i < self.state.cursor.row) : (i += 1) {
|
|
198
|
+
try tty.writeAll(ansi.ANSI.reverseIndex);
|
|
199
|
+
}
|
|
200
|
+
try tty.writeAll(ansi.ANSI.eraseBelowCursor);
|
|
201
|
+
},
|
|
202
|
+
else => {},
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
if (self.state.color_scheme_updates) {
|
|
207
|
+
try self.setColorSchemeUpdates(tty, false);
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
self.setTerminalTitle(tty, "");
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
pub fn enterAltScreen(self: *Terminal, tty: anytype) !void {
|
|
214
|
+
try tty.writeAll(ansi.ANSI.switchToAlternateScreen);
|
|
215
|
+
self.state.alt_screen = true;
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
pub fn exitAltScreen(self: *Terminal, tty: anytype) !void {
|
|
219
|
+
try tty.writeAll(ansi.ANSI.switchToMainScreen);
|
|
220
|
+
self.state.alt_screen = false;
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
pub fn queryTerminalSend(self: *Terminal, tty: anytype) !void {
|
|
224
|
+
self.checkEnvironmentOverrides();
|
|
225
|
+
self.graphics_query_pending = !self.skip_graphics_query;
|
|
226
|
+
self.capability_queries_pending = false;
|
|
227
|
+
|
|
228
|
+
// Send xtversion first (doesn't need DCS wrapping - used for tmux detection)
|
|
229
|
+
try tty.writeAll(ansi.ANSI.xtversion ++
|
|
230
|
+
ansi.ANSI.hideCursor ++
|
|
231
|
+
ansi.ANSI.saveCursorState);
|
|
232
|
+
|
|
233
|
+
if (self.in_tmux) {
|
|
234
|
+
try tty.writeAll(ansi.ANSI.capabilityQueriesTmux);
|
|
235
|
+
} else {
|
|
236
|
+
try tty.writeAll(ansi.ANSI.capabilityQueries);
|
|
237
|
+
self.capability_queries_pending = true;
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
if (!self.skip_explicit_width_query) {
|
|
241
|
+
try tty.writeAll(ansi.ANSI.home ++
|
|
242
|
+
ansi.ANSI.explicitWidthQuery ++
|
|
243
|
+
ansi.ANSI.cursorPositionRequest ++
|
|
244
|
+
ansi.ANSI.home ++
|
|
245
|
+
ansi.ANSI.scaledTextQuery ++
|
|
246
|
+
ansi.ANSI.cursorPositionRequest);
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
try tty.writeAll(ansi.ANSI.restoreCursorState);
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
pub fn sendPendingQueries(self: *Terminal, tty: anytype) !bool {
|
|
253
|
+
var sent = false;
|
|
254
|
+
const is_tmux = self.in_tmux or self.isXtversionTmux();
|
|
255
|
+
|
|
256
|
+
// Re-send capability queries DCS wrapped if tmux detected via xtversion
|
|
257
|
+
// Only needed if we got xtversion response indicating tmux
|
|
258
|
+
if (self.capability_queries_pending) {
|
|
259
|
+
if (self.term_info.from_xtversion and is_tmux) {
|
|
260
|
+
try tty.writeAll(ansi.ANSI.capabilityQueriesTmux);
|
|
261
|
+
sent = true;
|
|
262
|
+
}
|
|
263
|
+
// Clear pending flag regardless - non-tmux terminals already received unwrapped queries
|
|
264
|
+
self.capability_queries_pending = false;
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
if (self.graphics_query_pending and !self.skip_graphics_query) {
|
|
268
|
+
if (is_tmux) {
|
|
269
|
+
try tty.writeAll(ansi.ANSI.kittyGraphicsQueryTmux);
|
|
270
|
+
} else {
|
|
271
|
+
try tty.writeAll(ansi.ANSI.kittyGraphicsQuery);
|
|
272
|
+
}
|
|
273
|
+
self.graphics_query_pending = false;
|
|
274
|
+
sent = true;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
return sent;
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
pub fn enableDetectedFeatures(self: *Terminal, tty: anytype, use_kitty_keyboard: bool) !void {
|
|
281
|
+
if (builtin.os.tag == .windows) {
|
|
282
|
+
// Windows-specific defaults for ConPTY
|
|
283
|
+
self.caps.rgb = true;
|
|
284
|
+
self.caps.bracketed_paste = true;
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
self.checkEnvironmentOverrides();
|
|
288
|
+
|
|
289
|
+
if (!self.state.modify_other_keys and !self.state.kitty_keyboard) {
|
|
290
|
+
try self.setModifyOtherKeys(tty, true);
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
if (self.caps.kitty_keyboard and use_kitty_keyboard) {
|
|
294
|
+
if (self.state.modify_other_keys) {
|
|
295
|
+
try self.setModifyOtherKeys(tty, false);
|
|
296
|
+
}
|
|
297
|
+
try self.setKittyKeyboard(tty, true, self.opts.kitty_keyboard_flags);
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
if (self.caps.unicode == .unicode and !self.caps.explicit_width) {
|
|
301
|
+
try tty.writeAll(ansi.ANSI.unicodeSet);
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
if (self.caps.bracketed_paste) {
|
|
305
|
+
try self.setBracketedPaste(tty, true);
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
if (self.caps.focus_tracking) {
|
|
309
|
+
try self.setFocusTracking(tty, true);
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
if (!self.state.color_scheme_updates) {
|
|
313
|
+
try self.setColorSchemeUpdates(tty, true);
|
|
314
|
+
try tty.writeAll(ansi.ANSI.colorSchemeRequest);
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
fn checkEnvironmentOverrides(self: *Terminal) void {
|
|
319
|
+
self.in_tmux = false;
|
|
320
|
+
self.skip_graphics_query = false;
|
|
321
|
+
self.skip_explicit_width_query = false;
|
|
322
|
+
|
|
323
|
+
// Always just try to enable bracketed paste, even if it was reported as not supported
|
|
324
|
+
self.caps.bracketed_paste = true;
|
|
325
|
+
|
|
326
|
+
if (self.caps.rgb) {
|
|
327
|
+
self.caps.hyperlinks = true;
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
if (self.opts.remote) {
|
|
331
|
+
return;
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
var env_map_storage: ?std.process.EnvMap = null;
|
|
335
|
+
const env_map: *const std.process.EnvMap = self.opts.env_map orelse blk: {
|
|
336
|
+
env_map_storage = std.process.getEnvMap(std.heap.page_allocator) catch |err| {
|
|
337
|
+
logger.err("Failed to get environment map: {}", .{err});
|
|
338
|
+
return;
|
|
339
|
+
};
|
|
340
|
+
break :blk &env_map_storage.?;
|
|
341
|
+
};
|
|
342
|
+
defer if (env_map_storage) |*map| map.deinit();
|
|
343
|
+
|
|
344
|
+
if (!self.term_info.from_xtversion) {
|
|
345
|
+
if (env_map.get("TMUX")) |_| {
|
|
346
|
+
self.in_tmux = true;
|
|
347
|
+
self.caps.unicode = .wcwidth;
|
|
348
|
+
self.caps.explicit_cursor_positioning = true;
|
|
349
|
+
} else if (env_map.get("TERM")) |term| {
|
|
350
|
+
if (std.mem.startsWith(u8, term, "tmux")) {
|
|
351
|
+
self.in_tmux = true;
|
|
352
|
+
self.caps.unicode = .wcwidth;
|
|
353
|
+
self.caps.explicit_cursor_positioning = true;
|
|
354
|
+
} else if (std.mem.startsWith(u8, term, "screen")) {
|
|
355
|
+
self.skip_graphics_query = true;
|
|
356
|
+
self.caps.unicode = .wcwidth;
|
|
357
|
+
self.caps.explicit_cursor_positioning = true;
|
|
358
|
+
}
|
|
359
|
+
if (std.mem.indexOf(u8, term, "alacritty") != null) {
|
|
360
|
+
self.caps.explicit_cursor_positioning = true;
|
|
361
|
+
}
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
if (env_map.get("OPENTUI_GRAPHICS")) |val| {
|
|
366
|
+
if (std.mem.eql(u8, val, "false") or std.mem.eql(u8, val, "0")) {
|
|
367
|
+
self.skip_graphics_query = true;
|
|
368
|
+
} else if (std.mem.eql(u8, val, "true") or std.mem.eql(u8, val, "1")) {
|
|
369
|
+
self.skip_graphics_query = false;
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
if (!self.term_info.from_xtversion) {
|
|
374
|
+
if (env_map.get("TERM_PROGRAM")) |prog| {
|
|
375
|
+
const copy_len = @min(prog.len, self.term_info.name.len);
|
|
376
|
+
@memcpy(self.term_info.name[0..copy_len], prog[0..copy_len]);
|
|
377
|
+
self.term_info.name_len = copy_len;
|
|
378
|
+
|
|
379
|
+
if (env_map.get("TERM_PROGRAM_VERSION")) |ver| {
|
|
380
|
+
const ver_len = @min(ver.len, self.term_info.version.len);
|
|
381
|
+
@memcpy(self.term_info.version[0..ver_len], ver[0..ver_len]);
|
|
382
|
+
self.term_info.version_len = ver_len;
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
if (env_map.get("TERM_PROGRAM")) |prog| {
|
|
387
|
+
if (std.mem.eql(u8, prog, "vscode")) {
|
|
388
|
+
self.caps.kitty_keyboard = false;
|
|
389
|
+
self.caps.kitty_graphics = false;
|
|
390
|
+
self.caps.unicode = .unicode;
|
|
391
|
+
} else if (std.mem.eql(u8, prog, "Apple_Terminal")) {
|
|
392
|
+
self.caps.unicode = .wcwidth;
|
|
393
|
+
} else if (std.mem.eql(u8, prog, "Alacritty")) {
|
|
394
|
+
self.caps.explicit_cursor_positioning = true;
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
|
|
398
|
+
if (env_map.get("ALACRITTY_SOCKET") != null or env_map.get("ALACRITTY_LOG") != null) {
|
|
399
|
+
self.caps.explicit_cursor_positioning = true;
|
|
400
|
+
if (self.term_info.name_len == 0) {
|
|
401
|
+
const name = "Alacritty";
|
|
402
|
+
@memcpy(self.term_info.name[0..name.len], name);
|
|
403
|
+
self.term_info.name_len = name.len;
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
if (env_map.get("COLORTERM")) |colorterm| {
|
|
409
|
+
if (std.mem.eql(u8, colorterm, "truecolor") or
|
|
410
|
+
std.mem.eql(u8, colorterm, "24bit"))
|
|
411
|
+
{
|
|
412
|
+
self.caps.rgb = true;
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
if (!self.term_info.from_xtversion) {
|
|
417
|
+
if (env_map.get("TERMUX_VERSION")) |_| {
|
|
418
|
+
self.caps.unicode = .wcwidth;
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
if (env_map.get("VHS_RECORD")) |_| {
|
|
422
|
+
self.caps.unicode = .wcwidth;
|
|
423
|
+
self.caps.kitty_keyboard = false;
|
|
424
|
+
self.caps.kitty_graphics = false;
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
if (env_map.get("OPENTUI_FORCE_WCWIDTH")) |_| {
|
|
429
|
+
self.caps.unicode = .wcwidth;
|
|
430
|
+
}
|
|
431
|
+
if (env_map.get("OPENTUI_FORCE_UNICODE")) |_| {
|
|
432
|
+
self.caps.unicode = .unicode;
|
|
433
|
+
}
|
|
434
|
+
if (env_map.get("OPENTUI_FORCE_NOZWJ")) |_| {
|
|
435
|
+
self.caps.unicode = .no_zwj;
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
if (env_map.get("OPENTUI_FORCE_EXPLICIT_WIDTH")) |val| {
|
|
439
|
+
if (std.mem.eql(u8, val, "true") or std.mem.eql(u8, val, "1")) {
|
|
440
|
+
self.caps.explicit_width = true;
|
|
441
|
+
} else if (std.mem.eql(u8, val, "false") or std.mem.eql(u8, val, "0")) {
|
|
442
|
+
self.caps.explicit_width = false;
|
|
443
|
+
self.skip_explicit_width_query = true;
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
if (!self.caps.hyperlinks and self.term_info.from_xtversion) {
|
|
448
|
+
if (isHyperlinkTerm(self.getTerminalName())) {
|
|
449
|
+
self.caps.hyperlinks = true;
|
|
450
|
+
}
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
if (!self.caps.hyperlinks and !self.term_info.from_xtversion) {
|
|
454
|
+
if (env_map.get("TERM")) |term| {
|
|
455
|
+
if (isHyperlinkTerm(term)) {
|
|
456
|
+
self.caps.hyperlinks = true;
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
if (!self.caps.hyperlinks and !self.term_info.from_xtversion) {
|
|
462
|
+
const is_wsl = env_map.get("WSL_DISTRO_NAME") != null or env_map.get("WSL_INTEROP") != null;
|
|
463
|
+
const has_wt_session = env_map.get("WT_SESSION") != null;
|
|
464
|
+
if (is_wsl and has_wt_session) {
|
|
465
|
+
if (env_map.get("TERM")) |term| {
|
|
466
|
+
if (std.mem.startsWith(u8, term, "xterm")) {
|
|
467
|
+
self.caps.hyperlinks = true;
|
|
468
|
+
}
|
|
469
|
+
}
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
if (!self.caps.osc52 and !self.term_info.from_xtversion) {
|
|
474
|
+
if (env_map.get("WT_SESSION") != null) {
|
|
475
|
+
self.caps.osc52 = true;
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
if (!self.caps.osc52 and (self.in_tmux or env_map.get("STY") != null)) {
|
|
479
|
+
self.caps.osc52 = true;
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
if (!self.caps.osc52) {
|
|
483
|
+
if (env_map.get("TERM_PROGRAM")) |prog| {
|
|
484
|
+
if (isOsc52Term(prog)) {
|
|
485
|
+
self.caps.osc52 = true;
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
if (!self.caps.osc52) {
|
|
491
|
+
if (env_map.get("TERM")) |term| {
|
|
492
|
+
if (isOsc52Term(term) or std.mem.indexOf(u8, term, "256color") != null or std.mem.indexOf(u8, term, "xterm") != null) {
|
|
493
|
+
self.caps.osc52 = true;
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
// TODO: Allow pixel mouse mode to be enabled,
|
|
501
|
+
// currently does not make sense and is not supported by higher levels
|
|
502
|
+
pub fn setMouseMode(self: *Terminal, tty: anytype, enable: bool, enable_movement: bool) !void {
|
|
503
|
+
if (enable) {
|
|
504
|
+
if (self.state.mouse and self.state.mouse_movement == enable_movement) return;
|
|
505
|
+
} else if (!self.state.mouse) {
|
|
506
|
+
return;
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
if (enable) {
|
|
510
|
+
self.state.mouse = true;
|
|
511
|
+
self.state.mouse_movement = enable_movement;
|
|
512
|
+
if (!enable_movement) {
|
|
513
|
+
// Some terminals treat ?1000/?1002/?1003 as one family and let the
|
|
514
|
+
// last sequence win. Reset any-event tracking first, then enable
|
|
515
|
+
// click/drag modes so they remain active.
|
|
516
|
+
try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
|
|
517
|
+
}
|
|
518
|
+
try tty.writeAll(ansi.ANSI.enableMouseTracking);
|
|
519
|
+
try tty.writeAll(ansi.ANSI.enableButtonEventTracking);
|
|
520
|
+
if (enable_movement) {
|
|
521
|
+
try tty.writeAll(ansi.ANSI.enableAnyEventTracking);
|
|
522
|
+
}
|
|
523
|
+
try tty.writeAll(ansi.ANSI.enableSGRMouseMode);
|
|
524
|
+
} else {
|
|
525
|
+
self.state.mouse = false;
|
|
526
|
+
self.state.pixel_mouse = false;
|
|
527
|
+
try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
|
|
528
|
+
try tty.writeAll(ansi.ANSI.disableButtonEventTracking);
|
|
529
|
+
try tty.writeAll(ansi.ANSI.disableMouseTracking);
|
|
530
|
+
try tty.writeAll(ansi.ANSI.disableSGRMouseMode);
|
|
531
|
+
}
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
pub fn setBracketedPaste(self: *Terminal, tty: anytype, enable: bool) !void {
|
|
535
|
+
const seq = if (enable) ansi.ANSI.bracketedPasteSet else ansi.ANSI.bracketedPasteReset;
|
|
536
|
+
try tty.writeAll(seq);
|
|
537
|
+
self.state.bracketed_paste = enable;
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
pub fn setFocusTracking(self: *Terminal, tty: anytype, enable: bool) !void {
|
|
541
|
+
const seq = if (enable) ansi.ANSI.focusSet else ansi.ANSI.focusReset;
|
|
542
|
+
try tty.writeAll(seq);
|
|
543
|
+
self.state.focus_tracking = enable;
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
pub fn setKittyKeyboard(self: *Terminal, tty: anytype, enable: bool, flags: u8) !void {
|
|
547
|
+
if (enable) {
|
|
548
|
+
if (!self.state.kitty_keyboard) {
|
|
549
|
+
try tty.print(ansi.ANSI.csiUPush, .{flags});
|
|
550
|
+
self.state.kitty_keyboard = true;
|
|
551
|
+
self.state.kitty_keyboard_flags = flags;
|
|
552
|
+
}
|
|
553
|
+
} else {
|
|
554
|
+
if (self.state.kitty_keyboard) {
|
|
555
|
+
try tty.writeAll(ansi.ANSI.csiUPop);
|
|
556
|
+
self.state.kitty_keyboard = false;
|
|
557
|
+
self.state.kitty_keyboard_flags = 0;
|
|
558
|
+
}
|
|
559
|
+
}
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
pub fn setModifyOtherKeys(self: *Terminal, tty: anytype, enable: bool) !void {
|
|
563
|
+
const seq = if (enable) ansi.ANSI.modifyOtherKeysSet else ansi.ANSI.modifyOtherKeysReset;
|
|
564
|
+
try tty.writeAll(seq);
|
|
565
|
+
self.state.modify_other_keys = enable;
|
|
566
|
+
}
|
|
567
|
+
|
|
568
|
+
pub fn setColorSchemeUpdates(self: *Terminal, tty: anytype, enable: bool) !void {
|
|
569
|
+
const seq = if (enable) ansi.ANSI.colorSchemeSet else ansi.ANSI.colorSchemeReset;
|
|
570
|
+
try tty.writeAll(seq);
|
|
571
|
+
self.state.color_scheme_updates = enable;
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
/// Re-send all currently-active terminal mode escape sequences unconditionally.
|
|
575
|
+
///
|
|
576
|
+
/// When the terminal loses and regains focus (e.g. alt-tab, tab switch, minimize),
|
|
577
|
+
/// some terminal emulators (notably Windows Terminal / ConPTY) strip or reset
|
|
578
|
+
/// DEC private modes like mouse tracking (?1000/?1002/?1003/?1006), focus
|
|
579
|
+
/// tracking (?1004), and bracketed paste (?2004). This function re-emits the
|
|
580
|
+
/// enable sequences for every mode that our state tracking says is currently on,
|
|
581
|
+
/// without checking whether the mode "should" already be enabled — because the
|
|
582
|
+
/// terminal may have silently disabled it.
|
|
583
|
+
///
|
|
584
|
+
/// This should be called in response to the focus-in event (\x1b[I).
|
|
585
|
+
///
|
|
586
|
+
/// Per the xterm ctlseqs spec (Patch #401, 2025/06/22) and the Microsoft
|
|
587
|
+
/// Console Virtual Terminal Sequences documentation, the relevant DECSET
|
|
588
|
+
/// private modes are:
|
|
589
|
+
/// ?1000h - Normal mouse tracking (sends button press/release)
|
|
590
|
+
/// ?1002h - Button-event tracking (adds drag reporting)
|
|
591
|
+
/// ?1003h - Any-event tracking (adds all motion reporting)
|
|
592
|
+
/// ?1006h - SGR extended mouse mode (extended coordinate encoding)
|
|
593
|
+
/// ?1004h - Focus event tracking (sends \x1b[I / \x1b[O)
|
|
594
|
+
/// ?2004h - Bracketed paste mode (wraps pasted text in markers)
|
|
595
|
+
/// Kitty keyboard protocol (CSI > flags u) - progressive enhancement
|
|
596
|
+
/// modifyOtherKeys (CSI > 4 ; 1 m) - xterm key modification
|
|
597
|
+
pub fn restoreTerminalModes(self: *Terminal, tty: anytype) !void {
|
|
598
|
+
// Re-enable mouse tracking modes if active
|
|
599
|
+
if (self.state.mouse) {
|
|
600
|
+
if (!self.state.mouse_movement) {
|
|
601
|
+
try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
|
|
602
|
+
}
|
|
603
|
+
try tty.writeAll(ansi.ANSI.enableMouseTracking);
|
|
604
|
+
try tty.writeAll(ansi.ANSI.enableButtonEventTracking);
|
|
605
|
+
if (self.state.mouse_movement) {
|
|
606
|
+
try tty.writeAll(ansi.ANSI.enableAnyEventTracking);
|
|
607
|
+
}
|
|
608
|
+
try tty.writeAll(ansi.ANSI.enableSGRMouseMode);
|
|
609
|
+
}
|
|
610
|
+
|
|
611
|
+
// Re-enable focus tracking if active
|
|
612
|
+
if (self.state.focus_tracking) {
|
|
613
|
+
try tty.writeAll(ansi.ANSI.focusSet);
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
// Re-enable bracketed paste if active
|
|
617
|
+
if (self.state.bracketed_paste) {
|
|
618
|
+
try tty.writeAll(ansi.ANSI.bracketedPasteSet);
|
|
619
|
+
}
|
|
620
|
+
|
|
621
|
+
// Pop stale entry then re-push kitty keyboard protocol to avoid stack growth.
|
|
622
|
+
// Both sequences are in the same write buffer so the terminal processes them atomically.
|
|
623
|
+
if (self.state.kitty_keyboard) {
|
|
624
|
+
try tty.writeAll(ansi.ANSI.csiUPop);
|
|
625
|
+
try tty.print(ansi.ANSI.csiUPush, .{self.state.kitty_keyboard_flags});
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
// Re-enable modifyOtherKeys if active
|
|
629
|
+
if (self.state.modify_other_keys) {
|
|
630
|
+
try tty.writeAll(ansi.ANSI.modifyOtherKeysSet);
|
|
631
|
+
}
|
|
632
|
+
}
|
|
633
|
+
|
|
634
|
+
/// The responses look like these:
|
|
635
|
+
/// kitty - '\x1B[?1016;2$y\x1B[?2027;0$y\x1B[?2031;2$y\x1B[?1004;1$y\x1B[?2026;2$y\x1B[1;2R\x1B[1;3R\x1BP>|kitty(0.40.1)\x1B\\\x1B[?0u\x1B_Gi=1;EINVAL:Zero width/height not allowed\x1B\\\x1B[?62;c'
|
|
636
|
+
/// ghostty - '\x1B[?1016;1$y\x1B[?2027;1$y\x1B[?2031;2$y\x1B[?1004;1$y\x1B[?2004;2$y\x1B[?2026;2$y\x1B[1;1R\x1B[1;1R\x1BP>|ghostty 1.1.3\x1B\\\x1B[?0u\x1B_Gi=1;OK\x1B\\\x1B[?62;22c'
|
|
637
|
+
/// tmux - '\x1B[1;1R\x1B[1;1R\x1BP>|tmux 3.5a\x1B\\\x1B[?1;2;4c\x1B[?2;3;0S'
|
|
638
|
+
/// vscode - '\x1B[?1016;2$y'
|
|
639
|
+
/// alacritty - '\x1B[?1016;0$y\x1B[?2027;0$y\x1B[?2031;0$y\x1B[?1004;2$y\x1B[?2004;2$y\x1B[?2026;2$y\x1B[1;1R\x1B[1;1R\x1B[?0u\x1B[?6c'
|
|
640
|
+
///
|
|
641
|
+
/// Parsing these is not complete yet
|
|
642
|
+
pub fn processCapabilityResponse(self: *Terminal, response: []const u8) void {
|
|
643
|
+
// DECRPM responses
|
|
644
|
+
if (std.mem.indexOf(u8, response, "1016;2$y")) |_| {
|
|
645
|
+
self.caps.sgr_pixels = true;
|
|
646
|
+
}
|
|
647
|
+
if (std.mem.indexOf(u8, response, "2027;2$y")) |_| {
|
|
648
|
+
self.caps.unicode = .unicode;
|
|
649
|
+
}
|
|
650
|
+
if (std.mem.indexOf(u8, response, "2031;1$y") != null or std.mem.indexOf(u8, response, "2031;2$y") != null) {
|
|
651
|
+
self.caps.color_scheme_updates = true;
|
|
652
|
+
}
|
|
653
|
+
if (std.mem.indexOf(u8, response, "1004;1$y") != null or std.mem.indexOf(u8, response, "1004;2$y") != null) {
|
|
654
|
+
self.caps.focus_tracking = true;
|
|
655
|
+
}
|
|
656
|
+
if (std.mem.indexOf(u8, response, "2026;1$y") != null or std.mem.indexOf(u8, response, "2026;2$y") != null) {
|
|
657
|
+
self.caps.sync = true;
|
|
658
|
+
}
|
|
659
|
+
if (std.mem.indexOf(u8, response, "2004;1$y") != null or std.mem.indexOf(u8, response, "2004;2$y") != null) {
|
|
660
|
+
self.caps.bracketed_paste = true;
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
// Explicit width detection - cursor position report [1;NR where N >= 2 means explicit width supported
|
|
664
|
+
// We look for ESC[1; followed by a digit >= 2
|
|
665
|
+
// This handles cases where the cursor isn't at exact home position when queries are sent
|
|
666
|
+
if (std.mem.indexOf(u8, response, "\x1b[1;")) |pos| {
|
|
667
|
+
const after = response[pos + 4 ..];
|
|
668
|
+
if (after.len > 0) {
|
|
669
|
+
var end: usize = 0;
|
|
670
|
+
while (end < after.len and after[end] >= '0' and after[end] <= '9') : (end += 1) {}
|
|
671
|
+
if (end > 0 and end < after.len and after[end] == 'R') {
|
|
672
|
+
const col = std.fmt.parseInt(u16, after[0..end], 10) catch 0;
|
|
673
|
+
if (col >= 2) {
|
|
674
|
+
self.caps.explicit_width = true;
|
|
675
|
+
}
|
|
676
|
+
if (col >= 3) {
|
|
677
|
+
self.caps.scaled_text = true;
|
|
678
|
+
}
|
|
679
|
+
}
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
|
|
683
|
+
// Parse xtversion response: ESC P > | name version ESC \
|
|
684
|
+
// Examples: "\x1BP>|kitty(0.40.1)\x1B\\" or "\x1BP>|ghostty 1.1.3\x1B\\" or "\x1BP>|tmux 3.5a\x1B\\"
|
|
685
|
+
if (std.mem.indexOf(u8, response, "\x1bP>|")) |pos| {
|
|
686
|
+
const start = pos + 4; // Skip past "\x1BP>|"
|
|
687
|
+
if (std.mem.indexOf(u8, response[start..], "\x1b\\")) |end_offset| {
|
|
688
|
+
const term_str = response[start .. start + end_offset];
|
|
689
|
+
self.parseXtversion(term_str);
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
|
|
693
|
+
// Kitty detection
|
|
694
|
+
if (std.mem.indexOf(u8, response, "kitty")) |_| {
|
|
695
|
+
self.caps.kitty_keyboard = true;
|
|
696
|
+
self.caps.kitty_graphics = true;
|
|
697
|
+
self.caps.unicode = .unicode;
|
|
698
|
+
self.caps.rgb = true;
|
|
699
|
+
self.caps.sixel = true;
|
|
700
|
+
self.caps.bracketed_paste = true;
|
|
701
|
+
self.caps.hyperlinks = true;
|
|
702
|
+
}
|
|
703
|
+
|
|
704
|
+
// Kitty keyboard protocol detection via CSI ? u response
|
|
705
|
+
// Terminals supporting the protocol respond to CSI ? u with CSI ? <flags> u
|
|
706
|
+
// Examples: \x1b[?0u (ghostty, alacritty), \x1b[?1u, etc.
|
|
707
|
+
if (std.mem.indexOf(u8, response, "\x1b[?") != null and std.mem.indexOf(u8, response, "u") != null) {
|
|
708
|
+
// Look for pattern \x1b[?Nu where N is 0-31
|
|
709
|
+
var i: usize = 0;
|
|
710
|
+
while (i + 4 < response.len) : (i += 1) {
|
|
711
|
+
if (response[i] == '\x1b' and i + 1 < response.len and response[i + 1] == '[' and i + 2 < response.len and response[i + 2] == '?') {
|
|
712
|
+
var num_end = i + 3;
|
|
713
|
+
while (num_end < response.len and response[num_end] >= '0' and response[num_end] <= '9') : (num_end += 1) {}
|
|
714
|
+
if (num_end > i + 3 and num_end < response.len and response[num_end] == 'u') {
|
|
715
|
+
self.caps.kitty_keyboard = true;
|
|
716
|
+
break;
|
|
717
|
+
}
|
|
718
|
+
}
|
|
719
|
+
}
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
if (std.mem.indexOf(u8, response, "tmux")) |_| {
|
|
723
|
+
self.caps.unicode = .wcwidth;
|
|
724
|
+
self.caps.explicit_cursor_positioning = true;
|
|
725
|
+
}
|
|
726
|
+
|
|
727
|
+
if (std.mem.indexOf(u8, response, "alacritty")) |_| {
|
|
728
|
+
self.caps.explicit_cursor_positioning = true;
|
|
729
|
+
}
|
|
730
|
+
|
|
731
|
+
// Sixel detection via device attributes (capability 4 in DA1 response ending with 'c')
|
|
732
|
+
if (std.mem.indexOf(u8, response, ";c")) |pos| {
|
|
733
|
+
var start: usize = 0;
|
|
734
|
+
if (pos >= 4) {
|
|
735
|
+
start = pos;
|
|
736
|
+
while (start > 0 and response[start] != '\x1b') {
|
|
737
|
+
start -= 1;
|
|
738
|
+
}
|
|
739
|
+
|
|
740
|
+
const da_response = response[start .. pos + 2];
|
|
741
|
+
|
|
742
|
+
if (std.mem.indexOf(u8, da_response, "\x1b[?") == 0) {
|
|
743
|
+
if (std.mem.indexOf(u8, da_response, "4;") != null or std.mem.indexOf(u8, da_response, ";4;") != null or std.mem.indexOf(u8, da_response, ";4c") != null) {
|
|
744
|
+
self.caps.sixel = true;
|
|
745
|
+
}
|
|
746
|
+
}
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
// Kitty graphics response: ESC_Gi=31337;OK ESC\ or ESC_Gi=31337;EERROR... ESC\
|
|
751
|
+
// We look for our specific query ID (31337) to avoid false positives
|
|
752
|
+
if (std.mem.indexOf(u8, response, "\x1b_G")) |_| {
|
|
753
|
+
if (std.mem.indexOf(u8, response, "i=31337")) |_| {
|
|
754
|
+
// Got a response to our graphics query with our ID
|
|
755
|
+
// If it contains "OK" or even an error, the protocol is supported
|
|
756
|
+
// (errors mean the query was understood, just parameters were wrong)
|
|
757
|
+
self.caps.kitty_graphics = true;
|
|
758
|
+
}
|
|
759
|
+
}
|
|
760
|
+
|
|
761
|
+
if (!self.caps.osc52 and isOsc52Term(response)) {
|
|
762
|
+
self.caps.osc52 = true;
|
|
763
|
+
}
|
|
764
|
+
|
|
765
|
+
if (!self.caps.hyperlinks and isHyperlinkTerm(response)) {
|
|
766
|
+
self.caps.hyperlinks = true;
|
|
767
|
+
}
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
fn isOsc52Term(value: []const u8) bool {
|
|
771
|
+
return std.ascii.indexOfIgnoreCase(value, "iterm") != null or
|
|
772
|
+
std.ascii.indexOfIgnoreCase(value, "kitty") != null or
|
|
773
|
+
std.ascii.indexOfIgnoreCase(value, "alacritty") != null or
|
|
774
|
+
std.ascii.indexOfIgnoreCase(value, "wezterm") != null or
|
|
775
|
+
std.ascii.indexOfIgnoreCase(value, "contour") != null or
|
|
776
|
+
std.ascii.indexOfIgnoreCase(value, "foot") != null or
|
|
777
|
+
std.ascii.indexOfIgnoreCase(value, "rio") != null or
|
|
778
|
+
std.ascii.indexOfIgnoreCase(value, "ghostty") != null or
|
|
779
|
+
std.ascii.indexOfIgnoreCase(value, "tmux") != null or
|
|
780
|
+
std.ascii.indexOfIgnoreCase(value, "screen") != null;
|
|
781
|
+
}
|
|
782
|
+
|
|
783
|
+
fn isHyperlinkTerm(value: []const u8) bool {
|
|
784
|
+
return std.ascii.indexOfIgnoreCase(value, "ghostty") != null or
|
|
785
|
+
std.ascii.indexOfIgnoreCase(value, "kitty") != null or
|
|
786
|
+
std.ascii.indexOfIgnoreCase(value, "wezterm") != null or
|
|
787
|
+
std.ascii.indexOfIgnoreCase(value, "alacritty") != null or
|
|
788
|
+
std.ascii.indexOfIgnoreCase(value, "iterm") != null;
|
|
789
|
+
}
|
|
790
|
+
|
|
791
|
+
pub fn getCapabilities(self: *Terminal) Capabilities {
|
|
792
|
+
return self.caps;
|
|
793
|
+
}
|
|
794
|
+
|
|
795
|
+
pub fn setMousePointerStyle(self: *Terminal, style: MousePointerStyle) void {
|
|
796
|
+
self.state.mouse_pointer = style;
|
|
797
|
+
}
|
|
798
|
+
|
|
799
|
+
pub fn getMousePointer(self: *Terminal) MousePointerStyle {
|
|
800
|
+
return self.state.mouse_pointer;
|
|
801
|
+
}
|
|
802
|
+
|
|
803
|
+
pub fn setCursorPosition(self: *Terminal, x: u32, y: u32, visible: bool) void {
|
|
804
|
+
self.state.cursor.x = @max(1, x);
|
|
805
|
+
self.state.cursor.y = @max(1, y);
|
|
806
|
+
self.state.cursor.visible = visible;
|
|
807
|
+
|
|
808
|
+
// Update 0-based coordinates for terminal operations
|
|
809
|
+
self.state.cursor.col = @intCast(@max(0, x - 1));
|
|
810
|
+
self.state.cursor.row = @intCast(@max(0, y - 1));
|
|
811
|
+
}
|
|
812
|
+
|
|
813
|
+
pub fn setCursorStyle(self: *Terminal, style: CursorStyle, blinking: bool) void {
|
|
814
|
+
self.state.cursor.style = style;
|
|
815
|
+
self.state.cursor.blinking = blinking;
|
|
816
|
+
}
|
|
817
|
+
|
|
818
|
+
pub fn setCursorColor(self: *Terminal, color: [4]f32) void {
|
|
819
|
+
self.state.cursor.color = color;
|
|
820
|
+
}
|
|
821
|
+
|
|
822
|
+
pub fn getCursorPosition(self: *Terminal) struct { x: u32, y: u32, visible: bool } {
|
|
823
|
+
return .{
|
|
824
|
+
.x = self.state.cursor.x,
|
|
825
|
+
.y = self.state.cursor.y,
|
|
826
|
+
.visible = self.state.cursor.visible,
|
|
827
|
+
};
|
|
828
|
+
}
|
|
829
|
+
|
|
830
|
+
pub fn getCursorStyle(self: *Terminal) struct { style: CursorStyle, blinking: bool } {
|
|
831
|
+
return .{
|
|
832
|
+
.style = self.state.cursor.style,
|
|
833
|
+
.blinking = self.state.cursor.blinking,
|
|
834
|
+
};
|
|
835
|
+
}
|
|
836
|
+
|
|
837
|
+
pub fn getCursorColor(self: *Terminal) [4]f32 {
|
|
838
|
+
return self.state.cursor.color;
|
|
839
|
+
}
|
|
840
|
+
|
|
841
|
+
pub fn setKittyKeyboardFlags(self: *Terminal, flags: u8) void {
|
|
842
|
+
self.opts.kitty_keyboard_flags = flags;
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
pub fn setTerminalTitle(_: *Terminal, tty: anytype, title: []const u8) void {
|
|
846
|
+
// For Windows, we might need to use different approach, but ANSI sequences work in Windows Terminal, ConPTY, etc.
|
|
847
|
+
// For other platforms, ANSI OSC sequences work reliably
|
|
848
|
+
ansi.ANSI.setTerminalTitleOutput(tty, title) catch {};
|
|
849
|
+
}
|
|
850
|
+
|
|
851
|
+
/// Write OSC 52 clipboard sequence to the terminal
|
|
852
|
+
/// Supports tmux/screen passthrough, including nested tmux sessions
|
|
853
|
+
pub fn writeClipboard(self: *Terminal, tty: anytype, target: ClipboardTarget, payload: []const u8) !void {
|
|
854
|
+
if (!self.canWriteClipboard()) {
|
|
855
|
+
return error.NotSupported;
|
|
856
|
+
}
|
|
857
|
+
|
|
858
|
+
var buf: [1024]u8 = undefined;
|
|
859
|
+
var stream = std.io.fixedBufferStream(&buf);
|
|
860
|
+
const writer = stream.writer();
|
|
861
|
+
|
|
862
|
+
// Build OSC 52 sequence: ESC]52;<target>;<payload>ESC\
|
|
863
|
+
try writer.writeAll("\x1b]52;");
|
|
864
|
+
try writer.writeByte(target.toChar());
|
|
865
|
+
try writer.writeByte(';');
|
|
866
|
+
try writer.writeAll(payload);
|
|
867
|
+
try writer.writeAll("\x1b\\");
|
|
868
|
+
|
|
869
|
+
const osc52 = stream.getWritten();
|
|
870
|
+
|
|
871
|
+
// Use self.in_tmux which is set by checkEnvironmentOverrides() considering
|
|
872
|
+
// env vars, xtversion response, and remote option
|
|
873
|
+
const is_tmux = self.in_tmux or self.isXtversionTmux();
|
|
874
|
+
|
|
875
|
+
if (is_tmux) {
|
|
876
|
+
// For nested tmux, we use a fixed level of 1 as we don't have access
|
|
877
|
+
// to env vars here (by design - detection already happened in checkEnvironmentOverrides)
|
|
878
|
+
// In practice, single-level wrapping works for most cases
|
|
879
|
+
var wrapped_buf: [4096]u8 = undefined;
|
|
880
|
+
var wrapped_stream = std.io.fixedBufferStream(&wrapped_buf);
|
|
881
|
+
const wrap_writer = wrapped_stream.writer();
|
|
882
|
+
for (osc52) |c| {
|
|
883
|
+
if (c == '\x1b') {
|
|
884
|
+
try wrap_writer.writeByte('\x1b');
|
|
885
|
+
}
|
|
886
|
+
try wrap_writer.writeByte(c);
|
|
887
|
+
}
|
|
888
|
+
const doubled = wrapped_stream.getWritten();
|
|
889
|
+
|
|
890
|
+
try tty.writeAll(ansi.ANSI.tmuxDcsStart);
|
|
891
|
+
try tty.writeAll(doubled);
|
|
892
|
+
try tty.writeAll(ansi.ANSI.tmuxDcsEnd);
|
|
893
|
+
} else if (self.opts.remote) {
|
|
894
|
+
try tty.writeAll(osc52);
|
|
895
|
+
} else {
|
|
896
|
+
var env_map_storage: ?std.process.EnvMap = null;
|
|
897
|
+
const env_map: *const std.process.EnvMap = self.opts.env_map orelse blk: {
|
|
898
|
+
env_map_storage = std.process.getEnvMap(std.heap.page_allocator) catch |err| {
|
|
899
|
+
logger.err("Failed to get environment map: {}", .{err});
|
|
900
|
+
return;
|
|
901
|
+
};
|
|
902
|
+
break :blk &env_map_storage.?;
|
|
903
|
+
};
|
|
904
|
+
defer if (env_map_storage) |*map| map.deinit();
|
|
905
|
+
|
|
906
|
+
if (env_map.get("STY")) |_| {
|
|
907
|
+
var wrapped_buf: [2048]u8 = undefined;
|
|
908
|
+
var wrapped_stream = std.io.fixedBufferStream(&wrapped_buf);
|
|
909
|
+
const wrapped_writer = wrapped_stream.writer();
|
|
910
|
+
|
|
911
|
+
for (osc52) |c| {
|
|
912
|
+
if (c == '\x1b') {
|
|
913
|
+
try wrapped_writer.writeByte('\x1b');
|
|
914
|
+
}
|
|
915
|
+
try wrapped_writer.writeByte(c);
|
|
916
|
+
}
|
|
917
|
+
const doubled = wrapped_stream.getWritten();
|
|
918
|
+
|
|
919
|
+
try tty.writeAll(ansi.ANSI.screenDcsStart);
|
|
920
|
+
try tty.writeAll(doubled);
|
|
921
|
+
try tty.writeAll(ansi.ANSI.screenDcsEnd);
|
|
922
|
+
} else {
|
|
923
|
+
try tty.writeAll(osc52);
|
|
924
|
+
}
|
|
925
|
+
}
|
|
926
|
+
}
|
|
927
|
+
|
|
928
|
+
/// Check if we can write to the clipboard (TTY and OSC 52 supported)
|
|
929
|
+
fn canWriteClipboard(self: *Terminal) bool {
|
|
930
|
+
// In a real TTY environment, we'd check isTTY here
|
|
931
|
+
// For now, we just check if OSC 52 is supported
|
|
932
|
+
return self.caps.osc52;
|
|
933
|
+
}
|
|
934
|
+
|
|
935
|
+
/// Parse xtversion response string and extract terminal name and version
|
|
936
|
+
/// Examples: "kitty(0.40.1)", "ghostty 1.1.3", "tmux 3.5a"
|
|
937
|
+
fn parseXtversion(self: *Terminal, term_str: []const u8) void {
|
|
938
|
+
if (term_str.len == 0) return;
|
|
939
|
+
|
|
940
|
+
if (std.mem.indexOf(u8, term_str, "(")) |paren_pos| {
|
|
941
|
+
const name_len = @min(paren_pos, self.term_info.name.len);
|
|
942
|
+
@memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
|
|
943
|
+
self.term_info.name_len = name_len;
|
|
944
|
+
|
|
945
|
+
if (std.mem.indexOf(u8, term_str[paren_pos..], ")")) |close_offset| {
|
|
946
|
+
const ver_start = paren_pos + 1;
|
|
947
|
+
const ver_end = paren_pos + close_offset;
|
|
948
|
+
const ver_len = @min(ver_end - ver_start, self.term_info.version.len);
|
|
949
|
+
@memcpy(self.term_info.version[0..ver_len], term_str[ver_start .. ver_start + ver_len]);
|
|
950
|
+
self.term_info.version_len = ver_len;
|
|
951
|
+
}
|
|
952
|
+
} else {
|
|
953
|
+
if (std.mem.indexOf(u8, term_str, " ")) |space_pos| {
|
|
954
|
+
const name_len = @min(space_pos, self.term_info.name.len);
|
|
955
|
+
@memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
|
|
956
|
+
self.term_info.name_len = name_len;
|
|
957
|
+
|
|
958
|
+
const ver_start = space_pos + 1;
|
|
959
|
+
const ver_len = @min(term_str.len - ver_start, self.term_info.version.len);
|
|
960
|
+
@memcpy(self.term_info.version[0..ver_len], term_str[ver_start .. ver_start + ver_len]);
|
|
961
|
+
self.term_info.version_len = ver_len;
|
|
962
|
+
} else {
|
|
963
|
+
const name_len = @min(term_str.len, self.term_info.name.len);
|
|
964
|
+
@memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
|
|
965
|
+
self.term_info.name_len = name_len;
|
|
966
|
+
self.term_info.version_len = 0;
|
|
967
|
+
}
|
|
968
|
+
}
|
|
969
|
+
|
|
970
|
+
self.term_info.from_xtversion = true;
|
|
971
|
+
}
|
|
972
|
+
|
|
973
|
+
pub fn isXtversionTmux(self: *Terminal) bool {
|
|
974
|
+
return self.term_info.from_xtversion and std.mem.eql(u8, self.getTerminalName(), "tmux");
|
|
975
|
+
}
|
|
976
|
+
|
|
977
|
+
pub fn getTerminalInfo(self: *Terminal) TerminalInfo {
|
|
978
|
+
return self.term_info;
|
|
979
|
+
}
|
|
980
|
+
|
|
981
|
+
pub fn getTerminalName(self: *Terminal) []const u8 {
|
|
982
|
+
return self.term_info.name[0..self.term_info.name_len];
|
|
983
|
+
}
|
|
984
|
+
|
|
985
|
+
pub fn getTerminalVersion(self: *Terminal) []const u8 {
|
|
986
|
+
return self.term_info.version[0..self.term_info.version_len];
|
|
987
|
+
}
|