@fairyhunter13/opentui-core 0.1.113 β 0.1.114
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dev/keypress-debug-renderer.ts +148 -0
- package/dev/keypress-debug.ts +43 -0
- package/dev/print-env-vars.ts +32 -0
- package/dev/test-tmux-graphics-334.sh +68 -0
- package/dev/thai-debug-test.ts +68 -0
- package/docs/development.md +144 -0
- package/package.json +62 -53
- package/scripts/build.ts +400 -0
- package/scripts/publish.ts +60 -0
- package/src/3d/SpriteResourceManager.ts +286 -0
- package/src/3d/SpriteUtils.ts +70 -0
- package/src/3d/TextureUtils.ts +196 -0
- package/src/3d/ThreeRenderable.ts +197 -0
- package/src/3d/WGPURenderer.ts +294 -0
- package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
- package/src/3d/animation/SpriteAnimator.ts +633 -0
- package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
- package/src/3d/canvas.ts +464 -0
- package/src/3d/index.ts +12 -0
- package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
- package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
- package/src/3d/physics/physics-interface.ts +31 -0
- package/src/3d/shaders/supersampling.wgsl +201 -0
- package/src/3d.ts +3 -0
- package/src/NativeSpanFeed.ts +300 -0
- package/src/Renderable.ts +1704 -0
- package/src/__snapshots__/buffer.test.ts.snap +28 -0
- package/src/animation/Timeline.test.ts +2709 -0
- package/src/animation/Timeline.ts +598 -0
- package/src/ansi.ts +18 -0
- package/src/benchmark/attenuation-benchmark.ts +81 -0
- package/src/benchmark/colormatrix-benchmark.ts +128 -0
- package/src/benchmark/gain-benchmark.ts +80 -0
- package/src/benchmark/latest-all-bench-run.json +707 -0
- package/src/benchmark/latest-async-bench-run.json +336 -0
- package/src/benchmark/latest-default-bench-run.json +657 -0
- package/src/benchmark/latest-large-bench-run.json +707 -0
- package/src/benchmark/latest-quick-bench-run.json +207 -0
- package/src/benchmark/markdown-benchmark.ts +1796 -0
- package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
- package/src/benchmark/native-span-feed-benchmark.md +56 -0
- package/src/benchmark/native-span-feed-benchmark.ts +596 -0
- package/src/benchmark/native-span-feed-compare.ts +280 -0
- package/src/benchmark/renderer-benchmark.ts +754 -0
- package/src/benchmark/text-table-benchmark.ts +948 -0
- package/src/buffer.test.ts +291 -0
- package/src/buffer.ts +554 -0
- package/src/console.test.ts +612 -0
- package/src/console.ts +1254 -0
- package/src/edit-buffer.test.ts +1769 -0
- package/src/edit-buffer.ts +411 -0
- package/src/editor-view.test.ts +1032 -0
- package/src/editor-view.ts +284 -0
- package/src/examples/ascii-font-selection-demo.ts +245 -0
- package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
- package/src/examples/assets/concrete.png +0 -0
- package/src/examples/assets/crate.png +0 -0
- package/src/examples/assets/crate_emissive.png +0 -0
- package/src/examples/assets/forrest_background.png +0 -0
- package/src/examples/assets/hast-example.json +1018 -0
- package/src/examples/assets/heart.png +0 -0
- package/src/examples/assets/main_char_heavy_attack.png +0 -0
- package/src/examples/assets/main_char_idle.png +0 -0
- package/src/examples/assets/main_char_jump_end.png +0 -0
- package/src/examples/assets/main_char_jump_landing.png +0 -0
- package/src/examples/assets/main_char_jump_start.png +0 -0
- package/src/examples/assets/main_char_run_loop.png +0 -0
- package/src/examples/assets/roughness_map.jpg +0 -0
- package/src/examples/build.ts +115 -0
- package/src/examples/code-demo.ts +924 -0
- package/src/examples/console-demo.ts +358 -0
- package/src/examples/core-plugin-slots-demo.ts +759 -0
- package/src/examples/diff-demo.ts +701 -0
- package/src/examples/draggable-three-demo.ts +259 -0
- package/src/examples/editor-demo.ts +322 -0
- package/src/examples/extmarks-demo.ts +196 -0
- package/src/examples/focus-restore-demo.ts +310 -0
- package/src/examples/fonts.ts +245 -0
- package/src/examples/fractal-shader-demo.ts +268 -0
- package/src/examples/framebuffer-demo.ts +674 -0
- package/src/examples/full-unicode-demo.ts +241 -0
- package/src/examples/golden-star-demo.ts +933 -0
- package/src/examples/grayscale-buffer-demo.ts +249 -0
- package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
- package/src/examples/index.ts +926 -0
- package/src/examples/input-demo.ts +377 -0
- package/src/examples/input-select-layout-demo.ts +425 -0
- package/src/examples/install.sh +143 -0
- package/src/examples/keypress-debug-demo.ts +452 -0
- package/src/examples/lib/HexList.ts +122 -0
- package/src/examples/lib/PaletteGrid.ts +125 -0
- package/src/examples/lib/standalone-keys.ts +25 -0
- package/src/examples/lib/tab-controller.ts +243 -0
- package/src/examples/lights-phong-demo.ts +290 -0
- package/src/examples/link-demo.ts +220 -0
- package/src/examples/live-state-demo.ts +480 -0
- package/src/examples/markdown-demo.ts +725 -0
- package/src/examples/mouse-interaction-demo.ts +428 -0
- package/src/examples/nested-zindex-demo.ts +357 -0
- package/src/examples/opacity-example.ts +235 -0
- package/src/examples/opentui-demo.ts +1057 -0
- package/src/examples/physx-planck-2d-demo.ts +623 -0
- package/src/examples/physx-rapier-2d-demo.ts +655 -0
- package/src/examples/relative-positioning-demo.ts +323 -0
- package/src/examples/scroll-example.ts +214 -0
- package/src/examples/scrollbox-mouse-test.ts +112 -0
- package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
- package/src/examples/select-demo.ts +237 -0
- package/src/examples/shader-cube-demo.ts +1015 -0
- package/src/examples/simple-layout-example.ts +591 -0
- package/src/examples/slider-demo.ts +617 -0
- package/src/examples/split-mode-demo.ts +453 -0
- package/src/examples/sprite-animation-demo.ts +443 -0
- package/src/examples/sprite-particle-generator-demo.ts +486 -0
- package/src/examples/static-sprite-demo.ts +193 -0
- package/src/examples/sticky-scroll-example.ts +308 -0
- package/src/examples/styled-text-demo.ts +282 -0
- package/src/examples/tab-select-demo.ts +219 -0
- package/src/examples/terminal-title.ts +29 -0
- package/src/examples/terminal.ts +305 -0
- package/src/examples/text-node-demo.ts +416 -0
- package/src/examples/text-selection-demo.ts +377 -0
- package/src/examples/text-table-demo.ts +503 -0
- package/src/examples/text-truncation-demo.ts +481 -0
- package/src/examples/text-wrap.ts +757 -0
- package/src/examples/texture-loading-demo.ts +259 -0
- package/src/examples/timeline-example.ts +670 -0
- package/src/examples/transparency-demo.ts +400 -0
- package/src/examples/vnode-composition-demo.ts +404 -0
- package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
- package/src/index.ts +24 -0
- package/src/lib/KeyHandler.integration.test.ts +292 -0
- package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
- package/src/lib/KeyHandler.test.ts +662 -0
- package/src/lib/KeyHandler.ts +222 -0
- package/src/lib/RGBA.test.ts +984 -0
- package/src/lib/RGBA.ts +204 -0
- package/src/lib/ascii.font.ts +330 -0
- package/src/lib/border.test.ts +83 -0
- package/src/lib/border.ts +170 -0
- package/src/lib/bunfs.test.ts +27 -0
- package/src/lib/bunfs.ts +18 -0
- package/src/lib/clipboard.test.ts +41 -0
- package/src/lib/clipboard.ts +47 -0
- package/src/lib/clock.ts +35 -0
- package/src/lib/data-paths.test.ts +133 -0
- package/src/lib/data-paths.ts +109 -0
- package/src/lib/debounce.ts +106 -0
- package/src/lib/detect-links.test.ts +98 -0
- package/src/lib/detect-links.ts +56 -0
- package/src/lib/env.test.ts +228 -0
- package/src/lib/env.ts +209 -0
- package/src/lib/extmarks-history.ts +51 -0
- package/src/lib/extmarks-multiwidth.test.ts +322 -0
- package/src/lib/extmarks.test.ts +3457 -0
- package/src/lib/extmarks.ts +843 -0
- package/src/lib/fonts/block.json +405 -0
- package/src/lib/fonts/grid.json +265 -0
- package/src/lib/fonts/huge.json +741 -0
- package/src/lib/fonts/pallet.json +314 -0
- package/src/lib/fonts/shade.json +591 -0
- package/src/lib/fonts/slick.json +321 -0
- package/src/lib/fonts/tiny.json +69 -0
- package/src/lib/hast-styled-text.ts +59 -0
- package/src/lib/index.ts +21 -0
- package/src/lib/keymapping.test.ts +317 -0
- package/src/lib/keymapping.ts +115 -0
- package/src/lib/objects-in-viewport.test.ts +787 -0
- package/src/lib/objects-in-viewport.ts +153 -0
- package/src/lib/output.capture.ts +58 -0
- package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
- package/src/lib/parse.keypress-kitty.test.ts +663 -0
- package/src/lib/parse.keypress-kitty.ts +439 -0
- package/src/lib/parse.keypress.test.ts +1849 -0
- package/src/lib/parse.keypress.ts +397 -0
- package/src/lib/parse.mouse.test.ts +552 -0
- package/src/lib/parse.mouse.ts +232 -0
- package/src/lib/paste.ts +16 -0
- package/src/lib/queue.ts +65 -0
- package/src/lib/renderable.validations.test.ts +87 -0
- package/src/lib/renderable.validations.ts +83 -0
- package/src/lib/scroll-acceleration.ts +98 -0
- package/src/lib/selection.ts +240 -0
- package/src/lib/singleton.ts +28 -0
- package/src/lib/stdin-parser.test.ts +2290 -0
- package/src/lib/stdin-parser.ts +1810 -0
- package/src/lib/styled-text.ts +178 -0
- package/src/lib/terminal-capability-detection.test.ts +202 -0
- package/src/lib/terminal-capability-detection.ts +79 -0
- package/src/lib/terminal-palette.test.ts +878 -0
- package/src/lib/terminal-palette.ts +383 -0
- package/src/lib/tree-sitter/assets/README.md +118 -0
- package/src/lib/tree-sitter/assets/update.ts +334 -0
- package/src/lib/tree-sitter/assets.d.ts +9 -0
- package/src/lib/tree-sitter/cache.test.ts +273 -0
- package/src/lib/tree-sitter/client.test.ts +1165 -0
- package/src/lib/tree-sitter/client.ts +607 -0
- package/src/lib/tree-sitter/default-parsers.ts +86 -0
- package/src/lib/tree-sitter/download-utils.ts +148 -0
- package/src/lib/tree-sitter/index.ts +28 -0
- package/src/lib/tree-sitter/parser.worker.ts +1042 -0
- package/src/lib/tree-sitter/parsers-config.ts +81 -0
- package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
- package/src/lib/tree-sitter/resolve-ft.ts +189 -0
- package/src/lib/tree-sitter/types.ts +82 -0
- package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
- package/src/lib/tree-sitter-styled-text.ts +306 -0
- package/src/lib/validate-dir-name.ts +55 -0
- package/src/lib/yoga.options.test.ts +628 -0
- package/src/lib/yoga.options.ts +346 -0
- package/src/plugins/core-slot.ts +579 -0
- package/src/plugins/registry.ts +402 -0
- package/src/plugins/types.ts +46 -0
- package/src/post/effects.ts +930 -0
- package/src/post/filters.ts +489 -0
- package/src/post/matrices.ts +288 -0
- package/src/renderables/ASCIIFont.ts +219 -0
- package/src/renderables/Box.test.ts +205 -0
- package/src/renderables/Box.ts +326 -0
- package/src/renderables/Code.test.ts +2062 -0
- package/src/renderables/Code.ts +357 -0
- package/src/renderables/Diff.regression.test.ts +226 -0
- package/src/renderables/Diff.test.ts +3101 -0
- package/src/renderables/Diff.ts +1211 -0
- package/src/renderables/EditBufferRenderable.test.ts +288 -0
- package/src/renderables/EditBufferRenderable.ts +1166 -0
- package/src/renderables/FrameBuffer.ts +47 -0
- package/src/renderables/Input.test.ts +1228 -0
- package/src/renderables/Input.ts +247 -0
- package/src/renderables/LineNumberRenderable.ts +724 -0
- package/src/renderables/Markdown.ts +1393 -0
- package/src/renderables/ScrollBar.ts +422 -0
- package/src/renderables/ScrollBox.ts +883 -0
- package/src/renderables/Select.test.ts +1033 -0
- package/src/renderables/Select.ts +524 -0
- package/src/renderables/Slider.test.ts +456 -0
- package/src/renderables/Slider.ts +342 -0
- package/src/renderables/TabSelect.test.ts +197 -0
- package/src/renderables/TabSelect.ts +455 -0
- package/src/renderables/Text.selection-buffer.test.ts +123 -0
- package/src/renderables/Text.test.ts +2660 -0
- package/src/renderables/Text.ts +147 -0
- package/src/renderables/TextBufferRenderable.ts +518 -0
- package/src/renderables/TextNode.test.ts +1058 -0
- package/src/renderables/TextNode.ts +325 -0
- package/src/renderables/TextTable.test.ts +1421 -0
- package/src/renderables/TextTable.ts +1344 -0
- package/src/renderables/Textarea.ts +430 -0
- package/src/renderables/TimeToFirstDraw.ts +89 -0
- package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
- package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
- package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
- package/src/renderables/__tests__/Markdown.test.ts +2518 -0
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
- package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
- package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
- package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
- package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
- package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
- package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
- package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
- package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
- package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
- package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
- package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
- package/src/renderables/composition/README.md +8 -0
- package/src/renderables/composition/VRenderable.ts +32 -0
- package/src/renderables/composition/constructs.ts +127 -0
- package/src/renderables/composition/vnode.ts +289 -0
- package/src/renderables/index.ts +23 -0
- package/src/renderables/markdown-parser.ts +66 -0
- package/src/renderer.ts +2681 -0
- package/src/runtime-plugin-support.ts +39 -0
- package/src/runtime-plugin.ts +615 -0
- package/src/syntax-style.test.ts +841 -0
- package/src/syntax-style.ts +257 -0
- package/src/testing/README.md +210 -0
- package/src/testing/capture-spans.test.ts +194 -0
- package/src/testing/integration.test.ts +276 -0
- package/src/testing/manual-clock.ts +117 -0
- package/src/testing/mock-keys.test.ts +1378 -0
- package/src/testing/mock-keys.ts +457 -0
- package/src/testing/mock-mouse.test.ts +218 -0
- package/src/testing/mock-mouse.ts +247 -0
- package/src/testing/mock-tree-sitter-client.ts +73 -0
- package/src/testing/spy.ts +13 -0
- package/src/testing/test-recorder.test.ts +415 -0
- package/src/testing/test-recorder.ts +145 -0
- package/src/testing/test-renderer.ts +132 -0
- package/src/testing.ts +7 -0
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
- package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
- package/src/tests/allocator-stats.test.ts +38 -0
- package/src/tests/destroy-during-render.test.ts +200 -0
- package/src/tests/destroy-on-exit.fixture.ts +36 -0
- package/src/tests/destroy-on-exit.test.ts +41 -0
- package/src/tests/hover-cursor.test.ts +98 -0
- package/src/tests/native-span-feed-async.test.ts +173 -0
- package/src/tests/native-span-feed-close.test.ts +120 -0
- package/src/tests/native-span-feed-coverage.test.ts +227 -0
- package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
- package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
- package/src/tests/opacity.test.ts +123 -0
- package/src/tests/renderable.snapshot.test.ts +524 -0
- package/src/tests/renderable.test.ts +1281 -0
- package/src/tests/renderer.clock.test.ts +158 -0
- package/src/tests/renderer.console-startup.test.ts +185 -0
- package/src/tests/renderer.control.test.ts +425 -0
- package/src/tests/renderer.core-slot-binding.test.ts +952 -0
- package/src/tests/renderer.cursor.test.ts +26 -0
- package/src/tests/renderer.destroy-during-render.test.ts +147 -0
- package/src/tests/renderer.focus-restore.test.ts +257 -0
- package/src/tests/renderer.focus.test.ts +294 -0
- package/src/tests/renderer.idle.test.ts +219 -0
- package/src/tests/renderer.input.test.ts +2237 -0
- package/src/tests/renderer.kitty-flags.test.ts +195 -0
- package/src/tests/renderer.mouse.test.ts +1274 -0
- package/src/tests/renderer.palette.test.ts +629 -0
- package/src/tests/renderer.selection.test.ts +49 -0
- package/src/tests/renderer.slot-registry.test.ts +684 -0
- package/src/tests/renderer.useMouse.test.ts +47 -0
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
- package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
- package/src/tests/runtime-plugin-support.fixture.ts +11 -0
- package/src/tests/runtime-plugin-support.test.ts +19 -0
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
- package/src/tests/runtime-plugin.fixture.ts +40 -0
- package/src/tests/runtime-plugin.test.ts +354 -0
- package/src/tests/scrollbox-culling-bug.test.ts +114 -0
- package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
- package/src/tests/scrollbox-hitgrid.test.ts +909 -0
- package/src/tests/scrollbox.test.ts +1530 -0
- package/src/tests/wrap-resize-perf.test.ts +276 -0
- package/src/tests/yoga-setters.test.ts +921 -0
- package/src/text-buffer-view.test.ts +705 -0
- package/src/text-buffer-view.ts +189 -0
- package/src/text-buffer.test.ts +347 -0
- package/src/text-buffer.ts +250 -0
- package/src/types.ts +161 -0
- package/src/utils.ts +88 -0
- package/src/zig/ansi.zig +268 -0
- package/src/zig/bench/README.md +50 -0
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
- package/src/zig/bench/edit-buffer_bench.zig +476 -0
- package/src/zig/bench/native-span-feed_bench.zig +100 -0
- package/src/zig/bench/rope-markers_bench.zig +713 -0
- package/src/zig/bench/rope_bench.zig +514 -0
- package/src/zig/bench/styled-text_bench.zig +470 -0
- package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
- package/src/zig/bench/text-buffer-view_bench.zig +459 -0
- package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
- package/src/zig/bench/utf8_bench.zig +799 -0
- package/src/zig/bench-utils.zig +431 -0
- package/src/zig/bench.zig +217 -0
- package/src/zig/buffer-methods.zig +211 -0
- package/src/zig/buffer.zig +2281 -0
- package/src/zig/build.zig +289 -0
- package/src/zig/build.zig.zon +16 -0
- package/src/zig/edit-buffer.zig +825 -0
- package/src/zig/editor-view.zig +802 -0
- package/src/zig/event-bus.zig +13 -0
- package/src/zig/event-emitter.zig +65 -0
- package/src/zig/file-logger.zig +92 -0
- package/src/zig/grapheme.zig +599 -0
- package/src/zig/lib.zig +1854 -0
- package/src/zig/link.zig +333 -0
- package/src/zig/logger.zig +43 -0
- package/src/zig/mem-registry.zig +125 -0
- package/src/zig/native-span-feed-bench-lib.zig +7 -0
- package/src/zig/native-span-feed.zig +708 -0
- package/src/zig/renderer.zig +1393 -0
- package/src/zig/rope.zig +1220 -0
- package/src/zig/syntax-style.zig +161 -0
- package/src/zig/terminal.zig +987 -0
- package/src/zig/test.zig +72 -0
- package/src/zig/tests/README.md +18 -0
- package/src/zig/tests/buffer-methods_test.zig +1109 -0
- package/src/zig/tests/buffer_test.zig +2557 -0
- package/src/zig/tests/edit-buffer-history_test.zig +271 -0
- package/src/zig/tests/edit-buffer_test.zig +1689 -0
- package/src/zig/tests/editor-view_test.zig +3299 -0
- package/src/zig/tests/event-emitter_test.zig +249 -0
- package/src/zig/tests/grapheme_test.zig +1304 -0
- package/src/zig/tests/link_test.zig +190 -0
- package/src/zig/tests/mem-registry_test.zig +473 -0
- package/src/zig/tests/memory_leak_regression_test.zig +159 -0
- package/src/zig/tests/native-span-feed_test.zig +1264 -0
- package/src/zig/tests/renderer_test.zig +1017 -0
- package/src/zig/tests/rope-nested_test.zig +712 -0
- package/src/zig/tests/rope_fuzz_test.zig +238 -0
- package/src/zig/tests/rope_test.zig +2362 -0
- package/src/zig/tests/segment-merge.test.zig +148 -0
- package/src/zig/tests/syntax-style_test.zig +557 -0
- package/src/zig/tests/terminal_test.zig +754 -0
- package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
- package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
- package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
- package/src/zig/tests/text-buffer-segment_test.zig +320 -0
- package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
- package/src/zig/tests/text-buffer-view_test.zig +3649 -0
- package/src/zig/tests/text-buffer_test.zig +2191 -0
- package/src/zig/tests/unicode-width-map.zon +3909 -0
- package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
- package/src/zig/tests/utf8_test.zig +4057 -0
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
- package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
- package/src/zig/tests/word-wrap-editing_test.zig +498 -0
- package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
- package/src/zig/text-buffer-iterators.zig +499 -0
- package/src/zig/text-buffer-segment.zig +404 -0
- package/src/zig/text-buffer-view.zig +1371 -0
- package/src/zig/text-buffer.zig +1180 -0
- package/src/zig/utf8.zig +1948 -0
- package/src/zig/utils.zig +9 -0
- package/src/zig-structs.ts +261 -0
- package/src/zig.ts +3884 -0
- package/tsconfig.build.json +24 -0
- package/tsconfig.json +27 -0
- package/3d/SpriteResourceManager.d.ts +0 -74
- package/3d/SpriteUtils.d.ts +0 -13
- package/3d/TextureUtils.d.ts +0 -24
- package/3d/ThreeRenderable.d.ts +0 -40
- package/3d/WGPURenderer.d.ts +0 -61
- package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
- package/3d/animation/SpriteAnimator.d.ts +0 -124
- package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
- package/3d/canvas.d.ts +0 -44
- package/3d/index.d.ts +0 -12
- package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
- package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
- package/3d/physics/physics-interface.d.ts +0 -27
- package/3d.d.ts +0 -2
- package/3d.js +0 -34041
- package/3d.js.map +0 -155
- package/LICENSE +0 -21
- package/NativeSpanFeed.d.ts +0 -41
- package/Renderable.d.ts +0 -334
- package/animation/Timeline.d.ts +0 -126
- package/ansi.d.ts +0 -13
- package/buffer.d.ts +0 -111
- package/console.d.ts +0 -144
- package/edit-buffer.d.ts +0 -98
- package/editor-view.d.ts +0 -73
- package/index-9vwc3fg6.js +0 -12260
- package/index-9vwc3fg6.js.map +0 -42
- package/index-dcj62y8t.js +0 -20614
- package/index-dcj62y8t.js.map +0 -67
- package/index-f7n39gpy.js +0 -411
- package/index-f7n39gpy.js.map +0 -10
- package/index.d.ts +0 -23
- package/index.js +0 -478
- package/index.js.map +0 -9
- package/lib/KeyHandler.d.ts +0 -61
- package/lib/RGBA.d.ts +0 -25
- package/lib/ascii.font.d.ts +0 -508
- package/lib/border.d.ts +0 -51
- package/lib/bunfs.d.ts +0 -7
- package/lib/clipboard.d.ts +0 -17
- package/lib/clock.d.ts +0 -15
- package/lib/data-paths.d.ts +0 -26
- package/lib/debounce.d.ts +0 -42
- package/lib/detect-links.d.ts +0 -6
- package/lib/env.d.ts +0 -42
- package/lib/extmarks-history.d.ts +0 -17
- package/lib/extmarks.d.ts +0 -89
- package/lib/hast-styled-text.d.ts +0 -17
- package/lib/index.d.ts +0 -21
- package/lib/keymapping.d.ts +0 -25
- package/lib/objects-in-viewport.d.ts +0 -24
- package/lib/output.capture.d.ts +0 -24
- package/lib/parse.keypress-kitty.d.ts +0 -2
- package/lib/parse.keypress.d.ts +0 -26
- package/lib/parse.mouse.d.ts +0 -30
- package/lib/paste.d.ts +0 -7
- package/lib/queue.d.ts +0 -15
- package/lib/renderable.validations.d.ts +0 -12
- package/lib/scroll-acceleration.d.ts +0 -43
- package/lib/selection.d.ts +0 -63
- package/lib/singleton.d.ts +0 -7
- package/lib/stdin-parser.d.ts +0 -87
- package/lib/styled-text.d.ts +0 -63
- package/lib/terminal-capability-detection.d.ts +0 -30
- package/lib/terminal-palette.d.ts +0 -50
- package/lib/tree-sitter/assets/update.d.ts +0 -11
- package/lib/tree-sitter/client.d.ts +0 -47
- package/lib/tree-sitter/default-parsers.d.ts +0 -2
- package/lib/tree-sitter/download-utils.d.ts +0 -21
- package/lib/tree-sitter/index.d.ts +0 -8
- package/lib/tree-sitter/parser.worker.d.ts +0 -1
- package/lib/tree-sitter/parsers-config.d.ts +0 -53
- package/lib/tree-sitter/resolve-ft.d.ts +0 -5
- package/lib/tree-sitter/types.d.ts +0 -82
- package/lib/tree-sitter-styled-text.d.ts +0 -14
- package/lib/validate-dir-name.d.ts +0 -1
- package/lib/yoga.options.d.ts +0 -32
- package/parser.worker.js +0 -899
- package/parser.worker.js.map +0 -12
- package/plugins/core-slot.d.ts +0 -72
- package/plugins/registry.d.ts +0 -42
- package/plugins/types.d.ts +0 -34
- package/post/effects.d.ts +0 -147
- package/post/filters.d.ts +0 -65
- package/post/matrices.d.ts +0 -20
- package/renderables/ASCIIFont.d.ts +0 -52
- package/renderables/Box.d.ts +0 -81
- package/renderables/Code.d.ts +0 -78
- package/renderables/Diff.d.ts +0 -142
- package/renderables/EditBufferRenderable.d.ts +0 -237
- package/renderables/FrameBuffer.d.ts +0 -16
- package/renderables/Input.d.ts +0 -67
- package/renderables/LineNumberRenderable.d.ts +0 -78
- package/renderables/Markdown.d.ts +0 -185
- package/renderables/ScrollBar.d.ts +0 -77
- package/renderables/ScrollBox.d.ts +0 -124
- package/renderables/Select.d.ts +0 -115
- package/renderables/Slider.d.ts +0 -47
- package/renderables/TabSelect.d.ts +0 -96
- package/renderables/Text.d.ts +0 -36
- package/renderables/TextBufferRenderable.d.ts +0 -105
- package/renderables/TextNode.d.ts +0 -91
- package/renderables/TextTable.d.ts +0 -140
- package/renderables/Textarea.d.ts +0 -63
- package/renderables/TimeToFirstDraw.d.ts +0 -24
- package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
- package/renderables/composition/VRenderable.d.ts +0 -16
- package/renderables/composition/constructs.d.ts +0 -35
- package/renderables/composition/vnode.d.ts +0 -46
- package/renderables/index.d.ts +0 -23
- package/renderables/markdown-parser.d.ts +0 -10
- package/renderer.d.ts +0 -419
- package/runtime-plugin-support.d.ts +0 -3
- package/runtime-plugin-support.js +0 -29
- package/runtime-plugin-support.js.map +0 -10
- package/runtime-plugin.d.ts +0 -16
- package/runtime-plugin.js +0 -16
- package/runtime-plugin.js.map +0 -9
- package/syntax-style.d.ts +0 -54
- package/testing/manual-clock.d.ts +0 -17
- package/testing/mock-keys.d.ts +0 -81
- package/testing/mock-mouse.d.ts +0 -38
- package/testing/mock-tree-sitter-client.d.ts +0 -23
- package/testing/spy.d.ts +0 -7
- package/testing/test-recorder.d.ts +0 -61
- package/testing/test-renderer.d.ts +0 -23
- package/testing.d.ts +0 -6
- package/testing.js +0 -697
- package/testing.js.map +0 -15
- package/text-buffer-view.d.ts +0 -42
- package/text-buffer.d.ts +0 -67
- package/types.d.ts +0 -139
- package/utils.d.ts +0 -14
- package/zig-structs.d.ts +0 -155
- package/zig.d.ts +0 -353
- /package/{assets β src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
- /package/{assets β src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -0,0 +1,1769 @@
|
|
|
1
|
+
import { describe, expect, it, beforeEach, afterEach } from "bun:test"
|
|
2
|
+
import { EditBuffer } from "./edit-buffer.js"
|
|
3
|
+
|
|
4
|
+
describe("EditBuffer", () => {
|
|
5
|
+
let buffer: EditBuffer
|
|
6
|
+
|
|
7
|
+
beforeEach(() => {
|
|
8
|
+
buffer = EditBuffer.create("wcwidth")
|
|
9
|
+
})
|
|
10
|
+
|
|
11
|
+
afterEach(() => {
|
|
12
|
+
buffer.destroy()
|
|
13
|
+
})
|
|
14
|
+
|
|
15
|
+
describe("setText and getText", () => {
|
|
16
|
+
it("should set and retrieve text content", () => {
|
|
17
|
+
buffer.setText("Hello World")
|
|
18
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
it("should handle empty text", () => {
|
|
22
|
+
buffer.setText("")
|
|
23
|
+
expect(buffer.getText()).toBe("")
|
|
24
|
+
})
|
|
25
|
+
|
|
26
|
+
it("should handle text with newlines", () => {
|
|
27
|
+
const text = "Line 1\nLine 2\nLine 3"
|
|
28
|
+
buffer.setText(text)
|
|
29
|
+
expect(buffer.getText()).toBe(text)
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
it("should handle Unicode characters", () => {
|
|
33
|
+
const text = "Hello δΈη π"
|
|
34
|
+
buffer.setText(text)
|
|
35
|
+
expect(buffer.getText()).toBe(text)
|
|
36
|
+
})
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
describe("cursor position", () => {
|
|
40
|
+
it("should start cursor at beginning after setText", () => {
|
|
41
|
+
buffer.setText("Hello World")
|
|
42
|
+
const cursor = buffer.getCursorPosition()
|
|
43
|
+
|
|
44
|
+
expect(cursor.row).toBe(0)
|
|
45
|
+
expect(cursor.col).toBe(0)
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
it("should track cursor position after movements", () => {
|
|
49
|
+
buffer.setText("Hello World")
|
|
50
|
+
|
|
51
|
+
buffer.moveCursorRight()
|
|
52
|
+
let cursor = buffer.getCursorPosition()
|
|
53
|
+
expect(cursor.col).toBe(1)
|
|
54
|
+
|
|
55
|
+
buffer.moveCursorRight()
|
|
56
|
+
cursor = buffer.getCursorPosition()
|
|
57
|
+
expect(cursor.col).toBe(2)
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
it("should handle multi-line cursor positions", () => {
|
|
61
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
62
|
+
|
|
63
|
+
buffer.moveCursorDown()
|
|
64
|
+
let cursor = buffer.getCursorPosition()
|
|
65
|
+
expect(cursor.row).toBe(1)
|
|
66
|
+
|
|
67
|
+
buffer.moveCursorDown()
|
|
68
|
+
cursor = buffer.getCursorPosition()
|
|
69
|
+
expect(cursor.row).toBe(2)
|
|
70
|
+
})
|
|
71
|
+
})
|
|
72
|
+
|
|
73
|
+
describe("cursor movement", () => {
|
|
74
|
+
it("should move cursor left and right", () => {
|
|
75
|
+
buffer.setText("ABCDE")
|
|
76
|
+
|
|
77
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
78
|
+
expect(buffer.getCursorPosition().col).toBe(5)
|
|
79
|
+
|
|
80
|
+
buffer.moveCursorLeft()
|
|
81
|
+
expect(buffer.getCursorPosition().col).toBe(4)
|
|
82
|
+
|
|
83
|
+
buffer.moveCursorLeft()
|
|
84
|
+
expect(buffer.getCursorPosition().col).toBe(3)
|
|
85
|
+
})
|
|
86
|
+
|
|
87
|
+
it("should move cursor up and down", () => {
|
|
88
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
89
|
+
|
|
90
|
+
buffer.moveCursorDown()
|
|
91
|
+
expect(buffer.getCursorPosition().row).toBe(1)
|
|
92
|
+
|
|
93
|
+
buffer.moveCursorDown()
|
|
94
|
+
expect(buffer.getCursorPosition().row).toBe(2)
|
|
95
|
+
|
|
96
|
+
buffer.moveCursorUp()
|
|
97
|
+
expect(buffer.getCursorPosition().row).toBe(1)
|
|
98
|
+
})
|
|
99
|
+
|
|
100
|
+
it("should move to line start and end", () => {
|
|
101
|
+
buffer.setText("Hello World")
|
|
102
|
+
|
|
103
|
+
buffer.setCursorToLineCol(0, 11) // Move to end
|
|
104
|
+
expect(buffer.getCursorPosition().col).toBe(11)
|
|
105
|
+
|
|
106
|
+
const cursor = buffer.getCursorPosition()
|
|
107
|
+
buffer.setCursor(cursor.row, 0)
|
|
108
|
+
expect(buffer.getCursorPosition().col).toBe(0)
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
it("should goto specific line", () => {
|
|
112
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
113
|
+
|
|
114
|
+
buffer.gotoLine(1)
|
|
115
|
+
expect(buffer.getCursorPosition().row).toBe(1)
|
|
116
|
+
|
|
117
|
+
buffer.gotoLine(2)
|
|
118
|
+
expect(buffer.getCursorPosition().row).toBe(2)
|
|
119
|
+
})
|
|
120
|
+
|
|
121
|
+
it("should handle Unicode grapheme movement correctly", () => {
|
|
122
|
+
buffer.setText("AπB")
|
|
123
|
+
|
|
124
|
+
expect(buffer.getCursorPosition().col).toBe(0)
|
|
125
|
+
|
|
126
|
+
buffer.moveCursorRight() // Move to emoji
|
|
127
|
+
expect(buffer.getCursorPosition().col).toBe(1)
|
|
128
|
+
|
|
129
|
+
buffer.moveCursorRight() // Move past emoji (2 cells wide)
|
|
130
|
+
expect(buffer.getCursorPosition().col).toBe(3)
|
|
131
|
+
|
|
132
|
+
buffer.moveCursorRight() // Move to B
|
|
133
|
+
expect(buffer.getCursorPosition().col).toBe(4)
|
|
134
|
+
})
|
|
135
|
+
})
|
|
136
|
+
|
|
137
|
+
describe("text insertion", () => {
|
|
138
|
+
it("should insert single character", () => {
|
|
139
|
+
buffer.setText("Hello World")
|
|
140
|
+
|
|
141
|
+
buffer.setCursorToLineCol(0, 11) // Move to end
|
|
142
|
+
buffer.insertChar("!")
|
|
143
|
+
|
|
144
|
+
expect(buffer.getText()).toBe("Hello World!")
|
|
145
|
+
})
|
|
146
|
+
|
|
147
|
+
it("should insert text at cursor", () => {
|
|
148
|
+
buffer.setText("Hello")
|
|
149
|
+
|
|
150
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
151
|
+
buffer.insertText(" World")
|
|
152
|
+
|
|
153
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
154
|
+
})
|
|
155
|
+
|
|
156
|
+
it("should insert text in middle", () => {
|
|
157
|
+
buffer.setText("HelloWorld")
|
|
158
|
+
|
|
159
|
+
buffer.setCursorToLineCol(0, 5)
|
|
160
|
+
buffer.insertText(" ")
|
|
161
|
+
|
|
162
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
163
|
+
})
|
|
164
|
+
|
|
165
|
+
it("should handle continuous typing (edit session)", () => {
|
|
166
|
+
buffer.setText("")
|
|
167
|
+
|
|
168
|
+
buffer.insertText("Hello")
|
|
169
|
+
buffer.insertText(" ")
|
|
170
|
+
buffer.insertText("World")
|
|
171
|
+
|
|
172
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
173
|
+
})
|
|
174
|
+
|
|
175
|
+
it("should insert Unicode characters", () => {
|
|
176
|
+
buffer.setText("Hello")
|
|
177
|
+
|
|
178
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
179
|
+
buffer.insertText(" δΈη π")
|
|
180
|
+
|
|
181
|
+
expect(buffer.getText()).toBe("Hello δΈη π")
|
|
182
|
+
})
|
|
183
|
+
|
|
184
|
+
it("should handle newline insertion", () => {
|
|
185
|
+
buffer.setText("HelloWorld")
|
|
186
|
+
|
|
187
|
+
buffer.setCursorToLineCol(0, 5)
|
|
188
|
+
buffer.newLine()
|
|
189
|
+
|
|
190
|
+
expect(buffer.getText()).toBe("Hello\nWorld")
|
|
191
|
+
})
|
|
192
|
+
})
|
|
193
|
+
|
|
194
|
+
describe("text deletion", () => {
|
|
195
|
+
it("should delete character at cursor", () => {
|
|
196
|
+
buffer.setText("Hello World")
|
|
197
|
+
|
|
198
|
+
buffer.setCursorToLineCol(0, 6)
|
|
199
|
+
buffer.deleteChar()
|
|
200
|
+
|
|
201
|
+
expect(buffer.getText()).toBe("Hello orld")
|
|
202
|
+
})
|
|
203
|
+
|
|
204
|
+
it("should delete character backward", () => {
|
|
205
|
+
buffer.setText("")
|
|
206
|
+
|
|
207
|
+
buffer.insertText("test")
|
|
208
|
+
buffer.deleteCharBackward()
|
|
209
|
+
|
|
210
|
+
expect(buffer.getText()).toBe("tes")
|
|
211
|
+
})
|
|
212
|
+
|
|
213
|
+
it("should delete range within a single line", () => {
|
|
214
|
+
buffer.setText("Hello World")
|
|
215
|
+
|
|
216
|
+
buffer.deleteRange(0, 0, 0, 5)
|
|
217
|
+
|
|
218
|
+
expect(buffer.getText()).toBe(" World")
|
|
219
|
+
})
|
|
220
|
+
|
|
221
|
+
it("should delete range across multiple lines", () => {
|
|
222
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
223
|
+
|
|
224
|
+
buffer.deleteRange(0, 5, 2, 5)
|
|
225
|
+
|
|
226
|
+
expect(buffer.getText()).toBe("Line 3")
|
|
227
|
+
})
|
|
228
|
+
|
|
229
|
+
it("should handle deleteRange with start equal to end (no-op)", () => {
|
|
230
|
+
buffer.setText("Hello World")
|
|
231
|
+
|
|
232
|
+
buffer.deleteRange(0, 5, 0, 5)
|
|
233
|
+
|
|
234
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
235
|
+
})
|
|
236
|
+
|
|
237
|
+
it("should handle deleteRange with reversed start and end", () => {
|
|
238
|
+
buffer.setText("Hello World")
|
|
239
|
+
|
|
240
|
+
buffer.deleteRange(0, 10, 0, 5)
|
|
241
|
+
|
|
242
|
+
expect(buffer.getText()).toBe("Hellod")
|
|
243
|
+
})
|
|
244
|
+
|
|
245
|
+
it("should delete from middle of one line to middle of another", () => {
|
|
246
|
+
buffer.setText("AAAA\nBBBB\nCCCC")
|
|
247
|
+
|
|
248
|
+
buffer.deleteRange(0, 2, 2, 2)
|
|
249
|
+
|
|
250
|
+
expect(buffer.getText()).toBe("AACC")
|
|
251
|
+
})
|
|
252
|
+
|
|
253
|
+
it("should delete entire content with deleteRange", () => {
|
|
254
|
+
buffer.setText("Hello World")
|
|
255
|
+
|
|
256
|
+
buffer.deleteRange(0, 0, 0, 11)
|
|
257
|
+
|
|
258
|
+
expect(buffer.getText()).toBe("")
|
|
259
|
+
})
|
|
260
|
+
|
|
261
|
+
it("should handle deleteRange with Unicode characters", () => {
|
|
262
|
+
buffer.setText("Hello δΈη π")
|
|
263
|
+
|
|
264
|
+
buffer.deleteRange(0, 6, 0, 10)
|
|
265
|
+
|
|
266
|
+
expect(buffer.getText()).toBe("Hello π")
|
|
267
|
+
})
|
|
268
|
+
|
|
269
|
+
it("should delete entire line", () => {
|
|
270
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
271
|
+
|
|
272
|
+
buffer.gotoLine(1) // Go to Line 2
|
|
273
|
+
buffer.deleteLine()
|
|
274
|
+
|
|
275
|
+
expect(buffer.getText()).toBe("Line 1\nLine 3")
|
|
276
|
+
})
|
|
277
|
+
|
|
278
|
+
// TODO: Re-implement deleteToLineEnd as scripted method
|
|
279
|
+
it.skip("should delete to line end", () => {
|
|
280
|
+
buffer.setText("Hello World")
|
|
281
|
+
|
|
282
|
+
buffer.setCursorToLineCol(0, 6)
|
|
283
|
+
// buffer.deleteToLineEnd()
|
|
284
|
+
|
|
285
|
+
expect(buffer.getText()).toBe("Hello ")
|
|
286
|
+
})
|
|
287
|
+
|
|
288
|
+
it("should handle backspace in active edit session", () => {
|
|
289
|
+
buffer.setText("")
|
|
290
|
+
|
|
291
|
+
buffer.insertText("test")
|
|
292
|
+
buffer.deleteCharBackward()
|
|
293
|
+
buffer.deleteCharBackward()
|
|
294
|
+
|
|
295
|
+
expect(buffer.getText()).toBe("te")
|
|
296
|
+
})
|
|
297
|
+
})
|
|
298
|
+
|
|
299
|
+
describe("complex editing scenarios", () => {
|
|
300
|
+
it("should handle multiple edit operations in sequence", () => {
|
|
301
|
+
buffer.setText("Hello World")
|
|
302
|
+
|
|
303
|
+
buffer.setCursorToLineCol(0, 11) // Move to end
|
|
304
|
+
buffer.insertText("!")
|
|
305
|
+
|
|
306
|
+
buffer.setCursorToLineCol(0, 0) // Move to start
|
|
307
|
+
buffer.insertText(">> ")
|
|
308
|
+
|
|
309
|
+
buffer.setCursorToLineCol(0, 99) // Move to end of line
|
|
310
|
+
buffer.newLine()
|
|
311
|
+
buffer.insertText("New line")
|
|
312
|
+
|
|
313
|
+
expect(buffer.getText()).toBe(">> Hello World!\nNew line")
|
|
314
|
+
})
|
|
315
|
+
|
|
316
|
+
it("should handle insert, delete, and cursor movement", () => {
|
|
317
|
+
buffer.setText("AAAA\nBBBB\nCCCC")
|
|
318
|
+
|
|
319
|
+
buffer.gotoLine(1)
|
|
320
|
+
buffer.setCursorToLineCol(1, 4) // Move to end of line 1
|
|
321
|
+
buffer.insertText("X")
|
|
322
|
+
|
|
323
|
+
const text1 = buffer.getText()
|
|
324
|
+
expect(text1).toBe("AAAA\nBBBBX\nCCCC")
|
|
325
|
+
|
|
326
|
+
// After insert, cursor is at end, deleteCharBackward will delete X
|
|
327
|
+
buffer.deleteCharBackward()
|
|
328
|
+
|
|
329
|
+
expect(buffer.getText()).toBe("AAAA\nBBBB\nCCCC")
|
|
330
|
+
})
|
|
331
|
+
|
|
332
|
+
it("should handle line operations", () => {
|
|
333
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
334
|
+
|
|
335
|
+
buffer.gotoLine(1) // Go to Line 2
|
|
336
|
+
buffer.deleteLine()
|
|
337
|
+
|
|
338
|
+
// After deleting Line 2, we should have Line 1 and Line 3
|
|
339
|
+
const result = buffer.getText()
|
|
340
|
+
expect(result === "Line 1\nLine 3" || result === "Line 1\nLine 3\n").toBe(true)
|
|
341
|
+
})
|
|
342
|
+
})
|
|
343
|
+
|
|
344
|
+
describe("setCursor methods", () => {
|
|
345
|
+
it("should set cursor by line and byte offset", () => {
|
|
346
|
+
buffer.setText("Hello World")
|
|
347
|
+
|
|
348
|
+
buffer.setCursor(0, 6)
|
|
349
|
+
const cursor = buffer.getCursorPosition()
|
|
350
|
+
expect(cursor.col).toBe(6)
|
|
351
|
+
})
|
|
352
|
+
|
|
353
|
+
it("should set cursor by line and column", () => {
|
|
354
|
+
buffer.setText("Hello World")
|
|
355
|
+
|
|
356
|
+
buffer.setCursorToLineCol(0, 5)
|
|
357
|
+
const cursor = buffer.getCursorPosition()
|
|
358
|
+
expect(cursor.col).toBe(5)
|
|
359
|
+
})
|
|
360
|
+
|
|
361
|
+
it("should handle multi-line setCursorToLineCol", () => {
|
|
362
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
363
|
+
|
|
364
|
+
buffer.setCursorToLineCol(1, 3)
|
|
365
|
+
const cursor = buffer.getCursorPosition()
|
|
366
|
+
expect(cursor.row).toBe(1)
|
|
367
|
+
expect(cursor.col).toBe(3)
|
|
368
|
+
})
|
|
369
|
+
})
|
|
370
|
+
|
|
371
|
+
describe("word boundary navigation", () => {
|
|
372
|
+
it("should get next word boundary", () => {
|
|
373
|
+
buffer.setText("hello world foo")
|
|
374
|
+
buffer.setCursorToLineCol(0, 0)
|
|
375
|
+
|
|
376
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
377
|
+
expect(nextBoundary.col).toBeGreaterThan(0)
|
|
378
|
+
})
|
|
379
|
+
|
|
380
|
+
it("should get previous word boundary", () => {
|
|
381
|
+
buffer.setText("hello world foo")
|
|
382
|
+
buffer.setCursorToLineCol(0, 15)
|
|
383
|
+
|
|
384
|
+
const prevBoundary = buffer.getPrevWordBoundary()
|
|
385
|
+
expect(prevBoundary.col).toBeLessThan(15)
|
|
386
|
+
})
|
|
387
|
+
|
|
388
|
+
it("should handle word boundary at start", () => {
|
|
389
|
+
buffer.setText("hello world")
|
|
390
|
+
buffer.setCursorToLineCol(0, 0)
|
|
391
|
+
|
|
392
|
+
const prevBoundary = buffer.getPrevWordBoundary()
|
|
393
|
+
expect(prevBoundary.row).toBe(0)
|
|
394
|
+
expect(prevBoundary.col).toBe(0)
|
|
395
|
+
})
|
|
396
|
+
|
|
397
|
+
it("should handle word boundary at end", () => {
|
|
398
|
+
buffer.setText("hello world")
|
|
399
|
+
buffer.setCursorToLineCol(0, 11)
|
|
400
|
+
|
|
401
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
402
|
+
expect(nextBoundary.col).toBe(11)
|
|
403
|
+
})
|
|
404
|
+
|
|
405
|
+
it("should navigate across lines", () => {
|
|
406
|
+
buffer.setText("hello\nworld")
|
|
407
|
+
buffer.setCursorToLineCol(0, 5)
|
|
408
|
+
|
|
409
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
410
|
+
expect(nextBoundary.row).toBeGreaterThanOrEqual(0)
|
|
411
|
+
})
|
|
412
|
+
|
|
413
|
+
it("should handle punctuation boundaries", () => {
|
|
414
|
+
buffer.setText("hello-world test")
|
|
415
|
+
buffer.setCursorToLineCol(0, 0)
|
|
416
|
+
|
|
417
|
+
const next1 = buffer.getNextWordBoundary()
|
|
418
|
+
expect(next1.col).toBeGreaterThan(0)
|
|
419
|
+
})
|
|
420
|
+
|
|
421
|
+
it("should handle word boundaries after CJK graphemes", () => {
|
|
422
|
+
// "δ½ " = 2 cols, " " = 1 col, "ε₯½" = 2 cols
|
|
423
|
+
buffer.setText("δ½ ε₯½")
|
|
424
|
+
buffer.setCursorToLineCol(0, 0)
|
|
425
|
+
|
|
426
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
427
|
+
expect(nextBoundary.col).toBe(3)
|
|
428
|
+
|
|
429
|
+
buffer.setCursorToLineCol(0, 5)
|
|
430
|
+
const prevBoundary = buffer.getPrevWordBoundary()
|
|
431
|
+
expect(prevBoundary.col).toBe(3)
|
|
432
|
+
})
|
|
433
|
+
|
|
434
|
+
it("should handle word boundaries after emoji", () => {
|
|
435
|
+
// "π" = 2 cols, " " = 1 col, "ok" = 2 cols
|
|
436
|
+
buffer.setText("π ok")
|
|
437
|
+
buffer.setCursorToLineCol(0, 0)
|
|
438
|
+
|
|
439
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
440
|
+
expect(nextBoundary.col).toBe(3)
|
|
441
|
+
|
|
442
|
+
buffer.setCursorToLineCol(0, 5)
|
|
443
|
+
const prevBoundary = buffer.getPrevWordBoundary()
|
|
444
|
+
expect(prevBoundary.col).toBe(3)
|
|
445
|
+
})
|
|
446
|
+
|
|
447
|
+
it("should handle word boundaries around tabs", () => {
|
|
448
|
+
// tab = 2 cols
|
|
449
|
+
buffer.setText("Hello\tWorld")
|
|
450
|
+
buffer.setCursorToLineCol(0, 0)
|
|
451
|
+
|
|
452
|
+
const nextBoundary = buffer.getNextWordBoundary()
|
|
453
|
+
expect(nextBoundary.col).toBe(7)
|
|
454
|
+
|
|
455
|
+
buffer.setCursorToLineCol(0, 12)
|
|
456
|
+
const prevBoundary = buffer.getPrevWordBoundary()
|
|
457
|
+
expect(prevBoundary.col).toBe(7)
|
|
458
|
+
})
|
|
459
|
+
})
|
|
460
|
+
|
|
461
|
+
describe("native coordinate conversion methods", () => {
|
|
462
|
+
it("should convert offset to position", () => {
|
|
463
|
+
buffer.setText("Hello\nWorld")
|
|
464
|
+
|
|
465
|
+
const pos0 = buffer.offsetToPosition(0)
|
|
466
|
+
expect(pos0).toEqual({ row: 0, col: 0 })
|
|
467
|
+
|
|
468
|
+
const pos5 = buffer.offsetToPosition(5)
|
|
469
|
+
expect(pos5).toEqual({ row: 0, col: 5 })
|
|
470
|
+
|
|
471
|
+
const pos6 = buffer.offsetToPosition(6)
|
|
472
|
+
expect(pos6).toEqual({ row: 1, col: 0 })
|
|
473
|
+
|
|
474
|
+
const pos11 = buffer.offsetToPosition(11)
|
|
475
|
+
expect(pos11).toEqual({ row: 1, col: 5 })
|
|
476
|
+
})
|
|
477
|
+
|
|
478
|
+
it("should convert position to offset", () => {
|
|
479
|
+
buffer.setText("Hello\nWorld")
|
|
480
|
+
|
|
481
|
+
expect(buffer.positionToOffset(0, 0)).toBe(0)
|
|
482
|
+
expect(buffer.positionToOffset(0, 5)).toBe(5)
|
|
483
|
+
expect(buffer.positionToOffset(1, 0)).toBe(6)
|
|
484
|
+
expect(buffer.positionToOffset(1, 5)).toBe(11)
|
|
485
|
+
})
|
|
486
|
+
|
|
487
|
+
it("should get line start offset", () => {
|
|
488
|
+
buffer.setText("Line1\nLine2\nLine3")
|
|
489
|
+
|
|
490
|
+
expect(buffer.getLineStartOffset(0)).toBe(0)
|
|
491
|
+
expect(buffer.getLineStartOffset(1)).toBe(6)
|
|
492
|
+
expect(buffer.getLineStartOffset(2)).toBe(12)
|
|
493
|
+
})
|
|
494
|
+
|
|
495
|
+
it("should handle multiline text with varying lengths", () => {
|
|
496
|
+
buffer.setText("AAA\nBB\nCCCC")
|
|
497
|
+
|
|
498
|
+
expect(buffer.offsetToPosition(0)).toEqual({ row: 0, col: 0 })
|
|
499
|
+
expect(buffer.offsetToPosition(3)).toEqual({ row: 0, col: 3 })
|
|
500
|
+
expect(buffer.offsetToPosition(4)).toEqual({ row: 1, col: 0 })
|
|
501
|
+
expect(buffer.offsetToPosition(6)).toEqual({ row: 1, col: 2 })
|
|
502
|
+
expect(buffer.offsetToPosition(7)).toEqual({ row: 2, col: 0 })
|
|
503
|
+
|
|
504
|
+
expect(buffer.positionToOffset(0, 0)).toBe(0)
|
|
505
|
+
expect(buffer.positionToOffset(1, 0)).toBe(4)
|
|
506
|
+
expect(buffer.positionToOffset(2, 0)).toBe(7)
|
|
507
|
+
})
|
|
508
|
+
|
|
509
|
+
it("should return null for invalid offset", () => {
|
|
510
|
+
buffer.setText("Hello")
|
|
511
|
+
const result = buffer.offsetToPosition(1000)
|
|
512
|
+
expect(result).toBeNull()
|
|
513
|
+
})
|
|
514
|
+
|
|
515
|
+
it("should handle empty text", () => {
|
|
516
|
+
buffer.setText("")
|
|
517
|
+
|
|
518
|
+
const pos = buffer.offsetToPosition(0)
|
|
519
|
+
expect(pos).toEqual({ row: 0, col: 0 })
|
|
520
|
+
|
|
521
|
+
expect(buffer.positionToOffset(0, 0)).toBe(0)
|
|
522
|
+
expect(buffer.getLineStartOffset(0)).toBe(0)
|
|
523
|
+
})
|
|
524
|
+
})
|
|
525
|
+
|
|
526
|
+
describe("getEOL navigation", () => {
|
|
527
|
+
it("should get end of line from start", () => {
|
|
528
|
+
buffer.setText("Hello World")
|
|
529
|
+
buffer.setCursorToLineCol(0, 0)
|
|
530
|
+
|
|
531
|
+
const eol = buffer.getEOL()
|
|
532
|
+
expect(eol.row).toBe(0)
|
|
533
|
+
expect(eol.col).toBe(11)
|
|
534
|
+
})
|
|
535
|
+
|
|
536
|
+
it("should get end of line from middle", () => {
|
|
537
|
+
buffer.setText("Hello World")
|
|
538
|
+
buffer.setCursorToLineCol(0, 5)
|
|
539
|
+
|
|
540
|
+
const eol = buffer.getEOL()
|
|
541
|
+
expect(eol.row).toBe(0)
|
|
542
|
+
expect(eol.col).toBe(11)
|
|
543
|
+
})
|
|
544
|
+
|
|
545
|
+
it("should stay at end of line when already there", () => {
|
|
546
|
+
buffer.setText("Hello")
|
|
547
|
+
buffer.setCursorToLineCol(0, 5)
|
|
548
|
+
|
|
549
|
+
const eol = buffer.getEOL()
|
|
550
|
+
expect(eol.row).toBe(0)
|
|
551
|
+
expect(eol.col).toBe(5)
|
|
552
|
+
})
|
|
553
|
+
|
|
554
|
+
it("should handle multi-line text", () => {
|
|
555
|
+
buffer.setText("Hello\nWorld\nTest")
|
|
556
|
+
buffer.setCursorToLineCol(1, 0)
|
|
557
|
+
|
|
558
|
+
const eol = buffer.getEOL()
|
|
559
|
+
expect(eol.row).toBe(1)
|
|
560
|
+
expect(eol.col).toBe(5)
|
|
561
|
+
})
|
|
562
|
+
|
|
563
|
+
it("should handle empty lines", () => {
|
|
564
|
+
buffer.setText("Hello\n\nWorld")
|
|
565
|
+
buffer.setCursorToLineCol(1, 0)
|
|
566
|
+
|
|
567
|
+
const eol = buffer.getEOL()
|
|
568
|
+
expect(eol.row).toBe(1)
|
|
569
|
+
expect(eol.col).toBe(0)
|
|
570
|
+
})
|
|
571
|
+
|
|
572
|
+
it("should work on different lines", () => {
|
|
573
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
574
|
+
|
|
575
|
+
buffer.setCursorToLineCol(0, 0)
|
|
576
|
+
const eol0 = buffer.getEOL()
|
|
577
|
+
expect(eol0.row).toBe(0)
|
|
578
|
+
expect(eol0.col).toBe(6)
|
|
579
|
+
|
|
580
|
+
buffer.setCursorToLineCol(1, 0)
|
|
581
|
+
const eol1 = buffer.getEOL()
|
|
582
|
+
expect(eol1.row).toBe(1)
|
|
583
|
+
expect(eol1.col).toBe(6)
|
|
584
|
+
|
|
585
|
+
buffer.setCursorToLineCol(2, 0)
|
|
586
|
+
const eol2 = buffer.getEOL()
|
|
587
|
+
expect(eol2.row).toBe(2)
|
|
588
|
+
expect(eol2.col).toBe(6)
|
|
589
|
+
})
|
|
590
|
+
})
|
|
591
|
+
|
|
592
|
+
describe("error handling", () => {
|
|
593
|
+
it("should throw error when using destroyed buffer", () => {
|
|
594
|
+
buffer.setText("Test")
|
|
595
|
+
buffer.destroy()
|
|
596
|
+
|
|
597
|
+
expect(() => buffer.getText()).toThrow("EditBuffer is destroyed")
|
|
598
|
+
expect(() => buffer.insertText("x")).toThrow("EditBuffer is destroyed")
|
|
599
|
+
expect(() => buffer.moveCursorLeft()).toThrow("EditBuffer is destroyed")
|
|
600
|
+
})
|
|
601
|
+
})
|
|
602
|
+
|
|
603
|
+
describe("line boundary operations", () => {
|
|
604
|
+
it("should merge lines when backspacing at BOL", () => {
|
|
605
|
+
buffer.setText("Line 1\nLine 2")
|
|
606
|
+
buffer.setCursorToLineCol(1, 0) // Start of line 2
|
|
607
|
+
buffer.deleteCharBackward()
|
|
608
|
+
expect(buffer.getText()).toBe("Line 1Line 2")
|
|
609
|
+
const cursor = buffer.getCursorPosition()
|
|
610
|
+
expect(cursor.row).toBe(0)
|
|
611
|
+
expect(cursor.col).toBe(6)
|
|
612
|
+
})
|
|
613
|
+
|
|
614
|
+
it("should merge lines when deleting at EOL", () => {
|
|
615
|
+
buffer.setText("Line 1\nLine 2")
|
|
616
|
+
buffer.setCursorToLineCol(0, 6) // End of line 1
|
|
617
|
+
buffer.deleteChar()
|
|
618
|
+
expect(buffer.getText()).toBe("Line 1Line 2")
|
|
619
|
+
const cursor = buffer.getCursorPosition()
|
|
620
|
+
expect(cursor.row).toBe(0)
|
|
621
|
+
expect(cursor.col).toBe(6)
|
|
622
|
+
})
|
|
623
|
+
|
|
624
|
+
it("should handle newline insertion at BOL", () => {
|
|
625
|
+
buffer.setText("Hello")
|
|
626
|
+
buffer.setCursorToLineCol(0, 0)
|
|
627
|
+
buffer.newLine()
|
|
628
|
+
expect(buffer.getText()).toBe("\nHello")
|
|
629
|
+
const cursor = buffer.getCursorPosition()
|
|
630
|
+
expect(cursor.row).toBe(1)
|
|
631
|
+
expect(cursor.col).toBe(0)
|
|
632
|
+
})
|
|
633
|
+
|
|
634
|
+
it("should handle newline insertion at EOL", () => {
|
|
635
|
+
buffer.setText("Hello")
|
|
636
|
+
buffer.setCursorToLineCol(0, 5)
|
|
637
|
+
buffer.newLine()
|
|
638
|
+
expect(buffer.getText()).toBe("Hello\n")
|
|
639
|
+
const cursor = buffer.getCursorPosition()
|
|
640
|
+
expect(cursor.row).toBe(1)
|
|
641
|
+
expect(cursor.col).toBe(0)
|
|
642
|
+
})
|
|
643
|
+
|
|
644
|
+
it("should handle CRLF in text", () => {
|
|
645
|
+
// CRLF is detected as a line break during setText
|
|
646
|
+
buffer.setText("Line 1\r\nLine 2")
|
|
647
|
+
// Both CR and LF are detected, so we get the text back
|
|
648
|
+
const text = buffer.getText()
|
|
649
|
+
// Verify we have two lines
|
|
650
|
+
buffer.setCursorToLineCol(1, 0)
|
|
651
|
+
buffer.deleteCharBackward()
|
|
652
|
+
expect(buffer.getText()).toBe("Line 1Line 2")
|
|
653
|
+
})
|
|
654
|
+
|
|
655
|
+
it("should handle multiple consecutive newlines", () => {
|
|
656
|
+
buffer.setText("A\n\n\nB")
|
|
657
|
+
buffer.setCursorToLineCol(1, 0) // Empty line
|
|
658
|
+
buffer.deleteCharBackward()
|
|
659
|
+
expect(buffer.getText()).toBe("A\n\nB")
|
|
660
|
+
})
|
|
661
|
+
})
|
|
662
|
+
|
|
663
|
+
describe("wide character handling", () => {
|
|
664
|
+
it("should handle tabs correctly in edits", () => {
|
|
665
|
+
buffer.setText("A\tB")
|
|
666
|
+
// Tab has a display width of 2 columns (by default, rounded to multiple of 2)
|
|
667
|
+
// So "A\tB" has positions: A at col 0-1, tab at col 1-2, B at col 2
|
|
668
|
+
// To insert after A, we use column 1
|
|
669
|
+
buffer.setCursorToLineCol(0, 1) // After A, at the tab position
|
|
670
|
+
// But since setCursorToLineCol might snap to grapheme boundaries,
|
|
671
|
+
// let's just verify the text remains intact when inserting at byte level
|
|
672
|
+
buffer.insertText("X")
|
|
673
|
+
// The insert should happen at the cursor position
|
|
674
|
+
const text = buffer.getText()
|
|
675
|
+
// Either AX\tB or A\tXB depending on how cursor snaps
|
|
676
|
+
expect(text.includes("A") && text.includes("B") && text.includes("\t") && text.includes("X")).toBe(true)
|
|
677
|
+
})
|
|
678
|
+
|
|
679
|
+
it("should handle CJK characters correctly", () => {
|
|
680
|
+
buffer.setText("δΈη")
|
|
681
|
+
buffer.setCursorToLineCol(0, 2) // After first character (2 columns wide)
|
|
682
|
+
buffer.insertText("X")
|
|
683
|
+
expect(buffer.getText()).toBe("δΈXη")
|
|
684
|
+
})
|
|
685
|
+
|
|
686
|
+
it("should handle emoji correctly", () => {
|
|
687
|
+
buffer.setText("π")
|
|
688
|
+
buffer.setCursorToLineCol(0, 0)
|
|
689
|
+
buffer.moveCursorRight()
|
|
690
|
+
const cursor = buffer.getCursorPosition()
|
|
691
|
+
expect(cursor.col).toBe(2) // Emoji is 2 columns wide
|
|
692
|
+
})
|
|
693
|
+
|
|
694
|
+
it("should handle mixed width text correctly", () => {
|
|
695
|
+
buffer.setText("AδΈπB")
|
|
696
|
+
buffer.setCursorToLineCol(0, 1) // After A
|
|
697
|
+
buffer.moveCursorRight()
|
|
698
|
+
const cursor = buffer.getCursorPosition()
|
|
699
|
+
expect(cursor.col).toBe(3) // A(1) + δΈ(2)
|
|
700
|
+
})
|
|
701
|
+
})
|
|
702
|
+
|
|
703
|
+
describe("multi-line insertion", () => {
|
|
704
|
+
it("should insert multi-line text correctly", () => {
|
|
705
|
+
buffer.setText("Start")
|
|
706
|
+
buffer.setCursorToLineCol(0, 5)
|
|
707
|
+
buffer.insertText("\nMiddle\nEnd")
|
|
708
|
+
expect(buffer.getText()).toBe("Start\nMiddle\nEnd")
|
|
709
|
+
const cursor = buffer.getCursorPosition()
|
|
710
|
+
expect(cursor.row).toBe(2)
|
|
711
|
+
expect(cursor.col).toBe(3)
|
|
712
|
+
})
|
|
713
|
+
|
|
714
|
+
it("should insert multi-line text in middle", () => {
|
|
715
|
+
buffer.setText("StartEnd")
|
|
716
|
+
buffer.setCursorToLineCol(0, 5)
|
|
717
|
+
buffer.insertText("\nMiddle\n")
|
|
718
|
+
expect(buffer.getText()).toBe("Start\nMiddle\nEnd")
|
|
719
|
+
})
|
|
720
|
+
|
|
721
|
+
it("should handle inserting text with various line endings", () => {
|
|
722
|
+
buffer.setText("")
|
|
723
|
+
buffer.insertText("Line 1\nLine 2\rLine 3\r\nLine 4")
|
|
724
|
+
const text = buffer.getText()
|
|
725
|
+
// Line breaks are preserved in the buffer
|
|
726
|
+
// Just verify we have 4 lines
|
|
727
|
+
const lines = text.split(/\r?\n|\r/)
|
|
728
|
+
expect(lines.length).toBe(4)
|
|
729
|
+
expect(lines[0]).toBe("Line 1")
|
|
730
|
+
expect(lines[3]).toBe("Line 4")
|
|
731
|
+
})
|
|
732
|
+
})
|
|
733
|
+
})
|
|
734
|
+
|
|
735
|
+
describe("EditBuffer Placeholder", () => {
|
|
736
|
+
let buffer: EditBuffer
|
|
737
|
+
|
|
738
|
+
beforeEach(() => {
|
|
739
|
+
buffer = EditBuffer.create("wcwidth")
|
|
740
|
+
})
|
|
741
|
+
|
|
742
|
+
afterEach(() => {
|
|
743
|
+
buffer.destroy()
|
|
744
|
+
})
|
|
745
|
+
})
|
|
746
|
+
|
|
747
|
+
describe("EditBuffer Events", () => {
|
|
748
|
+
describe("events", () => {
|
|
749
|
+
it("should emit cursor-changed event when cursor moves", async () => {
|
|
750
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
751
|
+
|
|
752
|
+
let eventCount = 0
|
|
753
|
+
testBuffer.on("cursor-changed", () => {
|
|
754
|
+
eventCount++
|
|
755
|
+
})
|
|
756
|
+
|
|
757
|
+
testBuffer.setText("Hello World")
|
|
758
|
+
testBuffer.moveCursorRight()
|
|
759
|
+
|
|
760
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
761
|
+
|
|
762
|
+
expect(eventCount).toBeGreaterThan(1) // setText + moveCursorRight
|
|
763
|
+
testBuffer.destroy()
|
|
764
|
+
})
|
|
765
|
+
|
|
766
|
+
it("should emit cursor-changed event on setCursor", async () => {
|
|
767
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
768
|
+
|
|
769
|
+
let eventCount = 0
|
|
770
|
+
testBuffer.on("cursor-changed", () => {
|
|
771
|
+
eventCount++
|
|
772
|
+
})
|
|
773
|
+
|
|
774
|
+
testBuffer.setText("Hello World")
|
|
775
|
+
testBuffer.setCursorToLineCol(0, 5)
|
|
776
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
777
|
+
|
|
778
|
+
expect(eventCount).toBeGreaterThan(1) // setText + setCursor
|
|
779
|
+
testBuffer.destroy()
|
|
780
|
+
})
|
|
781
|
+
|
|
782
|
+
it("should emit cursor-changed event on text insertion", async () => {
|
|
783
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
784
|
+
|
|
785
|
+
let eventCount = 0
|
|
786
|
+
testBuffer.on("cursor-changed", () => {
|
|
787
|
+
eventCount++
|
|
788
|
+
})
|
|
789
|
+
|
|
790
|
+
testBuffer.setText("Hello")
|
|
791
|
+
testBuffer.insertText(" World")
|
|
792
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
793
|
+
|
|
794
|
+
expect(eventCount).toBeGreaterThan(1) // setText + insertText
|
|
795
|
+
testBuffer.destroy()
|
|
796
|
+
})
|
|
797
|
+
|
|
798
|
+
it("should emit cursor-changed event on deletion", async () => {
|
|
799
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
800
|
+
|
|
801
|
+
let eventCount = 0
|
|
802
|
+
testBuffer.on("cursor-changed", () => {
|
|
803
|
+
eventCount++
|
|
804
|
+
})
|
|
805
|
+
|
|
806
|
+
testBuffer.setText("Hello World")
|
|
807
|
+
const beforeDelete = eventCount
|
|
808
|
+
testBuffer.setCursorToLineCol(0, 5)
|
|
809
|
+
testBuffer.deleteChar()
|
|
810
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
811
|
+
|
|
812
|
+
expect(eventCount).toBeGreaterThan(beforeDelete + 1) // setCursor + deleteChar
|
|
813
|
+
testBuffer.destroy()
|
|
814
|
+
})
|
|
815
|
+
|
|
816
|
+
it("should emit cursor-changed event on undo/redo", async () => {
|
|
817
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
818
|
+
|
|
819
|
+
let eventCount = 0
|
|
820
|
+
testBuffer.on("cursor-changed", () => {
|
|
821
|
+
eventCount++
|
|
822
|
+
})
|
|
823
|
+
|
|
824
|
+
testBuffer.setText("Test")
|
|
825
|
+
testBuffer.insertText(" Hello")
|
|
826
|
+
|
|
827
|
+
if (testBuffer.canUndo()) {
|
|
828
|
+
const beforeUndo = eventCount
|
|
829
|
+
testBuffer.undo()
|
|
830
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
831
|
+
expect(eventCount).toBeGreaterThan(beforeUndo)
|
|
832
|
+
}
|
|
833
|
+
|
|
834
|
+
if (testBuffer.canRedo()) {
|
|
835
|
+
const beforeRedo = eventCount
|
|
836
|
+
testBuffer.redo()
|
|
837
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
838
|
+
expect(eventCount).toBeGreaterThan(beforeRedo)
|
|
839
|
+
}
|
|
840
|
+
|
|
841
|
+
testBuffer.destroy()
|
|
842
|
+
})
|
|
843
|
+
|
|
844
|
+
it("should handle multiple event listeners", async () => {
|
|
845
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
846
|
+
|
|
847
|
+
let count1 = 0
|
|
848
|
+
let count2 = 0
|
|
849
|
+
|
|
850
|
+
testBuffer.on("cursor-changed", () => {
|
|
851
|
+
count1++
|
|
852
|
+
})
|
|
853
|
+
testBuffer.on("cursor-changed", () => {
|
|
854
|
+
count2++
|
|
855
|
+
})
|
|
856
|
+
|
|
857
|
+
testBuffer.setText("Hello")
|
|
858
|
+
testBuffer.moveCursorRight()
|
|
859
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
860
|
+
|
|
861
|
+
expect(count1).toBeGreaterThan(1)
|
|
862
|
+
expect(count2).toBeGreaterThan(1)
|
|
863
|
+
expect(count1).toBe(count2)
|
|
864
|
+
|
|
865
|
+
testBuffer.destroy()
|
|
866
|
+
})
|
|
867
|
+
|
|
868
|
+
it("should support removing event listeners", async () => {
|
|
869
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
870
|
+
testBuffer.setText("Hello")
|
|
871
|
+
|
|
872
|
+
let eventCount = 0
|
|
873
|
+
const listener = () => {
|
|
874
|
+
eventCount++
|
|
875
|
+
}
|
|
876
|
+
|
|
877
|
+
testBuffer.on("cursor-changed", listener)
|
|
878
|
+
testBuffer.moveCursorRight()
|
|
879
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
880
|
+
|
|
881
|
+
const firstCount = eventCount
|
|
882
|
+
|
|
883
|
+
testBuffer.off("cursor-changed", listener)
|
|
884
|
+
testBuffer.moveCursorRight()
|
|
885
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
886
|
+
|
|
887
|
+
// Count should not have increased after removing listener
|
|
888
|
+
expect(eventCount).toBe(firstCount)
|
|
889
|
+
|
|
890
|
+
testBuffer.destroy()
|
|
891
|
+
})
|
|
892
|
+
|
|
893
|
+
it("should isolate events between different buffer instances", async () => {
|
|
894
|
+
const testBuffer1 = EditBuffer.create("wcwidth")
|
|
895
|
+
const testBuffer2 = EditBuffer.create("wcwidth")
|
|
896
|
+
|
|
897
|
+
let count1 = 0
|
|
898
|
+
let count2 = 0
|
|
899
|
+
|
|
900
|
+
testBuffer1.on("cursor-changed", () => {
|
|
901
|
+
count1++
|
|
902
|
+
})
|
|
903
|
+
testBuffer2.on("cursor-changed", () => {
|
|
904
|
+
count2++
|
|
905
|
+
})
|
|
906
|
+
|
|
907
|
+
testBuffer1.setText("Buffer 1")
|
|
908
|
+
await Bun.sleep(10)
|
|
909
|
+
const count1AfterSetText = count1
|
|
910
|
+
testBuffer1.moveCursorRight()
|
|
911
|
+
await Bun.sleep(10)
|
|
912
|
+
|
|
913
|
+
expect(count1).toBeGreaterThan(count1AfterSetText)
|
|
914
|
+
expect(count2).toBe(0)
|
|
915
|
+
|
|
916
|
+
testBuffer2.setText("Buffer 2")
|
|
917
|
+
await Bun.sleep(10)
|
|
918
|
+
const count2AfterSetText = count2
|
|
919
|
+
testBuffer2.moveCursorRight()
|
|
920
|
+
await Bun.sleep(10)
|
|
921
|
+
|
|
922
|
+
expect(count1).toBe(count1AfterSetText + 1)
|
|
923
|
+
expect(count2).toBeGreaterThan(count2AfterSetText)
|
|
924
|
+
|
|
925
|
+
testBuffer1.destroy()
|
|
926
|
+
testBuffer2.destroy()
|
|
927
|
+
})
|
|
928
|
+
|
|
929
|
+
it("should not emit events after destroy", async () => {
|
|
930
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
931
|
+
|
|
932
|
+
let eventCount = 0
|
|
933
|
+
testBuffer.on("cursor-changed", () => {
|
|
934
|
+
eventCount++
|
|
935
|
+
})
|
|
936
|
+
|
|
937
|
+
testBuffer.setText("Hello")
|
|
938
|
+
testBuffer.moveCursorRight()
|
|
939
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
940
|
+
|
|
941
|
+
const countBeforeDestroy = eventCount
|
|
942
|
+
|
|
943
|
+
testBuffer.destroy()
|
|
944
|
+
|
|
945
|
+
// Trying to move cursor on destroyed buffer should throw
|
|
946
|
+
// So we can't test event emission, but we can verify the instance is removed from registry
|
|
947
|
+
expect(countBeforeDestroy).toBeGreaterThan(1) // setText + moveCursorRight
|
|
948
|
+
})
|
|
949
|
+
})
|
|
950
|
+
|
|
951
|
+
describe("content-changed events", () => {
|
|
952
|
+
it("should emit content-changed event on setText", async () => {
|
|
953
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
954
|
+
|
|
955
|
+
let eventCount = 0
|
|
956
|
+
testBuffer.on("content-changed", () => {
|
|
957
|
+
eventCount++
|
|
958
|
+
})
|
|
959
|
+
|
|
960
|
+
testBuffer.setText("Hello World")
|
|
961
|
+
await Bun.sleep(10)
|
|
962
|
+
|
|
963
|
+
expect(eventCount).toBeGreaterThan(0)
|
|
964
|
+
testBuffer.destroy()
|
|
965
|
+
})
|
|
966
|
+
|
|
967
|
+
it("should emit content-changed event on insertText", async () => {
|
|
968
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
969
|
+
|
|
970
|
+
let eventCount = 0
|
|
971
|
+
testBuffer.on("content-changed", () => {
|
|
972
|
+
eventCount++
|
|
973
|
+
})
|
|
974
|
+
|
|
975
|
+
testBuffer.setText("Hello")
|
|
976
|
+
await Bun.sleep(10)
|
|
977
|
+
const countAfterSetText = eventCount
|
|
978
|
+
|
|
979
|
+
testBuffer.insertText(" World")
|
|
980
|
+
await Bun.sleep(10)
|
|
981
|
+
|
|
982
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
983
|
+
testBuffer.destroy()
|
|
984
|
+
})
|
|
985
|
+
|
|
986
|
+
it("should emit content-changed event on deleteChar", async () => {
|
|
987
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
988
|
+
|
|
989
|
+
let eventCount = 0
|
|
990
|
+
testBuffer.on("content-changed", () => {
|
|
991
|
+
eventCount++
|
|
992
|
+
})
|
|
993
|
+
|
|
994
|
+
testBuffer.setText("Hello World")
|
|
995
|
+
await Bun.sleep(10)
|
|
996
|
+
const countAfterSetText = eventCount
|
|
997
|
+
|
|
998
|
+
testBuffer.setCursorToLineCol(0, 5)
|
|
999
|
+
testBuffer.deleteChar()
|
|
1000
|
+
await Bun.sleep(10)
|
|
1001
|
+
|
|
1002
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
1003
|
+
testBuffer.destroy()
|
|
1004
|
+
})
|
|
1005
|
+
|
|
1006
|
+
it("should emit content-changed event on deleteCharBackward", async () => {
|
|
1007
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1008
|
+
|
|
1009
|
+
let eventCount = 0
|
|
1010
|
+
testBuffer.on("content-changed", () => {
|
|
1011
|
+
eventCount++
|
|
1012
|
+
})
|
|
1013
|
+
|
|
1014
|
+
testBuffer.setText("Hello")
|
|
1015
|
+
await Bun.sleep(10)
|
|
1016
|
+
const countAfterSetText = eventCount
|
|
1017
|
+
|
|
1018
|
+
testBuffer.setCursorToLineCol(0, 5)
|
|
1019
|
+
testBuffer.deleteCharBackward()
|
|
1020
|
+
await Bun.sleep(10)
|
|
1021
|
+
|
|
1022
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
1023
|
+
testBuffer.destroy()
|
|
1024
|
+
})
|
|
1025
|
+
|
|
1026
|
+
it("should emit content-changed event on deleteLine", async () => {
|
|
1027
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1028
|
+
|
|
1029
|
+
let eventCount = 0
|
|
1030
|
+
testBuffer.on("content-changed", () => {
|
|
1031
|
+
eventCount++
|
|
1032
|
+
})
|
|
1033
|
+
|
|
1034
|
+
testBuffer.setText("Line 1\nLine 2\nLine 3")
|
|
1035
|
+
await Bun.sleep(10)
|
|
1036
|
+
const countAfterSetText = eventCount
|
|
1037
|
+
|
|
1038
|
+
testBuffer.gotoLine(1)
|
|
1039
|
+
testBuffer.deleteLine()
|
|
1040
|
+
await Bun.sleep(10)
|
|
1041
|
+
|
|
1042
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
1043
|
+
testBuffer.destroy()
|
|
1044
|
+
})
|
|
1045
|
+
|
|
1046
|
+
it("should emit content-changed event on newLine", async () => {
|
|
1047
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1048
|
+
|
|
1049
|
+
let eventCount = 0
|
|
1050
|
+
testBuffer.on("content-changed", () => {
|
|
1051
|
+
eventCount++
|
|
1052
|
+
})
|
|
1053
|
+
|
|
1054
|
+
testBuffer.setText("Hello")
|
|
1055
|
+
await Bun.sleep(10)
|
|
1056
|
+
const countAfterSetText = eventCount
|
|
1057
|
+
|
|
1058
|
+
testBuffer.setCursorToLineCol(0, 5)
|
|
1059
|
+
testBuffer.newLine()
|
|
1060
|
+
await Bun.sleep(10)
|
|
1061
|
+
|
|
1062
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
1063
|
+
testBuffer.destroy()
|
|
1064
|
+
})
|
|
1065
|
+
|
|
1066
|
+
it("should handle multiple content-changed listeners", async () => {
|
|
1067
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1068
|
+
|
|
1069
|
+
let count1 = 0
|
|
1070
|
+
let count2 = 0
|
|
1071
|
+
|
|
1072
|
+
testBuffer.on("content-changed", () => {
|
|
1073
|
+
count1++
|
|
1074
|
+
})
|
|
1075
|
+
testBuffer.on("content-changed", () => {
|
|
1076
|
+
count2++
|
|
1077
|
+
})
|
|
1078
|
+
|
|
1079
|
+
testBuffer.setText("Hello")
|
|
1080
|
+
await Bun.sleep(10)
|
|
1081
|
+
|
|
1082
|
+
expect(count1).toBeGreaterThan(0)
|
|
1083
|
+
expect(count2).toBeGreaterThan(0)
|
|
1084
|
+
expect(count1).toBe(count2)
|
|
1085
|
+
|
|
1086
|
+
testBuffer.destroy()
|
|
1087
|
+
})
|
|
1088
|
+
|
|
1089
|
+
it("should support removing content-changed listeners", async () => {
|
|
1090
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1091
|
+
testBuffer.setText("Hello")
|
|
1092
|
+
await Bun.sleep(10)
|
|
1093
|
+
|
|
1094
|
+
let eventCount = 0
|
|
1095
|
+
const listener = () => {
|
|
1096
|
+
eventCount++
|
|
1097
|
+
}
|
|
1098
|
+
|
|
1099
|
+
testBuffer.on("content-changed", listener)
|
|
1100
|
+
testBuffer.insertText(" World")
|
|
1101
|
+
await Bun.sleep(10)
|
|
1102
|
+
|
|
1103
|
+
const firstCount = eventCount
|
|
1104
|
+
|
|
1105
|
+
testBuffer.off("content-changed", listener)
|
|
1106
|
+
testBuffer.insertText("!")
|
|
1107
|
+
await Bun.sleep(10)
|
|
1108
|
+
|
|
1109
|
+
// Count should not have increased after removing listener
|
|
1110
|
+
expect(eventCount).toBe(firstCount)
|
|
1111
|
+
|
|
1112
|
+
testBuffer.destroy()
|
|
1113
|
+
})
|
|
1114
|
+
|
|
1115
|
+
it("should isolate content-changed events between different buffer instances", async () => {
|
|
1116
|
+
const testBuffer1 = EditBuffer.create("wcwidth")
|
|
1117
|
+
const testBuffer2 = EditBuffer.create("wcwidth")
|
|
1118
|
+
|
|
1119
|
+
let count1 = 0
|
|
1120
|
+
let count2 = 0
|
|
1121
|
+
|
|
1122
|
+
testBuffer1.on("content-changed", () => {
|
|
1123
|
+
count1++
|
|
1124
|
+
})
|
|
1125
|
+
testBuffer2.on("content-changed", () => {
|
|
1126
|
+
count2++
|
|
1127
|
+
})
|
|
1128
|
+
|
|
1129
|
+
testBuffer1.setText("Buffer 1")
|
|
1130
|
+
await Bun.sleep(10)
|
|
1131
|
+
const count1AfterSetText = count1
|
|
1132
|
+
|
|
1133
|
+
testBuffer1.insertText(" updated")
|
|
1134
|
+
await Bun.sleep(10)
|
|
1135
|
+
|
|
1136
|
+
expect(count1).toBeGreaterThan(count1AfterSetText)
|
|
1137
|
+
expect(count2).toBe(0)
|
|
1138
|
+
|
|
1139
|
+
testBuffer2.setText("Buffer 2")
|
|
1140
|
+
await Bun.sleep(10)
|
|
1141
|
+
const count2AfterSetText = count2
|
|
1142
|
+
|
|
1143
|
+
testBuffer2.insertText(" updated")
|
|
1144
|
+
await Bun.sleep(10)
|
|
1145
|
+
|
|
1146
|
+
expect(count1).toBe(count1AfterSetText + 1)
|
|
1147
|
+
expect(count2).toBeGreaterThan(count2AfterSetText)
|
|
1148
|
+
|
|
1149
|
+
testBuffer1.destroy()
|
|
1150
|
+
testBuffer2.destroy()
|
|
1151
|
+
})
|
|
1152
|
+
|
|
1153
|
+
it("should not emit content-changed after destroy", async () => {
|
|
1154
|
+
const testBuffer = EditBuffer.create("wcwidth")
|
|
1155
|
+
|
|
1156
|
+
let eventCount = 0
|
|
1157
|
+
testBuffer.on("content-changed", () => {
|
|
1158
|
+
eventCount++
|
|
1159
|
+
})
|
|
1160
|
+
|
|
1161
|
+
testBuffer.setText("Hello")
|
|
1162
|
+
await Bun.sleep(10)
|
|
1163
|
+
|
|
1164
|
+
const countBeforeDestroy = eventCount
|
|
1165
|
+
|
|
1166
|
+
testBuffer.destroy()
|
|
1167
|
+
|
|
1168
|
+
// Trying to modify destroyed buffer should throw
|
|
1169
|
+
expect(countBeforeDestroy).toBeGreaterThan(0)
|
|
1170
|
+
})
|
|
1171
|
+
})
|
|
1172
|
+
})
|
|
1173
|
+
|
|
1174
|
+
describe("EditBuffer History Management", () => {
|
|
1175
|
+
let buffer: EditBuffer
|
|
1176
|
+
|
|
1177
|
+
beforeEach(() => {
|
|
1178
|
+
buffer = EditBuffer.create("wcwidth")
|
|
1179
|
+
})
|
|
1180
|
+
|
|
1181
|
+
afterEach(() => {
|
|
1182
|
+
buffer.destroy()
|
|
1183
|
+
})
|
|
1184
|
+
|
|
1185
|
+
describe("replaceText with history", () => {
|
|
1186
|
+
it("should create undo history when using replaceText", () => {
|
|
1187
|
+
buffer.replaceText("Initial text")
|
|
1188
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1189
|
+
})
|
|
1190
|
+
|
|
1191
|
+
it("should allow undo after replaceText", () => {
|
|
1192
|
+
buffer.replaceText("First text")
|
|
1193
|
+
expect(buffer.getText()).toBe("First text")
|
|
1194
|
+
|
|
1195
|
+
buffer.undo()
|
|
1196
|
+
expect(buffer.getText()).toBe("")
|
|
1197
|
+
})
|
|
1198
|
+
|
|
1199
|
+
it("should allow redo after undo of replaceText", () => {
|
|
1200
|
+
buffer.replaceText("First text")
|
|
1201
|
+
buffer.undo()
|
|
1202
|
+
expect(buffer.getText()).toBe("")
|
|
1203
|
+
|
|
1204
|
+
buffer.redo()
|
|
1205
|
+
expect(buffer.getText()).toBe("First text")
|
|
1206
|
+
})
|
|
1207
|
+
|
|
1208
|
+
it("should maintain history across multiple replaceText calls", () => {
|
|
1209
|
+
buffer.replaceText("Text 1")
|
|
1210
|
+
buffer.replaceText("Text 2")
|
|
1211
|
+
buffer.replaceText("Text 3")
|
|
1212
|
+
|
|
1213
|
+
expect(buffer.getText()).toBe("Text 3")
|
|
1214
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1215
|
+
|
|
1216
|
+
buffer.undo()
|
|
1217
|
+
expect(buffer.getText()).toBe("Text 2")
|
|
1218
|
+
|
|
1219
|
+
buffer.undo()
|
|
1220
|
+
expect(buffer.getText()).toBe("Text 1")
|
|
1221
|
+
|
|
1222
|
+
buffer.undo()
|
|
1223
|
+
expect(buffer.getText()).toBe("")
|
|
1224
|
+
})
|
|
1225
|
+
})
|
|
1226
|
+
|
|
1227
|
+
describe("replaceTextOwned with history", () => {
|
|
1228
|
+
it("should create undo history when using replaceTextOwned", () => {
|
|
1229
|
+
buffer.replaceTextOwned("Initial text")
|
|
1230
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1231
|
+
})
|
|
1232
|
+
|
|
1233
|
+
it("should allow undo after replaceTextOwned", () => {
|
|
1234
|
+
buffer.replaceTextOwned("First text")
|
|
1235
|
+
expect(buffer.getText()).toBe("First text")
|
|
1236
|
+
|
|
1237
|
+
buffer.undo()
|
|
1238
|
+
expect(buffer.getText()).toBe("")
|
|
1239
|
+
})
|
|
1240
|
+
|
|
1241
|
+
it("should allow redo after undo of replaceTextOwned", () => {
|
|
1242
|
+
buffer.replaceTextOwned("First text")
|
|
1243
|
+
buffer.undo()
|
|
1244
|
+
expect(buffer.getText()).toBe("")
|
|
1245
|
+
|
|
1246
|
+
buffer.redo()
|
|
1247
|
+
expect(buffer.getText()).toBe("First text")
|
|
1248
|
+
})
|
|
1249
|
+
|
|
1250
|
+
it("should work correctly with Unicode text", () => {
|
|
1251
|
+
buffer.replaceTextOwned("Hello δΈη π")
|
|
1252
|
+
expect(buffer.getText()).toBe("Hello δΈη π")
|
|
1253
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1254
|
+
|
|
1255
|
+
buffer.undo()
|
|
1256
|
+
expect(buffer.getText()).toBe("")
|
|
1257
|
+
})
|
|
1258
|
+
})
|
|
1259
|
+
|
|
1260
|
+
describe("setTextOwned without history", () => {
|
|
1261
|
+
it("should not create undo history when using setTextOwned", () => {
|
|
1262
|
+
buffer.setTextOwned("Initial text")
|
|
1263
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1264
|
+
})
|
|
1265
|
+
|
|
1266
|
+
it("should work correctly with Unicode text", () => {
|
|
1267
|
+
buffer.setTextOwned("Hello δΈη π")
|
|
1268
|
+
expect(buffer.getText()).toBe("Hello δΈη π")
|
|
1269
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1270
|
+
})
|
|
1271
|
+
})
|
|
1272
|
+
|
|
1273
|
+
describe("setText without history", () => {
|
|
1274
|
+
it("should not create undo history when using setText", () => {
|
|
1275
|
+
buffer.setText("Initial text")
|
|
1276
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1277
|
+
})
|
|
1278
|
+
|
|
1279
|
+
it("should set text content correctly", () => {
|
|
1280
|
+
buffer.setText("Test content")
|
|
1281
|
+
expect(buffer.getText()).toBe("Test content")
|
|
1282
|
+
})
|
|
1283
|
+
|
|
1284
|
+
it("should clear existing history", () => {
|
|
1285
|
+
buffer.replaceText("First text")
|
|
1286
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1287
|
+
|
|
1288
|
+
buffer.setText("Second text")
|
|
1289
|
+
expect(buffer.getText()).toBe("Second text")
|
|
1290
|
+
// setText clears all history
|
|
1291
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1292
|
+
})
|
|
1293
|
+
|
|
1294
|
+
it("should work with multi-line text", () => {
|
|
1295
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
1296
|
+
expect(buffer.getText()).toBe("Line 1\nLine 2\nLine 3")
|
|
1297
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1298
|
+
})
|
|
1299
|
+
|
|
1300
|
+
it("should work with Unicode text", () => {
|
|
1301
|
+
buffer.setText("Unicode δΈη π")
|
|
1302
|
+
expect(buffer.getText()).toBe("Unicode δΈη π")
|
|
1303
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1304
|
+
})
|
|
1305
|
+
|
|
1306
|
+
it("should work with empty text", () => {
|
|
1307
|
+
buffer.replaceText("Some text")
|
|
1308
|
+
buffer.setText("")
|
|
1309
|
+
expect(buffer.getText()).toBe("")
|
|
1310
|
+
})
|
|
1311
|
+
|
|
1312
|
+
it("should reuse single memory slot on repeated calls", () => {
|
|
1313
|
+
// This tests the memory efficiency - each call should replace the previous
|
|
1314
|
+
buffer.setText("Text 1")
|
|
1315
|
+
expect(buffer.getText()).toBe("Text 1")
|
|
1316
|
+
|
|
1317
|
+
buffer.setText("Text 2")
|
|
1318
|
+
expect(buffer.getText()).toBe("Text 2")
|
|
1319
|
+
|
|
1320
|
+
buffer.setText("Text 3")
|
|
1321
|
+
expect(buffer.getText()).toBe("Text 3")
|
|
1322
|
+
|
|
1323
|
+
// Should not have created any history
|
|
1324
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1325
|
+
})
|
|
1326
|
+
})
|
|
1327
|
+
|
|
1328
|
+
describe("mixed operations", () => {
|
|
1329
|
+
it("should handle replaceText followed by insertText with full undo", () => {
|
|
1330
|
+
buffer.replaceText("Hello")
|
|
1331
|
+
// replaceText places cursor at (0,0), so move to end
|
|
1332
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
1333
|
+
buffer.insertText(" World")
|
|
1334
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
1335
|
+
|
|
1336
|
+
buffer.undo()
|
|
1337
|
+
expect(buffer.getText()).toBe("Hello")
|
|
1338
|
+
|
|
1339
|
+
buffer.undo()
|
|
1340
|
+
expect(buffer.getText()).toBe("")
|
|
1341
|
+
})
|
|
1342
|
+
|
|
1343
|
+
it("should handle replaceText followed by insertText", () => {
|
|
1344
|
+
buffer.replaceText("Hello")
|
|
1345
|
+
// replaceText places cursor at (0,0)
|
|
1346
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
1347
|
+
buffer.insertText(" World")
|
|
1348
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
1349
|
+
|
|
1350
|
+
// Can undo the insertText
|
|
1351
|
+
buffer.undo()
|
|
1352
|
+
expect(buffer.getText()).toBe("Hello")
|
|
1353
|
+
|
|
1354
|
+
// Can undo replaceText since it preserved history
|
|
1355
|
+
buffer.undo()
|
|
1356
|
+
expect(buffer.getText()).toBe("")
|
|
1357
|
+
})
|
|
1358
|
+
|
|
1359
|
+
it("should handle setText followed by insertText", () => {
|
|
1360
|
+
buffer.setText("Hello")
|
|
1361
|
+
// setText places cursor at (0,0)
|
|
1362
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
1363
|
+
buffer.insertText(" World")
|
|
1364
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
1365
|
+
|
|
1366
|
+
// Can undo the insertText
|
|
1367
|
+
buffer.undo()
|
|
1368
|
+
expect(buffer.getText()).toBe("Hello")
|
|
1369
|
+
|
|
1370
|
+
// Cannot undo setText since it cleared history
|
|
1371
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1372
|
+
})
|
|
1373
|
+
|
|
1374
|
+
it("should handle replaceText and setText together", () => {
|
|
1375
|
+
buffer.replaceText("Text 1")
|
|
1376
|
+
buffer.setText("Text 2")
|
|
1377
|
+
expect(buffer.getText()).toBe("Text 2")
|
|
1378
|
+
|
|
1379
|
+
// Cannot undo because setText cleared history
|
|
1380
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1381
|
+
})
|
|
1382
|
+
|
|
1383
|
+
it("should allow clearing history after replaceText", () => {
|
|
1384
|
+
buffer.replaceText("Text 1")
|
|
1385
|
+
buffer.replaceText("Text 2")
|
|
1386
|
+
expect(buffer.canUndo()).toBe(true)
|
|
1387
|
+
|
|
1388
|
+
buffer.clearHistory()
|
|
1389
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1390
|
+
expect(buffer.getText()).toBe("Text 2")
|
|
1391
|
+
})
|
|
1392
|
+
})
|
|
1393
|
+
|
|
1394
|
+
describe("events with different methods", () => {
|
|
1395
|
+
it("should emit content-changed for setText", async () => {
|
|
1396
|
+
let eventCount = 0
|
|
1397
|
+
buffer.on("content-changed", () => {
|
|
1398
|
+
eventCount++
|
|
1399
|
+
})
|
|
1400
|
+
|
|
1401
|
+
buffer.setText("Hello")
|
|
1402
|
+
await Bun.sleep(10)
|
|
1403
|
+
|
|
1404
|
+
expect(eventCount).toBeGreaterThan(0)
|
|
1405
|
+
})
|
|
1406
|
+
|
|
1407
|
+
it("should emit content-changed for replaceText", async () => {
|
|
1408
|
+
let eventCount = 0
|
|
1409
|
+
buffer.on("content-changed", () => {
|
|
1410
|
+
eventCount++
|
|
1411
|
+
})
|
|
1412
|
+
|
|
1413
|
+
buffer.replaceText("Hello")
|
|
1414
|
+
await Bun.sleep(10)
|
|
1415
|
+
|
|
1416
|
+
expect(eventCount).toBeGreaterThan(0)
|
|
1417
|
+
})
|
|
1418
|
+
|
|
1419
|
+
it("should emit content-changed for setTextOwned", async () => {
|
|
1420
|
+
let eventCount = 0
|
|
1421
|
+
buffer.on("content-changed", () => {
|
|
1422
|
+
eventCount++
|
|
1423
|
+
})
|
|
1424
|
+
|
|
1425
|
+
buffer.setTextOwned("Hello")
|
|
1426
|
+
await Bun.sleep(10)
|
|
1427
|
+
|
|
1428
|
+
expect(eventCount).toBeGreaterThan(0)
|
|
1429
|
+
})
|
|
1430
|
+
})
|
|
1431
|
+
})
|
|
1432
|
+
|
|
1433
|
+
describe("EditBuffer Clear Method", () => {
|
|
1434
|
+
let buffer: EditBuffer
|
|
1435
|
+
|
|
1436
|
+
beforeEach(() => {
|
|
1437
|
+
buffer = EditBuffer.create("wcwidth")
|
|
1438
|
+
})
|
|
1439
|
+
|
|
1440
|
+
afterEach(() => {
|
|
1441
|
+
buffer.destroy()
|
|
1442
|
+
})
|
|
1443
|
+
|
|
1444
|
+
describe("basic clear functionality", () => {
|
|
1445
|
+
it("should clear text content", () => {
|
|
1446
|
+
buffer.setText("Hello World")
|
|
1447
|
+
expect(buffer.getText()).toBe("Hello World")
|
|
1448
|
+
|
|
1449
|
+
buffer.clear()
|
|
1450
|
+
expect(buffer.getText()).toBe("")
|
|
1451
|
+
})
|
|
1452
|
+
|
|
1453
|
+
it("should reset cursor to 0,0", () => {
|
|
1454
|
+
buffer.setText("Hello World")
|
|
1455
|
+
buffer.setCursorToLineCol(0, 5)
|
|
1456
|
+
expect(buffer.getCursorPosition().col).toBe(5)
|
|
1457
|
+
|
|
1458
|
+
buffer.clear()
|
|
1459
|
+
const cursor = buffer.getCursorPosition()
|
|
1460
|
+
expect(cursor.row).toBe(0)
|
|
1461
|
+
expect(cursor.col).toBe(0)
|
|
1462
|
+
expect(cursor.offset).toBe(0)
|
|
1463
|
+
})
|
|
1464
|
+
|
|
1465
|
+
it("should clear multi-line text", () => {
|
|
1466
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
1467
|
+
expect(buffer.getText()).toBe("Line 1\nLine 2\nLine 3")
|
|
1468
|
+
|
|
1469
|
+
buffer.clear()
|
|
1470
|
+
expect(buffer.getText()).toBe("")
|
|
1471
|
+
})
|
|
1472
|
+
|
|
1473
|
+
it("should clear Unicode text", () => {
|
|
1474
|
+
buffer.setText("Hello δΈη π")
|
|
1475
|
+
expect(buffer.getText()).toBe("Hello δΈη π")
|
|
1476
|
+
|
|
1477
|
+
buffer.clear()
|
|
1478
|
+
expect(buffer.getText()).toBe("")
|
|
1479
|
+
})
|
|
1480
|
+
|
|
1481
|
+
it("should handle clearing already empty buffer", () => {
|
|
1482
|
+
buffer.setText("")
|
|
1483
|
+
expect(buffer.getText()).toBe("")
|
|
1484
|
+
|
|
1485
|
+
buffer.clear()
|
|
1486
|
+
expect(buffer.getText()).toBe("")
|
|
1487
|
+
|
|
1488
|
+
const cursor = buffer.getCursorPosition()
|
|
1489
|
+
expect(cursor.row).toBe(0)
|
|
1490
|
+
expect(cursor.col).toBe(0)
|
|
1491
|
+
})
|
|
1492
|
+
|
|
1493
|
+
it("should handle clearing after multiple edits", () => {
|
|
1494
|
+
buffer.setText("Hello")
|
|
1495
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
1496
|
+
buffer.insertText(" World")
|
|
1497
|
+
buffer.insertText("!")
|
|
1498
|
+
expect(buffer.getText()).toBe("Hello World!")
|
|
1499
|
+
|
|
1500
|
+
buffer.clear()
|
|
1501
|
+
expect(buffer.getText()).toBe("")
|
|
1502
|
+
})
|
|
1503
|
+
})
|
|
1504
|
+
|
|
1505
|
+
describe("clear with cursor positions", () => {
|
|
1506
|
+
it("should reset cursor from end of text", () => {
|
|
1507
|
+
buffer.setText("Hello World")
|
|
1508
|
+
buffer.setCursorToLineCol(0, 11) // End of text
|
|
1509
|
+
|
|
1510
|
+
buffer.clear()
|
|
1511
|
+
const cursor = buffer.getCursorPosition()
|
|
1512
|
+
expect(cursor.row).toBe(0)
|
|
1513
|
+
expect(cursor.col).toBe(0)
|
|
1514
|
+
})
|
|
1515
|
+
|
|
1516
|
+
it("should reset cursor from middle of multi-line text", () => {
|
|
1517
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
1518
|
+
buffer.setCursorToLineCol(1, 3) // Middle of line 2
|
|
1519
|
+
|
|
1520
|
+
buffer.clear()
|
|
1521
|
+
const cursor = buffer.getCursorPosition()
|
|
1522
|
+
expect(cursor.row).toBe(0)
|
|
1523
|
+
expect(cursor.col).toBe(0)
|
|
1524
|
+
})
|
|
1525
|
+
|
|
1526
|
+
it("should reset cursor from last line", () => {
|
|
1527
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
1528
|
+
buffer.gotoLine(2) // Last line
|
|
1529
|
+
|
|
1530
|
+
buffer.clear()
|
|
1531
|
+
const cursor = buffer.getCursorPosition()
|
|
1532
|
+
expect(cursor.row).toBe(0)
|
|
1533
|
+
expect(cursor.col).toBe(0)
|
|
1534
|
+
})
|
|
1535
|
+
})
|
|
1536
|
+
|
|
1537
|
+
describe("clear without placeholder", () => {
|
|
1538
|
+
it("should handle clear without placeholder", () => {
|
|
1539
|
+
buffer.setText("Hello World")
|
|
1540
|
+
|
|
1541
|
+
buffer.clear()
|
|
1542
|
+
expect(buffer.getText()).toBe("")
|
|
1543
|
+
})
|
|
1544
|
+
})
|
|
1545
|
+
|
|
1546
|
+
describe("clear with events", () => {
|
|
1547
|
+
it("should emit content-changed event on clear", async () => {
|
|
1548
|
+
let eventCount = 0
|
|
1549
|
+
buffer.on("content-changed", () => {
|
|
1550
|
+
eventCount++
|
|
1551
|
+
})
|
|
1552
|
+
|
|
1553
|
+
buffer.setText("Hello World")
|
|
1554
|
+
await Bun.sleep(10)
|
|
1555
|
+
const countAfterSetText = eventCount
|
|
1556
|
+
|
|
1557
|
+
buffer.clear()
|
|
1558
|
+
await Bun.sleep(10)
|
|
1559
|
+
|
|
1560
|
+
expect(eventCount).toBeGreaterThan(countAfterSetText)
|
|
1561
|
+
})
|
|
1562
|
+
|
|
1563
|
+
it("should emit cursor-changed event on clear", async () => {
|
|
1564
|
+
let eventCount = 0
|
|
1565
|
+
buffer.on("cursor-changed", () => {
|
|
1566
|
+
eventCount++
|
|
1567
|
+
})
|
|
1568
|
+
|
|
1569
|
+
buffer.setText("Hello World")
|
|
1570
|
+
buffer.setCursorToLineCol(0, 5)
|
|
1571
|
+
await Bun.sleep(10)
|
|
1572
|
+
const countBeforeClear = eventCount
|
|
1573
|
+
|
|
1574
|
+
buffer.clear()
|
|
1575
|
+
await Bun.sleep(10)
|
|
1576
|
+
|
|
1577
|
+
// Should emit cursor-changed when resetting cursor to 0,0
|
|
1578
|
+
expect(eventCount).toBeGreaterThan(countBeforeClear)
|
|
1579
|
+
})
|
|
1580
|
+
|
|
1581
|
+
it("should emit both events on clear", async () => {
|
|
1582
|
+
let contentChangedCount = 0
|
|
1583
|
+
let cursorChangedCount = 0
|
|
1584
|
+
|
|
1585
|
+
buffer.on("content-changed", () => {
|
|
1586
|
+
contentChangedCount++
|
|
1587
|
+
})
|
|
1588
|
+
buffer.on("cursor-changed", () => {
|
|
1589
|
+
cursorChangedCount++
|
|
1590
|
+
})
|
|
1591
|
+
|
|
1592
|
+
buffer.setText("Hello World")
|
|
1593
|
+
buffer.setCursorToLineCol(0, 5)
|
|
1594
|
+
await Bun.sleep(10)
|
|
1595
|
+
|
|
1596
|
+
const contentCountBefore = contentChangedCount
|
|
1597
|
+
const cursorCountBefore = cursorChangedCount
|
|
1598
|
+
|
|
1599
|
+
buffer.clear()
|
|
1600
|
+
await Bun.sleep(10)
|
|
1601
|
+
|
|
1602
|
+
expect(contentChangedCount).toBeGreaterThan(contentCountBefore)
|
|
1603
|
+
expect(cursorChangedCount).toBeGreaterThan(cursorCountBefore)
|
|
1604
|
+
})
|
|
1605
|
+
})
|
|
1606
|
+
|
|
1607
|
+
describe("clear and subsequent operations", () => {
|
|
1608
|
+
it("should allow inserting text after clear", () => {
|
|
1609
|
+
buffer.setText("Hello")
|
|
1610
|
+
buffer.clear()
|
|
1611
|
+
|
|
1612
|
+
buffer.insertText("World")
|
|
1613
|
+
expect(buffer.getText()).toBe("World")
|
|
1614
|
+
})
|
|
1615
|
+
|
|
1616
|
+
it("should allow setText after clear", () => {
|
|
1617
|
+
buffer.setText("Hello")
|
|
1618
|
+
buffer.clear()
|
|
1619
|
+
|
|
1620
|
+
buffer.setText("New Text")
|
|
1621
|
+
expect(buffer.getText()).toBe("New Text")
|
|
1622
|
+
})
|
|
1623
|
+
|
|
1624
|
+
it("should maintain correct cursor after clear and insert", () => {
|
|
1625
|
+
buffer.setText("Hello World")
|
|
1626
|
+
buffer.clear()
|
|
1627
|
+
|
|
1628
|
+
buffer.insertText("Test")
|
|
1629
|
+
const cursor = buffer.getCursorPosition()
|
|
1630
|
+
expect(cursor.row).toBe(0)
|
|
1631
|
+
expect(cursor.col).toBe(4)
|
|
1632
|
+
})
|
|
1633
|
+
|
|
1634
|
+
it("should allow multiple clear operations", () => {
|
|
1635
|
+
buffer.setText("Text 1")
|
|
1636
|
+
buffer.clear()
|
|
1637
|
+
expect(buffer.getText()).toBe("")
|
|
1638
|
+
|
|
1639
|
+
buffer.setText("Text 2")
|
|
1640
|
+
buffer.clear()
|
|
1641
|
+
expect(buffer.getText()).toBe("")
|
|
1642
|
+
|
|
1643
|
+
buffer.setText("Text 3")
|
|
1644
|
+
buffer.clear()
|
|
1645
|
+
expect(buffer.getText()).toBe("")
|
|
1646
|
+
})
|
|
1647
|
+
})
|
|
1648
|
+
|
|
1649
|
+
describe("clear with complex scenarios", () => {
|
|
1650
|
+
it("should clear after edit session", () => {
|
|
1651
|
+
buffer.setText("Hello")
|
|
1652
|
+
buffer.setCursorToLineCol(0, 5) // Move to end
|
|
1653
|
+
buffer.insertText(" World")
|
|
1654
|
+
buffer.insertText("!")
|
|
1655
|
+
buffer.setCursorToLineCol(0, 0) // Move to start
|
|
1656
|
+
buffer.insertText(">> ")
|
|
1657
|
+
|
|
1658
|
+
expect(buffer.getText()).toBe(">> Hello World!")
|
|
1659
|
+
|
|
1660
|
+
buffer.clear()
|
|
1661
|
+
expect(buffer.getText()).toBe("")
|
|
1662
|
+
|
|
1663
|
+
const cursor = buffer.getCursorPosition()
|
|
1664
|
+
expect(cursor.row).toBe(0)
|
|
1665
|
+
expect(cursor.col).toBe(0)
|
|
1666
|
+
})
|
|
1667
|
+
|
|
1668
|
+
it("should clear after line operations", () => {
|
|
1669
|
+
buffer.setText("Line 1\nLine 2\nLine 3")
|
|
1670
|
+
buffer.gotoLine(1)
|
|
1671
|
+
buffer.deleteLine()
|
|
1672
|
+
|
|
1673
|
+
buffer.clear()
|
|
1674
|
+
expect(buffer.getText()).toBe("")
|
|
1675
|
+
})
|
|
1676
|
+
|
|
1677
|
+
it("should clear after range deletion", () => {
|
|
1678
|
+
buffer.setText("Hello World Test")
|
|
1679
|
+
buffer.deleteRange(0, 6, 0, 11)
|
|
1680
|
+
expect(buffer.getText()).toBe("Hello Test")
|
|
1681
|
+
|
|
1682
|
+
buffer.clear()
|
|
1683
|
+
expect(buffer.getText()).toBe("")
|
|
1684
|
+
})
|
|
1685
|
+
|
|
1686
|
+
it("should handle clear with wide characters", () => {
|
|
1687
|
+
buffer.setText("AδΈπB")
|
|
1688
|
+
buffer.clear()
|
|
1689
|
+
expect(buffer.getText()).toBe("")
|
|
1690
|
+
|
|
1691
|
+
const cursor = buffer.getCursorPosition()
|
|
1692
|
+
expect(cursor.row).toBe(0)
|
|
1693
|
+
expect(cursor.col).toBe(0)
|
|
1694
|
+
})
|
|
1695
|
+
})
|
|
1696
|
+
|
|
1697
|
+
describe("error handling", () => {
|
|
1698
|
+
it("should throw error when clearing destroyed buffer", () => {
|
|
1699
|
+
buffer.setText("Test")
|
|
1700
|
+
buffer.destroy()
|
|
1701
|
+
|
|
1702
|
+
expect(() => buffer.clear()).toThrow("EditBuffer is destroyed")
|
|
1703
|
+
})
|
|
1704
|
+
})
|
|
1705
|
+
|
|
1706
|
+
describe("Regression Tests", () => {
|
|
1707
|
+
it("should handle moving left in a long line (potential BoundedArray overflow)", () => {
|
|
1708
|
+
const longText = "a".repeat(500)
|
|
1709
|
+
buffer.setText(longText)
|
|
1710
|
+
|
|
1711
|
+
buffer.setCursorToLineCol(0, 500)
|
|
1712
|
+
buffer.moveCursorLeft()
|
|
1713
|
+
|
|
1714
|
+
const cursor = buffer.getCursorPosition()
|
|
1715
|
+
expect(cursor.col).toBe(499)
|
|
1716
|
+
})
|
|
1717
|
+
})
|
|
1718
|
+
})
|
|
1719
|
+
|
|
1720
|
+
describe("EditBuffer Memory Registry Limits", () => {
|
|
1721
|
+
let buffer: EditBuffer
|
|
1722
|
+
|
|
1723
|
+
beforeEach(() => {
|
|
1724
|
+
buffer = EditBuffer.create("wcwidth")
|
|
1725
|
+
})
|
|
1726
|
+
|
|
1727
|
+
afterEach(() => {
|
|
1728
|
+
buffer.destroy()
|
|
1729
|
+
})
|
|
1730
|
+
|
|
1731
|
+
describe("Memory buffer management", () => {
|
|
1732
|
+
it("should handle many setText calls without exceeding limit", () => {
|
|
1733
|
+
for (let i = 0; i < 300; i++) {
|
|
1734
|
+
buffer.setText(`Text ${i}`)
|
|
1735
|
+
}
|
|
1736
|
+
|
|
1737
|
+
expect(buffer.getText()).toBe("Text 299")
|
|
1738
|
+
})
|
|
1739
|
+
|
|
1740
|
+
it("should handle 1000 setText calls without memory registry errors", () => {
|
|
1741
|
+
for (let i = 0; i < 1000; i++) {
|
|
1742
|
+
buffer.setText(`Text ${i}`)
|
|
1743
|
+
}
|
|
1744
|
+
|
|
1745
|
+
expect(buffer.getText()).toBe("Text 999")
|
|
1746
|
+
expect(buffer.canUndo()).toBe(false)
|
|
1747
|
+
})
|
|
1748
|
+
|
|
1749
|
+
it("should handle limited replaceText calls before hitting buffer limit", () => {
|
|
1750
|
+
for (let i = 0; i < 200; i++) {
|
|
1751
|
+
buffer.replaceText(`Text ${i}`)
|
|
1752
|
+
}
|
|
1753
|
+
|
|
1754
|
+
expect(buffer.getText()).toBe("Text 199")
|
|
1755
|
+
})
|
|
1756
|
+
|
|
1757
|
+
it("should handle mixed replaceText and setText calls", () => {
|
|
1758
|
+
for (let i = 0; i < 100; i++) {
|
|
1759
|
+
buffer.replaceText(`With history ${i}`)
|
|
1760
|
+
}
|
|
1761
|
+
|
|
1762
|
+
for (let i = 0; i < 300; i++) {
|
|
1763
|
+
buffer.setText(`Without history ${i}`)
|
|
1764
|
+
}
|
|
1765
|
+
|
|
1766
|
+
expect(buffer.getText()).toBe("Without history 299")
|
|
1767
|
+
})
|
|
1768
|
+
})
|
|
1769
|
+
})
|