@fairyhunter13/opentui-core 0.1.112 → 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 +63 -51
- 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-8fks7yv1.js +0 -411
- package/index-8fks7yv1.js.map +0 -10
- package/index-egy5e2rs.js +0 -12267
- package/index-egy5e2rs.js.map +0 -42
- package/index-tse8gzh0.js +0 -20614
- package/index-tse8gzh0.js.map +0 -67
- 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,1378 @@
|
|
|
1
|
+
import { describe, test, expect } from "bun:test"
|
|
2
|
+
import { createMockKeys, KeyCodes } from "./mock-keys.js"
|
|
3
|
+
import { PassThrough } from "stream"
|
|
4
|
+
|
|
5
|
+
class MockRenderer {
|
|
6
|
+
public stdin: PassThrough
|
|
7
|
+
public emittedData: Buffer[] = []
|
|
8
|
+
|
|
9
|
+
constructor() {
|
|
10
|
+
this.stdin = new PassThrough()
|
|
11
|
+
|
|
12
|
+
this.stdin.on("data", (chunk: Buffer) => {
|
|
13
|
+
this.emittedData.push(chunk)
|
|
14
|
+
})
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
getEmittedData(): string {
|
|
18
|
+
return Buffer.concat(this.emittedData).toString()
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
describe("mock-keys", () => {
|
|
23
|
+
test("pressKeys with string keys", () => {
|
|
24
|
+
const mockRenderer = new MockRenderer()
|
|
25
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
26
|
+
|
|
27
|
+
mockKeys.pressKeys(["h", "e", "l", "l", "o"])
|
|
28
|
+
|
|
29
|
+
expect(mockRenderer.getEmittedData()).toBe("hello")
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
test("pressKeys with KeyCodes", () => {
|
|
33
|
+
const mockRenderer = new MockRenderer()
|
|
34
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
35
|
+
|
|
36
|
+
mockKeys.pressKeys([KeyCodes.RETURN, KeyCodes.TAB])
|
|
37
|
+
|
|
38
|
+
expect(mockRenderer.getEmittedData()).toBe("\r\t")
|
|
39
|
+
})
|
|
40
|
+
|
|
41
|
+
test("pressKey with string", () => {
|
|
42
|
+
const mockRenderer = new MockRenderer()
|
|
43
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
44
|
+
|
|
45
|
+
mockKeys.pressKey("a")
|
|
46
|
+
|
|
47
|
+
expect(mockRenderer.getEmittedData()).toBe("a")
|
|
48
|
+
})
|
|
49
|
+
|
|
50
|
+
test("pressKey with KeyCode", () => {
|
|
51
|
+
const mockRenderer = new MockRenderer()
|
|
52
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
53
|
+
|
|
54
|
+
mockKeys.pressKey(KeyCodes.ESCAPE)
|
|
55
|
+
|
|
56
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b")
|
|
57
|
+
})
|
|
58
|
+
|
|
59
|
+
test("typeText", () => {
|
|
60
|
+
const mockRenderer = new MockRenderer()
|
|
61
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
62
|
+
|
|
63
|
+
mockKeys.typeText("hello world")
|
|
64
|
+
|
|
65
|
+
expect(mockRenderer.getEmittedData()).toBe("hello world")
|
|
66
|
+
})
|
|
67
|
+
|
|
68
|
+
test("convenience methods", () => {
|
|
69
|
+
const mockRenderer = new MockRenderer()
|
|
70
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
71
|
+
|
|
72
|
+
mockKeys.pressEnter()
|
|
73
|
+
mockKeys.pressEscape()
|
|
74
|
+
mockKeys.pressTab()
|
|
75
|
+
mockKeys.pressBackspace()
|
|
76
|
+
|
|
77
|
+
expect(mockRenderer.getEmittedData()).toBe("\r\x1b\t\b")
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
test("pressArrow", () => {
|
|
81
|
+
const mockRenderer = new MockRenderer()
|
|
82
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
83
|
+
|
|
84
|
+
mockKeys.pressArrow("up")
|
|
85
|
+
mockKeys.pressArrow("down")
|
|
86
|
+
mockKeys.pressArrow("left")
|
|
87
|
+
mockKeys.pressArrow("right")
|
|
88
|
+
|
|
89
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[A\x1b[B\x1b[D\x1b[C")
|
|
90
|
+
})
|
|
91
|
+
|
|
92
|
+
test("pressCtrlC", () => {
|
|
93
|
+
const mockRenderer = new MockRenderer()
|
|
94
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
95
|
+
|
|
96
|
+
mockKeys.pressCtrlC()
|
|
97
|
+
|
|
98
|
+
expect(mockRenderer.getEmittedData()).toBe("\x03")
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
test("arbitrary string keys work", () => {
|
|
102
|
+
const mockRenderer = new MockRenderer()
|
|
103
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
104
|
+
|
|
105
|
+
mockKeys.pressKey("x")
|
|
106
|
+
mockKeys.pressKey("y")
|
|
107
|
+
mockKeys.pressKey("z")
|
|
108
|
+
|
|
109
|
+
expect(mockRenderer.getEmittedData()).toBe("xyz")
|
|
110
|
+
})
|
|
111
|
+
|
|
112
|
+
test("KeyCodes enum values work", () => {
|
|
113
|
+
const mockRenderer = new MockRenderer()
|
|
114
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
115
|
+
|
|
116
|
+
mockKeys.pressKey(KeyCodes.RETURN)
|
|
117
|
+
mockKeys.pressKey(KeyCodes.TAB)
|
|
118
|
+
mockKeys.pressKey(KeyCodes.ESCAPE)
|
|
119
|
+
|
|
120
|
+
expect(mockRenderer.getEmittedData()).toBe("\r\t\x1b")
|
|
121
|
+
})
|
|
122
|
+
|
|
123
|
+
test("data events are properly emitted", () => {
|
|
124
|
+
const mockRenderer = new MockRenderer()
|
|
125
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
126
|
+
|
|
127
|
+
const receivedData: Buffer[] = []
|
|
128
|
+
mockRenderer.stdin.on("data", (chunk: Buffer) => {
|
|
129
|
+
receivedData.push(chunk)
|
|
130
|
+
})
|
|
131
|
+
|
|
132
|
+
mockKeys.pressKey("a")
|
|
133
|
+
mockKeys.pressKey(KeyCodes.RETURN)
|
|
134
|
+
|
|
135
|
+
expect(receivedData).toHaveLength(2)
|
|
136
|
+
expect(receivedData[0].toString()).toBe("a")
|
|
137
|
+
expect(receivedData[1].toString()).toBe("\r")
|
|
138
|
+
})
|
|
139
|
+
|
|
140
|
+
test("multiple data events accumulate correctly", () => {
|
|
141
|
+
const mockRenderer = new MockRenderer()
|
|
142
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
143
|
+
|
|
144
|
+
const receivedData: string[] = []
|
|
145
|
+
mockRenderer.stdin.on("data", (chunk: Buffer) => {
|
|
146
|
+
receivedData.push(chunk.toString())
|
|
147
|
+
})
|
|
148
|
+
|
|
149
|
+
mockKeys.typeText("hello")
|
|
150
|
+
mockKeys.pressEnter()
|
|
151
|
+
|
|
152
|
+
expect(receivedData).toEqual(["h", "e", "l", "l", "o", "\r"])
|
|
153
|
+
})
|
|
154
|
+
|
|
155
|
+
test("stream write method emits data events correctly", () => {
|
|
156
|
+
const mockRenderer = new MockRenderer()
|
|
157
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
158
|
+
|
|
159
|
+
const emittedChunks: Buffer[] = []
|
|
160
|
+
mockRenderer.stdin.on("data", (chunk: Buffer) => {
|
|
161
|
+
emittedChunks.push(chunk)
|
|
162
|
+
})
|
|
163
|
+
|
|
164
|
+
// Directly test the stream write method that mock-keys uses
|
|
165
|
+
mockRenderer.stdin.write("test")
|
|
166
|
+
mockRenderer.stdin.write(KeyCodes.RETURN)
|
|
167
|
+
|
|
168
|
+
expect(emittedChunks).toHaveLength(2)
|
|
169
|
+
expect(emittedChunks[0].toString()).toBe("test")
|
|
170
|
+
expect(emittedChunks[1].toString()).toBe("\r")
|
|
171
|
+
})
|
|
172
|
+
|
|
173
|
+
test("pressKeys with delay works", async () => {
|
|
174
|
+
const mockRenderer = new MockRenderer()
|
|
175
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
176
|
+
|
|
177
|
+
const timestamps: number[] = []
|
|
178
|
+
mockRenderer.stdin.on("data", () => {
|
|
179
|
+
timestamps.push(Date.now())
|
|
180
|
+
})
|
|
181
|
+
|
|
182
|
+
const startTime = Date.now()
|
|
183
|
+
await mockKeys.pressKeys(["a", "b"], 10) // 10ms delay between keys
|
|
184
|
+
const totalElapsed = Date.now() - startTime
|
|
185
|
+
|
|
186
|
+
expect(timestamps).toHaveLength(2)
|
|
187
|
+
expect(timestamps[1] - timestamps[0]).toBeGreaterThanOrEqual(8) // Allow some tolerance
|
|
188
|
+
expect(totalElapsed).toBeGreaterThanOrEqual(16)
|
|
189
|
+
})
|
|
190
|
+
|
|
191
|
+
test("pressKey with shift modifier", () => {
|
|
192
|
+
const mockRenderer = new MockRenderer()
|
|
193
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
194
|
+
|
|
195
|
+
mockKeys.pressKey(KeyCodes.ARROW_RIGHT, { shift: true })
|
|
196
|
+
|
|
197
|
+
// Arrow right with shift: \x1b[1;2C
|
|
198
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;2C")
|
|
199
|
+
})
|
|
200
|
+
|
|
201
|
+
test("pressKey with ctrl modifier", () => {
|
|
202
|
+
const mockRenderer = new MockRenderer()
|
|
203
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
204
|
+
|
|
205
|
+
mockKeys.pressKey(KeyCodes.ARROW_LEFT, { ctrl: true })
|
|
206
|
+
|
|
207
|
+
// Arrow left with ctrl: \x1b[1;5D (1 base + 4 ctrl = 5)
|
|
208
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;5D")
|
|
209
|
+
})
|
|
210
|
+
|
|
211
|
+
test("pressKey with shift+ctrl modifiers", () => {
|
|
212
|
+
const mockRenderer = new MockRenderer()
|
|
213
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
214
|
+
|
|
215
|
+
mockKeys.pressKey(KeyCodes.ARROW_UP, { shift: true, ctrl: true })
|
|
216
|
+
|
|
217
|
+
// Arrow up with shift+ctrl: \x1b[1;6A (1 base + 1 shift + 4 ctrl = 6)
|
|
218
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;6A")
|
|
219
|
+
})
|
|
220
|
+
|
|
221
|
+
test("pressKey with meta modifier", () => {
|
|
222
|
+
const mockRenderer = new MockRenderer()
|
|
223
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
224
|
+
|
|
225
|
+
mockKeys.pressKey(KeyCodes.ARROW_DOWN, { meta: true })
|
|
226
|
+
|
|
227
|
+
// Arrow down with meta: \x1b[1;3B (1 base + 2 meta = 3)
|
|
228
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;3B")
|
|
229
|
+
})
|
|
230
|
+
|
|
231
|
+
test("pressKey with super modifier", () => {
|
|
232
|
+
const mockRenderer = new MockRenderer()
|
|
233
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
234
|
+
|
|
235
|
+
mockKeys.pressKey(KeyCodes.ARROW_UP, { super: true })
|
|
236
|
+
|
|
237
|
+
// Arrow up with super: \x1b[1;9A (1 base + 8 super = 9)
|
|
238
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;9A")
|
|
239
|
+
})
|
|
240
|
+
|
|
241
|
+
test("pressKey with hyper modifier", () => {
|
|
242
|
+
const mockRenderer = new MockRenderer()
|
|
243
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
244
|
+
|
|
245
|
+
mockKeys.pressKey(KeyCodes.ARROW_LEFT, { hyper: true })
|
|
246
|
+
|
|
247
|
+
// Arrow left with hyper: \x1b[1;17D (1 base + 16 hyper = 17)
|
|
248
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;17D")
|
|
249
|
+
})
|
|
250
|
+
|
|
251
|
+
test("pressKey with super+hyper modifiers", () => {
|
|
252
|
+
const mockRenderer = new MockRenderer()
|
|
253
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
254
|
+
|
|
255
|
+
mockKeys.pressKey(KeyCodes.ARROW_RIGHT, { super: true, hyper: true })
|
|
256
|
+
|
|
257
|
+
// Arrow right with super+hyper: \x1b[1;25C (1 base + 8 super + 16 hyper = 25)
|
|
258
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;25C")
|
|
259
|
+
})
|
|
260
|
+
|
|
261
|
+
test("pressArrow with shift modifier", () => {
|
|
262
|
+
const mockRenderer = new MockRenderer()
|
|
263
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
264
|
+
|
|
265
|
+
mockKeys.pressArrow("right", { shift: true })
|
|
266
|
+
|
|
267
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;2C")
|
|
268
|
+
})
|
|
269
|
+
|
|
270
|
+
test("pressArrow without modifiers still works", () => {
|
|
271
|
+
const mockRenderer = new MockRenderer()
|
|
272
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
273
|
+
|
|
274
|
+
mockKeys.pressArrow("left")
|
|
275
|
+
|
|
276
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[D")
|
|
277
|
+
})
|
|
278
|
+
|
|
279
|
+
test("pressKey with modifiers on HOME key", () => {
|
|
280
|
+
const mockRenderer = new MockRenderer()
|
|
281
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
282
|
+
|
|
283
|
+
mockKeys.pressKey(KeyCodes.HOME, { shift: true })
|
|
284
|
+
|
|
285
|
+
// HOME with shift: \x1b[1;2H
|
|
286
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;2H")
|
|
287
|
+
})
|
|
288
|
+
|
|
289
|
+
test("pressKey with modifiers on END key", () => {
|
|
290
|
+
const mockRenderer = new MockRenderer()
|
|
291
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
292
|
+
|
|
293
|
+
mockKeys.pressKey(KeyCodes.END, { shift: true })
|
|
294
|
+
|
|
295
|
+
// END with shift: \x1b[1;2F
|
|
296
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;2F")
|
|
297
|
+
})
|
|
298
|
+
|
|
299
|
+
test("pressKey with meta on regular character", () => {
|
|
300
|
+
const mockRenderer = new MockRenderer()
|
|
301
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
302
|
+
|
|
303
|
+
mockKeys.pressKey("a", { meta: true })
|
|
304
|
+
|
|
305
|
+
// Meta+a: \x1ba (escape + a)
|
|
306
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1ba")
|
|
307
|
+
})
|
|
308
|
+
|
|
309
|
+
test("pressKey with meta+shift on character", () => {
|
|
310
|
+
const mockRenderer = new MockRenderer()
|
|
311
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
312
|
+
|
|
313
|
+
mockKeys.pressKey("a", { meta: true, shift: true })
|
|
314
|
+
|
|
315
|
+
// Meta+Shift+a: \x1bA (escape + uppercase A)
|
|
316
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1bA")
|
|
317
|
+
})
|
|
318
|
+
|
|
319
|
+
test("pressKey with meta+ctrl on arrow", () => {
|
|
320
|
+
const mockRenderer = new MockRenderer()
|
|
321
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
322
|
+
|
|
323
|
+
mockKeys.pressKey(KeyCodes.ARROW_RIGHT, { meta: true, ctrl: true })
|
|
324
|
+
|
|
325
|
+
// Arrow right with meta+ctrl: \x1b[1;7C (1 base + 2 meta + 4 ctrl = 7)
|
|
326
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;7C")
|
|
327
|
+
})
|
|
328
|
+
|
|
329
|
+
test("pressKey with meta+shift+ctrl on arrow", () => {
|
|
330
|
+
const mockRenderer = new MockRenderer()
|
|
331
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
332
|
+
|
|
333
|
+
mockKeys.pressKey(KeyCodes.ARROW_UP, { meta: true, shift: true, ctrl: true })
|
|
334
|
+
|
|
335
|
+
// Arrow up with all modifiers: \x1b[1;8A (1 base + 1 shift + 2 meta + 4 ctrl = 8)
|
|
336
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;8A")
|
|
337
|
+
})
|
|
338
|
+
|
|
339
|
+
test("pressArrow with meta modifier", () => {
|
|
340
|
+
const mockRenderer = new MockRenderer()
|
|
341
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
342
|
+
|
|
343
|
+
mockKeys.pressArrow("left", { meta: true })
|
|
344
|
+
|
|
345
|
+
// Arrow left with meta: \x1b[1;3D
|
|
346
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;3D")
|
|
347
|
+
})
|
|
348
|
+
|
|
349
|
+
test("pressArrow with meta+shift modifiers", () => {
|
|
350
|
+
const mockRenderer = new MockRenderer()
|
|
351
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
352
|
+
|
|
353
|
+
mockKeys.pressArrow("down", { meta: true, shift: true })
|
|
354
|
+
|
|
355
|
+
// Arrow down with meta+shift: \x1b[1;4B (1 base + 1 shift + 2 meta = 4)
|
|
356
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;4B")
|
|
357
|
+
})
|
|
358
|
+
|
|
359
|
+
test("meta modifier produces escape sequences", () => {
|
|
360
|
+
const mockRenderer = new MockRenderer()
|
|
361
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
362
|
+
|
|
363
|
+
mockKeys.pressKey("a", { meta: true })
|
|
364
|
+
mockKeys.pressKey("z", { meta: true })
|
|
365
|
+
|
|
366
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1ba\x1bz")
|
|
367
|
+
})
|
|
368
|
+
|
|
369
|
+
test("pressEnter with modifiers", () => {
|
|
370
|
+
const mockRenderer = new MockRenderer()
|
|
371
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
372
|
+
|
|
373
|
+
mockKeys.pressEnter({ meta: true })
|
|
374
|
+
|
|
375
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b\r")
|
|
376
|
+
})
|
|
377
|
+
|
|
378
|
+
test("pressTab with shift modifier", () => {
|
|
379
|
+
const mockRenderer = new MockRenderer()
|
|
380
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
381
|
+
|
|
382
|
+
mockKeys.pressTab({ shift: true })
|
|
383
|
+
|
|
384
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[Z")
|
|
385
|
+
})
|
|
386
|
+
|
|
387
|
+
test("pressTab with shift modifier parses as shift+tab", async () => {
|
|
388
|
+
const { parseKeypress } = await import("../lib/parse.keypress")
|
|
389
|
+
const mockRenderer = new MockRenderer()
|
|
390
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
391
|
+
|
|
392
|
+
mockKeys.pressTab({ shift: true })
|
|
393
|
+
|
|
394
|
+
const result = parseKeypress(mockRenderer.getEmittedData())
|
|
395
|
+
expect(result).not.toBeNull()
|
|
396
|
+
expect(result?.name).toBe("tab")
|
|
397
|
+
expect(result?.shift).toBe(true)
|
|
398
|
+
})
|
|
399
|
+
|
|
400
|
+
test("pressTab without modifiers still sends raw tab", () => {
|
|
401
|
+
const mockRenderer = new MockRenderer()
|
|
402
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
403
|
+
|
|
404
|
+
mockKeys.pressTab()
|
|
405
|
+
|
|
406
|
+
expect(mockRenderer.getEmittedData()).toBe("\t")
|
|
407
|
+
})
|
|
408
|
+
|
|
409
|
+
test("pressEscape with ctrl modifier", () => {
|
|
410
|
+
const mockRenderer = new MockRenderer()
|
|
411
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
412
|
+
|
|
413
|
+
mockKeys.pressEscape({ ctrl: true })
|
|
414
|
+
|
|
415
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b")
|
|
416
|
+
})
|
|
417
|
+
|
|
418
|
+
test("pressBackspace with meta modifier", () => {
|
|
419
|
+
const mockRenderer = new MockRenderer()
|
|
420
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
421
|
+
|
|
422
|
+
mockKeys.pressBackspace({ meta: true })
|
|
423
|
+
|
|
424
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b\b")
|
|
425
|
+
})
|
|
426
|
+
|
|
427
|
+
test("pressKey with ctrl on letter produces control code", () => {
|
|
428
|
+
const mockRenderer = new MockRenderer()
|
|
429
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
430
|
+
|
|
431
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
432
|
+
mockKeys.pressKey("z", { ctrl: true })
|
|
433
|
+
|
|
434
|
+
expect(mockRenderer.getEmittedData()).toBe("\x01\x1a")
|
|
435
|
+
})
|
|
436
|
+
|
|
437
|
+
test("pressKey with ctrl on uppercase letter", () => {
|
|
438
|
+
const mockRenderer = new MockRenderer()
|
|
439
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
440
|
+
|
|
441
|
+
mockKeys.pressKey("A", { ctrl: true })
|
|
442
|
+
|
|
443
|
+
expect(mockRenderer.getEmittedData()).toBe("\x01")
|
|
444
|
+
})
|
|
445
|
+
|
|
446
|
+
test("pressKey with ctrl+meta combination", () => {
|
|
447
|
+
const mockRenderer = new MockRenderer()
|
|
448
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
449
|
+
|
|
450
|
+
mockKeys.pressKey("a", { ctrl: true, meta: true })
|
|
451
|
+
|
|
452
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b\x01")
|
|
453
|
+
})
|
|
454
|
+
|
|
455
|
+
test("ctrl modifier produces control codes", () => {
|
|
456
|
+
const mockRenderer = new MockRenderer()
|
|
457
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
458
|
+
|
|
459
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
460
|
+
mockKeys.pressKey("c", { ctrl: true })
|
|
461
|
+
mockKeys.pressKey("d", { ctrl: true })
|
|
462
|
+
|
|
463
|
+
expect(mockRenderer.getEmittedData()).toBe("\x01\x03\x04")
|
|
464
|
+
})
|
|
465
|
+
|
|
466
|
+
test("meta modifier produces escape sequences", () => {
|
|
467
|
+
const mockRenderer = new MockRenderer()
|
|
468
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
469
|
+
|
|
470
|
+
mockKeys.pressKey("a", { meta: true })
|
|
471
|
+
mockKeys.pressKey("z", { meta: true })
|
|
472
|
+
|
|
473
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1ba\x1bz")
|
|
474
|
+
})
|
|
475
|
+
|
|
476
|
+
test("all CTRL_* letters produce correct control codes", () => {
|
|
477
|
+
const mockRenderer = new MockRenderer()
|
|
478
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
479
|
+
|
|
480
|
+
const letters = "abcdefghijklmnopqrstuvwxyz"
|
|
481
|
+
for (const letter of letters) {
|
|
482
|
+
mockKeys.pressKey(letter, { ctrl: true })
|
|
483
|
+
}
|
|
484
|
+
|
|
485
|
+
const expected = letters
|
|
486
|
+
.split("")
|
|
487
|
+
.map((c) => String.fromCharCode(c.charCodeAt(0) - 96))
|
|
488
|
+
.join("")
|
|
489
|
+
expect(mockRenderer.getEmittedData()).toBe(expected)
|
|
490
|
+
})
|
|
491
|
+
|
|
492
|
+
test("pressKey with ctrl modifier produces control code", () => {
|
|
493
|
+
const mockRenderer = new MockRenderer()
|
|
494
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
495
|
+
mockKeys.pressKey("c", { ctrl: true })
|
|
496
|
+
|
|
497
|
+
expect(mockRenderer.getEmittedData()).toBe("\x03")
|
|
498
|
+
})
|
|
499
|
+
|
|
500
|
+
test("pressKey with meta modifier on letters produces escape sequences", () => {
|
|
501
|
+
const mockRenderer = new MockRenderer()
|
|
502
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
503
|
+
mockKeys.pressKey("x", { meta: true })
|
|
504
|
+
|
|
505
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1bx")
|
|
506
|
+
})
|
|
507
|
+
|
|
508
|
+
test("pressKey with ctrl modifier on special characters", () => {
|
|
509
|
+
const mockRenderer = new MockRenderer()
|
|
510
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
511
|
+
|
|
512
|
+
// Ctrl+- should produce \u001f (ASCII 31, Unit Separator)
|
|
513
|
+
mockRenderer.emittedData = []
|
|
514
|
+
mockKeys.pressKey("-", { ctrl: true })
|
|
515
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001f")
|
|
516
|
+
|
|
517
|
+
// Ctrl+. should produce \u001e (ASCII 30, Record Separator)
|
|
518
|
+
mockRenderer.emittedData = []
|
|
519
|
+
mockKeys.pressKey(".", { ctrl: true })
|
|
520
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001e")
|
|
521
|
+
|
|
522
|
+
// Ctrl+, should produce \u001c (ASCII 28, File Separator)
|
|
523
|
+
mockRenderer.emittedData = []
|
|
524
|
+
mockKeys.pressKey(",", { ctrl: true })
|
|
525
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001c")
|
|
526
|
+
|
|
527
|
+
// Ctrl+] should produce \u001d (ASCII 29, Group Separator)
|
|
528
|
+
mockRenderer.emittedData = []
|
|
529
|
+
mockKeys.pressKey("]", { ctrl: true })
|
|
530
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001d")
|
|
531
|
+
|
|
532
|
+
// Ctrl+[ should produce \x1b (ASCII 27, Escape)
|
|
533
|
+
mockRenderer.emittedData = []
|
|
534
|
+
mockKeys.pressKey("[", { ctrl: true })
|
|
535
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b")
|
|
536
|
+
|
|
537
|
+
// Ctrl+/ should produce \u001f (ASCII 31, same as Ctrl+-)
|
|
538
|
+
mockRenderer.emittedData = []
|
|
539
|
+
mockKeys.pressKey("/", { ctrl: true })
|
|
540
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001f")
|
|
541
|
+
|
|
542
|
+
// Ctrl+_ should also produce \u001f (ASCII 31)
|
|
543
|
+
mockRenderer.emittedData = []
|
|
544
|
+
mockKeys.pressKey("_", { ctrl: true })
|
|
545
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001f")
|
|
546
|
+
})
|
|
547
|
+
|
|
548
|
+
test("pressKey with ctrl modifier on all special control characters", () => {
|
|
549
|
+
const mockRenderer = new MockRenderer()
|
|
550
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
551
|
+
|
|
552
|
+
// Test all standard control character mappings
|
|
553
|
+
const tests = [
|
|
554
|
+
{ key: "[", expected: "\x1b" }, // ESC
|
|
555
|
+
{ key: "\\", expected: "\x1c" }, // FS
|
|
556
|
+
{ key: "]", expected: "\x1d" }, // GS
|
|
557
|
+
{ key: "^", expected: "\x1e" }, // RS
|
|
558
|
+
{ key: "_", expected: "\x1f" }, // US
|
|
559
|
+
{ key: "?", expected: "\x7f" }, // DEL
|
|
560
|
+
{ key: "@", expected: "\x00" }, // NUL
|
|
561
|
+
{ key: " ", expected: "\x00" }, // NUL (Ctrl+Space)
|
|
562
|
+
]
|
|
563
|
+
|
|
564
|
+
for (const { key, expected } of tests) {
|
|
565
|
+
mockRenderer.emittedData = []
|
|
566
|
+
mockKeys.pressKey(key, { ctrl: true })
|
|
567
|
+
expect(mockRenderer.getEmittedData()).toBe(expected)
|
|
568
|
+
}
|
|
569
|
+
})
|
|
570
|
+
|
|
571
|
+
test("pressKey with ctrl+meta on special characters", () => {
|
|
572
|
+
const mockRenderer = new MockRenderer()
|
|
573
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
574
|
+
|
|
575
|
+
// Ctrl+Meta+- should produce ESC + \u001f
|
|
576
|
+
mockRenderer.emittedData = []
|
|
577
|
+
mockKeys.pressKey("-", { ctrl: true, meta: true })
|
|
578
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b\u001f")
|
|
579
|
+
|
|
580
|
+
// Ctrl+Meta+] should produce ESC + \u001d
|
|
581
|
+
mockRenderer.emittedData = []
|
|
582
|
+
mockKeys.pressKey("]", { ctrl: true, meta: true })
|
|
583
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b\u001d")
|
|
584
|
+
})
|
|
585
|
+
|
|
586
|
+
test("pressKey with ctrl on special chars does NOT use kitty keyboard", () => {
|
|
587
|
+
const mockRenderer = new MockRenderer()
|
|
588
|
+
// Explicitly use non-kitty mode
|
|
589
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: false })
|
|
590
|
+
|
|
591
|
+
mockKeys.pressKey("-", { ctrl: true })
|
|
592
|
+
|
|
593
|
+
// Should produce raw control sequence, NOT kitty CSI u sequence
|
|
594
|
+
const data = mockRenderer.getEmittedData()
|
|
595
|
+
expect(data).toBe("\u001f")
|
|
596
|
+
expect(data).not.toContain("[") // Should not contain CSI
|
|
597
|
+
expect(data).not.toContain("u") // Should not contain kitty 'u' ending
|
|
598
|
+
})
|
|
599
|
+
|
|
600
|
+
test("comprehensive test: all punctuation keys work with ctrl modifier", () => {
|
|
601
|
+
const mockRenderer = new MockRenderer()
|
|
602
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
603
|
+
|
|
604
|
+
// Test multiple punctuation keys in sequence
|
|
605
|
+
mockKeys.pressKey("-", { ctrl: true })
|
|
606
|
+
mockKeys.pressKey(".", { ctrl: true })
|
|
607
|
+
mockKeys.pressKey(",", { ctrl: true })
|
|
608
|
+
mockKeys.pressKey("]", { ctrl: true })
|
|
609
|
+
mockKeys.pressKey("[", { ctrl: true })
|
|
610
|
+
|
|
611
|
+
const expected = "\u001f\u001e\u001c\u001d\x1b"
|
|
612
|
+
expect(mockRenderer.getEmittedData()).toBe(expected)
|
|
613
|
+
})
|
|
614
|
+
|
|
615
|
+
test("ctrl modifier with non-mapped characters preserves original", () => {
|
|
616
|
+
const mockRenderer = new MockRenderer()
|
|
617
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
618
|
+
|
|
619
|
+
// Characters without specific ctrl mappings should be preserved
|
|
620
|
+
// (though in real terminals they might not do anything)
|
|
621
|
+
mockKeys.pressKey("(", { ctrl: true })
|
|
622
|
+
|
|
623
|
+
// Should preserve the character since it's not in the mapping
|
|
624
|
+
expect(mockRenderer.getEmittedData()).toBe("(")
|
|
625
|
+
})
|
|
626
|
+
|
|
627
|
+
describe("Kitty Keyboard Protocol Mode", () => {
|
|
628
|
+
test("basic character in kitty mode", () => {
|
|
629
|
+
const mockRenderer = new MockRenderer()
|
|
630
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
631
|
+
|
|
632
|
+
mockKeys.pressKey("a")
|
|
633
|
+
|
|
634
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97u")
|
|
635
|
+
})
|
|
636
|
+
|
|
637
|
+
test("backspace without modifiers in kitty mode", () => {
|
|
638
|
+
const mockRenderer = new MockRenderer()
|
|
639
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
640
|
+
|
|
641
|
+
mockKeys.pressBackspace()
|
|
642
|
+
|
|
643
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[127u")
|
|
644
|
+
})
|
|
645
|
+
|
|
646
|
+
test("backspace with shift in kitty mode", () => {
|
|
647
|
+
const mockRenderer = new MockRenderer()
|
|
648
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
649
|
+
|
|
650
|
+
mockKeys.pressBackspace({ shift: true })
|
|
651
|
+
|
|
652
|
+
// Kitty protocol: backspace(127) with shift modifier (1+1=2)
|
|
653
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[127;2u")
|
|
654
|
+
})
|
|
655
|
+
|
|
656
|
+
test("backspace with ctrl in kitty mode", () => {
|
|
657
|
+
const mockRenderer = new MockRenderer()
|
|
658
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
659
|
+
|
|
660
|
+
mockKeys.pressBackspace({ ctrl: true })
|
|
661
|
+
|
|
662
|
+
// Kitty protocol: backspace(127) with ctrl modifier (4+1=5)
|
|
663
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[127;5u")
|
|
664
|
+
})
|
|
665
|
+
|
|
666
|
+
test("backspace with meta in kitty mode", () => {
|
|
667
|
+
const mockRenderer = new MockRenderer()
|
|
668
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
669
|
+
|
|
670
|
+
mockKeys.pressBackspace({ meta: true })
|
|
671
|
+
|
|
672
|
+
// Kitty protocol: backspace(127) with meta modifier (2+1=3)
|
|
673
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[127;3u")
|
|
674
|
+
})
|
|
675
|
+
|
|
676
|
+
test("backspace with shift+meta in kitty mode", () => {
|
|
677
|
+
const mockRenderer = new MockRenderer()
|
|
678
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
679
|
+
|
|
680
|
+
mockKeys.pressBackspace({ shift: true, meta: true })
|
|
681
|
+
|
|
682
|
+
// Kitty protocol: backspace(127) with shift+meta (1+2+1=4)
|
|
683
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[127;4u")
|
|
684
|
+
})
|
|
685
|
+
|
|
686
|
+
test("delete key in kitty mode", () => {
|
|
687
|
+
const mockRenderer = new MockRenderer()
|
|
688
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
689
|
+
|
|
690
|
+
mockKeys.pressKey("DELETE")
|
|
691
|
+
|
|
692
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57349u")
|
|
693
|
+
})
|
|
694
|
+
|
|
695
|
+
test("arrow keys in kitty mode", () => {
|
|
696
|
+
const mockRenderer = new MockRenderer()
|
|
697
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
698
|
+
|
|
699
|
+
mockKeys.pressArrow("up")
|
|
700
|
+
mockKeys.pressArrow("down")
|
|
701
|
+
mockKeys.pressArrow("left")
|
|
702
|
+
mockKeys.pressArrow("right")
|
|
703
|
+
|
|
704
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57352u\x1b[57353u\x1b[57350u\x1b[57351u")
|
|
705
|
+
})
|
|
706
|
+
|
|
707
|
+
test("arrow key with shift in kitty mode", () => {
|
|
708
|
+
const mockRenderer = new MockRenderer()
|
|
709
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
710
|
+
|
|
711
|
+
mockKeys.pressArrow("right", { shift: true })
|
|
712
|
+
|
|
713
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57351;2u")
|
|
714
|
+
})
|
|
715
|
+
|
|
716
|
+
test("arrow key with ctrl in kitty mode", () => {
|
|
717
|
+
const mockRenderer = new MockRenderer()
|
|
718
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
719
|
+
|
|
720
|
+
mockKeys.pressArrow("left", { ctrl: true })
|
|
721
|
+
|
|
722
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57350;5u")
|
|
723
|
+
})
|
|
724
|
+
|
|
725
|
+
test("arrow key with meta in kitty mode", () => {
|
|
726
|
+
const mockRenderer = new MockRenderer()
|
|
727
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
728
|
+
|
|
729
|
+
mockKeys.pressArrow("down", { meta: true })
|
|
730
|
+
|
|
731
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57353;3u")
|
|
732
|
+
})
|
|
733
|
+
|
|
734
|
+
test("arrow key with shift+ctrl+meta in kitty mode", () => {
|
|
735
|
+
const mockRenderer = new MockRenderer()
|
|
736
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
737
|
+
|
|
738
|
+
mockKeys.pressArrow("up", { shift: true, ctrl: true, meta: true })
|
|
739
|
+
|
|
740
|
+
// shift(1) + meta(2) + ctrl(4) = 7, plus 1 = 8
|
|
741
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57352;8u")
|
|
742
|
+
})
|
|
743
|
+
|
|
744
|
+
test("enter/return in kitty mode", () => {
|
|
745
|
+
const mockRenderer = new MockRenderer()
|
|
746
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
747
|
+
|
|
748
|
+
mockKeys.pressEnter()
|
|
749
|
+
|
|
750
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[13u")
|
|
751
|
+
})
|
|
752
|
+
|
|
753
|
+
test("enter with meta in kitty mode", () => {
|
|
754
|
+
const mockRenderer = new MockRenderer()
|
|
755
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
756
|
+
|
|
757
|
+
mockKeys.pressEnter({ meta: true })
|
|
758
|
+
|
|
759
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[13;3u")
|
|
760
|
+
})
|
|
761
|
+
|
|
762
|
+
test("tab in kitty mode", () => {
|
|
763
|
+
const mockRenderer = new MockRenderer()
|
|
764
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
765
|
+
|
|
766
|
+
mockKeys.pressTab()
|
|
767
|
+
|
|
768
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[9u")
|
|
769
|
+
})
|
|
770
|
+
|
|
771
|
+
test("tab with shift in kitty mode", () => {
|
|
772
|
+
const mockRenderer = new MockRenderer()
|
|
773
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
774
|
+
|
|
775
|
+
mockKeys.pressTab({ shift: true })
|
|
776
|
+
|
|
777
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[9;2u")
|
|
778
|
+
})
|
|
779
|
+
|
|
780
|
+
test("escape in kitty mode", () => {
|
|
781
|
+
const mockRenderer = new MockRenderer()
|
|
782
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
783
|
+
|
|
784
|
+
mockKeys.pressEscape()
|
|
785
|
+
|
|
786
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27u")
|
|
787
|
+
})
|
|
788
|
+
|
|
789
|
+
test("home key in kitty mode", () => {
|
|
790
|
+
const mockRenderer = new MockRenderer()
|
|
791
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
792
|
+
|
|
793
|
+
mockKeys.pressKey("HOME")
|
|
794
|
+
|
|
795
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57356u")
|
|
796
|
+
})
|
|
797
|
+
|
|
798
|
+
test("home with shift in kitty mode", () => {
|
|
799
|
+
const mockRenderer = new MockRenderer()
|
|
800
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
801
|
+
|
|
802
|
+
mockKeys.pressKey("HOME", { shift: true })
|
|
803
|
+
|
|
804
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57356;2u")
|
|
805
|
+
})
|
|
806
|
+
|
|
807
|
+
test("end key in kitty mode", () => {
|
|
808
|
+
const mockRenderer = new MockRenderer()
|
|
809
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
810
|
+
|
|
811
|
+
mockKeys.pressKey("END")
|
|
812
|
+
|
|
813
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57357u")
|
|
814
|
+
})
|
|
815
|
+
|
|
816
|
+
test("function keys in kitty mode", () => {
|
|
817
|
+
const mockRenderer = new MockRenderer()
|
|
818
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
819
|
+
|
|
820
|
+
mockKeys.pressKey("F1")
|
|
821
|
+
mockKeys.pressKey("F2")
|
|
822
|
+
mockKeys.pressKey("F12")
|
|
823
|
+
|
|
824
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[57364u\x1b[57365u\x1b[57375u")
|
|
825
|
+
})
|
|
826
|
+
|
|
827
|
+
test("regular characters with shift in kitty mode", () => {
|
|
828
|
+
const mockRenderer = new MockRenderer()
|
|
829
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
830
|
+
|
|
831
|
+
mockKeys.pressKey("a", { shift: true })
|
|
832
|
+
|
|
833
|
+
// 'a' (97) with shift modifier
|
|
834
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;2u")
|
|
835
|
+
})
|
|
836
|
+
|
|
837
|
+
test("regular characters with ctrl in kitty mode", () => {
|
|
838
|
+
const mockRenderer = new MockRenderer()
|
|
839
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
840
|
+
|
|
841
|
+
mockKeys.pressKey("c", { ctrl: true })
|
|
842
|
+
|
|
843
|
+
// 'c' (99) with ctrl modifier (4+1=5)
|
|
844
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[99;5u")
|
|
845
|
+
})
|
|
846
|
+
|
|
847
|
+
test("regular characters with meta in kitty mode", () => {
|
|
848
|
+
const mockRenderer = new MockRenderer()
|
|
849
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
850
|
+
|
|
851
|
+
mockKeys.pressKey("x", { meta: true })
|
|
852
|
+
|
|
853
|
+
// 'x' (120) with meta modifier (2+1=3)
|
|
854
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[120;3u")
|
|
855
|
+
})
|
|
856
|
+
|
|
857
|
+
test("multiple keys in sequence in kitty mode", () => {
|
|
858
|
+
const mockRenderer = new MockRenderer()
|
|
859
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
860
|
+
|
|
861
|
+
mockKeys.pressKey("h")
|
|
862
|
+
mockKeys.pressKey("i")
|
|
863
|
+
|
|
864
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[104u\x1b[105u")
|
|
865
|
+
})
|
|
866
|
+
|
|
867
|
+
test("mixed modifier combinations in kitty mode", () => {
|
|
868
|
+
const mockRenderer = new MockRenderer()
|
|
869
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
870
|
+
|
|
871
|
+
mockKeys.pressKey("a")
|
|
872
|
+
mockKeys.pressKey("a", { shift: true })
|
|
873
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
874
|
+
mockKeys.pressKey("a", { meta: true })
|
|
875
|
+
mockKeys.pressKey("a", { shift: true, ctrl: true })
|
|
876
|
+
|
|
877
|
+
expect(mockRenderer.getEmittedData()).toBe(
|
|
878
|
+
"\x1b[97u" + // no mods
|
|
879
|
+
"\x1b[97;2u" + // shift
|
|
880
|
+
"\x1b[97;5u" + // ctrl
|
|
881
|
+
"\x1b[97;3u" + // meta
|
|
882
|
+
"\x1b[97;6u", // shift+ctrl
|
|
883
|
+
)
|
|
884
|
+
})
|
|
885
|
+
|
|
886
|
+
test("character with super modifier in kitty mode", () => {
|
|
887
|
+
const mockRenderer = new MockRenderer()
|
|
888
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
889
|
+
|
|
890
|
+
mockKeys.pressKey("a", { super: true })
|
|
891
|
+
|
|
892
|
+
// 'a' (97) with super modifier (8+1=9)
|
|
893
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;9u")
|
|
894
|
+
})
|
|
895
|
+
|
|
896
|
+
test("character with hyper modifier in kitty mode", () => {
|
|
897
|
+
const mockRenderer = new MockRenderer()
|
|
898
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
899
|
+
|
|
900
|
+
mockKeys.pressKey("a", { hyper: true })
|
|
901
|
+
|
|
902
|
+
// 'a' (97) with hyper modifier (16+1=17)
|
|
903
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;17u")
|
|
904
|
+
})
|
|
905
|
+
|
|
906
|
+
test("character with super+hyper modifiers in kitty mode", () => {
|
|
907
|
+
const mockRenderer = new MockRenderer()
|
|
908
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
909
|
+
|
|
910
|
+
mockKeys.pressKey("a", { super: true, hyper: true })
|
|
911
|
+
|
|
912
|
+
// 'a' (97) with super+hyper (8+16+1=25)
|
|
913
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;25u")
|
|
914
|
+
})
|
|
915
|
+
|
|
916
|
+
test("character with all modifiers in kitty mode", () => {
|
|
917
|
+
const mockRenderer = new MockRenderer()
|
|
918
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
919
|
+
|
|
920
|
+
mockKeys.pressKey("a", { shift: true, ctrl: true, meta: true, super: true, hyper: true })
|
|
921
|
+
|
|
922
|
+
// 'a' (97) with all modifiers: shift(1) + meta(2) + ctrl(4) + super(8) + hyper(16) = 31, +1 = 32
|
|
923
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;32u")
|
|
924
|
+
})
|
|
925
|
+
|
|
926
|
+
test("kitty mode vs regular mode comparison", () => {
|
|
927
|
+
const kittyRenderer = new MockRenderer()
|
|
928
|
+
const regularRenderer = new MockRenderer()
|
|
929
|
+
const kittyKeys = createMockKeys(kittyRenderer as any, { kittyKeyboard: true })
|
|
930
|
+
const regularKeys = createMockKeys(regularRenderer as any, { kittyKeyboard: false })
|
|
931
|
+
|
|
932
|
+
kittyKeys.pressBackspace({ shift: true })
|
|
933
|
+
regularKeys.pressBackspace({ shift: true })
|
|
934
|
+
|
|
935
|
+
// Kitty should send the protocol sequence with modifier
|
|
936
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[127;2u")
|
|
937
|
+
// Regular should just send backspace (shift is ignored)
|
|
938
|
+
expect(regularRenderer.getEmittedData()).toBe("\b")
|
|
939
|
+
})
|
|
940
|
+
|
|
941
|
+
test("special characters with ctrl in kitty mode", () => {
|
|
942
|
+
const kittyRenderer = new MockRenderer()
|
|
943
|
+
const regularRenderer = new MockRenderer()
|
|
944
|
+
const kittyKeys = createMockKeys(kittyRenderer as any, { kittyKeyboard: true })
|
|
945
|
+
const regularKeys = createMockKeys(regularRenderer as any, { kittyKeyboard: false })
|
|
946
|
+
|
|
947
|
+
// Test Ctrl+- in both modes
|
|
948
|
+
kittyKeys.pressKey("-", { ctrl: true })
|
|
949
|
+
regularKeys.pressKey("-", { ctrl: true })
|
|
950
|
+
|
|
951
|
+
// Kitty should send the protocol sequence: '-' is codepoint 45, ctrl modifier is 4+1=5
|
|
952
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[45;5u")
|
|
953
|
+
// Regular should send raw control sequence \u001f
|
|
954
|
+
expect(regularRenderer.getEmittedData()).toBe("\u001f")
|
|
955
|
+
})
|
|
956
|
+
|
|
957
|
+
test("various special characters with ctrl in kitty mode", () => {
|
|
958
|
+
const kittyRenderer = new MockRenderer()
|
|
959
|
+
const kittyKeys = createMockKeys(kittyRenderer as any, { kittyKeyboard: true })
|
|
960
|
+
|
|
961
|
+
// Test multiple special characters
|
|
962
|
+
kittyRenderer.emittedData = []
|
|
963
|
+
kittyKeys.pressKey(".", { ctrl: true })
|
|
964
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[46;5u") // '.' is codepoint 46
|
|
965
|
+
|
|
966
|
+
kittyRenderer.emittedData = []
|
|
967
|
+
kittyKeys.pressKey(",", { ctrl: true })
|
|
968
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[44;5u") // ',' is codepoint 44
|
|
969
|
+
|
|
970
|
+
kittyRenderer.emittedData = []
|
|
971
|
+
kittyKeys.pressKey("]", { ctrl: true })
|
|
972
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[93;5u") // ']' is codepoint 93
|
|
973
|
+
})
|
|
974
|
+
})
|
|
975
|
+
|
|
976
|
+
describe("modifyOtherKeys Mode (CSI u variant)", () => {
|
|
977
|
+
test("modifyOtherKeys sequences can be parsed by parseKeypress", async () => {
|
|
978
|
+
const { parseKeypress } = await import("../lib/parse.keypress")
|
|
979
|
+
|
|
980
|
+
// Test that our generated sequences can be parsed correctly
|
|
981
|
+
const tests = [
|
|
982
|
+
{ seq: "\x1b[27;5;97~", expectedName: "a", expectedCtrl: true },
|
|
983
|
+
{ seq: "\x1b[27;2;13~", expectedName: "return", expectedShift: true },
|
|
984
|
+
{ seq: "\x1b[27;5;27~", expectedName: "escape", expectedCtrl: true },
|
|
985
|
+
{ seq: "\x1b[27;2;9~", expectedName: "tab", expectedShift: true },
|
|
986
|
+
{ seq: "\x1b[27;5;32~", expectedName: "space", expectedCtrl: true },
|
|
987
|
+
{ seq: "\x1b[27;6;97~", expectedName: "a", expectedShift: true, expectedCtrl: true },
|
|
988
|
+
]
|
|
989
|
+
|
|
990
|
+
for (const test of tests) {
|
|
991
|
+
const result = parseKeypress(test.seq)
|
|
992
|
+
expect(result).not.toBeNull()
|
|
993
|
+
expect(result?.name).toBe(test.expectedName)
|
|
994
|
+
if (test.expectedCtrl !== undefined) {
|
|
995
|
+
expect(result?.ctrl).toBe(test.expectedCtrl)
|
|
996
|
+
}
|
|
997
|
+
if (test.expectedShift !== undefined) {
|
|
998
|
+
expect(result?.shift).toBe(test.expectedShift)
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
1001
|
+
})
|
|
1002
|
+
|
|
1003
|
+
test("basic character without modifiers in modifyOtherKeys mode", () => {
|
|
1004
|
+
const mockRenderer = new MockRenderer()
|
|
1005
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1006
|
+
|
|
1007
|
+
mockKeys.pressKey("a")
|
|
1008
|
+
|
|
1009
|
+
// Without modifiers, should send plain character
|
|
1010
|
+
expect(mockRenderer.getEmittedData()).toBe("a")
|
|
1011
|
+
})
|
|
1012
|
+
|
|
1013
|
+
test("character with ctrl in modifyOtherKeys mode", () => {
|
|
1014
|
+
const mockRenderer = new MockRenderer()
|
|
1015
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1016
|
+
|
|
1017
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
1018
|
+
|
|
1019
|
+
// modifyOtherKeys format: CSI 27 ; modifier ; code ~
|
|
1020
|
+
// 'a' is charCode 97, ctrl is 4, modifier is 4+1=5
|
|
1021
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;97~")
|
|
1022
|
+
})
|
|
1023
|
+
|
|
1024
|
+
test("character with shift in modifyOtherKeys mode", () => {
|
|
1025
|
+
const mockRenderer = new MockRenderer()
|
|
1026
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1027
|
+
|
|
1028
|
+
mockKeys.pressKey("a", { shift: true })
|
|
1029
|
+
|
|
1030
|
+
// 'a' is charCode 97, shift is 1, modifier is 1+1=2
|
|
1031
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;2;97~")
|
|
1032
|
+
})
|
|
1033
|
+
|
|
1034
|
+
test("character with meta in modifyOtherKeys mode", () => {
|
|
1035
|
+
const mockRenderer = new MockRenderer()
|
|
1036
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1037
|
+
|
|
1038
|
+
mockKeys.pressKey("a", { meta: true })
|
|
1039
|
+
|
|
1040
|
+
// 'a' is charCode 97, meta is 2, modifier is 2+1=3
|
|
1041
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;3;97~")
|
|
1042
|
+
})
|
|
1043
|
+
|
|
1044
|
+
test("return/enter with ctrl in modifyOtherKeys mode", () => {
|
|
1045
|
+
const mockRenderer = new MockRenderer()
|
|
1046
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1047
|
+
|
|
1048
|
+
mockKeys.pressEnter({ ctrl: true })
|
|
1049
|
+
|
|
1050
|
+
// return is charCode 13, ctrl is 4, modifier is 4+1=5
|
|
1051
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;13~")
|
|
1052
|
+
})
|
|
1053
|
+
|
|
1054
|
+
test("return with shift in modifyOtherKeys mode", () => {
|
|
1055
|
+
const mockRenderer = new MockRenderer()
|
|
1056
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1057
|
+
|
|
1058
|
+
mockKeys.pressEnter({ shift: true })
|
|
1059
|
+
|
|
1060
|
+
// return is charCode 13, shift is 1, modifier is 1+1=2
|
|
1061
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;2;13~")
|
|
1062
|
+
})
|
|
1063
|
+
|
|
1064
|
+
test("escape with ctrl in modifyOtherKeys mode", () => {
|
|
1065
|
+
const mockRenderer = new MockRenderer()
|
|
1066
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1067
|
+
|
|
1068
|
+
mockKeys.pressEscape({ ctrl: true })
|
|
1069
|
+
|
|
1070
|
+
// escape is charCode 27, ctrl is 4, modifier is 4+1=5
|
|
1071
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;27~")
|
|
1072
|
+
})
|
|
1073
|
+
|
|
1074
|
+
test("tab with ctrl in modifyOtherKeys mode", () => {
|
|
1075
|
+
const mockRenderer = new MockRenderer()
|
|
1076
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1077
|
+
|
|
1078
|
+
mockKeys.pressTab({ ctrl: true })
|
|
1079
|
+
|
|
1080
|
+
// tab is charCode 9, ctrl is 4, modifier is 4+1=5
|
|
1081
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;9~")
|
|
1082
|
+
})
|
|
1083
|
+
|
|
1084
|
+
test("tab with shift in modifyOtherKeys mode", () => {
|
|
1085
|
+
const mockRenderer = new MockRenderer()
|
|
1086
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1087
|
+
|
|
1088
|
+
mockKeys.pressTab({ shift: true })
|
|
1089
|
+
|
|
1090
|
+
// tab is charCode 9, shift is 1, modifier is 1+1=2
|
|
1091
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;2;9~")
|
|
1092
|
+
})
|
|
1093
|
+
|
|
1094
|
+
test("backspace with ctrl in modifyOtherKeys mode", () => {
|
|
1095
|
+
const mockRenderer = new MockRenderer()
|
|
1096
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1097
|
+
|
|
1098
|
+
mockKeys.pressBackspace({ ctrl: true })
|
|
1099
|
+
|
|
1100
|
+
// backspace is charCode 127, ctrl is 4, modifier is 4+1=5
|
|
1101
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;127~")
|
|
1102
|
+
})
|
|
1103
|
+
|
|
1104
|
+
test("backspace with meta in modifyOtherKeys mode", () => {
|
|
1105
|
+
const mockRenderer = new MockRenderer()
|
|
1106
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1107
|
+
|
|
1108
|
+
mockKeys.pressBackspace({ meta: true })
|
|
1109
|
+
|
|
1110
|
+
// backspace is charCode 127, meta is 2, modifier is 2+1=3
|
|
1111
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;3;127~")
|
|
1112
|
+
})
|
|
1113
|
+
|
|
1114
|
+
test("space with ctrl in modifyOtherKeys mode", () => {
|
|
1115
|
+
const mockRenderer = new MockRenderer()
|
|
1116
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1117
|
+
|
|
1118
|
+
mockKeys.pressKey(" ", { ctrl: true })
|
|
1119
|
+
|
|
1120
|
+
// space is charCode 32, ctrl is 4, modifier is 4+1=5
|
|
1121
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;32~")
|
|
1122
|
+
})
|
|
1123
|
+
|
|
1124
|
+
test("special characters with ctrl in modifyOtherKeys mode", () => {
|
|
1125
|
+
const mockRenderer = new MockRenderer()
|
|
1126
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1127
|
+
|
|
1128
|
+
// Ctrl+- should use modifyOtherKeys format
|
|
1129
|
+
mockRenderer.emittedData = []
|
|
1130
|
+
mockKeys.pressKey("-", { ctrl: true })
|
|
1131
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;45~") // '-' is charCode 45
|
|
1132
|
+
|
|
1133
|
+
// Ctrl+. should use modifyOtherKeys format
|
|
1134
|
+
mockRenderer.emittedData = []
|
|
1135
|
+
mockKeys.pressKey(".", { ctrl: true })
|
|
1136
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;46~") // '.' is charCode 46
|
|
1137
|
+
|
|
1138
|
+
// Ctrl+, should use modifyOtherKeys format
|
|
1139
|
+
mockRenderer.emittedData = []
|
|
1140
|
+
mockKeys.pressKey(",", { ctrl: true })
|
|
1141
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;44~") // ',' is charCode 44
|
|
1142
|
+
|
|
1143
|
+
// Ctrl+] should use modifyOtherKeys format
|
|
1144
|
+
mockRenderer.emittedData = []
|
|
1145
|
+
mockKeys.pressKey("]", { ctrl: true })
|
|
1146
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;93~") // ']' is charCode 93
|
|
1147
|
+
})
|
|
1148
|
+
|
|
1149
|
+
test("multiple modifier combinations in modifyOtherKeys mode", () => {
|
|
1150
|
+
const mockRenderer = new MockRenderer()
|
|
1151
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1152
|
+
|
|
1153
|
+
// shift + ctrl: 1 + 4 = 5, + 1 = 6
|
|
1154
|
+
mockRenderer.emittedData = []
|
|
1155
|
+
mockKeys.pressKey("a", { shift: true, ctrl: true })
|
|
1156
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;6;97~")
|
|
1157
|
+
|
|
1158
|
+
// shift + meta: 1 + 2 = 3, + 1 = 4
|
|
1159
|
+
mockRenderer.emittedData = []
|
|
1160
|
+
mockKeys.pressKey("a", { shift: true, meta: true })
|
|
1161
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;4;97~")
|
|
1162
|
+
|
|
1163
|
+
// ctrl + meta: 4 + 2 = 6, + 1 = 7
|
|
1164
|
+
mockRenderer.emittedData = []
|
|
1165
|
+
mockKeys.pressKey("a", { ctrl: true, meta: true })
|
|
1166
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;7;97~")
|
|
1167
|
+
|
|
1168
|
+
// shift + ctrl + meta: 1 + 4 + 2 = 7, + 1 = 8
|
|
1169
|
+
mockRenderer.emittedData = []
|
|
1170
|
+
mockKeys.pressKey("a", { shift: true, ctrl: true, meta: true })
|
|
1171
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;8;97~")
|
|
1172
|
+
})
|
|
1173
|
+
|
|
1174
|
+
test("character with super modifier in modifyOtherKeys mode", () => {
|
|
1175
|
+
const mockRenderer = new MockRenderer()
|
|
1176
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1177
|
+
|
|
1178
|
+
mockKeys.pressKey("a", { super: true })
|
|
1179
|
+
|
|
1180
|
+
// 'a' is charCode 97, super is 8, modifier is 8+1=9
|
|
1181
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;9;97~")
|
|
1182
|
+
})
|
|
1183
|
+
|
|
1184
|
+
test("character with hyper modifier in modifyOtherKeys mode", () => {
|
|
1185
|
+
const mockRenderer = new MockRenderer()
|
|
1186
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1187
|
+
|
|
1188
|
+
mockKeys.pressKey("a", { hyper: true })
|
|
1189
|
+
|
|
1190
|
+
// 'a' is charCode 97, hyper is 16, modifier is 16+1=17
|
|
1191
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;17;97~")
|
|
1192
|
+
})
|
|
1193
|
+
|
|
1194
|
+
test("character with super+hyper modifiers in modifyOtherKeys mode", () => {
|
|
1195
|
+
const mockRenderer = new MockRenderer()
|
|
1196
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1197
|
+
|
|
1198
|
+
mockKeys.pressKey("a", { super: true, hyper: true })
|
|
1199
|
+
|
|
1200
|
+
// super(8) + hyper(16) = 24, +1 = 25
|
|
1201
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;25;97~")
|
|
1202
|
+
})
|
|
1203
|
+
|
|
1204
|
+
test("character with all modifiers in modifyOtherKeys mode", () => {
|
|
1205
|
+
const mockRenderer = new MockRenderer()
|
|
1206
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1207
|
+
|
|
1208
|
+
mockKeys.pressKey("a", { shift: true, ctrl: true, meta: true, super: true, hyper: true })
|
|
1209
|
+
|
|
1210
|
+
// shift(1) + meta(2) + ctrl(4) + super(8) + hyper(16) = 31, +1 = 32
|
|
1211
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;32;97~")
|
|
1212
|
+
})
|
|
1213
|
+
|
|
1214
|
+
test("arrow keys with modifiers fall through to regular mode in modifyOtherKeys", () => {
|
|
1215
|
+
const mockRenderer = new MockRenderer()
|
|
1216
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1217
|
+
|
|
1218
|
+
// Arrow keys should still use the standard CSI sequence with modifiers
|
|
1219
|
+
// not the modifyOtherKeys format
|
|
1220
|
+
mockKeys.pressArrow("right", { shift: true })
|
|
1221
|
+
|
|
1222
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[1;2C")
|
|
1223
|
+
})
|
|
1224
|
+
|
|
1225
|
+
test("kitty mode takes precedence over modifyOtherKeys mode", () => {
|
|
1226
|
+
const mockRenderer = new MockRenderer()
|
|
1227
|
+
const mockKeys = createMockKeys(mockRenderer as any, {
|
|
1228
|
+
kittyKeyboard: true,
|
|
1229
|
+
otherModifiersMode: true,
|
|
1230
|
+
})
|
|
1231
|
+
|
|
1232
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
1233
|
+
|
|
1234
|
+
// Should use kitty format, not modifyOtherKeys
|
|
1235
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;5u")
|
|
1236
|
+
})
|
|
1237
|
+
|
|
1238
|
+
test("modifyOtherKeys vs regular mode comparison", () => {
|
|
1239
|
+
const modifyOtherKeysRenderer = new MockRenderer()
|
|
1240
|
+
const regularRenderer = new MockRenderer()
|
|
1241
|
+
const modifyOtherKeysKeys = createMockKeys(modifyOtherKeysRenderer as any, { otherModifiersMode: true })
|
|
1242
|
+
const regularKeys = createMockKeys(regularRenderer as any, { otherModifiersMode: false })
|
|
1243
|
+
|
|
1244
|
+
modifyOtherKeysKeys.pressKey("-", { ctrl: true })
|
|
1245
|
+
regularKeys.pressKey("-", { ctrl: true })
|
|
1246
|
+
|
|
1247
|
+
// modifyOtherKeys should send CSI 27 format
|
|
1248
|
+
expect(modifyOtherKeysRenderer.getEmittedData()).toBe("\x1b[27;5;45~")
|
|
1249
|
+
// Regular should send raw control sequence
|
|
1250
|
+
expect(regularRenderer.getEmittedData()).toBe("\u001f")
|
|
1251
|
+
})
|
|
1252
|
+
|
|
1253
|
+
test("characters without modifiers don't use modifyOtherKeys format", () => {
|
|
1254
|
+
const mockRenderer = new MockRenderer()
|
|
1255
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1256
|
+
|
|
1257
|
+
// Without modifiers, should send plain characters
|
|
1258
|
+
mockKeys.pressKey("a")
|
|
1259
|
+
mockKeys.pressKey("b")
|
|
1260
|
+
mockKeys.pressEnter()
|
|
1261
|
+
|
|
1262
|
+
expect(mockRenderer.getEmittedData()).toBe("ab\r")
|
|
1263
|
+
})
|
|
1264
|
+
|
|
1265
|
+
test("modifyOtherKeys with all printable characters", () => {
|
|
1266
|
+
const mockRenderer = new MockRenderer()
|
|
1267
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1268
|
+
|
|
1269
|
+
const chars = "abcdefghijklmnopqrstuvwxyz0123456789-=[]\\;',./`"
|
|
1270
|
+
|
|
1271
|
+
for (const char of chars) {
|
|
1272
|
+
mockRenderer.emittedData = []
|
|
1273
|
+
mockKeys.pressKey(char, { ctrl: true })
|
|
1274
|
+
const charCode = char.charCodeAt(0)
|
|
1275
|
+
expect(mockRenderer.getEmittedData()).toBe(`\x1b[27;5;${charCode}~`)
|
|
1276
|
+
}
|
|
1277
|
+
})
|
|
1278
|
+
|
|
1279
|
+
test("modifyOtherKeys mode can be parsed back correctly", () => {
|
|
1280
|
+
const mockRenderer = new MockRenderer()
|
|
1281
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1282
|
+
|
|
1283
|
+
// Test various keys with modifiers
|
|
1284
|
+
const tests = [
|
|
1285
|
+
{ key: "a", mods: { ctrl: true }, expectedSeq: "\x1b[27;5;97~" },
|
|
1286
|
+
{ key: "-", mods: { ctrl: true }, expectedSeq: "\x1b[27;5;45~" },
|
|
1287
|
+
{ key: KeyCodes.RETURN, mods: { shift: true }, expectedSeq: "\x1b[27;2;13~" },
|
|
1288
|
+
{ key: KeyCodes.ESCAPE, mods: { ctrl: true }, expectedSeq: "\x1b[27;5;27~" },
|
|
1289
|
+
{ key: KeyCodes.TAB, mods: { shift: true }, expectedSeq: "\x1b[27;2;9~" },
|
|
1290
|
+
{ key: " ", mods: { ctrl: true }, expectedSeq: "\x1b[27;5;32~" },
|
|
1291
|
+
]
|
|
1292
|
+
|
|
1293
|
+
for (const { key, mods, expectedSeq } of tests) {
|
|
1294
|
+
mockRenderer.emittedData = []
|
|
1295
|
+
mockKeys.pressKey(key, mods)
|
|
1296
|
+
expect(mockRenderer.getEmittedData()).toBe(expectedSeq)
|
|
1297
|
+
}
|
|
1298
|
+
})
|
|
1299
|
+
|
|
1300
|
+
test("comprehensive three-mode comparison: regular vs modifyOtherKeys vs kitty", () => {
|
|
1301
|
+
const regularRenderer = new MockRenderer()
|
|
1302
|
+
const modifyOtherKeysRenderer = new MockRenderer()
|
|
1303
|
+
const kittyRenderer = new MockRenderer()
|
|
1304
|
+
|
|
1305
|
+
const regularKeys = createMockKeys(regularRenderer as any, { kittyKeyboard: false, otherModifiersMode: false })
|
|
1306
|
+
const modifyOtherKeysKeys = createMockKeys(modifyOtherKeysRenderer as any, { otherModifiersMode: true })
|
|
1307
|
+
const kittyKeys = createMockKeys(kittyRenderer as any, { kittyKeyboard: true })
|
|
1308
|
+
|
|
1309
|
+
// Test Ctrl+- in all three modes
|
|
1310
|
+
regularKeys.pressKey("-", { ctrl: true })
|
|
1311
|
+
modifyOtherKeysKeys.pressKey("-", { ctrl: true })
|
|
1312
|
+
kittyKeys.pressKey("-", { ctrl: true })
|
|
1313
|
+
|
|
1314
|
+
expect(regularRenderer.getEmittedData()).toBe("\u001f") // Raw control sequence
|
|
1315
|
+
expect(modifyOtherKeysRenderer.getEmittedData()).toBe("\x1b[27;5;45~") // modifyOtherKeys format
|
|
1316
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[45;5u") // Kitty format
|
|
1317
|
+
|
|
1318
|
+
// Test Shift+Enter in all three modes
|
|
1319
|
+
regularRenderer.emittedData = []
|
|
1320
|
+
modifyOtherKeysRenderer.emittedData = []
|
|
1321
|
+
kittyRenderer.emittedData = []
|
|
1322
|
+
|
|
1323
|
+
regularKeys.pressEnter({ shift: true })
|
|
1324
|
+
modifyOtherKeysKeys.pressEnter({ shift: true })
|
|
1325
|
+
kittyKeys.pressEnter({ shift: true })
|
|
1326
|
+
|
|
1327
|
+
expect(regularRenderer.getEmittedData()).toBe("\r") // Regular mode ignores shift on Enter
|
|
1328
|
+
expect(modifyOtherKeysRenderer.getEmittedData()).toBe("\x1b[27;2;13~") // modifyOtherKeys format
|
|
1329
|
+
expect(kittyRenderer.getEmittedData()).toBe("\x1b[13;2u") // Kitty format
|
|
1330
|
+
})
|
|
1331
|
+
})
|
|
1332
|
+
|
|
1333
|
+
describe("Mode selection and precedence", () => {
|
|
1334
|
+
test("default mode (no options)", () => {
|
|
1335
|
+
const mockRenderer = new MockRenderer()
|
|
1336
|
+
const mockKeys = createMockKeys(mockRenderer as any)
|
|
1337
|
+
|
|
1338
|
+
mockKeys.pressKey("-", { ctrl: true })
|
|
1339
|
+
|
|
1340
|
+
// Default should use raw control sequences
|
|
1341
|
+
expect(mockRenderer.getEmittedData()).toBe("\u001f")
|
|
1342
|
+
})
|
|
1343
|
+
|
|
1344
|
+
test("only kittyKeyboard enabled", () => {
|
|
1345
|
+
const mockRenderer = new MockRenderer()
|
|
1346
|
+
const mockKeys = createMockKeys(mockRenderer as any, { kittyKeyboard: true })
|
|
1347
|
+
|
|
1348
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
1349
|
+
|
|
1350
|
+
// Should use kitty format
|
|
1351
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;5u")
|
|
1352
|
+
})
|
|
1353
|
+
|
|
1354
|
+
test("only otherModifiersMode enabled", () => {
|
|
1355
|
+
const mockRenderer = new MockRenderer()
|
|
1356
|
+
const mockKeys = createMockKeys(mockRenderer as any, { otherModifiersMode: true })
|
|
1357
|
+
|
|
1358
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
1359
|
+
|
|
1360
|
+
// Should use modifyOtherKeys format
|
|
1361
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[27;5;97~")
|
|
1362
|
+
})
|
|
1363
|
+
|
|
1364
|
+
test("both kittyKeyboard and otherModifiersMode enabled (kitty wins)", () => {
|
|
1365
|
+
const mockRenderer = new MockRenderer()
|
|
1366
|
+
const mockKeys = createMockKeys(mockRenderer as any, {
|
|
1367
|
+
kittyKeyboard: true,
|
|
1368
|
+
otherModifiersMode: true,
|
|
1369
|
+
})
|
|
1370
|
+
|
|
1371
|
+
mockKeys.pressKey("a", { ctrl: true })
|
|
1372
|
+
|
|
1373
|
+
// Kitty should take precedence
|
|
1374
|
+
expect(mockRenderer.getEmittedData()).toBe("\x1b[97;5u")
|
|
1375
|
+
expect(mockRenderer.getEmittedData()).not.toContain("27;5;97~")
|
|
1376
|
+
})
|
|
1377
|
+
})
|
|
1378
|
+
})
|