@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,1866 @@
|
|
|
1
|
+
import { describe, expect, it, beforeEach, afterEach } from "bun:test"
|
|
2
|
+
import { createTestRenderer, type TestRenderer, type MockInput } from "../../testing/test-renderer.js"
|
|
3
|
+
import { createTextareaRenderable } from "./renderable-test-utils.js"
|
|
4
|
+
import { RGBA } from "../../lib/RGBA.js"
|
|
5
|
+
import { SyntaxStyle } from "../../syntax-style.js"
|
|
6
|
+
import { OptimizedBuffer } from "../../buffer.js"
|
|
7
|
+
import { fg, t } from "../../lib/index.js"
|
|
8
|
+
import { BoxRenderable } from "../Box.js"
|
|
9
|
+
import { TextareaRenderable } from "../Textarea.js"
|
|
10
|
+
import { TextRenderable } from "../Text.js"
|
|
11
|
+
|
|
12
|
+
let currentRenderer: TestRenderer
|
|
13
|
+
let renderOnce: () => Promise<void>
|
|
14
|
+
let currentMockInput: MockInput
|
|
15
|
+
let captureFrame: () => string
|
|
16
|
+
let resize: (width: number, height: number) => void
|
|
17
|
+
|
|
18
|
+
describe("Textarea - Rendering Tests", () => {
|
|
19
|
+
beforeEach(async () => {
|
|
20
|
+
;({
|
|
21
|
+
renderer: currentRenderer,
|
|
22
|
+
renderOnce,
|
|
23
|
+
captureCharFrame: captureFrame,
|
|
24
|
+
mockInput: currentMockInput,
|
|
25
|
+
resize,
|
|
26
|
+
} = await createTestRenderer({
|
|
27
|
+
width: 80,
|
|
28
|
+
height: 24,
|
|
29
|
+
}))
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
afterEach(() => {
|
|
33
|
+
currentRenderer.destroy()
|
|
34
|
+
})
|
|
35
|
+
|
|
36
|
+
describe("Wrapping", () => {
|
|
37
|
+
it("should move cursor down through all wrapped visual lines at column 0", async () => {
|
|
38
|
+
// Create a long line that will wrap into multiple visual lines
|
|
39
|
+
const longText =
|
|
40
|
+
"This is a very long line that will definitely wrap into multiple visual lines when the viewport is small"
|
|
41
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
42
|
+
initialValue: longText,
|
|
43
|
+
width: 20, // Small viewport to force wrapping
|
|
44
|
+
height: 10,
|
|
45
|
+
wrapMode: "word",
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
editor.focus()
|
|
49
|
+
|
|
50
|
+
// Set cursor at the beginning (0, 0) - logical position
|
|
51
|
+
editor.editBuffer.setCursor(0, 0)
|
|
52
|
+
await renderOnce()
|
|
53
|
+
|
|
54
|
+
// Get initial visual cursor position - should be at visual 0, 0
|
|
55
|
+
let visualCursor = editor.editorView.getVisualCursor()
|
|
56
|
+
expect(visualCursor.visualRow).toBe(0)
|
|
57
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
58
|
+
|
|
59
|
+
// Verify we have multiple wrapped lines (should be 7 for this text)
|
|
60
|
+
const vlineCount = editor.editorView.getVirtualLineCount()
|
|
61
|
+
expect(vlineCount).toBeGreaterThan(1)
|
|
62
|
+
|
|
63
|
+
// Move down through each wrapped line - cursor should stay at column 0
|
|
64
|
+
for (let i = 1; i < vlineCount; i++) {
|
|
65
|
+
currentMockInput.pressArrow("down")
|
|
66
|
+
await renderOnce()
|
|
67
|
+
|
|
68
|
+
visualCursor = editor.editorView.getVisualCursor()
|
|
69
|
+
|
|
70
|
+
// Cursor should have moved down to the next visual line
|
|
71
|
+
expect(visualCursor.visualRow).toBe(i)
|
|
72
|
+
|
|
73
|
+
// Cursor should be at column 0 (beginning of each wrapped line)
|
|
74
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// After moving through all wrapped lines, we should be at the last wrapped line
|
|
78
|
+
expect(visualCursor.visualRow).toBe(vlineCount - 1)
|
|
79
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
80
|
+
})
|
|
81
|
+
|
|
82
|
+
it("should move cursor up through all wrapped visual lines at column 0", async () => {
|
|
83
|
+
// Create a long line that will wrap into multiple visual lines
|
|
84
|
+
const longText =
|
|
85
|
+
"This is a very long line that will definitely wrap into multiple visual lines when the viewport is small"
|
|
86
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
87
|
+
initialValue: longText,
|
|
88
|
+
width: 20, // Small viewport to force wrapping
|
|
89
|
+
height: 10,
|
|
90
|
+
wrapMode: "word",
|
|
91
|
+
})
|
|
92
|
+
|
|
93
|
+
editor.focus()
|
|
94
|
+
|
|
95
|
+
// Verify we have multiple wrapped lines
|
|
96
|
+
const vlineCount = editor.editorView.getVirtualLineCount()
|
|
97
|
+
expect(vlineCount).toBeGreaterThan(1)
|
|
98
|
+
|
|
99
|
+
// Start at the END of the line (which will be on the last wrapped visual line)
|
|
100
|
+
const eol = editor.editBuffer.getEOL()
|
|
101
|
+
editor.editBuffer.setCursor(eol.row, eol.col)
|
|
102
|
+
await renderOnce()
|
|
103
|
+
|
|
104
|
+
// Move to the beginning of the last wrapped line (column 0 of last visual line)
|
|
105
|
+
let visualCursor = editor.editorView.getVisualCursor()
|
|
106
|
+
const lastVisualRow = visualCursor.visualRow
|
|
107
|
+
|
|
108
|
+
// Set cursor to column 0 of the last wrapped visual line by finding its logical column
|
|
109
|
+
// Last visual line starts at a specific logical column - we need to find it
|
|
110
|
+
const lastVlineStartCol = editor.logicalCursor.col - visualCursor.visualCol
|
|
111
|
+
editor.editBuffer.setCursor(0, lastVlineStartCol)
|
|
112
|
+
await renderOnce()
|
|
113
|
+
|
|
114
|
+
visualCursor = editor.editorView.getVisualCursor()
|
|
115
|
+
expect(visualCursor.visualRow).toBe(lastVisualRow)
|
|
116
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
117
|
+
|
|
118
|
+
// Now move UP through each wrapped line - cursor should stay at column 0
|
|
119
|
+
for (let i = lastVisualRow - 1; i >= 0; i--) {
|
|
120
|
+
currentMockInput.pressArrow("up")
|
|
121
|
+
await renderOnce()
|
|
122
|
+
|
|
123
|
+
visualCursor = editor.editorView.getVisualCursor()
|
|
124
|
+
|
|
125
|
+
// Cursor should have moved up to the previous visual line
|
|
126
|
+
expect(visualCursor.visualRow).toBe(i)
|
|
127
|
+
|
|
128
|
+
// Cursor should be at column 0 (beginning of each wrapped line)
|
|
129
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// After moving through all wrapped lines, we should be at the first wrapped line
|
|
133
|
+
expect(visualCursor.visualRow).toBe(0)
|
|
134
|
+
expect(visualCursor.visualCol).toBe(0)
|
|
135
|
+
})
|
|
136
|
+
|
|
137
|
+
it("should handle wrap mode property", async () => {
|
|
138
|
+
const longText = "A".repeat(100)
|
|
139
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
140
|
+
initialValue: longText,
|
|
141
|
+
width: 20,
|
|
142
|
+
height: 10,
|
|
143
|
+
wrapMode: "word",
|
|
144
|
+
})
|
|
145
|
+
|
|
146
|
+
expect(editor.wrapMode).toBe("word")
|
|
147
|
+
const wrappedCount = editor.editorView.getVirtualLineCount()
|
|
148
|
+
expect(wrappedCount).toBeGreaterThan(1)
|
|
149
|
+
|
|
150
|
+
editor.wrapMode = "none"
|
|
151
|
+
expect(editor.wrapMode).toBe("none")
|
|
152
|
+
const unwrappedCount = editor.editorView.getVirtualLineCount()
|
|
153
|
+
expect(unwrappedCount).toBe(1)
|
|
154
|
+
})
|
|
155
|
+
|
|
156
|
+
it("should handle wrapMode changes", async () => {
|
|
157
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
158
|
+
initialValue: "Hello wonderful world",
|
|
159
|
+
width: 12,
|
|
160
|
+
height: 10,
|
|
161
|
+
wrapMode: "char",
|
|
162
|
+
})
|
|
163
|
+
|
|
164
|
+
expect(editor.wrapMode).toBe("char")
|
|
165
|
+
|
|
166
|
+
editor.wrapMode = "word"
|
|
167
|
+
expect(editor.wrapMode).toBe("word")
|
|
168
|
+
})
|
|
169
|
+
|
|
170
|
+
it("should render with tab indicator correctly", async () => {
|
|
171
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
172
|
+
initialValue: "Line 1\tTabbed\nLine 2\t\tDouble tab",
|
|
173
|
+
tabIndicator: "→",
|
|
174
|
+
tabIndicatorColor: RGBA.fromValues(0.5, 0.5, 0.5, 1),
|
|
175
|
+
width: 40,
|
|
176
|
+
height: 10,
|
|
177
|
+
})
|
|
178
|
+
|
|
179
|
+
await renderOnce()
|
|
180
|
+
const frame = captureFrame()
|
|
181
|
+
expect(frame).toMatchSnapshot()
|
|
182
|
+
})
|
|
183
|
+
})
|
|
184
|
+
|
|
185
|
+
describe("Height and Width Measurement", () => {
|
|
186
|
+
it("should grow height for multiline text without wrapping", async () => {
|
|
187
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
188
|
+
initialValue: "Line 1\nLine 2\nLine 3\nLine 4\nLine 5",
|
|
189
|
+
wrapMode: "none",
|
|
190
|
+
width: 40,
|
|
191
|
+
})
|
|
192
|
+
|
|
193
|
+
await renderOnce()
|
|
194
|
+
|
|
195
|
+
expect(editor.height).toBe(5)
|
|
196
|
+
expect(editor.width).toBeGreaterThanOrEqual(6)
|
|
197
|
+
})
|
|
198
|
+
|
|
199
|
+
it("should grow height for wrapped text when wrapping enabled", async () => {
|
|
200
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
201
|
+
initialValue: "This is a very long line that will definitely wrap to multiple lines",
|
|
202
|
+
wrapMode: "word",
|
|
203
|
+
width: 15,
|
|
204
|
+
})
|
|
205
|
+
|
|
206
|
+
await renderOnce()
|
|
207
|
+
|
|
208
|
+
expect(editor.height).toBeGreaterThan(1)
|
|
209
|
+
expect(editor.width).toBeLessThanOrEqual(15)
|
|
210
|
+
})
|
|
211
|
+
|
|
212
|
+
it("should measure full width when wrapping is disabled and not constrained by parent", async () => {
|
|
213
|
+
const longLine = "This is a very long line that would wrap but wrapping is disabled"
|
|
214
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
215
|
+
initialValue: longLine,
|
|
216
|
+
wrapMode: "none",
|
|
217
|
+
position: "absolute",
|
|
218
|
+
})
|
|
219
|
+
|
|
220
|
+
await renderOnce()
|
|
221
|
+
|
|
222
|
+
expect(editor.height).toBe(1)
|
|
223
|
+
expect(editor.width).toBe(longLine.length)
|
|
224
|
+
})
|
|
225
|
+
|
|
226
|
+
it("should shrink height when deleting lines via value setter", async () => {
|
|
227
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
228
|
+
initialValue: "Line 1\nLine 2\nLine 3\nLine 4\nLine 5",
|
|
229
|
+
width: 40,
|
|
230
|
+
wrapMode: "none",
|
|
231
|
+
})
|
|
232
|
+
|
|
233
|
+
editor.focus()
|
|
234
|
+
await renderOnce()
|
|
235
|
+
expect(editor.height).toBe(5)
|
|
236
|
+
|
|
237
|
+
// Remove lines by setting new value
|
|
238
|
+
editor.setText("Line 1\nLine 2")
|
|
239
|
+
await renderOnce()
|
|
240
|
+
|
|
241
|
+
expect(editor.height).toBe(2)
|
|
242
|
+
expect(editor.plainText).toBe("Line 1\nLine 2")
|
|
243
|
+
})
|
|
244
|
+
|
|
245
|
+
it("should update height when content changes from single to multiline", async () => {
|
|
246
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
247
|
+
initialValue: "Single line",
|
|
248
|
+
wrapMode: "none",
|
|
249
|
+
})
|
|
250
|
+
|
|
251
|
+
await renderOnce()
|
|
252
|
+
expect(editor.height).toBe(1)
|
|
253
|
+
|
|
254
|
+
editor.setText("Line 1\nLine 2\nLine 3")
|
|
255
|
+
await renderOnce()
|
|
256
|
+
|
|
257
|
+
expect(editor.height).toBe(3)
|
|
258
|
+
})
|
|
259
|
+
|
|
260
|
+
it("should grow height when pressing Enter to add newlines", async () => {
|
|
261
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
262
|
+
initialValue: "Single line",
|
|
263
|
+
width: 40,
|
|
264
|
+
wrapMode: "none",
|
|
265
|
+
})
|
|
266
|
+
|
|
267
|
+
// Add a second textarea below to verify layout reflow
|
|
268
|
+
const { textarea: belowEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
269
|
+
initialValue: "Below",
|
|
270
|
+
width: 40,
|
|
271
|
+
})
|
|
272
|
+
|
|
273
|
+
await renderOnce()
|
|
274
|
+
expect(editor.height).toBe(1)
|
|
275
|
+
const initialHeight = editor.height
|
|
276
|
+
const initialBelowY = belowEditor.y
|
|
277
|
+
|
|
278
|
+
editor.focus()
|
|
279
|
+
editor.gotoLine(9999) // Move to end
|
|
280
|
+
|
|
281
|
+
// Press Enter 3 times to add 3 newlines
|
|
282
|
+
currentMockInput.pressEnter()
|
|
283
|
+
expect(editor.plainText).toBe("Single line\n")
|
|
284
|
+
await renderOnce() // Wait for layout recalculation
|
|
285
|
+
|
|
286
|
+
currentMockInput.pressEnter()
|
|
287
|
+
expect(editor.plainText).toBe("Single line\n\n")
|
|
288
|
+
await renderOnce() // Wait for layout recalculation
|
|
289
|
+
|
|
290
|
+
currentMockInput.pressEnter()
|
|
291
|
+
expect(editor.plainText).toBe("Single line\n\n\n")
|
|
292
|
+
await renderOnce() // Wait for layout recalculation
|
|
293
|
+
|
|
294
|
+
// The editor should have grown
|
|
295
|
+
expect(editor.height).toBeGreaterThan(initialHeight)
|
|
296
|
+
expect(editor.height).toBe(4) // 1 original line + 3 new lines
|
|
297
|
+
expect(editor.plainText).toBe("Single line\n\n\n")
|
|
298
|
+
|
|
299
|
+
// The element below should have moved down
|
|
300
|
+
expect(belowEditor.y).toBeGreaterThan(initialBelowY)
|
|
301
|
+
expect(belowEditor.y).toBe(4) // After the 4-line editor
|
|
302
|
+
})
|
|
303
|
+
})
|
|
304
|
+
|
|
305
|
+
describe("Unicode Support", () => {
|
|
306
|
+
it("should handle emoji insertion", async () => {
|
|
307
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
308
|
+
initialValue: "Hello",
|
|
309
|
+
width: 40,
|
|
310
|
+
height: 10,
|
|
311
|
+
})
|
|
312
|
+
|
|
313
|
+
editor.focus()
|
|
314
|
+
editor.gotoLine(9999) // Move to end
|
|
315
|
+
editor.insertText(" 🌟")
|
|
316
|
+
|
|
317
|
+
expect(editor.plainText).toBe("Hello 🌟")
|
|
318
|
+
})
|
|
319
|
+
|
|
320
|
+
it("should handle CJK characters", async () => {
|
|
321
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
322
|
+
initialValue: "Hello",
|
|
323
|
+
width: 40,
|
|
324
|
+
height: 10,
|
|
325
|
+
})
|
|
326
|
+
|
|
327
|
+
editor.focus()
|
|
328
|
+
editor.gotoLine(9999) // Move to end
|
|
329
|
+
editor.insertText(" 世界")
|
|
330
|
+
|
|
331
|
+
expect(editor.plainText).toBe("Hello 世界")
|
|
332
|
+
})
|
|
333
|
+
|
|
334
|
+
it("should handle emoji cursor movement", async () => {
|
|
335
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
336
|
+
initialValue: "A🌟B",
|
|
337
|
+
width: 40,
|
|
338
|
+
height: 10,
|
|
339
|
+
})
|
|
340
|
+
|
|
341
|
+
editor.focus()
|
|
342
|
+
expect(editor.logicalCursor.col).toBe(0)
|
|
343
|
+
|
|
344
|
+
currentMockInput.pressArrow("right") // Move past A
|
|
345
|
+
expect(editor.logicalCursor.col).toBe(1)
|
|
346
|
+
|
|
347
|
+
currentMockInput.pressArrow("right") // Move past emoji (2 cells)
|
|
348
|
+
expect(editor.logicalCursor.col).toBe(3)
|
|
349
|
+
|
|
350
|
+
currentMockInput.pressArrow("right") // Move past B
|
|
351
|
+
expect(editor.logicalCursor.col).toBe(4)
|
|
352
|
+
})
|
|
353
|
+
})
|
|
354
|
+
|
|
355
|
+
describe("Content Property", () => {
|
|
356
|
+
it("should update content programmatically", async () => {
|
|
357
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
358
|
+
initialValue: "Initial",
|
|
359
|
+
width: 40,
|
|
360
|
+
height: 10,
|
|
361
|
+
})
|
|
362
|
+
|
|
363
|
+
editor.setText("Updated")
|
|
364
|
+
expect(editor.plainText).toBe("Updated")
|
|
365
|
+
expect(editor.plainText).toBe("Updated")
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
it("should reset cursor when content changes", async () => {
|
|
369
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
370
|
+
initialValue: "Hello World",
|
|
371
|
+
width: 40,
|
|
372
|
+
height: 10,
|
|
373
|
+
})
|
|
374
|
+
|
|
375
|
+
editor.gotoLine(9999) // Move to end
|
|
376
|
+
expect(editor.logicalCursor.col).toBe(11)
|
|
377
|
+
|
|
378
|
+
editor.setText("New")
|
|
379
|
+
// Cursor should reset to start
|
|
380
|
+
expect(editor.logicalCursor.row).toBe(0)
|
|
381
|
+
expect(editor.logicalCursor.col).toBe(0)
|
|
382
|
+
})
|
|
383
|
+
|
|
384
|
+
it("should clear text with clear() method", async () => {
|
|
385
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
386
|
+
initialValue: "Hello World",
|
|
387
|
+
width: 40,
|
|
388
|
+
height: 10,
|
|
389
|
+
})
|
|
390
|
+
|
|
391
|
+
expect(editor.plainText).toBe("Hello World")
|
|
392
|
+
|
|
393
|
+
editor.clear()
|
|
394
|
+
expect(editor.plainText).toBe("")
|
|
395
|
+
})
|
|
396
|
+
|
|
397
|
+
it("should clear highlights with clear() method", async () => {
|
|
398
|
+
const style = SyntaxStyle.create()
|
|
399
|
+
const styleId = style.registerStyle("highlight", {
|
|
400
|
+
fg: RGBA.fromValues(1, 0, 0, 1),
|
|
401
|
+
})
|
|
402
|
+
|
|
403
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
404
|
+
initialValue: "Hello World",
|
|
405
|
+
width: 40,
|
|
406
|
+
height: 10,
|
|
407
|
+
syntaxStyle: style,
|
|
408
|
+
})
|
|
409
|
+
|
|
410
|
+
editor.addHighlightByCharRange({
|
|
411
|
+
start: 0,
|
|
412
|
+
end: 5,
|
|
413
|
+
styleId: styleId,
|
|
414
|
+
priority: 0,
|
|
415
|
+
})
|
|
416
|
+
|
|
417
|
+
const highlightsBefore = editor.getLineHighlights(0)
|
|
418
|
+
expect(highlightsBefore.length).toBeGreaterThan(0)
|
|
419
|
+
|
|
420
|
+
editor.clear()
|
|
421
|
+
|
|
422
|
+
expect(editor.plainText).toBe("")
|
|
423
|
+
const highlightsAfter = editor.getLineHighlights(0)
|
|
424
|
+
expect(highlightsAfter.length).toBe(0)
|
|
425
|
+
})
|
|
426
|
+
|
|
427
|
+
it("should clear both text and highlights together", async () => {
|
|
428
|
+
const style = SyntaxStyle.create()
|
|
429
|
+
const styleId = style.registerStyle("highlight", {
|
|
430
|
+
fg: RGBA.fromValues(1, 0, 0, 1),
|
|
431
|
+
})
|
|
432
|
+
|
|
433
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
434
|
+
initialValue: "Line 1\nLine 2\nLine 3",
|
|
435
|
+
width: 40,
|
|
436
|
+
height: 10,
|
|
437
|
+
syntaxStyle: style,
|
|
438
|
+
})
|
|
439
|
+
|
|
440
|
+
editor.addHighlight(0, { start: 0, end: 6, styleId: styleId, priority: 0 })
|
|
441
|
+
editor.addHighlight(1, { start: 0, end: 6, styleId: styleId, priority: 0 })
|
|
442
|
+
|
|
443
|
+
expect(editor.plainText).toBe("Line 1\nLine 2\nLine 3")
|
|
444
|
+
expect(editor.getLineHighlights(0).length).toBe(1)
|
|
445
|
+
expect(editor.getLineHighlights(1).length).toBe(1)
|
|
446
|
+
|
|
447
|
+
editor.clear()
|
|
448
|
+
|
|
449
|
+
expect(editor.plainText).toBe("")
|
|
450
|
+
expect(editor.getLineHighlights(0).length).toBe(0)
|
|
451
|
+
expect(editor.getLineHighlights(1).length).toBe(0)
|
|
452
|
+
})
|
|
453
|
+
|
|
454
|
+
it("should allow typing after clear()", async () => {
|
|
455
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
456
|
+
initialValue: "Hello World",
|
|
457
|
+
width: 40,
|
|
458
|
+
height: 10,
|
|
459
|
+
})
|
|
460
|
+
|
|
461
|
+
editor.focus()
|
|
462
|
+
expect(editor.plainText).toBe("Hello World")
|
|
463
|
+
|
|
464
|
+
currentMockInput.pressKey("!")
|
|
465
|
+
expect(editor.plainText).toBe("!Hello World")
|
|
466
|
+
|
|
467
|
+
editor.clear()
|
|
468
|
+
expect(editor.plainText).toBe("")
|
|
469
|
+
|
|
470
|
+
currentMockInput.pressKey("N")
|
|
471
|
+
currentMockInput.pressKey("e")
|
|
472
|
+
currentMockInput.pressKey("w")
|
|
473
|
+
expect(editor.plainText).toBe("New")
|
|
474
|
+
|
|
475
|
+
currentMockInput.pressKey(" ")
|
|
476
|
+
currentMockInput.pressKey("T")
|
|
477
|
+
currentMockInput.pressKey("e")
|
|
478
|
+
currentMockInput.pressKey("x")
|
|
479
|
+
currentMockInput.pressKey("t")
|
|
480
|
+
expect(editor.plainText).toBe("New Text")
|
|
481
|
+
})
|
|
482
|
+
})
|
|
483
|
+
|
|
484
|
+
describe("Rendering After Edits", () => {
|
|
485
|
+
it("should render correctly after insert text", async () => {
|
|
486
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
487
|
+
initialValue: "Test",
|
|
488
|
+
width: 40,
|
|
489
|
+
height: 10,
|
|
490
|
+
})
|
|
491
|
+
|
|
492
|
+
editor.focus()
|
|
493
|
+
editor.insertText("x")
|
|
494
|
+
|
|
495
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
496
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
497
|
+
|
|
498
|
+
expect(editor.plainText).toBe("xTest")
|
|
499
|
+
expect(editor.logicalCursor.col).toBe(1)
|
|
500
|
+
|
|
501
|
+
buffer.destroy()
|
|
502
|
+
})
|
|
503
|
+
|
|
504
|
+
it("should render correctly after rapid edits", async () => {
|
|
505
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
506
|
+
initialValue: "",
|
|
507
|
+
width: 40,
|
|
508
|
+
height: 10,
|
|
509
|
+
})
|
|
510
|
+
|
|
511
|
+
editor.focus()
|
|
512
|
+
|
|
513
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
514
|
+
|
|
515
|
+
for (let i = 0; i < 5; i++) {
|
|
516
|
+
editor.insertText("a")
|
|
517
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
expect(editor.plainText).toBe("aaaaa")
|
|
521
|
+
expect(editor.logicalCursor.col).toBe(5)
|
|
522
|
+
|
|
523
|
+
buffer.destroy()
|
|
524
|
+
})
|
|
525
|
+
|
|
526
|
+
it("should render correctly after newline", async () => {
|
|
527
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
528
|
+
initialValue: "Hello",
|
|
529
|
+
width: 40,
|
|
530
|
+
height: 10,
|
|
531
|
+
})
|
|
532
|
+
|
|
533
|
+
editor.focus()
|
|
534
|
+
editor.gotoLine(9999)
|
|
535
|
+
|
|
536
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
537
|
+
|
|
538
|
+
editor.newLine()
|
|
539
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
540
|
+
|
|
541
|
+
expect(editor.plainText).toBe("Hello\n")
|
|
542
|
+
expect(editor.logicalCursor.row).toBe(1)
|
|
543
|
+
expect(editor.logicalCursor.col).toBe(0)
|
|
544
|
+
|
|
545
|
+
buffer.destroy()
|
|
546
|
+
})
|
|
547
|
+
|
|
548
|
+
it("should render correctly after backspace", async () => {
|
|
549
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
550
|
+
initialValue: "Hello",
|
|
551
|
+
width: 40,
|
|
552
|
+
height: 10,
|
|
553
|
+
})
|
|
554
|
+
|
|
555
|
+
editor.focus()
|
|
556
|
+
editor.gotoLine(9999)
|
|
557
|
+
|
|
558
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
559
|
+
|
|
560
|
+
editor.deleteCharBackward()
|
|
561
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
562
|
+
|
|
563
|
+
expect(editor.plainText).toBe("Hell")
|
|
564
|
+
expect(editor.logicalCursor.col).toBe(4)
|
|
565
|
+
|
|
566
|
+
buffer.destroy()
|
|
567
|
+
})
|
|
568
|
+
|
|
569
|
+
it("should render correctly with draw-edit-draw pattern", async () => {
|
|
570
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
571
|
+
initialValue: "Test",
|
|
572
|
+
width: 40,
|
|
573
|
+
height: 10,
|
|
574
|
+
})
|
|
575
|
+
|
|
576
|
+
editor.focus()
|
|
577
|
+
|
|
578
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
579
|
+
|
|
580
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
581
|
+
editor.insertText("x")
|
|
582
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
583
|
+
|
|
584
|
+
expect(editor.plainText).toBe("xTest")
|
|
585
|
+
expect(editor.logicalCursor.col).toBe(1)
|
|
586
|
+
|
|
587
|
+
buffer.destroy()
|
|
588
|
+
})
|
|
589
|
+
|
|
590
|
+
it("should render correctly after multiple text buffer modifications", async () => {
|
|
591
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
592
|
+
initialValue: "Line1\nLine2\nLine3",
|
|
593
|
+
width: 40,
|
|
594
|
+
height: 10,
|
|
595
|
+
})
|
|
596
|
+
|
|
597
|
+
editor.focus()
|
|
598
|
+
|
|
599
|
+
const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
|
|
600
|
+
|
|
601
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
602
|
+
|
|
603
|
+
editor.insertText("X")
|
|
604
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
605
|
+
expect(editor.plainText).toBe("XLine1\nLine2\nLine3")
|
|
606
|
+
|
|
607
|
+
editor.newLine()
|
|
608
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
609
|
+
expect(editor.plainText).toBe("X\nLine1\nLine2\nLine3")
|
|
610
|
+
|
|
611
|
+
editor.deleteCharBackward()
|
|
612
|
+
buffer.drawEditorView(editor.editorView, 0, 0)
|
|
613
|
+
expect(editor.plainText).toBe("XLine1\nLine2\nLine3")
|
|
614
|
+
|
|
615
|
+
buffer.destroy()
|
|
616
|
+
})
|
|
617
|
+
})
|
|
618
|
+
|
|
619
|
+
describe("Viewport Scrolling", () => {
|
|
620
|
+
it("should scroll viewport down when cursor moves below visible area", async () => {
|
|
621
|
+
// Create editor with small viewport
|
|
622
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
623
|
+
initialValue: "Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9",
|
|
624
|
+
width: 40,
|
|
625
|
+
height: 5, // Only 5 lines visible
|
|
626
|
+
})
|
|
627
|
+
|
|
628
|
+
editor.focus()
|
|
629
|
+
|
|
630
|
+
// Initial viewport should show lines 0-4
|
|
631
|
+
let viewport = editor.editorView.getViewport()
|
|
632
|
+
expect(viewport.offsetY).toBe(0)
|
|
633
|
+
expect(viewport.height).toBe(5)
|
|
634
|
+
|
|
635
|
+
// Move cursor to line 7 (beyond viewport)
|
|
636
|
+
editor.gotoLine(7)
|
|
637
|
+
|
|
638
|
+
// Viewport should have scrolled to keep cursor visible
|
|
639
|
+
viewport = editor.editorView.getViewport()
|
|
640
|
+
// With scroll margin of 0.2 (20% = 1 line), viewport should scroll to show line 7
|
|
641
|
+
// Expected: offsetY should be at least 3 (to show lines 3-7)
|
|
642
|
+
expect(viewport.offsetY).toBeGreaterThanOrEqual(3)
|
|
643
|
+
})
|
|
644
|
+
|
|
645
|
+
it("should scroll viewport up when cursor moves above visible area", async () => {
|
|
646
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
647
|
+
initialValue: "Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9",
|
|
648
|
+
width: 40,
|
|
649
|
+
height: 5,
|
|
650
|
+
})
|
|
651
|
+
|
|
652
|
+
editor.focus()
|
|
653
|
+
|
|
654
|
+
// Start at line 8
|
|
655
|
+
editor.gotoLine(8)
|
|
656
|
+
|
|
657
|
+
let viewport = editor.editorView.getViewport()
|
|
658
|
+
// Viewport should have automatically scrolled to show line 8
|
|
659
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
660
|
+
|
|
661
|
+
// Now move to line 1 (above viewport)
|
|
662
|
+
editor.gotoLine(1)
|
|
663
|
+
|
|
664
|
+
viewport = editor.editorView.getViewport()
|
|
665
|
+
// Viewport should have scrolled up to show line 1
|
|
666
|
+
expect(viewport.offsetY).toBeLessThanOrEqual(1)
|
|
667
|
+
})
|
|
668
|
+
|
|
669
|
+
it("should scroll viewport when using arrow keys to move beyond visible area", async () => {
|
|
670
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
671
|
+
initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
|
|
672
|
+
width: 40,
|
|
673
|
+
height: 5,
|
|
674
|
+
})
|
|
675
|
+
|
|
676
|
+
editor.focus()
|
|
677
|
+
|
|
678
|
+
let viewport = editor.editorView.getViewport()
|
|
679
|
+
expect(viewport.offsetY).toBe(0)
|
|
680
|
+
|
|
681
|
+
// Press down arrow 6 times to move beyond initial viewport
|
|
682
|
+
for (let i = 0; i < 6; i++) {
|
|
683
|
+
currentMockInput.pressArrow("down")
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
viewport = editor.editorView.getViewport()
|
|
687
|
+
// Should have scrolled
|
|
688
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
689
|
+
})
|
|
690
|
+
|
|
691
|
+
it("should maintain scroll margin when moving cursor", async () => {
|
|
692
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
693
|
+
initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
|
|
694
|
+
width: 40,
|
|
695
|
+
height: 10,
|
|
696
|
+
scrollMargin: 0.2, // 20% = 2 lines margin
|
|
697
|
+
})
|
|
698
|
+
|
|
699
|
+
editor.focus()
|
|
700
|
+
|
|
701
|
+
// Move to line 8 (near bottom of initial viewport)
|
|
702
|
+
editor.gotoLine(8)
|
|
703
|
+
|
|
704
|
+
let viewport = editor.editorView.getViewport()
|
|
705
|
+
|
|
706
|
+
// With 2-line margin, cursor at line 8 should trigger scroll
|
|
707
|
+
// so that line 8 is at most at position 8 in viewport
|
|
708
|
+
expect(viewport.offsetY).toBeGreaterThanOrEqual(0)
|
|
709
|
+
})
|
|
710
|
+
|
|
711
|
+
it("should handle viewport scrolling with text wrapping", async () => {
|
|
712
|
+
const longLine = "word ".repeat(50) // Creates line that will wrap
|
|
713
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
714
|
+
initialValue: Array.from({ length: 10 }, (_, i) => (i === 5 ? longLine : `Line ${i}`)).join("\n"),
|
|
715
|
+
width: 20,
|
|
716
|
+
height: 5,
|
|
717
|
+
wrapMode: "word",
|
|
718
|
+
})
|
|
719
|
+
|
|
720
|
+
editor.focus()
|
|
721
|
+
|
|
722
|
+
// Move to the long line
|
|
723
|
+
editor.gotoLine(5)
|
|
724
|
+
|
|
725
|
+
const vlineCount = editor.editorView.getTotalVirtualLineCount()
|
|
726
|
+
expect(vlineCount).toBeGreaterThan(10) // Should be more due to wrapping
|
|
727
|
+
|
|
728
|
+
// Move to end of long line
|
|
729
|
+
const cursor = editor.logicalCursor
|
|
730
|
+
editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
|
|
731
|
+
|
|
732
|
+
let viewport = editor.editorView.getViewport()
|
|
733
|
+
|
|
734
|
+
// Viewport should have scrolled to show cursor
|
|
735
|
+
// This is complex with wrapping - we need virtual line scrolling
|
|
736
|
+
})
|
|
737
|
+
|
|
738
|
+
it("should verify viewport follows cursor to line 10", async () => {
|
|
739
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
740
|
+
initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
|
|
741
|
+
width: 40,
|
|
742
|
+
height: 8,
|
|
743
|
+
})
|
|
744
|
+
|
|
745
|
+
editor.focus()
|
|
746
|
+
|
|
747
|
+
// Move to line 10
|
|
748
|
+
editor.gotoLine(10)
|
|
749
|
+
|
|
750
|
+
const viewport = editor.editorView.getViewport()
|
|
751
|
+
|
|
752
|
+
// Viewport should have scrolled to show line 10
|
|
753
|
+
// With height=8 and scroll margin, line 10 should be visible
|
|
754
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
755
|
+
expect(viewport.offsetY).toBeLessThanOrEqual(10)
|
|
756
|
+
|
|
757
|
+
// Line 10 should be within the viewport range
|
|
758
|
+
const viewportEnd = viewport.offsetY + viewport.height
|
|
759
|
+
expect(10).toBeGreaterThanOrEqual(viewport.offsetY)
|
|
760
|
+
expect(10).toBeLessThan(viewportEnd)
|
|
761
|
+
})
|
|
762
|
+
|
|
763
|
+
it("should track viewport offset as cursor moves through document", async () => {
|
|
764
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
765
|
+
initialValue: Array.from({ length: 15 }, (_, i) => `Line ${i}`).join("\n"),
|
|
766
|
+
width: 30,
|
|
767
|
+
height: 5,
|
|
768
|
+
})
|
|
769
|
+
|
|
770
|
+
editor.focus()
|
|
771
|
+
|
|
772
|
+
const viewportOffsets: number[] = []
|
|
773
|
+
|
|
774
|
+
// Track viewport offset at different cursor positions
|
|
775
|
+
for (const line of [0, 2, 4, 6, 8, 10, 12]) {
|
|
776
|
+
editor.gotoLine(line)
|
|
777
|
+
const viewport = editor.editorView.getViewport()
|
|
778
|
+
viewportOffsets.push(viewport.offsetY)
|
|
779
|
+
}
|
|
780
|
+
|
|
781
|
+
// Viewport should generally increase as cursor moves down
|
|
782
|
+
// (with possible plateaus when cursor is already visible)
|
|
783
|
+
const lastOffset = viewportOffsets[viewportOffsets.length - 1]
|
|
784
|
+
const firstOffset = viewportOffsets[0]
|
|
785
|
+
expect(lastOffset).toBeGreaterThan(firstOffset)
|
|
786
|
+
|
|
787
|
+
// At line 0, viewport should be at 0
|
|
788
|
+
expect(viewportOffsets[0]).toBe(0)
|
|
789
|
+
|
|
790
|
+
// At line 12, viewport should have scrolled
|
|
791
|
+
expect(viewportOffsets[viewportOffsets.length - 1]).toBeGreaterThan(5)
|
|
792
|
+
})
|
|
793
|
+
|
|
794
|
+
it("should scroll viewport when cursor moves with Page Up/Page Down", async () => {
|
|
795
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
796
|
+
initialValue: Array.from({ length: 30 }, (_, i) => `Line ${i}`).join("\n"),
|
|
797
|
+
width: 40,
|
|
798
|
+
height: 10,
|
|
799
|
+
})
|
|
800
|
+
|
|
801
|
+
editor.focus()
|
|
802
|
+
|
|
803
|
+
let viewport = editor.editorView.getViewport()
|
|
804
|
+
expect(viewport.offsetY).toBe(0)
|
|
805
|
+
|
|
806
|
+
// Move down 15 lines (more than viewport height)
|
|
807
|
+
for (let i = 0; i < 15; i++) {
|
|
808
|
+
editor.moveCursorDown()
|
|
809
|
+
}
|
|
810
|
+
|
|
811
|
+
viewport = editor.editorView.getViewport()
|
|
812
|
+
|
|
813
|
+
// Should have scrolled
|
|
814
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
815
|
+
expect(editor.logicalCursor.row).toBe(15)
|
|
816
|
+
})
|
|
817
|
+
|
|
818
|
+
it("should scroll viewport down when pressing Enter repeatedly", async () => {
|
|
819
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
820
|
+
initialValue: "Start",
|
|
821
|
+
width: 40,
|
|
822
|
+
height: 5,
|
|
823
|
+
})
|
|
824
|
+
|
|
825
|
+
editor.focus()
|
|
826
|
+
editor.gotoLine(9999) // Move to end
|
|
827
|
+
|
|
828
|
+
let viewport = editor.editorView.getViewport()
|
|
829
|
+
expect(viewport.offsetY).toBe(0)
|
|
830
|
+
expect(editor.logicalCursor.row).toBe(0)
|
|
831
|
+
|
|
832
|
+
// Press Enter 8 times to create 8 new lines
|
|
833
|
+
for (let i = 0; i < 8; i++) {
|
|
834
|
+
currentMockInput.pressEnter()
|
|
835
|
+
}
|
|
836
|
+
|
|
837
|
+
// After 8 Enters, we should have 9 lines total (0-8)
|
|
838
|
+
expect(editor.logicalCursor.row).toBe(8)
|
|
839
|
+
|
|
840
|
+
// Viewport should have scrolled to keep cursor visible
|
|
841
|
+
viewport = editor.editorView.getViewport()
|
|
842
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
843
|
+
|
|
844
|
+
// Cursor should be visible in viewport
|
|
845
|
+
const cursorLine = editor.logicalCursor.row
|
|
846
|
+
expect(cursorLine).toBeGreaterThanOrEqual(viewport.offsetY)
|
|
847
|
+
expect(cursorLine).toBeLessThan(viewport.offsetY + viewport.height)
|
|
848
|
+
})
|
|
849
|
+
|
|
850
|
+
it("should scroll viewport up when pressing Backspace to delete characters and move up", async () => {
|
|
851
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
852
|
+
initialValue: Array.from({ length: 15 }, (_, i) => `Line ${i}`).join("\n"),
|
|
853
|
+
width: 40,
|
|
854
|
+
height: 5,
|
|
855
|
+
})
|
|
856
|
+
|
|
857
|
+
editor.focus()
|
|
858
|
+
|
|
859
|
+
// Start at line 10, move to end so we have characters to delete
|
|
860
|
+
editor.gotoLine(10)
|
|
861
|
+
let cursor = editor.logicalCursor
|
|
862
|
+
editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
|
|
863
|
+
|
|
864
|
+
let viewport = editor.editorView.getViewport()
|
|
865
|
+
expect(viewport.offsetY).toBeGreaterThan(0)
|
|
866
|
+
const initialOffset = viewport.offsetY
|
|
867
|
+
|
|
868
|
+
// Delete all text and move cursor up to line 0
|
|
869
|
+
// Press Ctrl+A to go to start, then move to line 2, then backspace repeatedly
|
|
870
|
+
editor.gotoLine(0) // Move to start
|
|
871
|
+
editor.gotoLine(2)
|
|
872
|
+
cursor = editor.logicalCursor
|
|
873
|
+
editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
|
|
874
|
+
|
|
875
|
+
// Now we're at line 2, and viewport should have scrolled up
|
|
876
|
+
viewport = editor.editorView.getViewport()
|
|
877
|
+
|
|
878
|
+
// Viewport should have scrolled up from initial position
|
|
879
|
+
expect(viewport.offsetY).toBeLessThan(initialOffset)
|
|
880
|
+
expect(editor.logicalCursor.row).toBe(2)
|
|
881
|
+
})
|
|
882
|
+
|
|
883
|
+
it("should scroll viewport when typing at end creates wrapped lines beyond viewport", async () => {
|
|
884
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
885
|
+
initialValue: "Start",
|
|
886
|
+
width: 20,
|
|
887
|
+
height: 5,
|
|
888
|
+
wrapMode: "word",
|
|
889
|
+
})
|
|
890
|
+
|
|
891
|
+
editor.focus()
|
|
892
|
+
editor.gotoLine(9999) // Move to end
|
|
893
|
+
|
|
894
|
+
let viewport = editor.editorView.getViewport()
|
|
895
|
+
expect(viewport.offsetY).toBe(0)
|
|
896
|
+
|
|
897
|
+
// Type enough to create multiple wrapped lines
|
|
898
|
+
const longText = " word".repeat(50)
|
|
899
|
+
for (const char of longText) {
|
|
900
|
+
currentMockInput.pressKey(char)
|
|
901
|
+
}
|
|
902
|
+
|
|
903
|
+
viewport = editor.editorView.getViewport()
|
|
904
|
+
const vlineCount = editor.editorView.getTotalVirtualLineCount()
|
|
905
|
+
|
|
906
|
+
// Should have created multiple virtual lines
|
|
907
|
+
expect(vlineCount).toBeGreaterThan(5)
|
|
908
|
+
|
|
909
|
+
// Viewport should have scrolled to keep cursor visible
|
|
910
|
+
// (This test may fail if virtual line scrolling isn't implemented yet)
|
|
911
|
+
expect(viewport.offsetY).toBeGreaterThanOrEqual(0)
|
|
912
|
+
})
|
|
913
|
+
|
|
914
|
+
it("should scroll viewport when using Enter to add lines, then Backspace to remove them", async () => {
|
|
915
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
916
|
+
initialValue: "Line 0\nLine 1\nLine 2",
|
|
917
|
+
width: 40,
|
|
918
|
+
height: 5,
|
|
919
|
+
})
|
|
920
|
+
|
|
921
|
+
editor.focus()
|
|
922
|
+
editor.gotoLine(9999) // Move to end
|
|
923
|
+
|
|
924
|
+
let viewport = editor.editorView.getViewport()
|
|
925
|
+
const initialOffset = viewport.offsetY
|
|
926
|
+
|
|
927
|
+
// Add 6 new lines
|
|
928
|
+
for (let i = 0; i < 6; i++) {
|
|
929
|
+
currentMockInput.pressEnter()
|
|
930
|
+
currentMockInput.pressKey("X")
|
|
931
|
+
}
|
|
932
|
+
|
|
933
|
+
// Should have scrolled down
|
|
934
|
+
viewport = editor.editorView.getViewport()
|
|
935
|
+
expect(viewport.offsetY).toBeGreaterThan(initialOffset)
|
|
936
|
+
const maxOffset = viewport.offsetY
|
|
937
|
+
|
|
938
|
+
// Now delete those lines by backspacing
|
|
939
|
+
for (let i = 0; i < 12; i++) {
|
|
940
|
+
// 12 backspaces to delete 6 "X\n" pairs
|
|
941
|
+
currentMockInput.pressBackspace()
|
|
942
|
+
}
|
|
943
|
+
|
|
944
|
+
// Should have scrolled back up
|
|
945
|
+
viewport = editor.editorView.getViewport()
|
|
946
|
+
expect(viewport.offsetY).toBeLessThan(maxOffset)
|
|
947
|
+
})
|
|
948
|
+
|
|
949
|
+
it("should show last line at bottom of viewport with no gap", async () => {
|
|
950
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
951
|
+
initialValue: Array.from({ length: 10 }, (_, i) => `Line ${i}`).join("\n"),
|
|
952
|
+
width: 40,
|
|
953
|
+
height: 5,
|
|
954
|
+
})
|
|
955
|
+
|
|
956
|
+
editor.focus()
|
|
957
|
+
|
|
958
|
+
// Move to last line (line 9)
|
|
959
|
+
editor.gotoLine(9)
|
|
960
|
+
|
|
961
|
+
let viewport = editor.editorView.getViewport()
|
|
962
|
+
|
|
963
|
+
// With 10 lines (0-9) and viewport height 5, max offset is 10 - 5 = 5
|
|
964
|
+
// Viewport should be at offset 5, showing lines 5-9 with line 9 at the bottom
|
|
965
|
+
expect(viewport.offsetY).toBe(5)
|
|
966
|
+
|
|
967
|
+
// Verify cursor line is visible
|
|
968
|
+
expect(9).toBeGreaterThanOrEqual(viewport.offsetY)
|
|
969
|
+
expect(9).toBeLessThan(viewport.offsetY + viewport.height)
|
|
970
|
+
|
|
971
|
+
// No gap - last visible line should be the last line of content
|
|
972
|
+
const lastVisibleLine = viewport.offsetY + viewport.height - 1
|
|
973
|
+
expect(lastVisibleLine).toBe(9)
|
|
974
|
+
})
|
|
975
|
+
|
|
976
|
+
it("should not scroll past end when document is smaller than viewport", async () => {
|
|
977
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
978
|
+
initialValue: "Line 0\nLine 1\nLine 2",
|
|
979
|
+
width: 40,
|
|
980
|
+
height: 10, // Viewport bigger than content
|
|
981
|
+
})
|
|
982
|
+
|
|
983
|
+
editor.focus()
|
|
984
|
+
|
|
985
|
+
// Move to last line
|
|
986
|
+
editor.gotoLine(2)
|
|
987
|
+
|
|
988
|
+
let viewport = editor.editorView.getViewport()
|
|
989
|
+
|
|
990
|
+
// Should NOT scroll at all - content fits in viewport
|
|
991
|
+
expect(viewport.offsetY).toBe(0)
|
|
992
|
+
})
|
|
993
|
+
})
|
|
994
|
+
|
|
995
|
+
describe("Placeholder Support", () => {
|
|
996
|
+
it("should display placeholder when empty", async () => {
|
|
997
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
998
|
+
initialValue: "",
|
|
999
|
+
width: 40,
|
|
1000
|
+
height: 10,
|
|
1001
|
+
placeholder: "Enter text here...",
|
|
1002
|
+
})
|
|
1003
|
+
|
|
1004
|
+
// plainText should return empty (placeholder is display-only)
|
|
1005
|
+
expect(editor.plainText).toBe("")
|
|
1006
|
+
expect(editor.placeholder).toBe("Enter text here...")
|
|
1007
|
+
})
|
|
1008
|
+
|
|
1009
|
+
it("should hide placeholder when text is inserted", async () => {
|
|
1010
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1011
|
+
initialValue: "",
|
|
1012
|
+
width: 40,
|
|
1013
|
+
height: 10,
|
|
1014
|
+
placeholder: "Type something...",
|
|
1015
|
+
})
|
|
1016
|
+
|
|
1017
|
+
editor.focus()
|
|
1018
|
+
expect(editor.plainText).toBe("")
|
|
1019
|
+
|
|
1020
|
+
currentMockInput.pressKey("H")
|
|
1021
|
+
currentMockInput.pressKey("i")
|
|
1022
|
+
|
|
1023
|
+
expect(editor.plainText).toBe("Hi")
|
|
1024
|
+
})
|
|
1025
|
+
|
|
1026
|
+
it("should reactivate placeholder when all text is deleted", async () => {
|
|
1027
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1028
|
+
initialValue: "Test",
|
|
1029
|
+
width: 40,
|
|
1030
|
+
height: 10,
|
|
1031
|
+
placeholder: "Empty buffer...",
|
|
1032
|
+
})
|
|
1033
|
+
|
|
1034
|
+
editor.focus()
|
|
1035
|
+
expect(editor.plainText).toBe("Test")
|
|
1036
|
+
|
|
1037
|
+
// Move to end, then delete all text
|
|
1038
|
+
editor.gotoLine(9999)
|
|
1039
|
+
for (let i = 0; i < 4; i++) {
|
|
1040
|
+
currentMockInput.pressBackspace()
|
|
1041
|
+
}
|
|
1042
|
+
|
|
1043
|
+
expect(editor.plainText).toBe("")
|
|
1044
|
+
})
|
|
1045
|
+
|
|
1046
|
+
it("should update placeholder text dynamically", async () => {
|
|
1047
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1048
|
+
initialValue: "",
|
|
1049
|
+
width: 40,
|
|
1050
|
+
height: 10,
|
|
1051
|
+
placeholder: "First placeholder",
|
|
1052
|
+
})
|
|
1053
|
+
|
|
1054
|
+
expect(editor.placeholder).toBe("First placeholder")
|
|
1055
|
+
expect(editor.plainText).toBe("")
|
|
1056
|
+
|
|
1057
|
+
editor.placeholder = "Second placeholder"
|
|
1058
|
+
expect(editor.placeholder).toBe("Second placeholder")
|
|
1059
|
+
expect(editor.plainText).toBe("")
|
|
1060
|
+
})
|
|
1061
|
+
|
|
1062
|
+
it("should update placeholder with styled text dynamically", async () => {
|
|
1063
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1064
|
+
initialValue: "",
|
|
1065
|
+
width: 40,
|
|
1066
|
+
height: 10,
|
|
1067
|
+
placeholder: "Colored placeholder",
|
|
1068
|
+
})
|
|
1069
|
+
|
|
1070
|
+
expect(editor.plainText).toBe("")
|
|
1071
|
+
|
|
1072
|
+
// Update placeholder with styled text
|
|
1073
|
+
editor.placeholder = t`${fg("#FF0000")("Red placeholder")}`
|
|
1074
|
+
expect(editor.plainText).toBe("")
|
|
1075
|
+
})
|
|
1076
|
+
|
|
1077
|
+
it("should work with value property setter", async () => {
|
|
1078
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1079
|
+
initialValue: "",
|
|
1080
|
+
width: 40,
|
|
1081
|
+
height: 10,
|
|
1082
|
+
placeholder: "Empty state",
|
|
1083
|
+
})
|
|
1084
|
+
|
|
1085
|
+
expect(editor.plainText).toBe("")
|
|
1086
|
+
|
|
1087
|
+
editor.setText("New content")
|
|
1088
|
+
expect(editor.plainText).toBe("New content")
|
|
1089
|
+
|
|
1090
|
+
editor.setText("")
|
|
1091
|
+
expect(editor.plainText).toBe("")
|
|
1092
|
+
})
|
|
1093
|
+
|
|
1094
|
+
it("should handle placeholder with focus changes", async () => {
|
|
1095
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1096
|
+
initialValue: "",
|
|
1097
|
+
width: 40,
|
|
1098
|
+
height: 10,
|
|
1099
|
+
placeholder: "Click to edit",
|
|
1100
|
+
})
|
|
1101
|
+
|
|
1102
|
+
// Placeholder should show regardless of focus
|
|
1103
|
+
expect(editor.plainText).toBe("")
|
|
1104
|
+
|
|
1105
|
+
editor.focus()
|
|
1106
|
+
expect(editor.plainText).toBe("")
|
|
1107
|
+
|
|
1108
|
+
editor.blur()
|
|
1109
|
+
expect(editor.plainText).toBe("")
|
|
1110
|
+
})
|
|
1111
|
+
|
|
1112
|
+
it("should handle typing after placeholder is shown", async () => {
|
|
1113
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1114
|
+
initialValue: "",
|
|
1115
|
+
width: 40,
|
|
1116
|
+
height: 10,
|
|
1117
|
+
placeholder: "Start typing...",
|
|
1118
|
+
})
|
|
1119
|
+
|
|
1120
|
+
editor.focus()
|
|
1121
|
+
expect(editor.plainText).toBe("")
|
|
1122
|
+
|
|
1123
|
+
currentMockInput.pressKey("H")
|
|
1124
|
+
currentMockInput.pressKey("e")
|
|
1125
|
+
currentMockInput.pressKey("l")
|
|
1126
|
+
currentMockInput.pressKey("l")
|
|
1127
|
+
currentMockInput.pressKey("o")
|
|
1128
|
+
|
|
1129
|
+
expect(editor.plainText).toBe("Hello")
|
|
1130
|
+
})
|
|
1131
|
+
|
|
1132
|
+
it("should show placeholder after deleting all typed text", async () => {
|
|
1133
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1134
|
+
initialValue: "",
|
|
1135
|
+
width: 40,
|
|
1136
|
+
height: 10,
|
|
1137
|
+
placeholder: "Type here",
|
|
1138
|
+
})
|
|
1139
|
+
|
|
1140
|
+
editor.focus()
|
|
1141
|
+
|
|
1142
|
+
// Type "Test"
|
|
1143
|
+
currentMockInput.pressKey("T")
|
|
1144
|
+
currentMockInput.pressKey("e")
|
|
1145
|
+
currentMockInput.pressKey("s")
|
|
1146
|
+
currentMockInput.pressKey("t")
|
|
1147
|
+
expect(editor.plainText).toBe("Test")
|
|
1148
|
+
|
|
1149
|
+
// Backspace all
|
|
1150
|
+
for (let i = 0; i < 4; i++) {
|
|
1151
|
+
currentMockInput.pressBackspace()
|
|
1152
|
+
}
|
|
1153
|
+
|
|
1154
|
+
expect(editor.plainText).toBe("")
|
|
1155
|
+
})
|
|
1156
|
+
|
|
1157
|
+
it("should handle placeholder with newlines", async () => {
|
|
1158
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1159
|
+
initialValue: "",
|
|
1160
|
+
width: 40,
|
|
1161
|
+
height: 10,
|
|
1162
|
+
placeholder: "Line 1\nLine 2",
|
|
1163
|
+
})
|
|
1164
|
+
|
|
1165
|
+
expect(editor.plainText).toBe("")
|
|
1166
|
+
|
|
1167
|
+
editor.insertText("Content")
|
|
1168
|
+
expect(editor.plainText).toBe("Content")
|
|
1169
|
+
})
|
|
1170
|
+
|
|
1171
|
+
it("should handle null placeholder (no placeholder)", async () => {
|
|
1172
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1173
|
+
initialValue: "",
|
|
1174
|
+
width: 40,
|
|
1175
|
+
height: 10,
|
|
1176
|
+
placeholder: null,
|
|
1177
|
+
})
|
|
1178
|
+
|
|
1179
|
+
expect(editor.placeholder).toBe(null)
|
|
1180
|
+
expect(editor.plainText).toBe("")
|
|
1181
|
+
|
|
1182
|
+
editor.insertText("Content")
|
|
1183
|
+
expect(editor.plainText).toBe("Content")
|
|
1184
|
+
})
|
|
1185
|
+
|
|
1186
|
+
it("should clear placeholder when set to null", async () => {
|
|
1187
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1188
|
+
initialValue: "",
|
|
1189
|
+
width: 40,
|
|
1190
|
+
height: 10,
|
|
1191
|
+
placeholder: "Initial placeholder",
|
|
1192
|
+
})
|
|
1193
|
+
|
|
1194
|
+
expect(editor.placeholder).toBe("Initial placeholder")
|
|
1195
|
+
expect(editor.plainText).toBe("")
|
|
1196
|
+
|
|
1197
|
+
editor.placeholder = null
|
|
1198
|
+
expect(editor.placeholder).toBe(null)
|
|
1199
|
+
expect(editor.plainText).toBe("")
|
|
1200
|
+
})
|
|
1201
|
+
|
|
1202
|
+
it("should reset placeholder when set to undefined", async () => {
|
|
1203
|
+
const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1204
|
+
initialValue: "",
|
|
1205
|
+
width: 40,
|
|
1206
|
+
height: 10,
|
|
1207
|
+
placeholder: "Initial placeholder",
|
|
1208
|
+
})
|
|
1209
|
+
|
|
1210
|
+
expect(editor.placeholder).toBe("Initial placeholder")
|
|
1211
|
+
|
|
1212
|
+
expect(() => {
|
|
1213
|
+
editor.placeholder = undefined
|
|
1214
|
+
}).not.toThrow()
|
|
1215
|
+
|
|
1216
|
+
expect(editor.placeholder).toBe(null)
|
|
1217
|
+
expect(editor.plainText).toBe("")
|
|
1218
|
+
})
|
|
1219
|
+
})
|
|
1220
|
+
|
|
1221
|
+
describe("Textarea Content Snapshots", () => {
|
|
1222
|
+
it("should render basic text content correctly", async () => {
|
|
1223
|
+
await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1224
|
+
initialValue: "Hello World",
|
|
1225
|
+
left: 5,
|
|
1226
|
+
top: 3,
|
|
1227
|
+
width: 20,
|
|
1228
|
+
height: 5,
|
|
1229
|
+
})
|
|
1230
|
+
|
|
1231
|
+
const frame = captureFrame()
|
|
1232
|
+
expect(frame).toMatchSnapshot()
|
|
1233
|
+
})
|
|
1234
|
+
|
|
1235
|
+
it("should render multiline text content correctly", async () => {
|
|
1236
|
+
await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1237
|
+
initialValue: "Line 1: Hello\nLine 2: World\nLine 3: Testing\nLine 4: Multiline",
|
|
1238
|
+
left: 1,
|
|
1239
|
+
top: 1,
|
|
1240
|
+
width: 30,
|
|
1241
|
+
height: 10,
|
|
1242
|
+
})
|
|
1243
|
+
|
|
1244
|
+
const frame = captureFrame()
|
|
1245
|
+
expect(frame).toMatchSnapshot()
|
|
1246
|
+
})
|
|
1247
|
+
|
|
1248
|
+
it("should render text with character wrapping correctly", async () => {
|
|
1249
|
+
await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1250
|
+
initialValue: "This is a very long text that should wrap to multiple lines when wrap is enabled",
|
|
1251
|
+
wrapMode: "char",
|
|
1252
|
+
width: 15,
|
|
1253
|
+
left: 0,
|
|
1254
|
+
top: 0,
|
|
1255
|
+
})
|
|
1256
|
+
|
|
1257
|
+
const frame = captureFrame()
|
|
1258
|
+
expect(frame).toMatchSnapshot()
|
|
1259
|
+
})
|
|
1260
|
+
|
|
1261
|
+
it("should render text with word wrapping and punctuation", async () => {
|
|
1262
|
+
await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1263
|
+
initialValue: "Hello,World.Test-Example/Path with various punctuation marks!",
|
|
1264
|
+
wrapMode: "word",
|
|
1265
|
+
width: 12,
|
|
1266
|
+
left: 0,
|
|
1267
|
+
top: 0,
|
|
1268
|
+
})
|
|
1269
|
+
|
|
1270
|
+
const frame = captureFrame()
|
|
1271
|
+
expect(frame).toMatchSnapshot()
|
|
1272
|
+
})
|
|
1273
|
+
|
|
1274
|
+
it("should render placeholder when creating textarea with placeholder directly", async () => {
|
|
1275
|
+
await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1276
|
+
placeholder: "Enter text here...",
|
|
1277
|
+
left: 1,
|
|
1278
|
+
top: 1,
|
|
1279
|
+
width: 30,
|
|
1280
|
+
height: 5,
|
|
1281
|
+
})
|
|
1282
|
+
|
|
1283
|
+
const frame = captureFrame()
|
|
1284
|
+
expect(frame).toMatchSnapshot()
|
|
1285
|
+
})
|
|
1286
|
+
|
|
1287
|
+
it("should render placeholder when set programmatically after creation", async () => {
|
|
1288
|
+
const { textarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1289
|
+
left: 1,
|
|
1290
|
+
top: 1,
|
|
1291
|
+
width: 30,
|
|
1292
|
+
height: 5,
|
|
1293
|
+
})
|
|
1294
|
+
|
|
1295
|
+
textarea.placeholder = "Type something..."
|
|
1296
|
+
await renderOnce()
|
|
1297
|
+
|
|
1298
|
+
const frame = captureFrame()
|
|
1299
|
+
expect(frame).toMatchSnapshot()
|
|
1300
|
+
})
|
|
1301
|
+
|
|
1302
|
+
it("should resize correctly when typing return as first input with placeholder", async () => {
|
|
1303
|
+
resize(40, 10)
|
|
1304
|
+
|
|
1305
|
+
const container = new BoxRenderable(currentRenderer, {
|
|
1306
|
+
border: true,
|
|
1307
|
+
left: 1,
|
|
1308
|
+
top: 1,
|
|
1309
|
+
})
|
|
1310
|
+
currentRenderer.root.add(container)
|
|
1311
|
+
|
|
1312
|
+
const textarea = new TextareaRenderable(currentRenderer, {
|
|
1313
|
+
placeholder: "Enter your message...",
|
|
1314
|
+
width: 30,
|
|
1315
|
+
minHeight: 1,
|
|
1316
|
+
maxHeight: 3,
|
|
1317
|
+
})
|
|
1318
|
+
container.add(textarea)
|
|
1319
|
+
|
|
1320
|
+
textarea.focus()
|
|
1321
|
+
await renderOnce()
|
|
1322
|
+
|
|
1323
|
+
const frameBeforeEnter = captureFrame()
|
|
1324
|
+
expect(textarea.height).toBe(1)
|
|
1325
|
+
|
|
1326
|
+
currentMockInput.pressEnter()
|
|
1327
|
+
await renderOnce()
|
|
1328
|
+
await renderOnce()
|
|
1329
|
+
|
|
1330
|
+
const frameAfterEnter = captureFrame()
|
|
1331
|
+
expect(frameAfterEnter).toMatchSnapshot()
|
|
1332
|
+
expect(textarea.height).toBe(2)
|
|
1333
|
+
expect(textarea.plainText).toBe("\n")
|
|
1334
|
+
})
|
|
1335
|
+
})
|
|
1336
|
+
|
|
1337
|
+
describe("Layout Reflow on Size Change", () => {
|
|
1338
|
+
it("should reflow subsequent elements when textarea grows and shrinks", async () => {
|
|
1339
|
+
const { textarea: firstEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1340
|
+
initialValue: "Short",
|
|
1341
|
+
width: 20,
|
|
1342
|
+
wrapMode: "word",
|
|
1343
|
+
})
|
|
1344
|
+
|
|
1345
|
+
const { textarea: secondEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1346
|
+
initialValue: "I am below the first textarea",
|
|
1347
|
+
width: 30,
|
|
1348
|
+
})
|
|
1349
|
+
|
|
1350
|
+
await renderOnce()
|
|
1351
|
+
|
|
1352
|
+
// Initially, first editor is 1 line high
|
|
1353
|
+
expect(firstEditor.height).toBe(1)
|
|
1354
|
+
const initialSecondY = secondEditor.y
|
|
1355
|
+
expect(initialSecondY).toBe(1) // Right after first editor
|
|
1356
|
+
|
|
1357
|
+
// Expand first editor with wrapped content
|
|
1358
|
+
firstEditor.setText("This is a very long line that will wrap to multiple lines and push the second textarea down")
|
|
1359
|
+
await renderOnce()
|
|
1360
|
+
|
|
1361
|
+
// First editor should now be taller
|
|
1362
|
+
expect(firstEditor.height).toBeGreaterThan(1)
|
|
1363
|
+
// Second editor should have moved down
|
|
1364
|
+
expect(secondEditor.y).toBeGreaterThan(initialSecondY)
|
|
1365
|
+
const expandedSecondY = secondEditor.y
|
|
1366
|
+
|
|
1367
|
+
// Shrink first editor back
|
|
1368
|
+
firstEditor.setText("Short again")
|
|
1369
|
+
await renderOnce()
|
|
1370
|
+
|
|
1371
|
+
// First editor should be 1 line again
|
|
1372
|
+
expect(firstEditor.height).toBe(1)
|
|
1373
|
+
// Second editor should have moved back up
|
|
1374
|
+
expect(secondEditor.y).toBeLessThan(expandedSecondY)
|
|
1375
|
+
expect(secondEditor.y).toBe(initialSecondY)
|
|
1376
|
+
})
|
|
1377
|
+
})
|
|
1378
|
+
|
|
1379
|
+
describe("Width/Height Setter Layout Tests", () => {
|
|
1380
|
+
it("should not shrink box when width is set via setter", async () => {
|
|
1381
|
+
resize(40, 10)
|
|
1382
|
+
|
|
1383
|
+
const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
|
|
1384
|
+
currentRenderer.root.add(container)
|
|
1385
|
+
|
|
1386
|
+
const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
|
|
1387
|
+
container.add(row)
|
|
1388
|
+
|
|
1389
|
+
const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00" })
|
|
1390
|
+
row.add(indicator)
|
|
1391
|
+
|
|
1392
|
+
const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
|
|
1393
|
+
indicator.add(indicatorText)
|
|
1394
|
+
|
|
1395
|
+
const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
|
|
1396
|
+
row.add(content)
|
|
1397
|
+
|
|
1398
|
+
const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
|
|
1399
|
+
content.add(contentText)
|
|
1400
|
+
|
|
1401
|
+
await renderOnce()
|
|
1402
|
+
|
|
1403
|
+
const initialIndicatorWidth = indicator.width
|
|
1404
|
+
|
|
1405
|
+
indicator.width = 5
|
|
1406
|
+
await renderOnce()
|
|
1407
|
+
|
|
1408
|
+
const frame = captureFrame()
|
|
1409
|
+
expect(frame).toMatchSnapshot()
|
|
1410
|
+
|
|
1411
|
+
expect(indicator.width).toBe(5)
|
|
1412
|
+
expect(content.width).toBeGreaterThan(0)
|
|
1413
|
+
expect(content.width).toBeLessThan(30)
|
|
1414
|
+
})
|
|
1415
|
+
|
|
1416
|
+
it("should not shrink box when height is set via setter in column layout with textarea", async () => {
|
|
1417
|
+
resize(30, 15)
|
|
1418
|
+
|
|
1419
|
+
const outerBox = new BoxRenderable(currentRenderer, { border: true, width: 25, height: 10 })
|
|
1420
|
+
currentRenderer.root.add(outerBox)
|
|
1421
|
+
|
|
1422
|
+
const column = new BoxRenderable(currentRenderer, { flexDirection: "column", height: "100%" })
|
|
1423
|
+
outerBox.add(column)
|
|
1424
|
+
|
|
1425
|
+
const header = new BoxRenderable(currentRenderer, { backgroundColor: "#f00" })
|
|
1426
|
+
column.add(header)
|
|
1427
|
+
|
|
1428
|
+
const headerText = new TextRenderable(currentRenderer, { content: "Header" })
|
|
1429
|
+
header.add(headerText)
|
|
1430
|
+
|
|
1431
|
+
const mainContent = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
|
|
1432
|
+
column.add(mainContent)
|
|
1433
|
+
|
|
1434
|
+
const { textarea: mainTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1435
|
+
initialValue: "Line1\nLine2\nLine3\nLine4\nLine5\nLine6\nLine7\nLine8",
|
|
1436
|
+
})
|
|
1437
|
+
mainContent.add(mainTextarea)
|
|
1438
|
+
|
|
1439
|
+
const footer = new BoxRenderable(currentRenderer, { height: 2, backgroundColor: "#00f" })
|
|
1440
|
+
column.add(footer)
|
|
1441
|
+
|
|
1442
|
+
const footerText = new TextRenderable(currentRenderer, { content: "Footer" })
|
|
1443
|
+
footer.add(footerText)
|
|
1444
|
+
|
|
1445
|
+
await renderOnce()
|
|
1446
|
+
|
|
1447
|
+
header.height = 3
|
|
1448
|
+
await renderOnce()
|
|
1449
|
+
|
|
1450
|
+
const frame = captureFrame()
|
|
1451
|
+
expect(frame).toMatchSnapshot()
|
|
1452
|
+
|
|
1453
|
+
expect(header.height).toBe(3)
|
|
1454
|
+
expect(mainContent.height).toBeGreaterThan(0)
|
|
1455
|
+
expect(footer.height).toBe(2)
|
|
1456
|
+
})
|
|
1457
|
+
|
|
1458
|
+
it("should not shrink box when minWidth is set via setter", async () => {
|
|
1459
|
+
resize(40, 10)
|
|
1460
|
+
|
|
1461
|
+
const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
|
|
1462
|
+
currentRenderer.root.add(container)
|
|
1463
|
+
|
|
1464
|
+
const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
|
|
1465
|
+
container.add(row)
|
|
1466
|
+
|
|
1467
|
+
const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
|
|
1468
|
+
row.add(indicator)
|
|
1469
|
+
|
|
1470
|
+
const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
|
|
1471
|
+
indicator.add(indicatorText)
|
|
1472
|
+
|
|
1473
|
+
const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
|
|
1474
|
+
row.add(content)
|
|
1475
|
+
|
|
1476
|
+
const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
|
|
1477
|
+
content.add(contentText)
|
|
1478
|
+
|
|
1479
|
+
await renderOnce()
|
|
1480
|
+
|
|
1481
|
+
indicator.minWidth = 5
|
|
1482
|
+
await renderOnce()
|
|
1483
|
+
|
|
1484
|
+
const frame = captureFrame()
|
|
1485
|
+
expect(frame).toMatchSnapshot()
|
|
1486
|
+
expect(indicator.width).toBeGreaterThanOrEqual(5)
|
|
1487
|
+
expect(content.width).toBeGreaterThan(0)
|
|
1488
|
+
})
|
|
1489
|
+
|
|
1490
|
+
it("should not shrink box when minHeight is set via setter in column layout with textarea", async () => {
|
|
1491
|
+
resize(30, 15)
|
|
1492
|
+
|
|
1493
|
+
const outerBox = new BoxRenderable(currentRenderer, { border: true, width: 25, height: 10 })
|
|
1494
|
+
currentRenderer.root.add(outerBox)
|
|
1495
|
+
|
|
1496
|
+
const column = new BoxRenderable(currentRenderer, { flexDirection: "column", height: "100%" })
|
|
1497
|
+
outerBox.add(column)
|
|
1498
|
+
|
|
1499
|
+
const header = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
|
|
1500
|
+
column.add(header)
|
|
1501
|
+
|
|
1502
|
+
const headerText = new TextRenderable(currentRenderer, { content: "Header" })
|
|
1503
|
+
header.add(headerText)
|
|
1504
|
+
|
|
1505
|
+
const mainContent = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
|
|
1506
|
+
column.add(mainContent)
|
|
1507
|
+
|
|
1508
|
+
const { textarea: mainTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1509
|
+
initialValue: "Line1\nLine2\nLine3\nLine4\nLine5\nLine6\nLine7\nLine8",
|
|
1510
|
+
})
|
|
1511
|
+
mainContent.add(mainTextarea)
|
|
1512
|
+
|
|
1513
|
+
const footer = new BoxRenderable(currentRenderer, { height: 2, backgroundColor: "#00f" })
|
|
1514
|
+
column.add(footer)
|
|
1515
|
+
|
|
1516
|
+
const footerText = new TextRenderable(currentRenderer, { content: "Footer" })
|
|
1517
|
+
footer.add(footerText)
|
|
1518
|
+
|
|
1519
|
+
await renderOnce()
|
|
1520
|
+
|
|
1521
|
+
header.minHeight = 3
|
|
1522
|
+
await renderOnce()
|
|
1523
|
+
|
|
1524
|
+
const frame = captureFrame()
|
|
1525
|
+
expect(frame).toMatchSnapshot()
|
|
1526
|
+
|
|
1527
|
+
expect(header.height).toBeGreaterThanOrEqual(3)
|
|
1528
|
+
expect(mainContent.height).toBeGreaterThan(0)
|
|
1529
|
+
expect(footer.height).toBe(2)
|
|
1530
|
+
})
|
|
1531
|
+
|
|
1532
|
+
it("should not shrink box when width is set from undefined via setter", async () => {
|
|
1533
|
+
resize(40, 10)
|
|
1534
|
+
|
|
1535
|
+
const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
|
|
1536
|
+
currentRenderer.root.add(container)
|
|
1537
|
+
|
|
1538
|
+
const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
|
|
1539
|
+
container.add(row)
|
|
1540
|
+
|
|
1541
|
+
const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
|
|
1542
|
+
row.add(indicator)
|
|
1543
|
+
|
|
1544
|
+
const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
|
|
1545
|
+
indicator.add(indicatorText)
|
|
1546
|
+
|
|
1547
|
+
const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
|
|
1548
|
+
row.add(content)
|
|
1549
|
+
|
|
1550
|
+
const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
|
|
1551
|
+
content.add(contentText)
|
|
1552
|
+
|
|
1553
|
+
await renderOnce()
|
|
1554
|
+
|
|
1555
|
+
indicator.width = 5
|
|
1556
|
+
await renderOnce()
|
|
1557
|
+
|
|
1558
|
+
const frame = captureFrame()
|
|
1559
|
+
expect(frame).toMatchSnapshot()
|
|
1560
|
+
|
|
1561
|
+
expect(indicator.width).toBe(5)
|
|
1562
|
+
expect(content.width).toBeGreaterThan(0)
|
|
1563
|
+
})
|
|
1564
|
+
|
|
1565
|
+
it("should verify dimensions are actually respected under extreme pressure", async () => {
|
|
1566
|
+
resize(30, 10)
|
|
1567
|
+
|
|
1568
|
+
const container = new BoxRenderable(currentRenderer, { border: true, width: 20 })
|
|
1569
|
+
currentRenderer.root.add(container)
|
|
1570
|
+
|
|
1571
|
+
const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
|
|
1572
|
+
container.add(row)
|
|
1573
|
+
|
|
1574
|
+
const box1 = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
|
|
1575
|
+
row.add(box1)
|
|
1576
|
+
const text1 = new TextRenderable(currentRenderer, { content: "AAA" })
|
|
1577
|
+
box1.add(text1)
|
|
1578
|
+
|
|
1579
|
+
const box2 = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexShrink: 1 })
|
|
1580
|
+
row.add(box2)
|
|
1581
|
+
const text2 = new TextRenderable(currentRenderer, { content: "BBB" })
|
|
1582
|
+
box2.add(text2)
|
|
1583
|
+
|
|
1584
|
+
const box3 = new BoxRenderable(currentRenderer, { backgroundColor: "#00f", flexGrow: 1 })
|
|
1585
|
+
row.add(box3)
|
|
1586
|
+
const text3 = new TextRenderable(currentRenderer, { content: "CCC" })
|
|
1587
|
+
box3.add(text3)
|
|
1588
|
+
|
|
1589
|
+
await renderOnce()
|
|
1590
|
+
|
|
1591
|
+
box1.width = 7
|
|
1592
|
+
box2.minWidth = 5
|
|
1593
|
+
await renderOnce()
|
|
1594
|
+
|
|
1595
|
+
expect(box1.width).toBe(7)
|
|
1596
|
+
expect(box2.width).toBeGreaterThanOrEqual(5)
|
|
1597
|
+
expect(box3.width).toBeGreaterThan(0)
|
|
1598
|
+
|
|
1599
|
+
const total = box1.width + box2.width + box3.width
|
|
1600
|
+
expect(total).toBeLessThanOrEqual(18)
|
|
1601
|
+
})
|
|
1602
|
+
})
|
|
1603
|
+
|
|
1604
|
+
describe("Absolute Positioned Box with Textarea", () => {
|
|
1605
|
+
it("should render textarea in absolute positioned box with padding and borders correctly", async () => {
|
|
1606
|
+
resize(80, 20)
|
|
1607
|
+
|
|
1608
|
+
const notificationBox = new BoxRenderable(currentRenderer, {
|
|
1609
|
+
position: "absolute",
|
|
1610
|
+
justifyContent: "center",
|
|
1611
|
+
alignItems: "flex-start",
|
|
1612
|
+
top: 2,
|
|
1613
|
+
right: 2,
|
|
1614
|
+
maxWidth: Math.min(60, 80 - 6),
|
|
1615
|
+
paddingLeft: 2,
|
|
1616
|
+
paddingRight: 2,
|
|
1617
|
+
paddingTop: 1,
|
|
1618
|
+
paddingBottom: 1,
|
|
1619
|
+
backgroundColor: "#1e293b",
|
|
1620
|
+
borderColor: "#3b82f6",
|
|
1621
|
+
border: ["left", "right"],
|
|
1622
|
+
})
|
|
1623
|
+
|
|
1624
|
+
currentRenderer.root.add(notificationBox)
|
|
1625
|
+
|
|
1626
|
+
const outerWrapperBox = new BoxRenderable(currentRenderer, {
|
|
1627
|
+
flexDirection: "row",
|
|
1628
|
+
paddingBottom: 1,
|
|
1629
|
+
paddingTop: 1,
|
|
1630
|
+
paddingLeft: 2,
|
|
1631
|
+
paddingRight: 2,
|
|
1632
|
+
gap: 2,
|
|
1633
|
+
})
|
|
1634
|
+
notificationBox.add(outerWrapperBox)
|
|
1635
|
+
|
|
1636
|
+
const innerContentBox = new BoxRenderable(currentRenderer, {
|
|
1637
|
+
flexGrow: 1,
|
|
1638
|
+
gap: 1,
|
|
1639
|
+
})
|
|
1640
|
+
outerWrapperBox.add(innerContentBox)
|
|
1641
|
+
|
|
1642
|
+
const titleText = new TextRenderable(currentRenderer, {
|
|
1643
|
+
content: "Important Notification",
|
|
1644
|
+
attributes: 1,
|
|
1645
|
+
marginBottom: 1,
|
|
1646
|
+
fg: "#f8fafc",
|
|
1647
|
+
})
|
|
1648
|
+
innerContentBox.add(titleText)
|
|
1649
|
+
|
|
1650
|
+
const { textarea: messageTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1651
|
+
initialValue:
|
|
1652
|
+
"This is a longer message that should wrap properly within the absolutely positioned box with appropriate width constraints and padding applied.",
|
|
1653
|
+
textColor: "#e2e8f0",
|
|
1654
|
+
wrapMode: "word",
|
|
1655
|
+
width: "100%",
|
|
1656
|
+
})
|
|
1657
|
+
innerContentBox.add(messageTextarea)
|
|
1658
|
+
|
|
1659
|
+
await renderOnce()
|
|
1660
|
+
|
|
1661
|
+
const frame = captureFrame()
|
|
1662
|
+
expect(frame).toMatchSnapshot()
|
|
1663
|
+
|
|
1664
|
+
expect(notificationBox.x).toBeGreaterThan(0)
|
|
1665
|
+
expect(notificationBox.y).toBe(2)
|
|
1666
|
+
expect(notificationBox.width).toBeGreaterThan(25)
|
|
1667
|
+
|
|
1668
|
+
expect(outerWrapperBox.width).toBeGreaterThan(15)
|
|
1669
|
+
expect(innerContentBox.width).toBeGreaterThan(15)
|
|
1670
|
+
|
|
1671
|
+
expect(titleText.width).toBeGreaterThan(15)
|
|
1672
|
+
expect(titleText.plainText).toBe("Important Notification")
|
|
1673
|
+
expect(titleText.height).toBe(1)
|
|
1674
|
+
|
|
1675
|
+
expect(messageTextarea.width).toBeGreaterThan(15)
|
|
1676
|
+
expect(messageTextarea.height).toBeGreaterThanOrEqual(1)
|
|
1677
|
+
expect(messageTextarea.plainText).toBe(
|
|
1678
|
+
"This is a longer message that should wrap properly within the absolutely positioned box with appropriate width constraints and padding applied.",
|
|
1679
|
+
)
|
|
1680
|
+
})
|
|
1681
|
+
|
|
1682
|
+
it("should render textarea fully visible in absolute positioned box at various positions", async () => {
|
|
1683
|
+
resize(100, 25)
|
|
1684
|
+
|
|
1685
|
+
const topRightBox = new BoxRenderable(currentRenderer, {
|
|
1686
|
+
position: "absolute",
|
|
1687
|
+
top: 1,
|
|
1688
|
+
right: 1,
|
|
1689
|
+
maxWidth: 40,
|
|
1690
|
+
paddingLeft: 1,
|
|
1691
|
+
paddingRight: 1,
|
|
1692
|
+
paddingTop: 0,
|
|
1693
|
+
paddingBottom: 0,
|
|
1694
|
+
backgroundColor: "#fef2f2",
|
|
1695
|
+
borderColor: "#ef4444",
|
|
1696
|
+
border: true,
|
|
1697
|
+
})
|
|
1698
|
+
currentRenderer.root.add(topRightBox)
|
|
1699
|
+
|
|
1700
|
+
const { textarea: topRightTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1701
|
+
initialValue: "Error: File not found in the specified directory path",
|
|
1702
|
+
textColor: "#991b1b",
|
|
1703
|
+
wrapMode: "word",
|
|
1704
|
+
width: "100%",
|
|
1705
|
+
})
|
|
1706
|
+
topRightBox.add(topRightTextarea)
|
|
1707
|
+
|
|
1708
|
+
const bottomLeftBox = new BoxRenderable(currentRenderer, {
|
|
1709
|
+
position: "absolute",
|
|
1710
|
+
bottom: 1,
|
|
1711
|
+
left: 1,
|
|
1712
|
+
maxWidth: 35,
|
|
1713
|
+
paddingLeft: 1,
|
|
1714
|
+
paddingRight: 1,
|
|
1715
|
+
backgroundColor: "#f0fdf4",
|
|
1716
|
+
borderColor: "#22c55e",
|
|
1717
|
+
border: ["top", "bottom"],
|
|
1718
|
+
})
|
|
1719
|
+
currentRenderer.root.add(bottomLeftBox)
|
|
1720
|
+
|
|
1721
|
+
const { textarea: bottomLeftTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1722
|
+
initialValue: "Success: Operation completed successfully!",
|
|
1723
|
+
textColor: "#166534",
|
|
1724
|
+
wrapMode: "word",
|
|
1725
|
+
width: "100%",
|
|
1726
|
+
})
|
|
1727
|
+
bottomLeftBox.add(bottomLeftTextarea)
|
|
1728
|
+
|
|
1729
|
+
await renderOnce()
|
|
1730
|
+
|
|
1731
|
+
const frame = captureFrame()
|
|
1732
|
+
expect(frame).toMatchSnapshot()
|
|
1733
|
+
|
|
1734
|
+
expect(topRightBox.y).toBe(1)
|
|
1735
|
+
expect(topRightBox.x).toBeGreaterThan(50)
|
|
1736
|
+
expect(topRightBox.width).toBeGreaterThan(30)
|
|
1737
|
+
expect(topRightBox.width).toBeLessThanOrEqual(40)
|
|
1738
|
+
|
|
1739
|
+
expect(topRightTextarea.plainText).toBe("Error: File not found in the specified directory path")
|
|
1740
|
+
expect(topRightTextarea.width).toBeGreaterThan(25)
|
|
1741
|
+
expect(topRightTextarea.width).toBeLessThanOrEqual(38)
|
|
1742
|
+
expect(topRightTextarea.height).toBeGreaterThan(1)
|
|
1743
|
+
|
|
1744
|
+
expect(bottomLeftBox.x).toBe(1)
|
|
1745
|
+
expect(bottomLeftBox.y).toBeGreaterThan(15)
|
|
1746
|
+
expect(bottomLeftBox.width).toBeGreaterThan(25)
|
|
1747
|
+
expect(bottomLeftBox.width).toBeLessThanOrEqual(35)
|
|
1748
|
+
|
|
1749
|
+
expect(bottomLeftTextarea.plainText).toBe("Success: Operation completed successfully!")
|
|
1750
|
+
expect(bottomLeftTextarea.width).toBeGreaterThan(25)
|
|
1751
|
+
expect(bottomLeftTextarea.width).toBeLessThanOrEqual(33)
|
|
1752
|
+
expect(bottomLeftTextarea.height).toBeGreaterThan(1)
|
|
1753
|
+
})
|
|
1754
|
+
|
|
1755
|
+
it("should handle width:100% textarea in absolute positioned box with constrained maxWidth", async () => {
|
|
1756
|
+
resize(70, 15)
|
|
1757
|
+
|
|
1758
|
+
const constrainedBox = new BoxRenderable(currentRenderer, {
|
|
1759
|
+
position: "absolute",
|
|
1760
|
+
top: 5,
|
|
1761
|
+
left: 10,
|
|
1762
|
+
maxWidth: 50,
|
|
1763
|
+
paddingLeft: 3,
|
|
1764
|
+
paddingRight: 3,
|
|
1765
|
+
paddingTop: 2,
|
|
1766
|
+
paddingBottom: 2,
|
|
1767
|
+
backgroundColor: "#1e1e2e",
|
|
1768
|
+
})
|
|
1769
|
+
currentRenderer.root.add(constrainedBox)
|
|
1770
|
+
|
|
1771
|
+
const { textarea: longTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1772
|
+
initialValue:
|
|
1773
|
+
"This is an extremely long piece of text that needs to wrap multiple times within the constrained width of the absolutely positioned container box with significant padding on all sides.",
|
|
1774
|
+
textColor: "#cdd6f4",
|
|
1775
|
+
wrapMode: "word",
|
|
1776
|
+
width: "100%",
|
|
1777
|
+
})
|
|
1778
|
+
constrainedBox.add(longTextarea)
|
|
1779
|
+
|
|
1780
|
+
await renderOnce()
|
|
1781
|
+
|
|
1782
|
+
const frame = captureFrame()
|
|
1783
|
+
expect(frame).toMatchSnapshot()
|
|
1784
|
+
|
|
1785
|
+
expect(constrainedBox.width).toBeLessThanOrEqual(50)
|
|
1786
|
+
expect(constrainedBox.width).toBeGreaterThan(40)
|
|
1787
|
+
expect(constrainedBox.x).toBe(10)
|
|
1788
|
+
expect(constrainedBox.y).toBe(5)
|
|
1789
|
+
|
|
1790
|
+
expect(longTextarea.width).toBeGreaterThan(35)
|
|
1791
|
+
expect(longTextarea.width).toBeLessThanOrEqual(44)
|
|
1792
|
+
expect(longTextarea.height).toBeGreaterThanOrEqual(5)
|
|
1793
|
+
expect(longTextarea.plainText).toBe(
|
|
1794
|
+
"This is an extremely long piece of text that needs to wrap multiple times within the constrained width of the absolutely positioned container box with significant padding on all sides.",
|
|
1795
|
+
)
|
|
1796
|
+
})
|
|
1797
|
+
|
|
1798
|
+
it("should render multiple textarea elements in absolute positioned box with proper spacing", async () => {
|
|
1799
|
+
resize(90, 20)
|
|
1800
|
+
|
|
1801
|
+
const infoBox = new BoxRenderable(currentRenderer, {
|
|
1802
|
+
position: "absolute",
|
|
1803
|
+
justifyContent: "flex-start",
|
|
1804
|
+
alignItems: "flex-start",
|
|
1805
|
+
top: 3,
|
|
1806
|
+
right: 5,
|
|
1807
|
+
maxWidth: 45,
|
|
1808
|
+
paddingLeft: 2,
|
|
1809
|
+
paddingRight: 2,
|
|
1810
|
+
paddingTop: 1,
|
|
1811
|
+
paddingBottom: 1,
|
|
1812
|
+
backgroundColor: "#eff6ff",
|
|
1813
|
+
borderColor: "#3b82f6",
|
|
1814
|
+
border: true,
|
|
1815
|
+
})
|
|
1816
|
+
currentRenderer.root.add(infoBox)
|
|
1817
|
+
|
|
1818
|
+
const headerText = new TextRenderable(currentRenderer, {
|
|
1819
|
+
content: "System Update",
|
|
1820
|
+
attributes: 1,
|
|
1821
|
+
fg: "#1e40af",
|
|
1822
|
+
})
|
|
1823
|
+
infoBox.add(headerText)
|
|
1824
|
+
|
|
1825
|
+
const { textarea: bodyTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
|
|
1826
|
+
initialValue: "A new version is available with bug fixes and performance improvements.",
|
|
1827
|
+
textColor: "#1e3a8a",
|
|
1828
|
+
wrapMode: "word",
|
|
1829
|
+
width: "100%",
|
|
1830
|
+
marginTop: 1,
|
|
1831
|
+
})
|
|
1832
|
+
infoBox.add(bodyTextarea)
|
|
1833
|
+
|
|
1834
|
+
const footerText = new TextRenderable(currentRenderer, {
|
|
1835
|
+
content: "Click to install",
|
|
1836
|
+
fg: "#60a5fa",
|
|
1837
|
+
marginTop: 1,
|
|
1838
|
+
})
|
|
1839
|
+
infoBox.add(footerText)
|
|
1840
|
+
|
|
1841
|
+
await renderOnce()
|
|
1842
|
+
|
|
1843
|
+
const frame = captureFrame()
|
|
1844
|
+
expect(frame).toMatchSnapshot()
|
|
1845
|
+
|
|
1846
|
+
expect(headerText.plainText).toBe("System Update")
|
|
1847
|
+
expect(bodyTextarea.plainText).toBe("A new version is available with bug fixes and performance improvements.")
|
|
1848
|
+
expect(footerText.plainText).toBe("Click to install")
|
|
1849
|
+
|
|
1850
|
+
expect(infoBox.width).toBeGreaterThan(35)
|
|
1851
|
+
expect(infoBox.width).toBeLessThanOrEqual(45)
|
|
1852
|
+
|
|
1853
|
+
expect(headerText.width).toBeGreaterThan(10)
|
|
1854
|
+
expect(headerText.height).toBe(1)
|
|
1855
|
+
|
|
1856
|
+
expect(bodyTextarea.width).toBeGreaterThan(30)
|
|
1857
|
+
expect(bodyTextarea.height).toBeGreaterThanOrEqual(2)
|
|
1858
|
+
|
|
1859
|
+
expect(footerText.width).toBeGreaterThan(10)
|
|
1860
|
+
expect(footerText.height).toBe(1)
|
|
1861
|
+
|
|
1862
|
+
expect(bodyTextarea.y).toBeGreaterThan(headerText.y)
|
|
1863
|
+
expect(footerText.y).toBeGreaterThan(bodyTextarea.y)
|
|
1864
|
+
})
|
|
1865
|
+
})
|
|
1866
|
+
})
|