@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,629 @@
|
|
|
1
|
+
import { test, expect, describe } from "bun:test"
|
|
2
|
+
import { createTestRenderer, type TestRendererOptions } from "../testing/test-renderer.js"
|
|
3
|
+
import { EventEmitter } from "events"
|
|
4
|
+
import { Buffer } from "node:buffer"
|
|
5
|
+
import { Readable } from "node:stream"
|
|
6
|
+
import tty from "tty"
|
|
7
|
+
import { ManualClock } from "../testing/manual-clock.js"
|
|
8
|
+
import type { GetPaletteOptions, TerminalColors } from "../lib/terminal-palette.js"
|
|
9
|
+
|
|
10
|
+
const OSC_SUPPORT_TIMEOUT_MS = 300
|
|
11
|
+
|
|
12
|
+
function flushAsync(): Promise<void> {
|
|
13
|
+
return Promise.resolve().then(() => Promise.resolve())
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function schedule(clock: ManualClock | undefined, fn: () => void): void {
|
|
17
|
+
if (clock) {
|
|
18
|
+
clock.setTimeout(fn, 0)
|
|
19
|
+
return
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
process.nextTick(fn)
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
function createMockStreams(clock?: ManualClock) {
|
|
26
|
+
const mockStdin = new Readable({ read() {} }) as tty.ReadStream
|
|
27
|
+
mockStdin.isTTY = true
|
|
28
|
+
mockStdin.setRawMode = () => mockStdin
|
|
29
|
+
mockStdin.resume = () => mockStdin
|
|
30
|
+
mockStdin.pause = () => mockStdin
|
|
31
|
+
mockStdin.setEncoding = () => mockStdin
|
|
32
|
+
|
|
33
|
+
const writes: string[] = []
|
|
34
|
+
const mockStdout = {
|
|
35
|
+
isTTY: true,
|
|
36
|
+
columns: 80,
|
|
37
|
+
rows: 24,
|
|
38
|
+
write: (data: string | Buffer) => {
|
|
39
|
+
writes.push(data.toString())
|
|
40
|
+
const dataStr = data.toString()
|
|
41
|
+
if (dataStr === "\x1b]4;0;?\x07") {
|
|
42
|
+
schedule(clock, () => {
|
|
43
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;0;rgb:0000/0000/0000\x07"))
|
|
44
|
+
})
|
|
45
|
+
} else if (dataStr.includes("\x1b]4;")) {
|
|
46
|
+
schedule(clock, () => {
|
|
47
|
+
for (let i = 0; i < 16; i++) {
|
|
48
|
+
mockStdin.emit("data", Buffer.from(`\x1b]4;${i};rgb:1000/2000/3000\x07`))
|
|
49
|
+
}
|
|
50
|
+
})
|
|
51
|
+
} else if (dataStr.includes("\x1b]10;?")) {
|
|
52
|
+
schedule(clock, () => {
|
|
53
|
+
mockStdin.emit("data", Buffer.from("\x1b]10;#ffffff\x07"))
|
|
54
|
+
mockStdin.emit("data", Buffer.from("\x1b]11;#000000\x07"))
|
|
55
|
+
mockStdin.emit("data", Buffer.from("\x1b]12;#00ff00\x07"))
|
|
56
|
+
mockStdin.emit("data", Buffer.from("\x1b]13;#ffffff\x07"))
|
|
57
|
+
mockStdin.emit("data", Buffer.from("\x1b]14;#000000\x07"))
|
|
58
|
+
mockStdin.emit("data", Buffer.from("\x1b]15;#ffffff\x07"))
|
|
59
|
+
mockStdin.emit("data", Buffer.from("\x1b]16;#000000\x07"))
|
|
60
|
+
mockStdin.emit("data", Buffer.from("\x1b]17;#333333\x07"))
|
|
61
|
+
mockStdin.emit("data", Buffer.from("\x1b]19;#cccccc\x07"))
|
|
62
|
+
})
|
|
63
|
+
}
|
|
64
|
+
return true
|
|
65
|
+
},
|
|
66
|
+
} as any
|
|
67
|
+
|
|
68
|
+
return { mockStdin, mockStdout, writes }
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
async function advancePaletteClock(clock: ManualClock, ms: number): Promise<void> {
|
|
72
|
+
await flushAsync()
|
|
73
|
+
// Flush queued 0ms mock terminal responses before advancing the real timeout window.
|
|
74
|
+
clock.advance(0)
|
|
75
|
+
await flushAsync()
|
|
76
|
+
clock.advance(ms)
|
|
77
|
+
await flushAsync()
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
async function detectPaletteAndAdvanceClock(
|
|
81
|
+
renderer: {
|
|
82
|
+
getPalette(options?: GetPaletteOptions): Promise<TerminalColors>
|
|
83
|
+
paletteDetectionStatus: "idle" | "detecting" | "cached"
|
|
84
|
+
},
|
|
85
|
+
clock: ManualClock,
|
|
86
|
+
options?: GetPaletteOptions,
|
|
87
|
+
): Promise<TerminalColors> {
|
|
88
|
+
const palettePromise = renderer.getPalette(options)
|
|
89
|
+
|
|
90
|
+
if (renderer.paletteDetectionStatus === "detecting") {
|
|
91
|
+
const detectionTimeoutMs = Math.max(options?.timeout ?? 5000, OSC_SUPPORT_TIMEOUT_MS)
|
|
92
|
+
await advancePaletteClock(clock, detectionTimeoutMs)
|
|
93
|
+
} else {
|
|
94
|
+
await flushAsync()
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
return palettePromise
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
async function createPaletteRenderer(options: Partial<TestRendererOptions> = {}) {
|
|
101
|
+
const clock = options.clock instanceof ManualClock ? options.clock : new ManualClock()
|
|
102
|
+
const { mockStdin, mockStdout, writes } = createMockStreams(clock)
|
|
103
|
+
const { renderer } = await createTestRenderer({
|
|
104
|
+
stdin: mockStdin,
|
|
105
|
+
stdout: mockStdout,
|
|
106
|
+
...options,
|
|
107
|
+
clock,
|
|
108
|
+
})
|
|
109
|
+
|
|
110
|
+
return { renderer, mockStdin, mockStdout, writes, clock }
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
describe("Palette caching behavior", () => {
|
|
114
|
+
test("getPalette returns cached palette on subsequent calls", async () => {
|
|
115
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
116
|
+
|
|
117
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
118
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
119
|
+
|
|
120
|
+
expect(palette1).toBe(palette2)
|
|
121
|
+
expect(palette1).toEqual(palette2)
|
|
122
|
+
|
|
123
|
+
renderer.destroy()
|
|
124
|
+
})
|
|
125
|
+
|
|
126
|
+
test("getPalette caches correctly with non-256 size parameter", async () => {
|
|
127
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
128
|
+
|
|
129
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
130
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
131
|
+
|
|
132
|
+
expect(palette1).toBe(palette2)
|
|
133
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
134
|
+
|
|
135
|
+
renderer.destroy()
|
|
136
|
+
})
|
|
137
|
+
|
|
138
|
+
test("cached palette is returned instantly", async () => {
|
|
139
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer()
|
|
140
|
+
|
|
141
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
142
|
+
const writeCountAfterFirst = writes.length
|
|
143
|
+
|
|
144
|
+
const timeAfterFirstDetection = clock.now()
|
|
145
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
146
|
+
|
|
147
|
+
expect(clock.now()).toBe(timeAfterFirstDetection)
|
|
148
|
+
expect(writes.length).toBe(writeCountAfterFirst)
|
|
149
|
+
|
|
150
|
+
renderer.destroy()
|
|
151
|
+
})
|
|
152
|
+
|
|
153
|
+
test("multiple concurrent calls share same detection", async () => {
|
|
154
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer()
|
|
155
|
+
|
|
156
|
+
const palettePromises = [
|
|
157
|
+
renderer.getPalette({ timeout: 300 }),
|
|
158
|
+
renderer.getPalette({ timeout: 300 }),
|
|
159
|
+
renderer.getPalette({ timeout: 300 }),
|
|
160
|
+
]
|
|
161
|
+
|
|
162
|
+
await advancePaletteClock(clock, 300)
|
|
163
|
+
|
|
164
|
+
const [palette1, palette2, palette3] = await Promise.all(palettePromises)
|
|
165
|
+
|
|
166
|
+
expect(palette1).toBe(palette2)
|
|
167
|
+
expect(palette2).toBe(palette3)
|
|
168
|
+
|
|
169
|
+
const oscSupportChecks = writes.filter((w) => w.includes("\x1b]4;0;?"))
|
|
170
|
+
expect(oscSupportChecks.length).toBeLessThanOrEqual(2)
|
|
171
|
+
|
|
172
|
+
renderer.destroy()
|
|
173
|
+
})
|
|
174
|
+
|
|
175
|
+
test("palette detector created only once", async () => {
|
|
176
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
177
|
+
|
|
178
|
+
// @ts-expect-error - accessing private property for testing
|
|
179
|
+
expect(renderer._paletteDetector).toBeNull()
|
|
180
|
+
|
|
181
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
182
|
+
|
|
183
|
+
// @ts-expect-error - accessing private property for testing
|
|
184
|
+
const detector1 = renderer._paletteDetector
|
|
185
|
+
expect(detector1).not.toBeNull()
|
|
186
|
+
|
|
187
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
188
|
+
|
|
189
|
+
// @ts-expect-error - accessing private property for testing
|
|
190
|
+
const detector2 = renderer._paletteDetector
|
|
191
|
+
expect(detector1).toBe(detector2)
|
|
192
|
+
|
|
193
|
+
renderer.destroy()
|
|
194
|
+
})
|
|
195
|
+
|
|
196
|
+
test("cache persists with different timeout values", async () => {
|
|
197
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer()
|
|
198
|
+
|
|
199
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
200
|
+
const writeCountAfterFirst = writes.length
|
|
201
|
+
|
|
202
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 5000 })
|
|
203
|
+
|
|
204
|
+
expect(writes.length).toBe(writeCountAfterFirst)
|
|
205
|
+
expect(palette1).toBe(palette2)
|
|
206
|
+
|
|
207
|
+
renderer.destroy()
|
|
208
|
+
})
|
|
209
|
+
|
|
210
|
+
test("cache persists across renderer lifecycle", async () => {
|
|
211
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
212
|
+
|
|
213
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
214
|
+
|
|
215
|
+
renderer.start()
|
|
216
|
+
await flushAsync()
|
|
217
|
+
renderer.pause()
|
|
218
|
+
renderer.suspend()
|
|
219
|
+
renderer.resume()
|
|
220
|
+
renderer.stop()
|
|
221
|
+
|
|
222
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
223
|
+
expect(palette1).toBe(palette2)
|
|
224
|
+
|
|
225
|
+
renderer.destroy()
|
|
226
|
+
})
|
|
227
|
+
})
|
|
228
|
+
|
|
229
|
+
describe("Palette detection with non-TTY", () => {
|
|
230
|
+
test("handles non-TTY streams gracefully", async () => {
|
|
231
|
+
const clock = new ManualClock()
|
|
232
|
+
const mockStdin = new EventEmitter() as any
|
|
233
|
+
mockStdin.isTTY = false
|
|
234
|
+
mockStdin.setRawMode = () => {}
|
|
235
|
+
mockStdin.resume = () => {}
|
|
236
|
+
mockStdin.pause = () => {}
|
|
237
|
+
mockStdin.setEncoding = () => {}
|
|
238
|
+
|
|
239
|
+
const mockStdout = {
|
|
240
|
+
isTTY: false,
|
|
241
|
+
columns: 80,
|
|
242
|
+
rows: 24,
|
|
243
|
+
write: () => true,
|
|
244
|
+
} as any
|
|
245
|
+
|
|
246
|
+
const { renderer } = await createTestRenderer({
|
|
247
|
+
stdin: mockStdin,
|
|
248
|
+
stdout: mockStdout,
|
|
249
|
+
clock,
|
|
250
|
+
})
|
|
251
|
+
|
|
252
|
+
const palette = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
253
|
+
|
|
254
|
+
expect(typeof palette === "object" && palette !== null && Array.isArray(palette.palette)).toBe(true)
|
|
255
|
+
|
|
256
|
+
const cached = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
257
|
+
expect(palette).toBe(cached)
|
|
258
|
+
|
|
259
|
+
renderer.destroy()
|
|
260
|
+
})
|
|
261
|
+
})
|
|
262
|
+
|
|
263
|
+
describe("Palette detection with OSC responses", () => {
|
|
264
|
+
test("detects colors from OSC responses", async () => {
|
|
265
|
+
const clock = new ManualClock()
|
|
266
|
+
const mockStdin = new EventEmitter() as any
|
|
267
|
+
mockStdin.isTTY = true
|
|
268
|
+
mockStdin.setRawMode = () => {}
|
|
269
|
+
mockStdin.resume = () => {}
|
|
270
|
+
mockStdin.pause = () => {}
|
|
271
|
+
mockStdin.setEncoding = () => {}
|
|
272
|
+
|
|
273
|
+
const mockStdout = {
|
|
274
|
+
isTTY: true,
|
|
275
|
+
columns: 80,
|
|
276
|
+
rows: 24,
|
|
277
|
+
write: (data: string | Buffer) => {
|
|
278
|
+
const dataStr = data.toString()
|
|
279
|
+
clock.setTimeout(() => {
|
|
280
|
+
if (dataStr.includes("\x1b]4;0;?")) {
|
|
281
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;0;#000000\x07"))
|
|
282
|
+
}
|
|
283
|
+
if (dataStr.match(/\x1b\]4;\d+;/g)) {
|
|
284
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;0;#000000\x07"))
|
|
285
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;1;#ff0000\x07"))
|
|
286
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;2;#00ff00\x07"))
|
|
287
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;3;#0000ff\x07"))
|
|
288
|
+
for (let i = 4; i < 256; i++) {
|
|
289
|
+
mockStdin.emit("data", Buffer.from(`\x1b]4;${i};#808080\x07`))
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
}, 0)
|
|
293
|
+
return true
|
|
294
|
+
},
|
|
295
|
+
} as any
|
|
296
|
+
|
|
297
|
+
const { renderer } = await createTestRenderer({
|
|
298
|
+
stdin: mockStdin,
|
|
299
|
+
stdout: mockStdout,
|
|
300
|
+
useThread: false,
|
|
301
|
+
clock,
|
|
302
|
+
})
|
|
303
|
+
|
|
304
|
+
const palette = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
305
|
+
|
|
306
|
+
expect(typeof palette === "object" && palette !== null && Array.isArray(palette.palette)).toBe(true)
|
|
307
|
+
expect(palette.palette.length).toBeGreaterThanOrEqual(16)
|
|
308
|
+
expect(palette.palette[0]).toBe("#000000")
|
|
309
|
+
expect(palette.palette[1]).toBe("#ff0000")
|
|
310
|
+
expect(palette.palette[2]).toBe("#00ff00")
|
|
311
|
+
expect(palette.palette[3]).toBe("#0000ff")
|
|
312
|
+
|
|
313
|
+
const cached = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
314
|
+
expect(palette).toBe(cached)
|
|
315
|
+
|
|
316
|
+
renderer.destroy()
|
|
317
|
+
})
|
|
318
|
+
|
|
319
|
+
test("handles RGB format responses", async () => {
|
|
320
|
+
const clock = new ManualClock()
|
|
321
|
+
const mockStdin = new EventEmitter() as any
|
|
322
|
+
mockStdin.isTTY = true
|
|
323
|
+
mockStdin.setRawMode = () => {}
|
|
324
|
+
mockStdin.resume = () => {}
|
|
325
|
+
mockStdin.pause = () => {}
|
|
326
|
+
mockStdin.setEncoding = () => {}
|
|
327
|
+
|
|
328
|
+
const mockStdout = {
|
|
329
|
+
isTTY: true,
|
|
330
|
+
columns: 80,
|
|
331
|
+
rows: 24,
|
|
332
|
+
write: (data: string | Buffer) => {
|
|
333
|
+
const dataStr = data.toString()
|
|
334
|
+
clock.setTimeout(() => {
|
|
335
|
+
if (dataStr.includes("\x1b]4;0;?")) {
|
|
336
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;0;rgb:0000/0000/0000\x07"))
|
|
337
|
+
}
|
|
338
|
+
if (dataStr.match(/\x1b\]4;\d+;/g)) {
|
|
339
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;0;rgb:0000/0000/0000\x07"))
|
|
340
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;1;rgb:ffff/0000/0000\x07"))
|
|
341
|
+
mockStdin.emit("data", Buffer.from("\x1b]4;2;rgb:8000/8000/8000\x07"))
|
|
342
|
+
for (let i = 3; i < 256; i++) {
|
|
343
|
+
mockStdin.emit("data", Buffer.from(`\x1b]4;${i};rgb:1111/1111/1111\x07`))
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
}, 0)
|
|
347
|
+
return true
|
|
348
|
+
},
|
|
349
|
+
} as any
|
|
350
|
+
|
|
351
|
+
const { renderer } = await createTestRenderer({
|
|
352
|
+
stdin: mockStdin,
|
|
353
|
+
stdout: mockStdout,
|
|
354
|
+
useThread: false,
|
|
355
|
+
clock,
|
|
356
|
+
})
|
|
357
|
+
|
|
358
|
+
const palette = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
359
|
+
|
|
360
|
+
expect(palette.palette[0]).toBe("#000000")
|
|
361
|
+
expect(palette.palette[1]).toBe("#ff0000")
|
|
362
|
+
expect(palette.palette[2]).toBe("#808080")
|
|
363
|
+
|
|
364
|
+
renderer.destroy()
|
|
365
|
+
})
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
describe("Palette integration tests", () => {
|
|
369
|
+
test("palette detection does not interfere with input handling", async () => {
|
|
370
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
371
|
+
|
|
372
|
+
const keysReceived: string[] = []
|
|
373
|
+
renderer.keyInput.on("keypress", (event) => {
|
|
374
|
+
keysReceived.push(event.name || "unknown")
|
|
375
|
+
})
|
|
376
|
+
|
|
377
|
+
const palettePromise = renderer.getPalette({ timeout: 300 })
|
|
378
|
+
|
|
379
|
+
mockStdin.emit("data", Buffer.from("a"))
|
|
380
|
+
mockStdin.emit("data", Buffer.from("b"))
|
|
381
|
+
mockStdin.emit("data", Buffer.from("c"))
|
|
382
|
+
|
|
383
|
+
await flushAsync()
|
|
384
|
+
|
|
385
|
+
expect(keysReceived.length).toBeGreaterThanOrEqual(3)
|
|
386
|
+
|
|
387
|
+
await advancePaletteClock(clock, 300)
|
|
388
|
+
await palettePromise
|
|
389
|
+
|
|
390
|
+
renderer.destroy()
|
|
391
|
+
})
|
|
392
|
+
|
|
393
|
+
test("getPalette works with different renderer configurations", async () => {
|
|
394
|
+
const configs = [{ width: 40, height: 10 }, { width: 120, height: 40 }, { useMouse: false }]
|
|
395
|
+
|
|
396
|
+
for (const config of configs) {
|
|
397
|
+
const { renderer: testRenderer, clock, mockStdin, mockStdout } = await createPaletteRenderer(config)
|
|
398
|
+
|
|
399
|
+
const palette = await detectPaletteAndAdvanceClock(testRenderer, clock, { timeout: 300 })
|
|
400
|
+
expect(typeof palette === "object" && palette !== null && Array.isArray(palette.palette)).toBe(true)
|
|
401
|
+
|
|
402
|
+
const cached = await detectPaletteAndAdvanceClock(testRenderer, clock, { timeout: 100 })
|
|
403
|
+
expect(palette).toBe(cached)
|
|
404
|
+
|
|
405
|
+
testRenderer.destroy()
|
|
406
|
+
}
|
|
407
|
+
})
|
|
408
|
+
})
|
|
409
|
+
|
|
410
|
+
describe("Palette cache invalidation", () => {
|
|
411
|
+
test("clearPaletteCache invalidates cache", async () => {
|
|
412
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
413
|
+
|
|
414
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
415
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
416
|
+
|
|
417
|
+
renderer.clearPaletteCache()
|
|
418
|
+
expect(renderer.paletteDetectionStatus).toBe("idle")
|
|
419
|
+
|
|
420
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
421
|
+
|
|
422
|
+
expect(palette1).not.toBe(palette2)
|
|
423
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
424
|
+
|
|
425
|
+
renderer.destroy()
|
|
426
|
+
})
|
|
427
|
+
|
|
428
|
+
test("paletteDetectionStatus tracks detection lifecycle", async () => {
|
|
429
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
430
|
+
|
|
431
|
+
expect(renderer.paletteDetectionStatus).toBe("idle")
|
|
432
|
+
|
|
433
|
+
const palettePromise = renderer.getPalette({ timeout: 300 })
|
|
434
|
+
expect(renderer.paletteDetectionStatus).toBe("detecting")
|
|
435
|
+
|
|
436
|
+
await advancePaletteClock(clock, 300)
|
|
437
|
+
await palettePromise
|
|
438
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
439
|
+
|
|
440
|
+
renderer.destroy()
|
|
441
|
+
})
|
|
442
|
+
})
|
|
443
|
+
|
|
444
|
+
describe("Palette detection with suspended renderer", () => {
|
|
445
|
+
test("getPalette throws error when renderer is suspended", async () => {
|
|
446
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
447
|
+
|
|
448
|
+
renderer.suspend()
|
|
449
|
+
|
|
450
|
+
await expect(renderer.getPalette({ timeout: 300 })).rejects.toThrow(
|
|
451
|
+
"Cannot detect palette while renderer is suspended",
|
|
452
|
+
)
|
|
453
|
+
|
|
454
|
+
renderer.destroy()
|
|
455
|
+
})
|
|
456
|
+
|
|
457
|
+
test("getPalette works after resume", async () => {
|
|
458
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
459
|
+
|
|
460
|
+
renderer.suspend()
|
|
461
|
+
renderer.resume()
|
|
462
|
+
|
|
463
|
+
const palette = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
464
|
+
expect(typeof palette === "object" && palette !== null && Array.isArray(palette.palette)).toBe(true)
|
|
465
|
+
|
|
466
|
+
renderer.destroy()
|
|
467
|
+
})
|
|
468
|
+
})
|
|
469
|
+
|
|
470
|
+
describe("Palette detector cleanup", () => {
|
|
471
|
+
test("destroy cleans up palette detector", async () => {
|
|
472
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
473
|
+
|
|
474
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
475
|
+
|
|
476
|
+
renderer.destroy()
|
|
477
|
+
|
|
478
|
+
// @ts-expect-error - accessing private property for testing
|
|
479
|
+
expect(renderer._paletteDetector).toBeNull()
|
|
480
|
+
// @ts-expect-error - accessing private property for testing
|
|
481
|
+
expect(renderer._paletteDetectionPromise).toBeNull()
|
|
482
|
+
// @ts-expect-error - accessing private property for testing
|
|
483
|
+
expect(renderer._cachedPalette).toBeNull()
|
|
484
|
+
})
|
|
485
|
+
|
|
486
|
+
test("multiple destroy calls don't cause errors", async () => {
|
|
487
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
488
|
+
|
|
489
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 300 })
|
|
490
|
+
|
|
491
|
+
expect(() => {
|
|
492
|
+
renderer.destroy()
|
|
493
|
+
renderer.destroy()
|
|
494
|
+
renderer.destroy()
|
|
495
|
+
}).not.toThrow()
|
|
496
|
+
})
|
|
497
|
+
|
|
498
|
+
test("palette detection uses router OSC source without extra stdin listeners", async () => {
|
|
499
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
500
|
+
|
|
501
|
+
const baselineListenerCount = mockStdin.listenerCount("data")
|
|
502
|
+
const palettePromise = renderer.getPalette({ timeout: 300 })
|
|
503
|
+
|
|
504
|
+
const duringDetectionCount = mockStdin.listenerCount("data")
|
|
505
|
+
expect(duringDetectionCount).toBe(baselineListenerCount)
|
|
506
|
+
|
|
507
|
+
await advancePaletteClock(clock, 300)
|
|
508
|
+
await palettePromise
|
|
509
|
+
|
|
510
|
+
const afterDetectionCount = mockStdin.listenerCount("data")
|
|
511
|
+
expect(afterDetectionCount).toBe(baselineListenerCount)
|
|
512
|
+
|
|
513
|
+
renderer.destroy()
|
|
514
|
+
})
|
|
515
|
+
})
|
|
516
|
+
|
|
517
|
+
describe("Palette detection error handling", () => {
|
|
518
|
+
test("handles timeout gracefully", async () => {
|
|
519
|
+
const clock = new ManualClock()
|
|
520
|
+
const mockStdin = new EventEmitter() as any
|
|
521
|
+
mockStdin.isTTY = true
|
|
522
|
+
mockStdin.setRawMode = () => {}
|
|
523
|
+
mockStdin.resume = () => {}
|
|
524
|
+
mockStdin.pause = () => {}
|
|
525
|
+
mockStdin.setEncoding = () => {}
|
|
526
|
+
|
|
527
|
+
const mockStdout = {
|
|
528
|
+
isTTY: true,
|
|
529
|
+
columns: 80,
|
|
530
|
+
rows: 24,
|
|
531
|
+
write: () => true,
|
|
532
|
+
} as any
|
|
533
|
+
|
|
534
|
+
const { renderer } = await createTestRenderer({
|
|
535
|
+
stdin: mockStdin,
|
|
536
|
+
stdout: mockStdout,
|
|
537
|
+
clock,
|
|
538
|
+
})
|
|
539
|
+
|
|
540
|
+
const palette = await detectPaletteAndAdvanceClock(renderer, clock, { timeout: 100 })
|
|
541
|
+
expect(typeof palette === "object" && palette !== null && Array.isArray(palette.palette)).toBe(true)
|
|
542
|
+
expect(palette.palette.every((c) => c === null)).toBe(true)
|
|
543
|
+
|
|
544
|
+
renderer.destroy()
|
|
545
|
+
})
|
|
546
|
+
|
|
547
|
+
test("handles stdin listener restoration on error", async () => {
|
|
548
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
549
|
+
|
|
550
|
+
try {
|
|
551
|
+
const palettePromise = renderer.getPalette({ timeout: 300 })
|
|
552
|
+
await advancePaletteClock(clock, 300)
|
|
553
|
+
await palettePromise
|
|
554
|
+
} catch (error) {}
|
|
555
|
+
|
|
556
|
+
const listenerCount = mockStdin.listenerCount("data")
|
|
557
|
+
expect(listenerCount).toBeGreaterThan(0)
|
|
558
|
+
|
|
559
|
+
renderer.destroy()
|
|
560
|
+
})
|
|
561
|
+
})
|
|
562
|
+
|
|
563
|
+
describe("Palette cache with different sizes", () => {
|
|
564
|
+
test("cache works correctly when requesting size=16 twice", async () => {
|
|
565
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer()
|
|
566
|
+
|
|
567
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
568
|
+
const writeCountAfterFirst = writes.length
|
|
569
|
+
|
|
570
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
571
|
+
expect(palette1.palette.length).toBe(16)
|
|
572
|
+
|
|
573
|
+
const timeAfterFirstDetection = clock.now()
|
|
574
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
575
|
+
|
|
576
|
+
expect(clock.now()).toBe(timeAfterFirstDetection)
|
|
577
|
+
expect(writes.length).toBe(writeCountAfterFirst)
|
|
578
|
+
expect(palette1).toBe(palette2)
|
|
579
|
+
expect(renderer.paletteDetectionStatus).toBe("cached")
|
|
580
|
+
|
|
581
|
+
renderer.destroy()
|
|
582
|
+
})
|
|
583
|
+
|
|
584
|
+
test("cache is invalidated when requesting different size", async () => {
|
|
585
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer({ useThread: false })
|
|
586
|
+
|
|
587
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
588
|
+
const writeCountAfter16 = writes.length
|
|
589
|
+
|
|
590
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 256, timeout: 300 })
|
|
591
|
+
const writeCountAfter256 = writes.length
|
|
592
|
+
|
|
593
|
+
expect(writeCountAfter256).toBeGreaterThan(writeCountAfter16)
|
|
594
|
+
expect(palette1).not.toBe(palette2)
|
|
595
|
+
|
|
596
|
+
renderer.destroy()
|
|
597
|
+
})
|
|
598
|
+
|
|
599
|
+
test("cache persists across multiple identical size requests", async () => {
|
|
600
|
+
const { renderer, clock, mockStdin, mockStdout, writes } = await createPaletteRenderer()
|
|
601
|
+
|
|
602
|
+
const palette1 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
603
|
+
const writeCountAfterFirst = writes.length
|
|
604
|
+
|
|
605
|
+
const palette2 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
606
|
+
const palette3 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
607
|
+
const palette4 = await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
608
|
+
|
|
609
|
+
expect(writes.length).toBe(writeCountAfterFirst)
|
|
610
|
+
expect(palette1).toBe(palette2)
|
|
611
|
+
expect(palette2).toBe(palette3)
|
|
612
|
+
expect(palette3).toBe(palette4)
|
|
613
|
+
|
|
614
|
+
renderer.destroy()
|
|
615
|
+
})
|
|
616
|
+
|
|
617
|
+
test("cached call is significantly faster than initial detection", async () => {
|
|
618
|
+
const { renderer, clock, mockStdin, mockStdout } = await createPaletteRenderer()
|
|
619
|
+
|
|
620
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
621
|
+
const timeAfterFirstDetection = clock.now()
|
|
622
|
+
|
|
623
|
+
await detectPaletteAndAdvanceClock(renderer, clock, { size: 16, timeout: 300 })
|
|
624
|
+
|
|
625
|
+
expect(clock.now()).toBe(timeAfterFirstDetection)
|
|
626
|
+
|
|
627
|
+
renderer.destroy()
|
|
628
|
+
})
|
|
629
|
+
})
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { test, expect, beforeEach, afterEach } from "bun:test"
|
|
2
|
+
import { createTestRenderer, type TestRenderer } from "../testing/test-renderer.js"
|
|
3
|
+
import { TextRenderable } from "../renderables/Text.js"
|
|
4
|
+
|
|
5
|
+
let renderer: TestRenderer
|
|
6
|
+
let renderOnce: () => void
|
|
7
|
+
|
|
8
|
+
beforeEach(async () => {
|
|
9
|
+
;({ renderer, renderOnce } = await createTestRenderer({}))
|
|
10
|
+
})
|
|
11
|
+
|
|
12
|
+
afterEach(() => {
|
|
13
|
+
renderer.destroy()
|
|
14
|
+
})
|
|
15
|
+
|
|
16
|
+
test("selection on destroyed renderable should not throw", () => {
|
|
17
|
+
const text = new TextRenderable(renderer, {
|
|
18
|
+
content: "Hello World",
|
|
19
|
+
width: 20,
|
|
20
|
+
height: 1,
|
|
21
|
+
})
|
|
22
|
+
|
|
23
|
+
renderer.root.add(text)
|
|
24
|
+
renderOnce()
|
|
25
|
+
|
|
26
|
+
// Start selection
|
|
27
|
+
renderer.startSelection(text, 0, 0)
|
|
28
|
+
|
|
29
|
+
// Update selection - this should not throw
|
|
30
|
+
renderer.updateSelection(text, 5, 1)
|
|
31
|
+
|
|
32
|
+
expect(renderer.getSelection()).not.toBeNull()
|
|
33
|
+
|
|
34
|
+
// Destroy the text renderable
|
|
35
|
+
text.destroy()
|
|
36
|
+
|
|
37
|
+
expect(text.isDestroyed).toBe(true)
|
|
38
|
+
|
|
39
|
+
// Get selection - this should not throw
|
|
40
|
+
expect(renderer.getSelection()!.getSelectedText()).toBe("")
|
|
41
|
+
|
|
42
|
+
// Update selection - this should not throw
|
|
43
|
+
renderer.updateSelection(text, 8, 1)
|
|
44
|
+
|
|
45
|
+
// Clear selection - this should not throw
|
|
46
|
+
renderer.clearSelection()
|
|
47
|
+
|
|
48
|
+
expect(renderer.getSelection()).toBeNull()
|
|
49
|
+
})
|