@fairyhunter13/opentui-core 0.1.114 → 0.1.115
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/3d/SpriteResourceManager.d.ts +74 -0
- package/3d/SpriteUtils.d.ts +13 -0
- package/3d/TextureUtils.d.ts +24 -0
- package/3d/ThreeRenderable.d.ts +40 -0
- package/3d/WGPURenderer.d.ts +61 -0
- package/3d/animation/ExplodingSpriteEffect.d.ts +71 -0
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +76 -0
- package/3d/animation/SpriteAnimator.d.ts +124 -0
- package/3d/animation/SpriteParticleGenerator.d.ts +62 -0
- package/3d/canvas.d.ts +44 -0
- package/3d/index.d.ts +12 -0
- package/3d/physics/PlanckPhysicsAdapter.d.ts +19 -0
- package/3d/physics/RapierPhysicsAdapter.d.ts +19 -0
- package/3d/physics/physics-interface.d.ts +27 -0
- package/3d.d.ts +2 -0
- package/3d.js +34041 -0
- package/3d.js.map +155 -0
- package/LICENSE +21 -0
- package/NativeSpanFeed.d.ts +41 -0
- package/Renderable.d.ts +334 -0
- package/animation/Timeline.d.ts +126 -0
- package/ansi.d.ts +13 -0
- package/buffer.d.ts +111 -0
- package/console.d.ts +144 -0
- package/edit-buffer.d.ts +98 -0
- package/editor-view.d.ts +73 -0
- package/index-j4m38kjn.js +411 -0
- package/index-j4m38kjn.js.map +10 -0
- package/index-tse8gzh0.js +20614 -0
- package/index-tse8gzh0.js.map +67 -0
- package/index-vv2jcd4r.js +12299 -0
- package/index-vv2jcd4r.js.map +42 -0
- package/index.d.ts +23 -0
- package/index.js +478 -0
- package/index.js.map +9 -0
- package/lib/KeyHandler.d.ts +61 -0
- package/lib/RGBA.d.ts +25 -0
- package/lib/ascii.font.d.ts +508 -0
- package/lib/border.d.ts +51 -0
- package/lib/bunfs.d.ts +7 -0
- package/lib/clipboard.d.ts +17 -0
- package/lib/clock.d.ts +15 -0
- package/lib/data-paths.d.ts +26 -0
- package/lib/debounce.d.ts +42 -0
- package/lib/detect-links.d.ts +6 -0
- package/lib/env.d.ts +42 -0
- package/lib/extmarks-history.d.ts +17 -0
- package/lib/extmarks.d.ts +89 -0
- package/lib/hast-styled-text.d.ts +17 -0
- package/lib/index.d.ts +21 -0
- package/lib/keymapping.d.ts +25 -0
- package/lib/objects-in-viewport.d.ts +24 -0
- package/lib/output.capture.d.ts +24 -0
- package/lib/parse.keypress-kitty.d.ts +2 -0
- package/lib/parse.keypress.d.ts +26 -0
- package/lib/parse.mouse.d.ts +30 -0
- package/lib/paste.d.ts +7 -0
- package/lib/queue.d.ts +15 -0
- package/lib/renderable.validations.d.ts +12 -0
- package/lib/scroll-acceleration.d.ts +43 -0
- package/lib/selection.d.ts +63 -0
- package/lib/singleton.d.ts +7 -0
- package/lib/stdin-parser.d.ts +87 -0
- package/lib/styled-text.d.ts +63 -0
- package/lib/terminal-capability-detection.d.ts +30 -0
- package/lib/terminal-palette.d.ts +50 -0
- package/lib/tree-sitter/assets/update.d.ts +11 -0
- package/lib/tree-sitter/client.d.ts +47 -0
- package/lib/tree-sitter/default-parsers.d.ts +2 -0
- package/lib/tree-sitter/download-utils.d.ts +21 -0
- package/lib/tree-sitter/index.d.ts +8 -0
- package/lib/tree-sitter/parser.worker.d.ts +1 -0
- package/lib/tree-sitter/parsers-config.d.ts +53 -0
- package/lib/tree-sitter/resolve-ft.d.ts +5 -0
- package/lib/tree-sitter/types.d.ts +82 -0
- package/lib/tree-sitter-styled-text.d.ts +14 -0
- package/lib/validate-dir-name.d.ts +1 -0
- package/lib/yoga.options.d.ts +32 -0
- package/package.json +50 -62
- package/parser.worker.js +899 -0
- package/parser.worker.js.map +12 -0
- package/plugins/core-slot.d.ts +72 -0
- package/plugins/registry.d.ts +42 -0
- package/plugins/types.d.ts +34 -0
- package/post/effects.d.ts +147 -0
- package/post/filters.d.ts +65 -0
- package/post/matrices.d.ts +20 -0
- package/renderables/ASCIIFont.d.ts +52 -0
- package/renderables/Box.d.ts +81 -0
- package/renderables/Code.d.ts +78 -0
- package/renderables/Diff.d.ts +142 -0
- package/renderables/EditBufferRenderable.d.ts +237 -0
- package/renderables/FrameBuffer.d.ts +16 -0
- package/renderables/Input.d.ts +67 -0
- package/renderables/LineNumberRenderable.d.ts +78 -0
- package/renderables/Markdown.d.ts +185 -0
- package/renderables/ScrollBar.d.ts +77 -0
- package/renderables/ScrollBox.d.ts +124 -0
- package/renderables/Select.d.ts +115 -0
- package/renderables/Slider.d.ts +47 -0
- package/renderables/TabSelect.d.ts +96 -0
- package/renderables/Text.d.ts +36 -0
- package/renderables/TextBufferRenderable.d.ts +105 -0
- package/renderables/TextNode.d.ts +91 -0
- package/renderables/TextTable.d.ts +140 -0
- package/renderables/Textarea.d.ts +63 -0
- package/renderables/TimeToFirstDraw.d.ts +24 -0
- package/renderables/__tests__/renderable-test-utils.d.ts +12 -0
- package/renderables/composition/VRenderable.d.ts +16 -0
- package/renderables/composition/constructs.d.ts +35 -0
- package/renderables/composition/vnode.d.ts +46 -0
- package/renderables/index.d.ts +23 -0
- package/renderables/markdown-parser.d.ts +10 -0
- package/renderer.d.ts +419 -0
- package/runtime-plugin-support.d.ts +3 -0
- package/runtime-plugin-support.js +29 -0
- package/runtime-plugin-support.js.map +10 -0
- package/runtime-plugin.d.ts +16 -0
- package/runtime-plugin.js +16 -0
- package/runtime-plugin.js.map +9 -0
- package/syntax-style.d.ts +54 -0
- package/testing/manual-clock.d.ts +17 -0
- package/testing/mock-keys.d.ts +81 -0
- package/testing/mock-mouse.d.ts +38 -0
- package/testing/mock-tree-sitter-client.d.ts +23 -0
- package/testing/spy.d.ts +7 -0
- package/testing/test-recorder.d.ts +61 -0
- package/testing/test-renderer.d.ts +23 -0
- package/testing.d.ts +6 -0
- package/testing.js +697 -0
- package/testing.js.map +15 -0
- package/text-buffer-view.d.ts +42 -0
- package/text-buffer.d.ts +67 -0
- package/types.d.ts +139 -0
- package/utils.d.ts +14 -0
- package/zig-structs.d.ts +155 -0
- package/zig.d.ts +353 -0
- package/dev/keypress-debug-renderer.ts +0 -148
- package/dev/keypress-debug.ts +0 -43
- package/dev/print-env-vars.ts +0 -32
- package/dev/test-tmux-graphics-334.sh +0 -68
- package/dev/thai-debug-test.ts +0 -68
- package/docs/development.md +0 -144
- package/scripts/build.ts +0 -400
- package/scripts/publish.ts +0 -60
- package/src/3d/SpriteResourceManager.ts +0 -286
- package/src/3d/SpriteUtils.ts +0 -70
- package/src/3d/TextureUtils.ts +0 -196
- package/src/3d/ThreeRenderable.ts +0 -197
- package/src/3d/WGPURenderer.ts +0 -294
- package/src/3d/animation/ExplodingSpriteEffect.ts +0 -513
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +0 -429
- package/src/3d/animation/SpriteAnimator.ts +0 -633
- package/src/3d/animation/SpriteParticleGenerator.ts +0 -435
- package/src/3d/canvas.ts +0 -464
- package/src/3d/index.ts +0 -12
- package/src/3d/physics/PlanckPhysicsAdapter.ts +0 -72
- package/src/3d/physics/RapierPhysicsAdapter.ts +0 -66
- package/src/3d/physics/physics-interface.ts +0 -31
- package/src/3d/shaders/supersampling.wgsl +0 -201
- package/src/3d.ts +0 -3
- package/src/NativeSpanFeed.ts +0 -300
- package/src/Renderable.ts +0 -1704
- package/src/__snapshots__/buffer.test.ts.snap +0 -28
- package/src/animation/Timeline.test.ts +0 -2709
- package/src/animation/Timeline.ts +0 -598
- package/src/ansi.ts +0 -18
- package/src/benchmark/attenuation-benchmark.ts +0 -81
- package/src/benchmark/colormatrix-benchmark.ts +0 -128
- package/src/benchmark/gain-benchmark.ts +0 -80
- package/src/benchmark/latest-all-bench-run.json +0 -707
- package/src/benchmark/latest-async-bench-run.json +0 -336
- package/src/benchmark/latest-default-bench-run.json +0 -657
- package/src/benchmark/latest-large-bench-run.json +0 -707
- package/src/benchmark/latest-quick-bench-run.json +0 -207
- package/src/benchmark/markdown-benchmark.ts +0 -1796
- package/src/benchmark/native-span-feed-async-benchmark.ts +0 -355
- package/src/benchmark/native-span-feed-benchmark.md +0 -56
- package/src/benchmark/native-span-feed-benchmark.ts +0 -596
- package/src/benchmark/native-span-feed-compare.ts +0 -280
- package/src/benchmark/renderer-benchmark.ts +0 -754
- package/src/benchmark/text-table-benchmark.ts +0 -948
- package/src/buffer.test.ts +0 -291
- package/src/buffer.ts +0 -554
- package/src/console.test.ts +0 -612
- package/src/console.ts +0 -1254
- package/src/edit-buffer.test.ts +0 -1769
- package/src/edit-buffer.ts +0 -411
- package/src/editor-view.test.ts +0 -1032
- package/src/editor-view.ts +0 -284
- package/src/examples/ascii-font-selection-demo.ts +0 -245
- 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 +0 -1018
- 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 +0 -115
- package/src/examples/code-demo.ts +0 -924
- package/src/examples/console-demo.ts +0 -358
- package/src/examples/core-plugin-slots-demo.ts +0 -759
- package/src/examples/diff-demo.ts +0 -701
- package/src/examples/draggable-three-demo.ts +0 -259
- package/src/examples/editor-demo.ts +0 -322
- package/src/examples/extmarks-demo.ts +0 -196
- package/src/examples/focus-restore-demo.ts +0 -310
- package/src/examples/fonts.ts +0 -245
- package/src/examples/fractal-shader-demo.ts +0 -268
- package/src/examples/framebuffer-demo.ts +0 -674
- package/src/examples/full-unicode-demo.ts +0 -241
- package/src/examples/golden-star-demo.ts +0 -933
- package/src/examples/grayscale-buffer-demo.ts +0 -249
- package/src/examples/hast-syntax-highlighting-demo.ts +0 -129
- package/src/examples/index.ts +0 -926
- package/src/examples/input-demo.ts +0 -377
- package/src/examples/input-select-layout-demo.ts +0 -425
- package/src/examples/install.sh +0 -143
- package/src/examples/keypress-debug-demo.ts +0 -452
- package/src/examples/lib/HexList.ts +0 -122
- package/src/examples/lib/PaletteGrid.ts +0 -125
- package/src/examples/lib/standalone-keys.ts +0 -25
- package/src/examples/lib/tab-controller.ts +0 -243
- package/src/examples/lights-phong-demo.ts +0 -290
- package/src/examples/link-demo.ts +0 -220
- package/src/examples/live-state-demo.ts +0 -480
- package/src/examples/markdown-demo.ts +0 -725
- package/src/examples/mouse-interaction-demo.ts +0 -428
- package/src/examples/nested-zindex-demo.ts +0 -357
- package/src/examples/opacity-example.ts +0 -235
- package/src/examples/opentui-demo.ts +0 -1057
- package/src/examples/physx-planck-2d-demo.ts +0 -623
- package/src/examples/physx-rapier-2d-demo.ts +0 -655
- package/src/examples/relative-positioning-demo.ts +0 -323
- package/src/examples/scroll-example.ts +0 -214
- package/src/examples/scrollbox-mouse-test.ts +0 -112
- package/src/examples/scrollbox-overlay-hit-test.ts +0 -206
- package/src/examples/select-demo.ts +0 -237
- package/src/examples/shader-cube-demo.ts +0 -1015
- package/src/examples/simple-layout-example.ts +0 -591
- package/src/examples/slider-demo.ts +0 -617
- package/src/examples/split-mode-demo.ts +0 -453
- package/src/examples/sprite-animation-demo.ts +0 -443
- package/src/examples/sprite-particle-generator-demo.ts +0 -486
- package/src/examples/static-sprite-demo.ts +0 -193
- package/src/examples/sticky-scroll-example.ts +0 -308
- package/src/examples/styled-text-demo.ts +0 -282
- package/src/examples/tab-select-demo.ts +0 -219
- package/src/examples/terminal-title.ts +0 -29
- package/src/examples/terminal.ts +0 -305
- package/src/examples/text-node-demo.ts +0 -416
- package/src/examples/text-selection-demo.ts +0 -377
- package/src/examples/text-table-demo.ts +0 -503
- package/src/examples/text-truncation-demo.ts +0 -481
- package/src/examples/text-wrap.ts +0 -757
- package/src/examples/texture-loading-demo.ts +0 -259
- package/src/examples/timeline-example.ts +0 -670
- package/src/examples/transparency-demo.ts +0 -400
- package/src/examples/vnode-composition-demo.ts +0 -404
- package/src/examples/wide-grapheme-overlay-demo.ts +0 -280
- package/src/index.ts +0 -24
- package/src/lib/KeyHandler.integration.test.ts +0 -292
- package/src/lib/KeyHandler.stopPropagation.test.ts +0 -289
- package/src/lib/KeyHandler.test.ts +0 -662
- package/src/lib/KeyHandler.ts +0 -222
- package/src/lib/RGBA.test.ts +0 -984
- package/src/lib/RGBA.ts +0 -204
- package/src/lib/ascii.font.ts +0 -330
- package/src/lib/border.test.ts +0 -83
- package/src/lib/border.ts +0 -170
- package/src/lib/bunfs.test.ts +0 -27
- package/src/lib/bunfs.ts +0 -18
- package/src/lib/clipboard.test.ts +0 -41
- package/src/lib/clipboard.ts +0 -47
- package/src/lib/clock.ts +0 -35
- package/src/lib/data-paths.test.ts +0 -133
- package/src/lib/data-paths.ts +0 -109
- package/src/lib/debounce.ts +0 -106
- package/src/lib/detect-links.test.ts +0 -98
- package/src/lib/detect-links.ts +0 -56
- package/src/lib/env.test.ts +0 -228
- package/src/lib/env.ts +0 -209
- package/src/lib/extmarks-history.ts +0 -51
- package/src/lib/extmarks-multiwidth.test.ts +0 -322
- package/src/lib/extmarks.test.ts +0 -3457
- package/src/lib/extmarks.ts +0 -843
- package/src/lib/fonts/block.json +0 -405
- package/src/lib/fonts/grid.json +0 -265
- package/src/lib/fonts/huge.json +0 -741
- package/src/lib/fonts/pallet.json +0 -314
- package/src/lib/fonts/shade.json +0 -591
- package/src/lib/fonts/slick.json +0 -321
- package/src/lib/fonts/tiny.json +0 -69
- package/src/lib/hast-styled-text.ts +0 -59
- package/src/lib/index.ts +0 -21
- package/src/lib/keymapping.test.ts +0 -317
- package/src/lib/keymapping.ts +0 -115
- package/src/lib/objects-in-viewport.test.ts +0 -787
- package/src/lib/objects-in-viewport.ts +0 -153
- package/src/lib/output.capture.ts +0 -58
- package/src/lib/parse.keypress-kitty.protocol.test.ts +0 -340
- package/src/lib/parse.keypress-kitty.test.ts +0 -663
- package/src/lib/parse.keypress-kitty.ts +0 -439
- package/src/lib/parse.keypress.test.ts +0 -1849
- package/src/lib/parse.keypress.ts +0 -397
- package/src/lib/parse.mouse.test.ts +0 -552
- package/src/lib/parse.mouse.ts +0 -232
- package/src/lib/paste.ts +0 -16
- package/src/lib/queue.ts +0 -65
- package/src/lib/renderable.validations.test.ts +0 -87
- package/src/lib/renderable.validations.ts +0 -83
- package/src/lib/scroll-acceleration.ts +0 -98
- package/src/lib/selection.ts +0 -240
- package/src/lib/singleton.ts +0 -28
- package/src/lib/stdin-parser.test.ts +0 -2290
- package/src/lib/stdin-parser.ts +0 -1810
- package/src/lib/styled-text.ts +0 -178
- package/src/lib/terminal-capability-detection.test.ts +0 -202
- package/src/lib/terminal-capability-detection.ts +0 -79
- package/src/lib/terminal-palette.test.ts +0 -878
- package/src/lib/terminal-palette.ts +0 -383
- package/src/lib/tree-sitter/assets/README.md +0 -118
- package/src/lib/tree-sitter/assets/update.ts +0 -334
- package/src/lib/tree-sitter/assets.d.ts +0 -9
- package/src/lib/tree-sitter/cache.test.ts +0 -273
- package/src/lib/tree-sitter/client.test.ts +0 -1165
- package/src/lib/tree-sitter/client.ts +0 -607
- package/src/lib/tree-sitter/default-parsers.ts +0 -86
- package/src/lib/tree-sitter/download-utils.ts +0 -148
- package/src/lib/tree-sitter/index.ts +0 -28
- package/src/lib/tree-sitter/parser.worker.ts +0 -1042
- package/src/lib/tree-sitter/parsers-config.ts +0 -81
- package/src/lib/tree-sitter/resolve-ft.test.ts +0 -55
- package/src/lib/tree-sitter/resolve-ft.ts +0 -189
- package/src/lib/tree-sitter/types.ts +0 -82
- package/src/lib/tree-sitter-styled-text.test.ts +0 -1253
- package/src/lib/tree-sitter-styled-text.ts +0 -306
- package/src/lib/validate-dir-name.ts +0 -55
- package/src/lib/yoga.options.test.ts +0 -628
- package/src/lib/yoga.options.ts +0 -346
- package/src/plugins/core-slot.ts +0 -579
- package/src/plugins/registry.ts +0 -402
- package/src/plugins/types.ts +0 -46
- package/src/post/effects.ts +0 -930
- package/src/post/filters.ts +0 -489
- package/src/post/matrices.ts +0 -288
- package/src/renderables/ASCIIFont.ts +0 -219
- package/src/renderables/Box.test.ts +0 -205
- package/src/renderables/Box.ts +0 -326
- package/src/renderables/Code.test.ts +0 -2062
- package/src/renderables/Code.ts +0 -357
- package/src/renderables/Diff.regression.test.ts +0 -226
- package/src/renderables/Diff.test.ts +0 -3101
- package/src/renderables/Diff.ts +0 -1211
- package/src/renderables/EditBufferRenderable.test.ts +0 -288
- package/src/renderables/EditBufferRenderable.ts +0 -1166
- package/src/renderables/FrameBuffer.ts +0 -47
- package/src/renderables/Input.test.ts +0 -1228
- package/src/renderables/Input.ts +0 -247
- package/src/renderables/LineNumberRenderable.ts +0 -724
- package/src/renderables/Markdown.ts +0 -1393
- package/src/renderables/ScrollBar.ts +0 -422
- package/src/renderables/ScrollBox.ts +0 -883
- package/src/renderables/Select.test.ts +0 -1033
- package/src/renderables/Select.ts +0 -524
- package/src/renderables/Slider.test.ts +0 -456
- package/src/renderables/Slider.ts +0 -342
- package/src/renderables/TabSelect.test.ts +0 -197
- package/src/renderables/TabSelect.ts +0 -455
- package/src/renderables/Text.selection-buffer.test.ts +0 -123
- package/src/renderables/Text.test.ts +0 -2660
- package/src/renderables/Text.ts +0 -147
- package/src/renderables/TextBufferRenderable.ts +0 -518
- package/src/renderables/TextNode.test.ts +0 -1058
- package/src/renderables/TextNode.ts +0 -325
- package/src/renderables/TextTable.test.ts +0 -1421
- package/src/renderables/TextTable.ts +0 -1344
- package/src/renderables/Textarea.ts +0 -430
- package/src/renderables/TimeToFirstDraw.ts +0 -89
- package/src/renderables/__snapshots__/Code.test.ts.snap +0 -13
- package/src/renderables/__snapshots__/Diff.test.ts.snap +0 -785
- package/src/renderables/__snapshots__/Text.test.ts.snap +0 -421
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +0 -215
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +0 -144
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +0 -816
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +0 -1865
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +0 -85
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +0 -242
- package/src/renderables/__tests__/Markdown.test.ts +0 -2518
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +0 -87
- package/src/renderables/__tests__/Textarea.buffer.test.ts +0 -682
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +0 -675
- package/src/renderables/__tests__/Textarea.editing.test.ts +0 -2041
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +0 -35
- package/src/renderables/__tests__/Textarea.events.test.ts +0 -738
- package/src/renderables/__tests__/Textarea.highlights.test.ts +0 -590
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +0 -3149
- package/src/renderables/__tests__/Textarea.paste.test.ts +0 -357
- package/src/renderables/__tests__/Textarea.rendering.test.ts +0 -1866
- package/src/renderables/__tests__/Textarea.scroll.test.ts +0 -733
- package/src/renderables/__tests__/Textarea.selection.test.ts +0 -1590
- package/src/renderables/__tests__/Textarea.stress.test.ts +0 -670
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +0 -383
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +0 -310
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +0 -221
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +0 -89
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +0 -457
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +0 -158
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +0 -387
- package/src/renderables/__tests__/markdown-parser.test.ts +0 -217
- package/src/renderables/__tests__/renderable-test-utils.ts +0 -60
- package/src/renderables/composition/README.md +0 -8
- package/src/renderables/composition/VRenderable.ts +0 -32
- package/src/renderables/composition/constructs.ts +0 -127
- package/src/renderables/composition/vnode.ts +0 -289
- package/src/renderables/index.ts +0 -23
- package/src/renderables/markdown-parser.ts +0 -66
- package/src/renderer.ts +0 -2681
- package/src/runtime-plugin-support.ts +0 -39
- package/src/runtime-plugin.ts +0 -615
- package/src/syntax-style.test.ts +0 -841
- package/src/syntax-style.ts +0 -257
- package/src/testing/README.md +0 -210
- package/src/testing/capture-spans.test.ts +0 -194
- package/src/testing/integration.test.ts +0 -276
- package/src/testing/manual-clock.ts +0 -117
- package/src/testing/mock-keys.test.ts +0 -1378
- package/src/testing/mock-keys.ts +0 -457
- package/src/testing/mock-mouse.test.ts +0 -218
- package/src/testing/mock-mouse.ts +0 -247
- package/src/testing/mock-tree-sitter-client.ts +0 -73
- package/src/testing/spy.ts +0 -13
- package/src/testing/test-recorder.test.ts +0 -415
- package/src/testing/test-recorder.ts +0 -145
- package/src/testing/test-renderer.ts +0 -132
- package/src/testing.ts +0 -7
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +0 -481
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +0 -19
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +0 -29
- package/src/tests/absolute-positioning.snapshot.test.ts +0 -638
- package/src/tests/allocator-stats.test.ts +0 -38
- package/src/tests/destroy-during-render.test.ts +0 -200
- package/src/tests/destroy-on-exit.fixture.ts +0 -36
- package/src/tests/destroy-on-exit.test.ts +0 -41
- package/src/tests/hover-cursor.test.ts +0 -98
- package/src/tests/native-span-feed-async.test.ts +0 -173
- package/src/tests/native-span-feed-close.test.ts +0 -120
- package/src/tests/native-span-feed-coverage.test.ts +0 -227
- package/src/tests/native-span-feed-edge-cases.test.ts +0 -352
- package/src/tests/native-span-feed-use-after-free.test.ts +0 -45
- package/src/tests/opacity.test.ts +0 -123
- package/src/tests/renderable.snapshot.test.ts +0 -524
- package/src/tests/renderable.test.ts +0 -1281
- package/src/tests/renderer.clock.test.ts +0 -158
- package/src/tests/renderer.console-startup.test.ts +0 -185
- package/src/tests/renderer.control.test.ts +0 -425
- package/src/tests/renderer.core-slot-binding.test.ts +0 -952
- package/src/tests/renderer.cursor.test.ts +0 -26
- package/src/tests/renderer.destroy-during-render.test.ts +0 -147
- package/src/tests/renderer.focus-restore.test.ts +0 -257
- package/src/tests/renderer.focus.test.ts +0 -294
- package/src/tests/renderer.idle.test.ts +0 -219
- package/src/tests/renderer.input.test.ts +0 -2237
- package/src/tests/renderer.kitty-flags.test.ts +0 -195
- package/src/tests/renderer.mouse.test.ts +0 -1274
- package/src/tests/renderer.palette.test.ts +0 -629
- package/src/tests/renderer.selection.test.ts +0 -49
- package/src/tests/renderer.slot-registry.test.ts +0 -684
- package/src/tests/renderer.useMouse.test.ts +0 -47
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +0 -76
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +0 -43
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +0 -67
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +0 -72
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +0 -44
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +0 -85
- package/src/tests/runtime-plugin-path-alias.fixture.ts +0 -43
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +0 -65
- package/src/tests/runtime-plugin-support.fixture.ts +0 -11
- package/src/tests/runtime-plugin-support.test.ts +0 -19
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +0 -30
- package/src/tests/runtime-plugin.fixture.ts +0 -40
- package/src/tests/runtime-plugin.test.ts +0 -354
- package/src/tests/scrollbox-culling-bug.test.ts +0 -114
- package/src/tests/scrollbox-hitgrid-resize.test.ts +0 -136
- package/src/tests/scrollbox-hitgrid.test.ts +0 -909
- package/src/tests/scrollbox.test.ts +0 -1530
- package/src/tests/wrap-resize-perf.test.ts +0 -276
- package/src/tests/yoga-setters.test.ts +0 -921
- package/src/text-buffer-view.test.ts +0 -705
- package/src/text-buffer-view.ts +0 -189
- package/src/text-buffer.test.ts +0 -347
- package/src/text-buffer.ts +0 -250
- package/src/types.ts +0 -161
- package/src/utils.ts +0 -88
- package/src/zig/ansi.zig +0 -268
- package/src/zig/bench/README.md +0 -50
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +0 -887
- package/src/zig/bench/edit-buffer_bench.zig +0 -476
- package/src/zig/bench/native-span-feed_bench.zig +0 -100
- package/src/zig/bench/rope-markers_bench.zig +0 -713
- package/src/zig/bench/rope_bench.zig +0 -514
- package/src/zig/bench/styled-text_bench.zig +0 -470
- package/src/zig/bench/text-buffer-coords_bench.zig +0 -362
- package/src/zig/bench/text-buffer-view_bench.zig +0 -459
- package/src/zig/bench/text-chunk-graphemes_bench.zig +0 -273
- package/src/zig/bench/utf8_bench.zig +0 -799
- package/src/zig/bench-utils.zig +0 -431
- package/src/zig/bench.zig +0 -217
- package/src/zig/buffer-methods.zig +0 -211
- package/src/zig/buffer.zig +0 -2281
- package/src/zig/build.zig +0 -289
- package/src/zig/build.zig.zon +0 -16
- package/src/zig/edit-buffer.zig +0 -825
- package/src/zig/editor-view.zig +0 -802
- package/src/zig/event-bus.zig +0 -13
- package/src/zig/event-emitter.zig +0 -65
- package/src/zig/file-logger.zig +0 -92
- package/src/zig/grapheme.zig +0 -599
- package/src/zig/lib.zig +0 -1854
- package/src/zig/link.zig +0 -333
- package/src/zig/logger.zig +0 -43
- package/src/zig/mem-registry.zig +0 -125
- package/src/zig/native-span-feed-bench-lib.zig +0 -7
- package/src/zig/native-span-feed.zig +0 -708
- package/src/zig/renderer.zig +0 -1393
- package/src/zig/rope.zig +0 -1220
- package/src/zig/syntax-style.zig +0 -161
- package/src/zig/terminal.zig +0 -987
- package/src/zig/test.zig +0 -72
- package/src/zig/tests/README.md +0 -18
- package/src/zig/tests/buffer-methods_test.zig +0 -1109
- package/src/zig/tests/buffer_test.zig +0 -2557
- package/src/zig/tests/edit-buffer-history_test.zig +0 -271
- package/src/zig/tests/edit-buffer_test.zig +0 -1689
- package/src/zig/tests/editor-view_test.zig +0 -3299
- package/src/zig/tests/event-emitter_test.zig +0 -249
- package/src/zig/tests/grapheme_test.zig +0 -1304
- package/src/zig/tests/link_test.zig +0 -190
- package/src/zig/tests/mem-registry_test.zig +0 -473
- package/src/zig/tests/memory_leak_regression_test.zig +0 -159
- package/src/zig/tests/native-span-feed_test.zig +0 -1264
- package/src/zig/tests/renderer_test.zig +0 -1017
- package/src/zig/tests/rope-nested_test.zig +0 -712
- package/src/zig/tests/rope_fuzz_test.zig +0 -238
- package/src/zig/tests/rope_test.zig +0 -2362
- package/src/zig/tests/segment-merge.test.zig +0 -148
- package/src/zig/tests/syntax-style_test.zig +0 -557
- package/src/zig/tests/terminal_test.zig +0 -754
- package/src/zig/tests/text-buffer-drawing_test.zig +0 -3237
- package/src/zig/tests/text-buffer-highlights_test.zig +0 -666
- package/src/zig/tests/text-buffer-iterators_test.zig +0 -776
- package/src/zig/tests/text-buffer-segment_test.zig +0 -320
- package/src/zig/tests/text-buffer-selection_test.zig +0 -1035
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +0 -358
- package/src/zig/tests/text-buffer-view_test.zig +0 -3649
- package/src/zig/tests/text-buffer_test.zig +0 -2191
- package/src/zig/tests/unicode-width-map.zon +0 -3909
- package/src/zig/tests/utf8_no_zwj_test.zig +0 -260
- package/src/zig/tests/utf8_test.zig +0 -4057
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +0 -267
- package/src/zig/tests/utf8_wcwidth_test.zig +0 -357
- package/src/zig/tests/word-wrap-editing_test.zig +0 -498
- package/src/zig/tests/wrap-cache-perf_test.zig +0 -113
- package/src/zig/text-buffer-iterators.zig +0 -499
- package/src/zig/text-buffer-segment.zig +0 -404
- package/src/zig/text-buffer-view.zig +0 -1371
- package/src/zig/text-buffer.zig +0 -1180
- package/src/zig/utf8.zig +0 -1948
- package/src/zig/utils.zig +0 -9
- package/src/zig-structs.ts +0 -261
- package/src/zig.ts +0 -3884
- package/tsconfig.build.json +0 -24
- package/tsconfig.json +0 -27
- /package/{src/lib/tree-sitter/assets → assets}/javascript/highlights.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/markdown/highlights.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/markdown/injections.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/highlights.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/typescript/highlights.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/zig/highlights.scm +0 -0
- /package/{src/lib/tree-sitter/assets → assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -1,2062 +0,0 @@
|
|
|
1
|
-
import { test, expect, beforeEach, afterEach } from "bun:test"
|
|
2
|
-
import { CodeRenderable } from "./Code.js"
|
|
3
|
-
import { SyntaxStyle } from "../syntax-style.js"
|
|
4
|
-
import { RGBA } from "../lib/RGBA.js"
|
|
5
|
-
import { createTestRenderer, type TestRenderer, MockTreeSitterClient, type MockMouse } from "../testing.js"
|
|
6
|
-
import { TreeSitterClient } from "../lib/tree-sitter/index.js"
|
|
7
|
-
import type { SimpleHighlight } from "../lib/tree-sitter/types.js"
|
|
8
|
-
import { BoxRenderable } from "./Box.js"
|
|
9
|
-
|
|
10
|
-
let currentRenderer: TestRenderer
|
|
11
|
-
let renderOnce: () => Promise<void>
|
|
12
|
-
let captureFrame: () => string
|
|
13
|
-
let mockMouse: MockMouse
|
|
14
|
-
let resize: (width: number, height: number) => void
|
|
15
|
-
|
|
16
|
-
beforeEach(async () => {
|
|
17
|
-
const testRenderer = await createTestRenderer({ width: 80, height: 24 })
|
|
18
|
-
currentRenderer = testRenderer.renderer
|
|
19
|
-
renderOnce = testRenderer.renderOnce
|
|
20
|
-
captureFrame = testRenderer.captureCharFrame
|
|
21
|
-
mockMouse = testRenderer.mockMouse
|
|
22
|
-
resize = testRenderer.resize
|
|
23
|
-
})
|
|
24
|
-
|
|
25
|
-
afterEach(async () => {
|
|
26
|
-
if (currentRenderer) {
|
|
27
|
-
currentRenderer.destroy()
|
|
28
|
-
}
|
|
29
|
-
})
|
|
30
|
-
|
|
31
|
-
test("CodeRenderable - basic construction", async () => {
|
|
32
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
33
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
34
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
35
|
-
string: { fg: RGBA.fromValues(0, 1, 0, 1) },
|
|
36
|
-
})
|
|
37
|
-
|
|
38
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
39
|
-
id: "test-code",
|
|
40
|
-
content: 'const message = "Hello, world!";',
|
|
41
|
-
filetype: "javascript",
|
|
42
|
-
syntaxStyle,
|
|
43
|
-
conceal: false,
|
|
44
|
-
})
|
|
45
|
-
|
|
46
|
-
expect(codeRenderable.content).toBe('const message = "Hello, world!";')
|
|
47
|
-
expect(codeRenderable.filetype).toBe("javascript")
|
|
48
|
-
expect(codeRenderable.syntaxStyle).toBe(syntaxStyle)
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
test("CodeRenderable - content updates", async () => {
|
|
52
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
53
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
54
|
-
})
|
|
55
|
-
|
|
56
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
57
|
-
id: "test-code",
|
|
58
|
-
content: "original content",
|
|
59
|
-
filetype: "javascript",
|
|
60
|
-
syntaxStyle,
|
|
61
|
-
conceal: false,
|
|
62
|
-
})
|
|
63
|
-
|
|
64
|
-
expect(codeRenderable.content).toBe("original content")
|
|
65
|
-
|
|
66
|
-
codeRenderable.content = "updated content"
|
|
67
|
-
expect(codeRenderable.content).toBe("updated content")
|
|
68
|
-
})
|
|
69
|
-
|
|
70
|
-
test("CodeRenderable - filetype updates", async () => {
|
|
71
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
72
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
73
|
-
})
|
|
74
|
-
|
|
75
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
76
|
-
id: "test-code",
|
|
77
|
-
content: "console.log('test');",
|
|
78
|
-
filetype: "javascript",
|
|
79
|
-
syntaxStyle,
|
|
80
|
-
conceal: false,
|
|
81
|
-
})
|
|
82
|
-
|
|
83
|
-
expect(codeRenderable.filetype).toBe("javascript")
|
|
84
|
-
|
|
85
|
-
codeRenderable.filetype = "typescript"
|
|
86
|
-
expect(codeRenderable.filetype).toBe("typescript")
|
|
87
|
-
})
|
|
88
|
-
|
|
89
|
-
test("CodeRenderable - re-highlights when content changes during active highlighting", async () => {
|
|
90
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
91
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
92
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
93
|
-
})
|
|
94
|
-
|
|
95
|
-
const mockClient = new MockTreeSitterClient()
|
|
96
|
-
mockClient.setMockResult({
|
|
97
|
-
highlights: [
|
|
98
|
-
[0, 5, "keyword"],
|
|
99
|
-
[6, 13, "identifier"],
|
|
100
|
-
] as SimpleHighlight[],
|
|
101
|
-
})
|
|
102
|
-
|
|
103
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
104
|
-
id: "test-code",
|
|
105
|
-
content: "const message = 'hello';",
|
|
106
|
-
filetype: "javascript",
|
|
107
|
-
syntaxStyle,
|
|
108
|
-
treeSitterClient: mockClient,
|
|
109
|
-
conceal: false,
|
|
110
|
-
})
|
|
111
|
-
|
|
112
|
-
currentRenderer.root.add(codeRenderable)
|
|
113
|
-
await renderOnce()
|
|
114
|
-
|
|
115
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
116
|
-
|
|
117
|
-
codeRenderable.content = "let newMessage = 'world';"
|
|
118
|
-
|
|
119
|
-
expect(codeRenderable.content).toBe("let newMessage = 'world';")
|
|
120
|
-
|
|
121
|
-
await renderOnce()
|
|
122
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
123
|
-
|
|
124
|
-
mockClient.resolveHighlightOnce(0)
|
|
125
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
126
|
-
|
|
127
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
128
|
-
|
|
129
|
-
mockClient.resolveHighlightOnce(0)
|
|
130
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
131
|
-
|
|
132
|
-
expect(mockClient.isHighlighting()).toBe(false)
|
|
133
|
-
})
|
|
134
|
-
|
|
135
|
-
test("CodeRenderable - multiple content changes during highlighting", async () => {
|
|
136
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
137
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
const mockClient = new MockTreeSitterClient()
|
|
141
|
-
mockClient.setMockResult({ highlights: [] })
|
|
142
|
-
|
|
143
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
144
|
-
id: "test-code",
|
|
145
|
-
content: "original content",
|
|
146
|
-
filetype: "javascript",
|
|
147
|
-
syntaxStyle,
|
|
148
|
-
treeSitterClient: mockClient,
|
|
149
|
-
conceal: false,
|
|
150
|
-
})
|
|
151
|
-
|
|
152
|
-
currentRenderer.root.add(codeRenderable)
|
|
153
|
-
await renderOnce()
|
|
154
|
-
|
|
155
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
156
|
-
|
|
157
|
-
codeRenderable.content = "first change"
|
|
158
|
-
codeRenderable.content = "second change"
|
|
159
|
-
codeRenderable.content = "final content"
|
|
160
|
-
|
|
161
|
-
expect(codeRenderable.content).toBe("final content")
|
|
162
|
-
|
|
163
|
-
await renderOnce()
|
|
164
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
165
|
-
|
|
166
|
-
mockClient.resolveHighlightOnce(0)
|
|
167
|
-
|
|
168
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
169
|
-
|
|
170
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
171
|
-
|
|
172
|
-
mockClient.resolveHighlightOnce(0)
|
|
173
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
174
|
-
|
|
175
|
-
expect(mockClient.isHighlighting()).toBe(false)
|
|
176
|
-
})
|
|
177
|
-
|
|
178
|
-
test("CodeRenderable - uses fallback rendering when no filetype provided", async () => {
|
|
179
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
180
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
181
|
-
})
|
|
182
|
-
|
|
183
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
184
|
-
id: "test-code",
|
|
185
|
-
content: "const message = 'hello world';",
|
|
186
|
-
syntaxStyle,
|
|
187
|
-
conceal: false,
|
|
188
|
-
})
|
|
189
|
-
|
|
190
|
-
currentRenderer.root.add(codeRenderable)
|
|
191
|
-
await renderOnce()
|
|
192
|
-
|
|
193
|
-
expect(codeRenderable.content).toBe("const message = 'hello world';")
|
|
194
|
-
expect(codeRenderable.filetype).toBeUndefined()
|
|
195
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello world';")
|
|
196
|
-
})
|
|
197
|
-
|
|
198
|
-
test("CodeRenderable - uses fallback rendering when highlighting throws error", async () => {
|
|
199
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
200
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
201
|
-
})
|
|
202
|
-
|
|
203
|
-
const mockClient = new MockTreeSitterClient()
|
|
204
|
-
|
|
205
|
-
mockClient.highlightOnce = async () => {
|
|
206
|
-
throw new Error("Highlighting failed")
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
210
|
-
id: "test-code",
|
|
211
|
-
content: "const message = 'hello world';",
|
|
212
|
-
filetype: "javascript",
|
|
213
|
-
syntaxStyle,
|
|
214
|
-
treeSitterClient: mockClient,
|
|
215
|
-
conceal: false,
|
|
216
|
-
})
|
|
217
|
-
|
|
218
|
-
currentRenderer.root.add(codeRenderable)
|
|
219
|
-
await renderOnce()
|
|
220
|
-
|
|
221
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
222
|
-
await renderOnce()
|
|
223
|
-
|
|
224
|
-
expect(codeRenderable.content).toBe("const message = 'hello world';")
|
|
225
|
-
expect(codeRenderable.filetype).toBe("javascript")
|
|
226
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello world';")
|
|
227
|
-
})
|
|
228
|
-
|
|
229
|
-
test("CodeRenderable - handles empty content", async () => {
|
|
230
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
231
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
232
|
-
})
|
|
233
|
-
|
|
234
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
235
|
-
id: "test-code",
|
|
236
|
-
content: "",
|
|
237
|
-
filetype: "javascript",
|
|
238
|
-
syntaxStyle,
|
|
239
|
-
conceal: false,
|
|
240
|
-
})
|
|
241
|
-
|
|
242
|
-
await renderOnce()
|
|
243
|
-
|
|
244
|
-
expect(codeRenderable.content).toBe("")
|
|
245
|
-
expect(codeRenderable.filetype).toBe("javascript")
|
|
246
|
-
expect(codeRenderable.plainText).toBe("")
|
|
247
|
-
})
|
|
248
|
-
|
|
249
|
-
test("CodeRenderable - empty content does not trigger highlighting", async () => {
|
|
250
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
251
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
252
|
-
})
|
|
253
|
-
|
|
254
|
-
const mockClient = new MockTreeSitterClient()
|
|
255
|
-
mockClient.setMockResult({ highlights: [] })
|
|
256
|
-
|
|
257
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
258
|
-
id: "test-code",
|
|
259
|
-
content: "const message = 'hello';",
|
|
260
|
-
filetype: "javascript",
|
|
261
|
-
syntaxStyle,
|
|
262
|
-
treeSitterClient: mockClient,
|
|
263
|
-
conceal: false,
|
|
264
|
-
})
|
|
265
|
-
|
|
266
|
-
currentRenderer.root.add(codeRenderable)
|
|
267
|
-
await renderOnce()
|
|
268
|
-
|
|
269
|
-
mockClient.resolveHighlightOnce(0)
|
|
270
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
271
|
-
await renderOnce()
|
|
272
|
-
|
|
273
|
-
expect(codeRenderable.content).toBe("const message = 'hello';")
|
|
274
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
275
|
-
|
|
276
|
-
codeRenderable.content = ""
|
|
277
|
-
await renderOnce()
|
|
278
|
-
|
|
279
|
-
expect(mockClient.isHighlighting()).toBe(false)
|
|
280
|
-
expect(codeRenderable.content).toBe("")
|
|
281
|
-
})
|
|
282
|
-
|
|
283
|
-
test("CodeRenderable - text renders immediately before highlighting completes", async () => {
|
|
284
|
-
resize(32, 2)
|
|
285
|
-
|
|
286
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
287
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
288
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
289
|
-
})
|
|
290
|
-
|
|
291
|
-
const mockClient = new MockTreeSitterClient()
|
|
292
|
-
mockClient.setMockResult({
|
|
293
|
-
highlights: [
|
|
294
|
-
[0, 5, "keyword"],
|
|
295
|
-
[6, 13, "identifier"],
|
|
296
|
-
] as SimpleHighlight[],
|
|
297
|
-
})
|
|
298
|
-
|
|
299
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
300
|
-
id: "test-code",
|
|
301
|
-
content: "const message = 'hello world';",
|
|
302
|
-
filetype: "javascript",
|
|
303
|
-
syntaxStyle,
|
|
304
|
-
treeSitterClient: mockClient,
|
|
305
|
-
conceal: false,
|
|
306
|
-
left: 0,
|
|
307
|
-
top: 0,
|
|
308
|
-
})
|
|
309
|
-
|
|
310
|
-
currentRenderer.root.add(codeRenderable)
|
|
311
|
-
await renderOnce()
|
|
312
|
-
|
|
313
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
314
|
-
|
|
315
|
-
const frameBeforeHighlighting = captureFrame()
|
|
316
|
-
expect(frameBeforeHighlighting).toMatchSnapshot("text visible before highlighting completes")
|
|
317
|
-
|
|
318
|
-
mockClient.resolveHighlightOnce(0)
|
|
319
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
320
|
-
await renderOnce()
|
|
321
|
-
|
|
322
|
-
const frameAfterHighlighting = captureFrame()
|
|
323
|
-
expect(frameAfterHighlighting).toMatchSnapshot("text visible after highlighting completes")
|
|
324
|
-
})
|
|
325
|
-
|
|
326
|
-
test("CodeRenderable - batches concurrent content and filetype updates", async () => {
|
|
327
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
328
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
329
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
330
|
-
})
|
|
331
|
-
|
|
332
|
-
let highlightCount = 0
|
|
333
|
-
const mockClient = new MockTreeSitterClient()
|
|
334
|
-
const originalHighlightOnce = mockClient.highlightOnce.bind(mockClient)
|
|
335
|
-
|
|
336
|
-
mockClient.highlightOnce = async (content: string, filetype: string) => {
|
|
337
|
-
highlightCount++
|
|
338
|
-
return originalHighlightOnce(content, filetype)
|
|
339
|
-
}
|
|
340
|
-
|
|
341
|
-
mockClient.setMockResult({
|
|
342
|
-
highlights: [[0, 3, "keyword"]] as SimpleHighlight[],
|
|
343
|
-
})
|
|
344
|
-
|
|
345
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
346
|
-
id: "test-code",
|
|
347
|
-
content: "const message = 'hello';",
|
|
348
|
-
filetype: "javascript",
|
|
349
|
-
syntaxStyle,
|
|
350
|
-
treeSitterClient: mockClient,
|
|
351
|
-
conceal: false,
|
|
352
|
-
})
|
|
353
|
-
|
|
354
|
-
currentRenderer.root.add(codeRenderable)
|
|
355
|
-
await renderOnce()
|
|
356
|
-
|
|
357
|
-
mockClient.resolveHighlightOnce(0)
|
|
358
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
359
|
-
|
|
360
|
-
highlightCount = 0
|
|
361
|
-
|
|
362
|
-
codeRenderable.content = "let newMessage = 'world';"
|
|
363
|
-
codeRenderable.filetype = "typescript"
|
|
364
|
-
|
|
365
|
-
await renderOnce()
|
|
366
|
-
|
|
367
|
-
mockClient.resolveAllHighlightOnce()
|
|
368
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
369
|
-
|
|
370
|
-
expect(highlightCount).toBe(1)
|
|
371
|
-
expect(codeRenderable.content).toBe("let newMessage = 'world';")
|
|
372
|
-
expect(codeRenderable.filetype).toBe("typescript")
|
|
373
|
-
})
|
|
374
|
-
|
|
375
|
-
test("CodeRenderable - batches multiple updates in same tick into single highlight", async () => {
|
|
376
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
377
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
378
|
-
})
|
|
379
|
-
|
|
380
|
-
let highlightCount = 0
|
|
381
|
-
const highlightCalls: Array<{ content: string; filetype: string }> = []
|
|
382
|
-
const mockClient = new MockTreeSitterClient()
|
|
383
|
-
const originalHighlightOnce = mockClient.highlightOnce.bind(mockClient)
|
|
384
|
-
|
|
385
|
-
mockClient.highlightOnce = async (content: string, filetype: string) => {
|
|
386
|
-
highlightCount++
|
|
387
|
-
highlightCalls.push({ content, filetype })
|
|
388
|
-
return originalHighlightOnce(content, filetype)
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
mockClient.setMockResult({ highlights: [] })
|
|
392
|
-
|
|
393
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
394
|
-
id: "test-code",
|
|
395
|
-
content: "initial",
|
|
396
|
-
filetype: "javascript",
|
|
397
|
-
syntaxStyle,
|
|
398
|
-
treeSitterClient: mockClient,
|
|
399
|
-
conceal: false,
|
|
400
|
-
})
|
|
401
|
-
|
|
402
|
-
currentRenderer.root.add(codeRenderable)
|
|
403
|
-
await renderOnce()
|
|
404
|
-
|
|
405
|
-
mockClient.resolveHighlightOnce(0)
|
|
406
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
407
|
-
|
|
408
|
-
highlightCount = 0
|
|
409
|
-
highlightCalls.length = 0
|
|
410
|
-
|
|
411
|
-
codeRenderable.content = "first content change"
|
|
412
|
-
codeRenderable.filetype = "typescript"
|
|
413
|
-
codeRenderable.content = "second content change"
|
|
414
|
-
|
|
415
|
-
await renderOnce()
|
|
416
|
-
|
|
417
|
-
mockClient.resolveAllHighlightOnce()
|
|
418
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
419
|
-
|
|
420
|
-
expect(highlightCount).toBe(1)
|
|
421
|
-
expect(highlightCalls[0]?.content).toBe("second content change")
|
|
422
|
-
expect(highlightCalls[0]?.filetype).toBe("typescript")
|
|
423
|
-
})
|
|
424
|
-
|
|
425
|
-
test("CodeRenderable - renders markdown with TypeScript injection correctly", async () => {
|
|
426
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
427
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
428
|
-
keyword: { fg: RGBA.fromValues(1, 0, 0, 1) }, // Red
|
|
429
|
-
string: { fg: RGBA.fromValues(0, 1, 0, 1) }, // Green
|
|
430
|
-
"markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) }, // Blue
|
|
431
|
-
})
|
|
432
|
-
|
|
433
|
-
const markdownCode = `# Hello\n\n\`\`\`typescript\nconst msg: string = "hi";\n\`\`\``
|
|
434
|
-
|
|
435
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
436
|
-
id: "test-markdown",
|
|
437
|
-
content: markdownCode,
|
|
438
|
-
filetype: "markdown",
|
|
439
|
-
syntaxStyle,
|
|
440
|
-
conceal: false,
|
|
441
|
-
left: 0,
|
|
442
|
-
top: 0,
|
|
443
|
-
})
|
|
444
|
-
|
|
445
|
-
currentRenderer.root.add(codeRenderable)
|
|
446
|
-
await renderOnce()
|
|
447
|
-
|
|
448
|
-
await new Promise((resolve) => setTimeout(resolve, 100))
|
|
449
|
-
await renderOnce()
|
|
450
|
-
|
|
451
|
-
expect(codeRenderable.plainText).toContain("# Hello")
|
|
452
|
-
expect(codeRenderable.plainText).toContain("const msg")
|
|
453
|
-
expect(codeRenderable.plainText).toContain("typescript")
|
|
454
|
-
})
|
|
455
|
-
|
|
456
|
-
test("CodeRenderable - continues highlighting after unresolved promise", async () => {
|
|
457
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
458
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
459
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
460
|
-
})
|
|
461
|
-
|
|
462
|
-
let highlightCount = 0
|
|
463
|
-
const pendingPromises: Array<{ content: string; filetype: string; never: boolean }> = []
|
|
464
|
-
|
|
465
|
-
class HangingMockClient extends TreeSitterClient {
|
|
466
|
-
constructor() {
|
|
467
|
-
super({ dataPath: "/tmp/mock" })
|
|
468
|
-
}
|
|
469
|
-
|
|
470
|
-
override async highlightOnce(
|
|
471
|
-
content: string,
|
|
472
|
-
filetype: string,
|
|
473
|
-
): Promise<{ highlights?: SimpleHighlight[]; warning?: string; error?: string }> {
|
|
474
|
-
highlightCount++
|
|
475
|
-
|
|
476
|
-
const shouldHang = highlightCount === 4 && filetype === "typescript"
|
|
477
|
-
|
|
478
|
-
pendingPromises.push({ content, filetype, never: shouldHang })
|
|
479
|
-
|
|
480
|
-
if (shouldHang) {
|
|
481
|
-
return new Promise(() => {})
|
|
482
|
-
}
|
|
483
|
-
|
|
484
|
-
return Promise.resolve({ highlights: [] })
|
|
485
|
-
}
|
|
486
|
-
}
|
|
487
|
-
|
|
488
|
-
const mockClient = new HangingMockClient()
|
|
489
|
-
|
|
490
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
491
|
-
id: "test-code",
|
|
492
|
-
content: "interface User { name: string; }",
|
|
493
|
-
filetype: "typescript",
|
|
494
|
-
syntaxStyle,
|
|
495
|
-
treeSitterClient: mockClient,
|
|
496
|
-
conceal: false,
|
|
497
|
-
})
|
|
498
|
-
|
|
499
|
-
currentRenderer.root.add(codeRenderable)
|
|
500
|
-
await renderOnce()
|
|
501
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
502
|
-
|
|
503
|
-
highlightCount = 0
|
|
504
|
-
pendingPromises.length = 0
|
|
505
|
-
|
|
506
|
-
codeRenderable.content = "const message = 'hello';"
|
|
507
|
-
codeRenderable.filetype = "javascript"
|
|
508
|
-
await renderOnce()
|
|
509
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
510
|
-
|
|
511
|
-
codeRenderable.content = "# Documentation"
|
|
512
|
-
codeRenderable.filetype = "markdown"
|
|
513
|
-
await renderOnce()
|
|
514
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
515
|
-
|
|
516
|
-
codeRenderable.content = "const message = 'world';"
|
|
517
|
-
codeRenderable.filetype = "javascript"
|
|
518
|
-
await renderOnce()
|
|
519
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
520
|
-
|
|
521
|
-
codeRenderable.content = "interface User { name: string; }"
|
|
522
|
-
codeRenderable.filetype = "typescript"
|
|
523
|
-
await renderOnce()
|
|
524
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
525
|
-
|
|
526
|
-
codeRenderable.content = "# New Documentation"
|
|
527
|
-
codeRenderable.filetype = "markdown"
|
|
528
|
-
await renderOnce()
|
|
529
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
530
|
-
|
|
531
|
-
const markdownHighlightHappened = pendingPromises.some(
|
|
532
|
-
(p) => p.content === "# New Documentation" && p.filetype === "markdown",
|
|
533
|
-
)
|
|
534
|
-
|
|
535
|
-
expect(codeRenderable.content).toBe("# New Documentation")
|
|
536
|
-
expect(codeRenderable.filetype).toBe("markdown")
|
|
537
|
-
expect(markdownHighlightHappened).toBe(true)
|
|
538
|
-
expect(highlightCount).toBe(5)
|
|
539
|
-
})
|
|
540
|
-
|
|
541
|
-
test("CodeRenderable - concealment is enabled by default", async () => {
|
|
542
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
543
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
544
|
-
})
|
|
545
|
-
|
|
546
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
547
|
-
id: "test-code",
|
|
548
|
-
content: "const message = 'hello';",
|
|
549
|
-
filetype: "javascript",
|
|
550
|
-
syntaxStyle,
|
|
551
|
-
})
|
|
552
|
-
|
|
553
|
-
expect(codeRenderable.conceal).toBe(true)
|
|
554
|
-
})
|
|
555
|
-
|
|
556
|
-
test("CodeRenderable - concealment can be disabled explicitly", async () => {
|
|
557
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
558
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
559
|
-
})
|
|
560
|
-
|
|
561
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
562
|
-
id: "test-code",
|
|
563
|
-
content: "const message = 'hello';",
|
|
564
|
-
filetype: "javascript",
|
|
565
|
-
syntaxStyle,
|
|
566
|
-
conceal: false,
|
|
567
|
-
})
|
|
568
|
-
|
|
569
|
-
expect(codeRenderable.conceal).toBe(false)
|
|
570
|
-
})
|
|
571
|
-
|
|
572
|
-
test("CodeRenderable - applies concealment to styled text", async () => {
|
|
573
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
574
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
575
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
576
|
-
})
|
|
577
|
-
|
|
578
|
-
const mockClient = new MockTreeSitterClient()
|
|
579
|
-
mockClient.setMockResult({
|
|
580
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
581
|
-
})
|
|
582
|
-
|
|
583
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
584
|
-
id: "test-code",
|
|
585
|
-
content: "const message = 'hello';",
|
|
586
|
-
filetype: "javascript",
|
|
587
|
-
syntaxStyle,
|
|
588
|
-
treeSitterClient: mockClient,
|
|
589
|
-
conceal: true,
|
|
590
|
-
left: 0,
|
|
591
|
-
top: 0,
|
|
592
|
-
})
|
|
593
|
-
|
|
594
|
-
currentRenderer.root.add(codeRenderable)
|
|
595
|
-
|
|
596
|
-
expect(codeRenderable.conceal).toBe(true)
|
|
597
|
-
|
|
598
|
-
mockClient.resolveHighlightOnce(0)
|
|
599
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
600
|
-
await renderOnce()
|
|
601
|
-
|
|
602
|
-
expect(codeRenderable.content).toBe("const message = 'hello';")
|
|
603
|
-
})
|
|
604
|
-
|
|
605
|
-
test("CodeRenderable - updating conceal triggers re-highlighting", async () => {
|
|
606
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
607
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
608
|
-
})
|
|
609
|
-
|
|
610
|
-
const mockClient = new MockTreeSitterClient()
|
|
611
|
-
mockClient.setMockResult({ highlights: [] })
|
|
612
|
-
|
|
613
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
614
|
-
id: "test-code",
|
|
615
|
-
content: "const message = 'hello';",
|
|
616
|
-
filetype: "javascript",
|
|
617
|
-
syntaxStyle,
|
|
618
|
-
treeSitterClient: mockClient,
|
|
619
|
-
conceal: true,
|
|
620
|
-
})
|
|
621
|
-
|
|
622
|
-
currentRenderer.root.add(codeRenderable)
|
|
623
|
-
await renderOnce()
|
|
624
|
-
|
|
625
|
-
expect(codeRenderable.conceal).toBe(true)
|
|
626
|
-
|
|
627
|
-
mockClient.resolveHighlightOnce(0)
|
|
628
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
629
|
-
|
|
630
|
-
codeRenderable.conceal = false
|
|
631
|
-
expect(codeRenderable.conceal).toBe(false)
|
|
632
|
-
|
|
633
|
-
await renderOnce()
|
|
634
|
-
|
|
635
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
636
|
-
mockClient.resolveHighlightOnce(0)
|
|
637
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
638
|
-
})
|
|
639
|
-
|
|
640
|
-
test("CodeRenderable - drawUnstyledText is true by default", async () => {
|
|
641
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
642
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
643
|
-
})
|
|
644
|
-
|
|
645
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
646
|
-
id: "test-code",
|
|
647
|
-
content: "const message = 'hello';",
|
|
648
|
-
filetype: "javascript",
|
|
649
|
-
syntaxStyle,
|
|
650
|
-
})
|
|
651
|
-
|
|
652
|
-
expect(codeRenderable.drawUnstyledText).toBe(true)
|
|
653
|
-
})
|
|
654
|
-
|
|
655
|
-
test("CodeRenderable - drawUnstyledText can be set to false", async () => {
|
|
656
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
657
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
658
|
-
})
|
|
659
|
-
|
|
660
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
661
|
-
id: "test-code",
|
|
662
|
-
content: "const message = 'hello';",
|
|
663
|
-
filetype: "javascript",
|
|
664
|
-
syntaxStyle,
|
|
665
|
-
drawUnstyledText: false,
|
|
666
|
-
})
|
|
667
|
-
|
|
668
|
-
expect(codeRenderable.drawUnstyledText).toBe(false)
|
|
669
|
-
})
|
|
670
|
-
|
|
671
|
-
test("CodeRenderable - with drawUnstyledText=true, text renders before highlighting", async () => {
|
|
672
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
673
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
674
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
675
|
-
})
|
|
676
|
-
|
|
677
|
-
const mockClient = new MockTreeSitterClient()
|
|
678
|
-
mockClient.setMockResult({
|
|
679
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
680
|
-
})
|
|
681
|
-
|
|
682
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
683
|
-
id: "test-code",
|
|
684
|
-
content: "const message = 'hello';",
|
|
685
|
-
filetype: "javascript",
|
|
686
|
-
syntaxStyle,
|
|
687
|
-
treeSitterClient: mockClient,
|
|
688
|
-
drawUnstyledText: true,
|
|
689
|
-
left: 0,
|
|
690
|
-
top: 0,
|
|
691
|
-
})
|
|
692
|
-
|
|
693
|
-
currentRenderer.root.add(codeRenderable)
|
|
694
|
-
await renderOnce()
|
|
695
|
-
|
|
696
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
697
|
-
|
|
698
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
699
|
-
|
|
700
|
-
mockClient.resolveHighlightOnce(0)
|
|
701
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
702
|
-
await renderOnce()
|
|
703
|
-
|
|
704
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
705
|
-
})
|
|
706
|
-
|
|
707
|
-
test("CodeRenderable - with drawUnstyledText=false, text does not render before highlighting but lineCount is correct", async () => {
|
|
708
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
709
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
710
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
711
|
-
})
|
|
712
|
-
|
|
713
|
-
const mockClient = new MockTreeSitterClient()
|
|
714
|
-
mockClient.setMockResult({
|
|
715
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
716
|
-
})
|
|
717
|
-
|
|
718
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
719
|
-
id: "test-code",
|
|
720
|
-
content: "const message = 'hello';",
|
|
721
|
-
filetype: "javascript",
|
|
722
|
-
syntaxStyle,
|
|
723
|
-
treeSitterClient: mockClient,
|
|
724
|
-
drawUnstyledText: false,
|
|
725
|
-
left: 0,
|
|
726
|
-
top: 0,
|
|
727
|
-
})
|
|
728
|
-
|
|
729
|
-
currentRenderer.root.add(codeRenderable)
|
|
730
|
-
await renderOnce()
|
|
731
|
-
|
|
732
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
733
|
-
|
|
734
|
-
// Text buffer has content (for lineCount), but nothing renders yet
|
|
735
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
736
|
-
expect(codeRenderable.lineCount).toBe(1)
|
|
737
|
-
const frameBeforeHighlighting = captureFrame()
|
|
738
|
-
expect(frameBeforeHighlighting.trim()).toBe("")
|
|
739
|
-
|
|
740
|
-
mockClient.resolveHighlightOnce(0)
|
|
741
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
742
|
-
await renderOnce()
|
|
743
|
-
|
|
744
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
745
|
-
const frameAfterHighlighting = captureFrame()
|
|
746
|
-
expect(frameAfterHighlighting).toContain("const message")
|
|
747
|
-
})
|
|
748
|
-
|
|
749
|
-
test("CodeRenderable - updating drawUnstyledText from false to true triggers re-highlighting", async () => {
|
|
750
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
751
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
752
|
-
})
|
|
753
|
-
|
|
754
|
-
const mockClient = new MockTreeSitterClient()
|
|
755
|
-
mockClient.setMockResult({ highlights: [] })
|
|
756
|
-
|
|
757
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
758
|
-
id: "test-code",
|
|
759
|
-
content: "const message = 'hello';",
|
|
760
|
-
filetype: "javascript",
|
|
761
|
-
syntaxStyle,
|
|
762
|
-
treeSitterClient: mockClient,
|
|
763
|
-
drawUnstyledText: false,
|
|
764
|
-
left: 0,
|
|
765
|
-
top: 0,
|
|
766
|
-
})
|
|
767
|
-
|
|
768
|
-
currentRenderer.root.add(codeRenderable)
|
|
769
|
-
|
|
770
|
-
expect(codeRenderable.drawUnstyledText).toBe(false)
|
|
771
|
-
|
|
772
|
-
await renderOnce()
|
|
773
|
-
// Text buffer has content for lineCount, but we can verify nothing renders
|
|
774
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
775
|
-
expect(codeRenderable.lineCount).toBe(1)
|
|
776
|
-
|
|
777
|
-
mockClient.resolveHighlightOnce(0)
|
|
778
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
779
|
-
|
|
780
|
-
codeRenderable.drawUnstyledText = true
|
|
781
|
-
expect(codeRenderable.drawUnstyledText).toBe(true)
|
|
782
|
-
|
|
783
|
-
await renderOnce()
|
|
784
|
-
|
|
785
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
786
|
-
|
|
787
|
-
mockClient.resolveHighlightOnce(0)
|
|
788
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
789
|
-
await renderOnce()
|
|
790
|
-
|
|
791
|
-
expect(mockClient.isHighlighting()).toBe(false)
|
|
792
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
793
|
-
})
|
|
794
|
-
|
|
795
|
-
test("CodeRenderable - updating drawUnstyledText from true to false triggers re-highlighting", async () => {
|
|
796
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
797
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
798
|
-
})
|
|
799
|
-
|
|
800
|
-
const mockClient = new MockTreeSitterClient()
|
|
801
|
-
mockClient.setMockResult({ highlights: [] })
|
|
802
|
-
|
|
803
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
804
|
-
id: "test-code",
|
|
805
|
-
content: "const message = 'hello';",
|
|
806
|
-
filetype: "javascript",
|
|
807
|
-
syntaxStyle,
|
|
808
|
-
treeSitterClient: mockClient,
|
|
809
|
-
drawUnstyledText: true,
|
|
810
|
-
})
|
|
811
|
-
|
|
812
|
-
currentRenderer.root.add(codeRenderable)
|
|
813
|
-
await renderOnce()
|
|
814
|
-
|
|
815
|
-
expect(codeRenderable.drawUnstyledText).toBe(true)
|
|
816
|
-
|
|
817
|
-
mockClient.resolveHighlightOnce(0)
|
|
818
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
819
|
-
|
|
820
|
-
codeRenderable.drawUnstyledText = false
|
|
821
|
-
expect(codeRenderable.drawUnstyledText).toBe(false)
|
|
822
|
-
|
|
823
|
-
await renderOnce()
|
|
824
|
-
|
|
825
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
826
|
-
mockClient.resolveHighlightOnce(0)
|
|
827
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
828
|
-
})
|
|
829
|
-
|
|
830
|
-
test("CodeRenderable - uses fallback rendering on error even with drawUnstyledText=false", async () => {
|
|
831
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
832
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
833
|
-
})
|
|
834
|
-
|
|
835
|
-
const mockClient = new MockTreeSitterClient()
|
|
836
|
-
|
|
837
|
-
mockClient.highlightOnce = async () => {
|
|
838
|
-
throw new Error("Highlighting failed")
|
|
839
|
-
}
|
|
840
|
-
|
|
841
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
842
|
-
id: "test-code",
|
|
843
|
-
content: "const message = 'hello world';",
|
|
844
|
-
filetype: "javascript",
|
|
845
|
-
syntaxStyle,
|
|
846
|
-
treeSitterClient: mockClient,
|
|
847
|
-
drawUnstyledText: false,
|
|
848
|
-
left: 0,
|
|
849
|
-
top: 0,
|
|
850
|
-
})
|
|
851
|
-
|
|
852
|
-
currentRenderer.root.add(codeRenderable)
|
|
853
|
-
|
|
854
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
855
|
-
await renderOnce()
|
|
856
|
-
|
|
857
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello world';")
|
|
858
|
-
})
|
|
859
|
-
|
|
860
|
-
test("CodeRenderable - with drawUnstyledText=false and no filetype, fallback is used", async () => {
|
|
861
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
862
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
863
|
-
})
|
|
864
|
-
|
|
865
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
866
|
-
id: "test-code",
|
|
867
|
-
content: "const message = 'hello world';",
|
|
868
|
-
syntaxStyle,
|
|
869
|
-
drawUnstyledText: false,
|
|
870
|
-
left: 0,
|
|
871
|
-
top: 0,
|
|
872
|
-
})
|
|
873
|
-
|
|
874
|
-
currentRenderer.root.add(codeRenderable)
|
|
875
|
-
|
|
876
|
-
await renderOnce()
|
|
877
|
-
|
|
878
|
-
expect(codeRenderable.filetype).toBeUndefined()
|
|
879
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello world';")
|
|
880
|
-
})
|
|
881
|
-
|
|
882
|
-
test("CodeRenderable - with drawUnstyledText=false, multiple updates only render final highlighted text", async () => {
|
|
883
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
884
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
885
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
886
|
-
})
|
|
887
|
-
|
|
888
|
-
const mockClient = new MockTreeSitterClient()
|
|
889
|
-
mockClient.setMockResult({
|
|
890
|
-
highlights: [[0, 3, "keyword"]] as SimpleHighlight[],
|
|
891
|
-
})
|
|
892
|
-
|
|
893
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
894
|
-
id: "test-code",
|
|
895
|
-
content: "const message = 'hello';",
|
|
896
|
-
filetype: "javascript",
|
|
897
|
-
syntaxStyle,
|
|
898
|
-
treeSitterClient: mockClient,
|
|
899
|
-
drawUnstyledText: false,
|
|
900
|
-
left: 0,
|
|
901
|
-
top: 0,
|
|
902
|
-
})
|
|
903
|
-
|
|
904
|
-
currentRenderer.root.add(codeRenderable)
|
|
905
|
-
await renderOnce()
|
|
906
|
-
|
|
907
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
908
|
-
|
|
909
|
-
// Text buffer has content (for lineCount), but nothing renders yet
|
|
910
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
911
|
-
expect(codeRenderable.lineCount).toBe(1)
|
|
912
|
-
const frameBeforeHighlighting = captureFrame()
|
|
913
|
-
expect(frameBeforeHighlighting.trim()).toBe("")
|
|
914
|
-
|
|
915
|
-
codeRenderable.content = "let newMessage = 'world';"
|
|
916
|
-
await renderOnce()
|
|
917
|
-
|
|
918
|
-
// Text buffer updated immediately, but still no rendering
|
|
919
|
-
expect(codeRenderable.plainText).toBe("let newMessage = 'world';")
|
|
920
|
-
expect(codeRenderable.lineCount).toBe(1)
|
|
921
|
-
const frameAfterUpdate = captureFrame()
|
|
922
|
-
expect(frameAfterUpdate.trim()).toBe("")
|
|
923
|
-
|
|
924
|
-
mockClient.resolveAllHighlightOnce()
|
|
925
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
926
|
-
await renderOnce()
|
|
927
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
928
|
-
|
|
929
|
-
expect(mockClient.isHighlighting()).toBe(false)
|
|
930
|
-
expect(codeRenderable.plainText).toBe("let newMessage = 'world';")
|
|
931
|
-
const frameAfterHighlighting = captureFrame()
|
|
932
|
-
expect(frameAfterHighlighting).toContain("let newMessage")
|
|
933
|
-
})
|
|
934
|
-
|
|
935
|
-
// TODO: flaky in CI because it needs to finish in time
|
|
936
|
-
// lib/tree-sitter/client.ts needs a way to check if the queue is empty
|
|
937
|
-
// then this can wait for all tree-sitter operations to complete
|
|
938
|
-
// instead of the arbitrary 500ms wait
|
|
939
|
-
// it worked before because text was set anyway for drawUnstyledText=false
|
|
940
|
-
test.skip("CodeRenderable - simulates markdown stream from LLM with async updates", async () => {
|
|
941
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
942
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
943
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
944
|
-
string: { fg: RGBA.fromValues(0, 1, 0, 1) },
|
|
945
|
-
"markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
946
|
-
})
|
|
947
|
-
|
|
948
|
-
// Base markdown content that we'll repeat to grow to ~1MB
|
|
949
|
-
const baseMarkdownContent = `# Code Example
|
|
950
|
-
|
|
951
|
-
Here's a simple TypeScript function:
|
|
952
|
-
|
|
953
|
-
\`\`\`typescript
|
|
954
|
-
function greet(name: string): string {
|
|
955
|
-
return \`Hello, \${name}!\`;
|
|
956
|
-
}
|
|
957
|
-
|
|
958
|
-
const message = greet("World");
|
|
959
|
-
console.log(message);
|
|
960
|
-
\`\`\`
|
|
961
|
-
`
|
|
962
|
-
|
|
963
|
-
const targetSize = 64 * 128
|
|
964
|
-
let fullMarkdownContent = ""
|
|
965
|
-
let iteration = 0
|
|
966
|
-
while (fullMarkdownContent.length < targetSize) {
|
|
967
|
-
fullMarkdownContent += `\n--- Iteration ${iteration} ---\n\n` + baseMarkdownContent
|
|
968
|
-
iteration++
|
|
969
|
-
}
|
|
970
|
-
|
|
971
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
972
|
-
id: "test-markdown-stream",
|
|
973
|
-
content: "",
|
|
974
|
-
filetype: "markdown",
|
|
975
|
-
syntaxStyle,
|
|
976
|
-
conceal: false,
|
|
977
|
-
left: 0,
|
|
978
|
-
top: 0,
|
|
979
|
-
drawUnstyledText: false,
|
|
980
|
-
})
|
|
981
|
-
await codeRenderable.treeSitterClient.initialize()
|
|
982
|
-
await codeRenderable.treeSitterClient.preloadParser("markdown")
|
|
983
|
-
|
|
984
|
-
currentRenderer.root.add(codeRenderable)
|
|
985
|
-
currentRenderer.start()
|
|
986
|
-
|
|
987
|
-
let currentContent = ""
|
|
988
|
-
|
|
989
|
-
const chunkSize = 64
|
|
990
|
-
const chunks: string[] = []
|
|
991
|
-
for (let i = 0; i < fullMarkdownContent.length; i += chunkSize) {
|
|
992
|
-
chunks.push(fullMarkdownContent.slice(i, Math.min(i + chunkSize, fullMarkdownContent.length)))
|
|
993
|
-
}
|
|
994
|
-
|
|
995
|
-
for (let i = 0; i < chunks.length; i++) {
|
|
996
|
-
const chunk = chunks[i]
|
|
997
|
-
currentContent += chunk
|
|
998
|
-
codeRenderable.content = currentContent
|
|
999
|
-
await new Promise((resolve) => setTimeout(resolve, Math.floor(Math.random() * 25) + 1))
|
|
1000
|
-
}
|
|
1001
|
-
|
|
1002
|
-
// wait for highlighting to complete (long for slow machines/CI)
|
|
1003
|
-
await new Promise((resolve) => setTimeout(resolve, 500))
|
|
1004
|
-
|
|
1005
|
-
expect(codeRenderable.content).toBe(fullMarkdownContent)
|
|
1006
|
-
expect(codeRenderable.content.length).toBeGreaterThanOrEqual(targetSize)
|
|
1007
|
-
expect(codeRenderable.plainText).toContain("# Code Example")
|
|
1008
|
-
expect(codeRenderable.plainText).toContain("function greet")
|
|
1009
|
-
expect(codeRenderable.plainText).toContain("typescript")
|
|
1010
|
-
expect(codeRenderable.plainText).toContain("Hello")
|
|
1011
|
-
|
|
1012
|
-
const plainText = codeRenderable.plainText
|
|
1013
|
-
expect(plainText.length).toBeGreaterThan(targetSize * 0.9)
|
|
1014
|
-
expect(plainText).toContain("Code Example")
|
|
1015
|
-
expect(plainText).toContain("const message = greet")
|
|
1016
|
-
})
|
|
1017
|
-
|
|
1018
|
-
test("CodeRenderable - streaming option is false by default", async () => {
|
|
1019
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1020
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1021
|
-
})
|
|
1022
|
-
|
|
1023
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1024
|
-
id: "test-code",
|
|
1025
|
-
content: "const message = 'hello';",
|
|
1026
|
-
filetype: "javascript",
|
|
1027
|
-
syntaxStyle,
|
|
1028
|
-
})
|
|
1029
|
-
|
|
1030
|
-
expect(codeRenderable.streaming).toBe(false)
|
|
1031
|
-
})
|
|
1032
|
-
|
|
1033
|
-
test("CodeRenderable - streaming can be enabled", async () => {
|
|
1034
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1035
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1036
|
-
})
|
|
1037
|
-
|
|
1038
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1039
|
-
id: "test-code",
|
|
1040
|
-
content: "const message = 'hello';",
|
|
1041
|
-
filetype: "javascript",
|
|
1042
|
-
syntaxStyle,
|
|
1043
|
-
streaming: true,
|
|
1044
|
-
})
|
|
1045
|
-
|
|
1046
|
-
expect(codeRenderable.streaming).toBe(true)
|
|
1047
|
-
})
|
|
1048
|
-
|
|
1049
|
-
test("CodeRenderable - streaming mode respects drawUnstyledText only for initial content", async () => {
|
|
1050
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1051
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1052
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1053
|
-
})
|
|
1054
|
-
|
|
1055
|
-
const mockClient = new MockTreeSitterClient()
|
|
1056
|
-
mockClient.setMockResult({
|
|
1057
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1058
|
-
})
|
|
1059
|
-
|
|
1060
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1061
|
-
id: "test-code",
|
|
1062
|
-
content: "const initial = 'hello';",
|
|
1063
|
-
filetype: "javascript",
|
|
1064
|
-
syntaxStyle,
|
|
1065
|
-
treeSitterClient: mockClient,
|
|
1066
|
-
streaming: true,
|
|
1067
|
-
drawUnstyledText: true,
|
|
1068
|
-
left: 0,
|
|
1069
|
-
top: 0,
|
|
1070
|
-
})
|
|
1071
|
-
|
|
1072
|
-
currentRenderer.root.add(codeRenderable)
|
|
1073
|
-
|
|
1074
|
-
await renderOnce()
|
|
1075
|
-
expect(codeRenderable.plainText).toBe("const initial = 'hello';")
|
|
1076
|
-
|
|
1077
|
-
mockClient.resolveHighlightOnce(0)
|
|
1078
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1079
|
-
|
|
1080
|
-
codeRenderable.content = "const updated = 'world';"
|
|
1081
|
-
await new Promise((resolve) => queueMicrotask(resolve))
|
|
1082
|
-
|
|
1083
|
-
expect(codeRenderable.content).toBe("const updated = 'world';")
|
|
1084
|
-
})
|
|
1085
|
-
|
|
1086
|
-
test("CodeRenderable - streaming mode with drawUnstyledText=false waits for new highlights", async () => {
|
|
1087
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1088
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1089
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1090
|
-
})
|
|
1091
|
-
|
|
1092
|
-
const mockClient = new MockTreeSitterClient({ autoResolveTimeout: 10 })
|
|
1093
|
-
mockClient.setMockResult({
|
|
1094
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1095
|
-
})
|
|
1096
|
-
|
|
1097
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1098
|
-
id: "test-code",
|
|
1099
|
-
content: "const initial = 'hello';",
|
|
1100
|
-
filetype: "javascript",
|
|
1101
|
-
syntaxStyle,
|
|
1102
|
-
treeSitterClient: mockClient,
|
|
1103
|
-
streaming: true,
|
|
1104
|
-
drawUnstyledText: false,
|
|
1105
|
-
left: 0,
|
|
1106
|
-
top: 0,
|
|
1107
|
-
})
|
|
1108
|
-
|
|
1109
|
-
currentRenderer.root.add(codeRenderable)
|
|
1110
|
-
currentRenderer.start()
|
|
1111
|
-
|
|
1112
|
-
await Bun.sleep(30)
|
|
1113
|
-
|
|
1114
|
-
expect(codeRenderable.plainText).toBe("const initial = 'hello';")
|
|
1115
|
-
|
|
1116
|
-
codeRenderable.content = "const updated = 'world';"
|
|
1117
|
-
expect(codeRenderable.plainText).toBe("const initial = 'hello';")
|
|
1118
|
-
|
|
1119
|
-
await Bun.sleep(30)
|
|
1120
|
-
|
|
1121
|
-
expect(codeRenderable.plainText).toBe("const updated = 'world';")
|
|
1122
|
-
|
|
1123
|
-
currentRenderer.stop()
|
|
1124
|
-
})
|
|
1125
|
-
|
|
1126
|
-
test("CodeRenderable - onChunks callback can transform chunks when highlights are empty", async () => {
|
|
1127
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1128
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1129
|
-
})
|
|
1130
|
-
|
|
1131
|
-
const mockClient = new MockTreeSitterClient()
|
|
1132
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1133
|
-
|
|
1134
|
-
let callbackInvoked = false
|
|
1135
|
-
|
|
1136
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1137
|
-
id: "test-code",
|
|
1138
|
-
content: "hello",
|
|
1139
|
-
filetype: "plaintext",
|
|
1140
|
-
syntaxStyle,
|
|
1141
|
-
treeSitterClient: mockClient,
|
|
1142
|
-
onChunks: (chunks) => {
|
|
1143
|
-
callbackInvoked = true
|
|
1144
|
-
return chunks.map((chunk) => ({
|
|
1145
|
-
...chunk,
|
|
1146
|
-
text: chunk.text.toUpperCase(),
|
|
1147
|
-
}))
|
|
1148
|
-
},
|
|
1149
|
-
})
|
|
1150
|
-
|
|
1151
|
-
currentRenderer.root.add(codeRenderable)
|
|
1152
|
-
await renderOnce()
|
|
1153
|
-
|
|
1154
|
-
mockClient.resolveHighlightOnce(0)
|
|
1155
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1156
|
-
await renderOnce()
|
|
1157
|
-
|
|
1158
|
-
expect(callbackInvoked).toBe(true)
|
|
1159
|
-
expect(codeRenderable.plainText).toBe("HELLO")
|
|
1160
|
-
})
|
|
1161
|
-
|
|
1162
|
-
test("CodeRenderable - onHighlight callback receives highlights and context", async () => {
|
|
1163
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1164
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1165
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1166
|
-
})
|
|
1167
|
-
|
|
1168
|
-
const mockClient = new MockTreeSitterClient()
|
|
1169
|
-
mockClient.setMockResult({
|
|
1170
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1171
|
-
})
|
|
1172
|
-
|
|
1173
|
-
let callbackInvoked = false
|
|
1174
|
-
let receivedHighlights: SimpleHighlight[] | null = null
|
|
1175
|
-
let receivedContext: { content: string; filetype: string | undefined; syntaxStyle: SyntaxStyle } | null = null
|
|
1176
|
-
|
|
1177
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1178
|
-
id: "test-code",
|
|
1179
|
-
content: "const message = 'hello';",
|
|
1180
|
-
filetype: "javascript",
|
|
1181
|
-
syntaxStyle,
|
|
1182
|
-
treeSitterClient: mockClient,
|
|
1183
|
-
onHighlight: (highlights, context) => {
|
|
1184
|
-
callbackInvoked = true
|
|
1185
|
-
receivedHighlights = [...highlights]
|
|
1186
|
-
receivedContext = { ...context }
|
|
1187
|
-
return highlights
|
|
1188
|
-
},
|
|
1189
|
-
})
|
|
1190
|
-
|
|
1191
|
-
currentRenderer.root.add(codeRenderable)
|
|
1192
|
-
await renderOnce()
|
|
1193
|
-
|
|
1194
|
-
mockClient.resolveHighlightOnce(0)
|
|
1195
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1196
|
-
await renderOnce()
|
|
1197
|
-
|
|
1198
|
-
expect(callbackInvoked).toBe(true)
|
|
1199
|
-
expect(receivedHighlights).not.toBeNull()
|
|
1200
|
-
expect(receivedHighlights?.length).toBe(1)
|
|
1201
|
-
expect(receivedHighlights?.[0]).toEqual([0, 5, "keyword"])
|
|
1202
|
-
expect(receivedContext?.content).toBe("const message = 'hello';")
|
|
1203
|
-
expect(receivedContext?.filetype).toBe("javascript")
|
|
1204
|
-
expect(receivedContext?.syntaxStyle).toBe(syntaxStyle)
|
|
1205
|
-
})
|
|
1206
|
-
|
|
1207
|
-
test("CodeRenderable - onHighlight callback can add custom highlights", async () => {
|
|
1208
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1209
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1210
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1211
|
-
"custom.highlight": { fg: RGBA.fromValues(1, 0, 0, 1) },
|
|
1212
|
-
})
|
|
1213
|
-
|
|
1214
|
-
const mockClient = new MockTreeSitterClient()
|
|
1215
|
-
mockClient.setMockResult({
|
|
1216
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1217
|
-
})
|
|
1218
|
-
|
|
1219
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1220
|
-
id: "test-code",
|
|
1221
|
-
content: "const message = 'hello';",
|
|
1222
|
-
filetype: "javascript",
|
|
1223
|
-
syntaxStyle,
|
|
1224
|
-
treeSitterClient: mockClient,
|
|
1225
|
-
onHighlight: (highlights) => {
|
|
1226
|
-
highlights.push([6, 13, "custom.highlight", {}])
|
|
1227
|
-
return highlights
|
|
1228
|
-
},
|
|
1229
|
-
})
|
|
1230
|
-
|
|
1231
|
-
currentRenderer.root.add(codeRenderable)
|
|
1232
|
-
await renderOnce()
|
|
1233
|
-
|
|
1234
|
-
mockClient.resolveHighlightOnce(0)
|
|
1235
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1236
|
-
await renderOnce()
|
|
1237
|
-
|
|
1238
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
1239
|
-
|
|
1240
|
-
// Verify both the original keyword highlight and the custom highlight are applied
|
|
1241
|
-
const lineHighlights = codeRenderable.getLineHighlights(0)
|
|
1242
|
-
expect(lineHighlights.length).toBeGreaterThanOrEqual(2)
|
|
1243
|
-
|
|
1244
|
-
// Check keyword highlight exists with the correct styleId
|
|
1245
|
-
const keywordStyleId = syntaxStyle.getStyleId("keyword")
|
|
1246
|
-
const keywordHighlight = lineHighlights.find((h) => h.styleId === keywordStyleId)
|
|
1247
|
-
expect(keywordHighlight).toBeDefined()
|
|
1248
|
-
|
|
1249
|
-
// Check custom highlight exists with the correct styleId
|
|
1250
|
-
const customStyleId = syntaxStyle.getStyleId("custom.highlight")
|
|
1251
|
-
const customHighlight = lineHighlights.find((h) => h.styleId === customStyleId)
|
|
1252
|
-
expect(customHighlight).toBeDefined()
|
|
1253
|
-
})
|
|
1254
|
-
|
|
1255
|
-
test("CodeRenderable - onHighlight callback returning undefined uses original highlights", async () => {
|
|
1256
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1257
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1258
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1259
|
-
})
|
|
1260
|
-
|
|
1261
|
-
const mockClient = new MockTreeSitterClient()
|
|
1262
|
-
mockClient.setMockResult({
|
|
1263
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1264
|
-
})
|
|
1265
|
-
|
|
1266
|
-
let callbackInvoked = false
|
|
1267
|
-
|
|
1268
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1269
|
-
id: "test-code",
|
|
1270
|
-
content: "const message = 'hello';",
|
|
1271
|
-
filetype: "javascript",
|
|
1272
|
-
syntaxStyle,
|
|
1273
|
-
treeSitterClient: mockClient,
|
|
1274
|
-
onHighlight: (highlights) => {
|
|
1275
|
-
callbackInvoked = true
|
|
1276
|
-
return undefined as unknown as SimpleHighlight[]
|
|
1277
|
-
},
|
|
1278
|
-
})
|
|
1279
|
-
|
|
1280
|
-
currentRenderer.root.add(codeRenderable)
|
|
1281
|
-
await renderOnce()
|
|
1282
|
-
|
|
1283
|
-
mockClient.resolveHighlightOnce(0)
|
|
1284
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1285
|
-
await renderOnce()
|
|
1286
|
-
|
|
1287
|
-
expect(callbackInvoked).toBe(true)
|
|
1288
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
1289
|
-
})
|
|
1290
|
-
|
|
1291
|
-
test("CodeRenderable - onHighlight callback is called on re-highlighting when content changes", async () => {
|
|
1292
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1293
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1294
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1295
|
-
})
|
|
1296
|
-
|
|
1297
|
-
const mockClient = new MockTreeSitterClient()
|
|
1298
|
-
mockClient.setMockResult({
|
|
1299
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1300
|
-
})
|
|
1301
|
-
|
|
1302
|
-
let callbackCount = 0
|
|
1303
|
-
|
|
1304
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1305
|
-
id: "test-code",
|
|
1306
|
-
content: "const message = 'hello';",
|
|
1307
|
-
filetype: "javascript",
|
|
1308
|
-
syntaxStyle,
|
|
1309
|
-
treeSitterClient: mockClient,
|
|
1310
|
-
onHighlight: (highlights) => {
|
|
1311
|
-
callbackCount++
|
|
1312
|
-
return highlights
|
|
1313
|
-
},
|
|
1314
|
-
})
|
|
1315
|
-
|
|
1316
|
-
currentRenderer.root.add(codeRenderable)
|
|
1317
|
-
await renderOnce()
|
|
1318
|
-
|
|
1319
|
-
mockClient.resolveHighlightOnce(0)
|
|
1320
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1321
|
-
await renderOnce()
|
|
1322
|
-
|
|
1323
|
-
expect(callbackCount).toBe(1)
|
|
1324
|
-
|
|
1325
|
-
codeRenderable.content = "let newMessage = 'world';"
|
|
1326
|
-
await renderOnce()
|
|
1327
|
-
|
|
1328
|
-
mockClient.resolveHighlightOnce(0)
|
|
1329
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1330
|
-
await renderOnce()
|
|
1331
|
-
|
|
1332
|
-
expect(callbackCount).toBe(2)
|
|
1333
|
-
})
|
|
1334
|
-
|
|
1335
|
-
test("CodeRenderable - onHighlight callback supports async functions", async () => {
|
|
1336
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1337
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1338
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1339
|
-
"async.highlight": { fg: RGBA.fromValues(0, 1, 0, 1) },
|
|
1340
|
-
})
|
|
1341
|
-
|
|
1342
|
-
const mockClient = new MockTreeSitterClient()
|
|
1343
|
-
mockClient.setMockResult({
|
|
1344
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1345
|
-
})
|
|
1346
|
-
|
|
1347
|
-
let asyncCallbackCompleted = false
|
|
1348
|
-
|
|
1349
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1350
|
-
id: "test-code",
|
|
1351
|
-
content: "const message = 'hello';",
|
|
1352
|
-
filetype: "javascript",
|
|
1353
|
-
syntaxStyle,
|
|
1354
|
-
treeSitterClient: mockClient,
|
|
1355
|
-
onHighlight: async (highlights) => {
|
|
1356
|
-
// Simulate async operation (e.g., fetching additional highlight data)
|
|
1357
|
-
await new Promise((resolve) => setTimeout(resolve, 5))
|
|
1358
|
-
highlights.push([6, 13, "async.highlight", {}])
|
|
1359
|
-
asyncCallbackCompleted = true
|
|
1360
|
-
return highlights
|
|
1361
|
-
},
|
|
1362
|
-
})
|
|
1363
|
-
|
|
1364
|
-
currentRenderer.root.add(codeRenderable)
|
|
1365
|
-
await renderOnce()
|
|
1366
|
-
|
|
1367
|
-
mockClient.resolveHighlightOnce(0)
|
|
1368
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
1369
|
-
await renderOnce()
|
|
1370
|
-
|
|
1371
|
-
expect(asyncCallbackCompleted).toBe(true)
|
|
1372
|
-
expect(codeRenderable.plainText).toBe("const message = 'hello';")
|
|
1373
|
-
|
|
1374
|
-
// Verify the async highlight was applied
|
|
1375
|
-
const lineHighlights = codeRenderable.getLineHighlights(0)
|
|
1376
|
-
expect(lineHighlights.length).toBeGreaterThanOrEqual(2)
|
|
1377
|
-
|
|
1378
|
-
const asyncStyleId = syntaxStyle.getStyleId("async.highlight")
|
|
1379
|
-
const asyncHighlight = lineHighlights.find((h) => h.styleId === asyncStyleId && h.start === 6 && h.end === 13)
|
|
1380
|
-
expect(asyncHighlight).toBeDefined()
|
|
1381
|
-
})
|
|
1382
|
-
|
|
1383
|
-
test("CodeRenderable - streaming mode caches highlights between updates", async () => {
|
|
1384
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1385
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1386
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1387
|
-
})
|
|
1388
|
-
|
|
1389
|
-
const mockClient = new MockTreeSitterClient()
|
|
1390
|
-
mockClient.setMockResult({
|
|
1391
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1392
|
-
})
|
|
1393
|
-
|
|
1394
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1395
|
-
id: "test-code",
|
|
1396
|
-
content: "const initial = 'hello';",
|
|
1397
|
-
filetype: "javascript",
|
|
1398
|
-
syntaxStyle,
|
|
1399
|
-
treeSitterClient: mockClient,
|
|
1400
|
-
streaming: true,
|
|
1401
|
-
left: 0,
|
|
1402
|
-
top: 0,
|
|
1403
|
-
})
|
|
1404
|
-
|
|
1405
|
-
currentRenderer.root.add(codeRenderable)
|
|
1406
|
-
|
|
1407
|
-
mockClient.resolveHighlightOnce(0)
|
|
1408
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1409
|
-
|
|
1410
|
-
codeRenderable.content = "const updated = 'world';"
|
|
1411
|
-
await new Promise((resolve) => queueMicrotask(resolve))
|
|
1412
|
-
|
|
1413
|
-
codeRenderable.content = "const updated2 = 'test';"
|
|
1414
|
-
await new Promise((resolve) => queueMicrotask(resolve))
|
|
1415
|
-
|
|
1416
|
-
codeRenderable.content = "const final = 'done';"
|
|
1417
|
-
await new Promise((resolve) => queueMicrotask(resolve))
|
|
1418
|
-
|
|
1419
|
-
await renderOnce()
|
|
1420
|
-
|
|
1421
|
-
expect(codeRenderable.content).toBe("const final = 'done';")
|
|
1422
|
-
expect(codeRenderable.plainText).toBe("const final = 'done';")
|
|
1423
|
-
})
|
|
1424
|
-
|
|
1425
|
-
test("CodeRenderable - streaming mode works with large content updates", async () => {
|
|
1426
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1427
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1428
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1429
|
-
})
|
|
1430
|
-
|
|
1431
|
-
const mockClient = new MockTreeSitterClient()
|
|
1432
|
-
mockClient.setMockResult({
|
|
1433
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1434
|
-
})
|
|
1435
|
-
|
|
1436
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1437
|
-
id: "test-code",
|
|
1438
|
-
content: "const x = 1;",
|
|
1439
|
-
filetype: "javascript",
|
|
1440
|
-
syntaxStyle,
|
|
1441
|
-
treeSitterClient: mockClient,
|
|
1442
|
-
streaming: true,
|
|
1443
|
-
drawUnstyledText: true,
|
|
1444
|
-
left: 0,
|
|
1445
|
-
top: 0,
|
|
1446
|
-
})
|
|
1447
|
-
|
|
1448
|
-
currentRenderer.root.add(codeRenderable)
|
|
1449
|
-
await renderOnce()
|
|
1450
|
-
|
|
1451
|
-
// Wait for initial highlighting
|
|
1452
|
-
mockClient.resolveHighlightOnce(0)
|
|
1453
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1454
|
-
|
|
1455
|
-
// Simulate streaming with progressively larger content
|
|
1456
|
-
let content = "const x = 1;"
|
|
1457
|
-
for (let i = 0; i < 10; i++) {
|
|
1458
|
-
content += `\nconst var${i} = ${i};`
|
|
1459
|
-
codeRenderable.content = content
|
|
1460
|
-
await new Promise((resolve) => setTimeout(resolve, 5))
|
|
1461
|
-
}
|
|
1462
|
-
|
|
1463
|
-
await renderOnce()
|
|
1464
|
-
mockClient.resolveAllHighlightOnce()
|
|
1465
|
-
await new Promise((resolve) => setTimeout(resolve, 20))
|
|
1466
|
-
await renderOnce()
|
|
1467
|
-
|
|
1468
|
-
expect(codeRenderable.content).toContain("const var9 = 9;")
|
|
1469
|
-
expect(codeRenderable.plainText).toContain("const var9 = 9;")
|
|
1470
|
-
})
|
|
1471
|
-
|
|
1472
|
-
test("CodeRenderable - disabling streaming clears cached highlights", async () => {
|
|
1473
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1474
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1475
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1476
|
-
})
|
|
1477
|
-
|
|
1478
|
-
const mockClient = new MockTreeSitterClient()
|
|
1479
|
-
mockClient.setMockResult({
|
|
1480
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1481
|
-
})
|
|
1482
|
-
|
|
1483
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1484
|
-
id: "test-code",
|
|
1485
|
-
content: "const initial = 'hello';",
|
|
1486
|
-
filetype: "javascript",
|
|
1487
|
-
syntaxStyle,
|
|
1488
|
-
treeSitterClient: mockClient,
|
|
1489
|
-
streaming: true,
|
|
1490
|
-
})
|
|
1491
|
-
|
|
1492
|
-
currentRenderer.root.add(codeRenderable)
|
|
1493
|
-
await renderOnce()
|
|
1494
|
-
|
|
1495
|
-
expect(codeRenderable.streaming).toBe(true)
|
|
1496
|
-
|
|
1497
|
-
mockClient.resolveHighlightOnce(0)
|
|
1498
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1499
|
-
|
|
1500
|
-
codeRenderable.streaming = false
|
|
1501
|
-
expect(codeRenderable.streaming).toBe(false)
|
|
1502
|
-
|
|
1503
|
-
await renderOnce()
|
|
1504
|
-
|
|
1505
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1506
|
-
})
|
|
1507
|
-
|
|
1508
|
-
test("CodeRenderable - streaming mode with drawUnstyledText=false shows nothing initially", async () => {
|
|
1509
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1510
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1511
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1512
|
-
})
|
|
1513
|
-
|
|
1514
|
-
const mockClient = new MockTreeSitterClient()
|
|
1515
|
-
mockClient.setMockResult({
|
|
1516
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1517
|
-
})
|
|
1518
|
-
|
|
1519
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1520
|
-
id: "test-code",
|
|
1521
|
-
content: "const initial = 'hello';",
|
|
1522
|
-
filetype: "javascript",
|
|
1523
|
-
syntaxStyle,
|
|
1524
|
-
treeSitterClient: mockClient,
|
|
1525
|
-
streaming: true,
|
|
1526
|
-
drawUnstyledText: false,
|
|
1527
|
-
left: 0,
|
|
1528
|
-
top: 0,
|
|
1529
|
-
})
|
|
1530
|
-
|
|
1531
|
-
currentRenderer.root.add(codeRenderable)
|
|
1532
|
-
|
|
1533
|
-
await renderOnce()
|
|
1534
|
-
const frameBeforeHighlighting = captureFrame()
|
|
1535
|
-
expect(frameBeforeHighlighting.trim()).toBe("")
|
|
1536
|
-
|
|
1537
|
-
mockClient.resolveHighlightOnce(0)
|
|
1538
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1539
|
-
await renderOnce()
|
|
1540
|
-
|
|
1541
|
-
const frameAfterHighlighting = captureFrame()
|
|
1542
|
-
expect(frameAfterHighlighting).toContain("const initial")
|
|
1543
|
-
})
|
|
1544
|
-
|
|
1545
|
-
test("CodeRenderable - streaming mode handles empty cached highlights gracefully", async () => {
|
|
1546
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1547
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1548
|
-
})
|
|
1549
|
-
|
|
1550
|
-
const mockClient = new MockTreeSitterClient()
|
|
1551
|
-
mockClient.setMockResult({
|
|
1552
|
-
highlights: [],
|
|
1553
|
-
})
|
|
1554
|
-
|
|
1555
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1556
|
-
id: "test-code",
|
|
1557
|
-
content: "plain text",
|
|
1558
|
-
filetype: "javascript",
|
|
1559
|
-
syntaxStyle,
|
|
1560
|
-
treeSitterClient: mockClient,
|
|
1561
|
-
streaming: true,
|
|
1562
|
-
drawUnstyledText: true,
|
|
1563
|
-
})
|
|
1564
|
-
|
|
1565
|
-
currentRenderer.root.add(codeRenderable)
|
|
1566
|
-
await renderOnce()
|
|
1567
|
-
|
|
1568
|
-
mockClient.resolveHighlightOnce(0)
|
|
1569
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1570
|
-
|
|
1571
|
-
codeRenderable.content = "more plain text"
|
|
1572
|
-
await renderOnce()
|
|
1573
|
-
|
|
1574
|
-
expect(codeRenderable.content).toBe("more plain text")
|
|
1575
|
-
expect(codeRenderable.plainText).toBe("more plain text")
|
|
1576
|
-
})
|
|
1577
|
-
|
|
1578
|
-
test("CodeRenderable - selection across two Code renderables in flex row", async () => {
|
|
1579
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1580
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1581
|
-
})
|
|
1582
|
-
|
|
1583
|
-
const container = new BoxRenderable(currentRenderer, {
|
|
1584
|
-
id: "container",
|
|
1585
|
-
width: 80,
|
|
1586
|
-
height: 10,
|
|
1587
|
-
flexDirection: "row",
|
|
1588
|
-
left: 0,
|
|
1589
|
-
top: 0,
|
|
1590
|
-
})
|
|
1591
|
-
currentRenderer.root.add(container)
|
|
1592
|
-
|
|
1593
|
-
const leftCode = new CodeRenderable(currentRenderer, {
|
|
1594
|
-
id: "left-code",
|
|
1595
|
-
content: "line1\nline2\nline3\nline4\nline5",
|
|
1596
|
-
syntaxStyle,
|
|
1597
|
-
selectable: true,
|
|
1598
|
-
wrapMode: "none",
|
|
1599
|
-
width: 20,
|
|
1600
|
-
height: 5,
|
|
1601
|
-
})
|
|
1602
|
-
|
|
1603
|
-
const rightCode = new CodeRenderable(currentRenderer, {
|
|
1604
|
-
id: "right-code",
|
|
1605
|
-
content: "lineA\nlineB\nlineC\nlineD\nlineE",
|
|
1606
|
-
syntaxStyle,
|
|
1607
|
-
selectable: true,
|
|
1608
|
-
wrapMode: "none",
|
|
1609
|
-
width: 20,
|
|
1610
|
-
height: 5,
|
|
1611
|
-
})
|
|
1612
|
-
|
|
1613
|
-
container.add(leftCode)
|
|
1614
|
-
container.add(rightCode)
|
|
1615
|
-
|
|
1616
|
-
await renderOnce()
|
|
1617
|
-
|
|
1618
|
-
expect(leftCode.x).toBe(0)
|
|
1619
|
-
expect(rightCode.x).toBeGreaterThan(leftCode.x)
|
|
1620
|
-
|
|
1621
|
-
const startX = leftCode.x + 2
|
|
1622
|
-
const startY = leftCode.y + 2
|
|
1623
|
-
const endX = rightCode.x + 3
|
|
1624
|
-
const endY = rightCode.y + rightCode.height + 2
|
|
1625
|
-
|
|
1626
|
-
await mockMouse.drag(startX, startY, endX, endY)
|
|
1627
|
-
await renderOnce()
|
|
1628
|
-
|
|
1629
|
-
expect(leftCode.hasSelection()).toBe(true)
|
|
1630
|
-
expect(rightCode.hasSelection()).toBe(true)
|
|
1631
|
-
|
|
1632
|
-
const leftSelection = leftCode.getSelectedText()
|
|
1633
|
-
const rightSelection = rightCode.getSelectedText()
|
|
1634
|
-
const leftSelectionObj = leftCode.getSelection()
|
|
1635
|
-
const rightSelectionObj = rightCode.getSelection()
|
|
1636
|
-
|
|
1637
|
-
expect(leftSelectionObj).not.toBeNull()
|
|
1638
|
-
expect(rightSelectionObj).not.toBeNull()
|
|
1639
|
-
|
|
1640
|
-
if (leftSelectionObj && rightSelectionObj) {
|
|
1641
|
-
expect(leftSelectionObj.start).toBeGreaterThan(0)
|
|
1642
|
-
expect(leftSelectionObj.end).toBe(29)
|
|
1643
|
-
expect(rightSelectionObj.start).toBe(0)
|
|
1644
|
-
expect(rightSelectionObj.end).toBe(29)
|
|
1645
|
-
expect(leftSelection).toBe("ne3\nline4\nline5")
|
|
1646
|
-
expect(rightSelection).toBe("lineA\nlineB\nlineC\nlineD\nlineE")
|
|
1647
|
-
}
|
|
1648
|
-
})
|
|
1649
|
-
|
|
1650
|
-
test("CodeRenderable - content update during async highlighting does not get overwritten by stale highlight result", async () => {
|
|
1651
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1652
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1653
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1654
|
-
})
|
|
1655
|
-
|
|
1656
|
-
const mockClient = new MockTreeSitterClient()
|
|
1657
|
-
mockClient.setMockResult({
|
|
1658
|
-
highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
|
|
1659
|
-
})
|
|
1660
|
-
|
|
1661
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1662
|
-
id: "test-code",
|
|
1663
|
-
content: "line1\nline2\nline3",
|
|
1664
|
-
filetype: "javascript",
|
|
1665
|
-
syntaxStyle,
|
|
1666
|
-
treeSitterClient: mockClient,
|
|
1667
|
-
drawUnstyledText: true,
|
|
1668
|
-
})
|
|
1669
|
-
|
|
1670
|
-
currentRenderer.root.add(codeRenderable)
|
|
1671
|
-
await renderOnce()
|
|
1672
|
-
|
|
1673
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1674
|
-
expect(codeRenderable.lineCount).toBe(3)
|
|
1675
|
-
|
|
1676
|
-
codeRenderable.content = "line1\nline2\nline3\nline4\nline5"
|
|
1677
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1678
|
-
|
|
1679
|
-
mockClient.resolveHighlightOnce(0)
|
|
1680
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1681
|
-
|
|
1682
|
-
expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
|
|
1683
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1684
|
-
|
|
1685
|
-
await renderOnce()
|
|
1686
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1687
|
-
|
|
1688
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1689
|
-
|
|
1690
|
-
mockClient.resolveHighlightOnce(0)
|
|
1691
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1692
|
-
await renderOnce()
|
|
1693
|
-
|
|
1694
|
-
expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
|
|
1695
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1696
|
-
expect(codeRenderable.plainText).toBe("line1\nline2\nline3\nline4\nline5")
|
|
1697
|
-
})
|
|
1698
|
-
|
|
1699
|
-
test("CodeRenderable - lineCount is correct immediately with drawUnstyledText=false", async () => {
|
|
1700
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1701
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1702
|
-
})
|
|
1703
|
-
|
|
1704
|
-
const mockClient = new MockTreeSitterClient()
|
|
1705
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1706
|
-
|
|
1707
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1708
|
-
id: "test-code",
|
|
1709
|
-
content: "line1\nline2\nline3\nline4\nline5",
|
|
1710
|
-
filetype: "javascript",
|
|
1711
|
-
syntaxStyle,
|
|
1712
|
-
treeSitterClient: mockClient,
|
|
1713
|
-
drawUnstyledText: false,
|
|
1714
|
-
})
|
|
1715
|
-
|
|
1716
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1717
|
-
expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
|
|
1718
|
-
|
|
1719
|
-
currentRenderer.root.add(codeRenderable)
|
|
1720
|
-
await renderOnce()
|
|
1721
|
-
|
|
1722
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1723
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1724
|
-
|
|
1725
|
-
const frameBeforeHighlighting = captureFrame()
|
|
1726
|
-
expect(frameBeforeHighlighting.trim()).toBe("")
|
|
1727
|
-
|
|
1728
|
-
mockClient.resolveHighlightOnce(0)
|
|
1729
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1730
|
-
await renderOnce()
|
|
1731
|
-
|
|
1732
|
-
expect(codeRenderable.lineCount).toBe(5)
|
|
1733
|
-
const frameAfterHighlighting = captureFrame()
|
|
1734
|
-
expect(frameAfterHighlighting).toContain("line1")
|
|
1735
|
-
})
|
|
1736
|
-
|
|
1737
|
-
test("CodeRenderable - lineCount updates correctly when content changes with drawUnstyledText=false", async () => {
|
|
1738
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1739
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1740
|
-
})
|
|
1741
|
-
|
|
1742
|
-
const mockClient = new MockTreeSitterClient()
|
|
1743
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1744
|
-
|
|
1745
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1746
|
-
id: "test-code",
|
|
1747
|
-
content: "line1\nline2\nline3",
|
|
1748
|
-
filetype: "javascript",
|
|
1749
|
-
syntaxStyle,
|
|
1750
|
-
treeSitterClient: mockClient,
|
|
1751
|
-
drawUnstyledText: false,
|
|
1752
|
-
})
|
|
1753
|
-
|
|
1754
|
-
expect(codeRenderable.lineCount).toBe(3)
|
|
1755
|
-
|
|
1756
|
-
currentRenderer.root.add(codeRenderable)
|
|
1757
|
-
await renderOnce()
|
|
1758
|
-
|
|
1759
|
-
codeRenderable.content = "line1\nline2\nline3\nline4\nline5\nline6\nline7"
|
|
1760
|
-
expect(codeRenderable.lineCount).toBe(7)
|
|
1761
|
-
|
|
1762
|
-
await renderOnce()
|
|
1763
|
-
expect(codeRenderable.lineCount).toBe(7)
|
|
1764
|
-
|
|
1765
|
-
codeRenderable.content = "line1\nline2"
|
|
1766
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1767
|
-
|
|
1768
|
-
await renderOnce()
|
|
1769
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1770
|
-
})
|
|
1771
|
-
|
|
1772
|
-
test("CodeRenderable - lineInfo is accessible with drawUnstyledText=false before highlighting", async () => {
|
|
1773
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1774
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1775
|
-
})
|
|
1776
|
-
|
|
1777
|
-
const mockClient = new MockTreeSitterClient()
|
|
1778
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1779
|
-
|
|
1780
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1781
|
-
id: "test-code",
|
|
1782
|
-
content: "short\nlonger line here\nmed",
|
|
1783
|
-
filetype: "javascript",
|
|
1784
|
-
syntaxStyle,
|
|
1785
|
-
treeSitterClient: mockClient,
|
|
1786
|
-
drawUnstyledText: false,
|
|
1787
|
-
})
|
|
1788
|
-
|
|
1789
|
-
currentRenderer.root.add(codeRenderable)
|
|
1790
|
-
|
|
1791
|
-
expect(codeRenderable.lineCount).toBe(3)
|
|
1792
|
-
expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
|
|
1793
|
-
|
|
1794
|
-
await renderOnce()
|
|
1795
|
-
|
|
1796
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1797
|
-
expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
|
|
1798
|
-
expect(codeRenderable.lineInfo.lineSources.length).toBe(3)
|
|
1799
|
-
|
|
1800
|
-
mockClient.resolveHighlightOnce(0)
|
|
1801
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1802
|
-
await renderOnce()
|
|
1803
|
-
|
|
1804
|
-
expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
|
|
1805
|
-
expect(codeRenderable.lineInfo.lineSources.length).toBe(3)
|
|
1806
|
-
})
|
|
1807
|
-
|
|
1808
|
-
test("CodeRenderable - plainText reflects content immediately with drawUnstyledText=false", async () => {
|
|
1809
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1810
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1811
|
-
})
|
|
1812
|
-
|
|
1813
|
-
const mockClient = new MockTreeSitterClient()
|
|
1814
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1815
|
-
|
|
1816
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1817
|
-
id: "test-code",
|
|
1818
|
-
content: "initial content",
|
|
1819
|
-
filetype: "javascript",
|
|
1820
|
-
syntaxStyle,
|
|
1821
|
-
treeSitterClient: mockClient,
|
|
1822
|
-
drawUnstyledText: false,
|
|
1823
|
-
})
|
|
1824
|
-
|
|
1825
|
-
expect(codeRenderable.plainText).toBe("initial content")
|
|
1826
|
-
|
|
1827
|
-
currentRenderer.root.add(codeRenderable)
|
|
1828
|
-
await renderOnce()
|
|
1829
|
-
|
|
1830
|
-
expect(mockClient.isHighlighting()).toBe(true)
|
|
1831
|
-
expect(codeRenderable.plainText).toBe("initial content")
|
|
1832
|
-
|
|
1833
|
-
codeRenderable.content = "updated content"
|
|
1834
|
-
expect(codeRenderable.plainText).toBe("updated content")
|
|
1835
|
-
|
|
1836
|
-
await renderOnce()
|
|
1837
|
-
const frame = captureFrame()
|
|
1838
|
-
expect(frame.trim()).toBe("")
|
|
1839
|
-
|
|
1840
|
-
mockClient.resolveAllHighlightOnce()
|
|
1841
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1842
|
-
await renderOnce()
|
|
1843
|
-
|
|
1844
|
-
expect(codeRenderable.plainText).toBe("updated content")
|
|
1845
|
-
const finalFrame = captureFrame()
|
|
1846
|
-
expect(finalFrame).toContain("updated content")
|
|
1847
|
-
})
|
|
1848
|
-
|
|
1849
|
-
test("CodeRenderable - textLength is correct with drawUnstyledText=false", async () => {
|
|
1850
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1851
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1852
|
-
})
|
|
1853
|
-
|
|
1854
|
-
const mockClient = new MockTreeSitterClient()
|
|
1855
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1856
|
-
|
|
1857
|
-
const content = "hello world test"
|
|
1858
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1859
|
-
id: "test-code",
|
|
1860
|
-
content,
|
|
1861
|
-
filetype: "javascript",
|
|
1862
|
-
syntaxStyle,
|
|
1863
|
-
treeSitterClient: mockClient,
|
|
1864
|
-
drawUnstyledText: false,
|
|
1865
|
-
})
|
|
1866
|
-
|
|
1867
|
-
expect(codeRenderable.textLength).toBe(content.length)
|
|
1868
|
-
|
|
1869
|
-
currentRenderer.root.add(codeRenderable)
|
|
1870
|
-
await renderOnce()
|
|
1871
|
-
|
|
1872
|
-
expect(codeRenderable.textLength).toBe(content.length)
|
|
1873
|
-
|
|
1874
|
-
const newContent = "longer content here"
|
|
1875
|
-
codeRenderable.content = newContent
|
|
1876
|
-
expect(codeRenderable.textLength).toBe(newContent.length)
|
|
1877
|
-
})
|
|
1878
|
-
|
|
1879
|
-
test("CodeRenderable - streaming mode with drawUnstyledText=false has correct lineCount", async () => {
|
|
1880
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1881
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1882
|
-
})
|
|
1883
|
-
|
|
1884
|
-
const mockClient = new MockTreeSitterClient()
|
|
1885
|
-
mockClient.setMockResult({ highlights: [] })
|
|
1886
|
-
|
|
1887
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1888
|
-
id: "test-code",
|
|
1889
|
-
content: "line1\nline2",
|
|
1890
|
-
filetype: "javascript",
|
|
1891
|
-
syntaxStyle,
|
|
1892
|
-
treeSitterClient: mockClient,
|
|
1893
|
-
streaming: true,
|
|
1894
|
-
drawUnstyledText: false,
|
|
1895
|
-
})
|
|
1896
|
-
|
|
1897
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1898
|
-
|
|
1899
|
-
currentRenderer.root.add(codeRenderable)
|
|
1900
|
-
await renderOnce()
|
|
1901
|
-
|
|
1902
|
-
const frameBeforeHighlighting = captureFrame()
|
|
1903
|
-
expect(frameBeforeHighlighting.trim()).toBe("")
|
|
1904
|
-
|
|
1905
|
-
mockClient.resolveHighlightOnce(0)
|
|
1906
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1907
|
-
await renderOnce()
|
|
1908
|
-
|
|
1909
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1910
|
-
|
|
1911
|
-
codeRenderable.content = "line1\nline2\nline3\nline4"
|
|
1912
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1913
|
-
|
|
1914
|
-
codeRenderable.content = "line1\nline2\nline3\nline4\nline5\nline6"
|
|
1915
|
-
expect(codeRenderable.lineCount).toBe(2)
|
|
1916
|
-
|
|
1917
|
-
await renderOnce()
|
|
1918
|
-
mockClient.resolveAllHighlightOnce()
|
|
1919
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1920
|
-
await renderOnce()
|
|
1921
|
-
|
|
1922
|
-
expect(codeRenderable.lineCount).toBe(6)
|
|
1923
|
-
const finalFrame = captureFrame()
|
|
1924
|
-
expect(finalFrame).toContain("line1")
|
|
1925
|
-
})
|
|
1926
|
-
|
|
1927
|
-
test("CodeRenderable - streaming with conceal and drawUnstyledText=false should not jump when fenced code blocks are concealed", async () => {
|
|
1928
|
-
resize(80, 20)
|
|
1929
|
-
|
|
1930
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
1931
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
1932
|
-
keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1933
|
-
string: { fg: RGBA.fromValues(0, 1, 0, 1) },
|
|
1934
|
-
"markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) },
|
|
1935
|
-
"markup.raw.block": { fg: RGBA.fromValues(0.5, 0.5, 0.5, 1) },
|
|
1936
|
-
})
|
|
1937
|
-
|
|
1938
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
1939
|
-
id: "test-markdown",
|
|
1940
|
-
content: "# Example",
|
|
1941
|
-
filetype: "markdown",
|
|
1942
|
-
syntaxStyle,
|
|
1943
|
-
streaming: true,
|
|
1944
|
-
conceal: true,
|
|
1945
|
-
drawUnstyledText: false,
|
|
1946
|
-
left: 0,
|
|
1947
|
-
top: 0,
|
|
1948
|
-
})
|
|
1949
|
-
|
|
1950
|
-
currentRenderer.root.add(codeRenderable)
|
|
1951
|
-
|
|
1952
|
-
const waitForHighlightingCycle = async (timeout = 2000) => {
|
|
1953
|
-
const start = Date.now()
|
|
1954
|
-
await renderOnce()
|
|
1955
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1956
|
-
while (codeRenderable.isHighlighting && Date.now() - start < timeout) {
|
|
1957
|
-
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
1958
|
-
}
|
|
1959
|
-
await renderOnce()
|
|
1960
|
-
}
|
|
1961
|
-
|
|
1962
|
-
// Use TestRecorder to capture frames
|
|
1963
|
-
const { TestRecorder } = await import("../testing/test-recorder.js")
|
|
1964
|
-
const recorder = new TestRecorder(currentRenderer)
|
|
1965
|
-
|
|
1966
|
-
// Start renderer and recorder
|
|
1967
|
-
currentRenderer.start()
|
|
1968
|
-
recorder.rec()
|
|
1969
|
-
|
|
1970
|
-
// Wait for initial highlighting to complete
|
|
1971
|
-
await waitForHighlightingCycle()
|
|
1972
|
-
|
|
1973
|
-
// Now simulate streaming: add more content including fenced code block
|
|
1974
|
-
codeRenderable.content = `# Example\n\nHere's some code:\n\n\`\`\`typescript\nconst x = 1;\n\`\`\``
|
|
1975
|
-
|
|
1976
|
-
// Wait for highlighting to process the update
|
|
1977
|
-
await waitForHighlightingCycle()
|
|
1978
|
-
|
|
1979
|
-
// Stop everything
|
|
1980
|
-
currentRenderer.stop()
|
|
1981
|
-
recorder.stop()
|
|
1982
|
-
|
|
1983
|
-
const frames = recorder.recordedFrames
|
|
1984
|
-
|
|
1985
|
-
// Analyze frames to detect the presence of backticks
|
|
1986
|
-
const frameAnalysis: Array<{ hasBackticks: boolean; lineCount: number; isEmpty: boolean }> = []
|
|
1987
|
-
|
|
1988
|
-
for (const recordedFrame of frames) {
|
|
1989
|
-
const frame = recordedFrame.frame
|
|
1990
|
-
const hasBackticks = frame.includes("```")
|
|
1991
|
-
const lines = frame.split("\n").filter((line) => line.trim().length > 0)
|
|
1992
|
-
const isEmpty = frame.trim().length === 0
|
|
1993
|
-
|
|
1994
|
-
frameAnalysis.push({
|
|
1995
|
-
hasBackticks,
|
|
1996
|
-
lineCount: lines.length,
|
|
1997
|
-
isEmpty,
|
|
1998
|
-
})
|
|
1999
|
-
}
|
|
2000
|
-
|
|
2001
|
-
let hasFlickering = false
|
|
2002
|
-
for (let i = 2; i < frameAnalysis.length; i++) {
|
|
2003
|
-
const prev = frameAnalysis[i - 1]
|
|
2004
|
-
const curr = frameAnalysis[i]
|
|
2005
|
-
if (!prev.isEmpty && curr.isEmpty) {
|
|
2006
|
-
hasFlickering = true
|
|
2007
|
-
}
|
|
2008
|
-
}
|
|
2009
|
-
|
|
2010
|
-
const framesWithBackticks = frameAnalysis.filter((f) => f.hasBackticks && !f.isEmpty)
|
|
2011
|
-
|
|
2012
|
-
expect(framesWithBackticks.length).toBe(0)
|
|
2013
|
-
expect(hasFlickering).toBe(false)
|
|
2014
|
-
|
|
2015
|
-
const finalFrame = frameAnalysis[frameAnalysis.length - 1]
|
|
2016
|
-
expect(finalFrame.isEmpty).toBe(false)
|
|
2017
|
-
expect(finalFrame.hasBackticks).toBe(false)
|
|
2018
|
-
expect(finalFrame.lineCount).toBe(3)
|
|
2019
|
-
|
|
2020
|
-
const finalFrameText = frames[frames.length - 1].frame
|
|
2021
|
-
expect(finalFrameText).toContain("Example")
|
|
2022
|
-
expect(finalFrameText).toContain("Here's some code")
|
|
2023
|
-
expect(finalFrameText).toContain("const x = 1")
|
|
2024
|
-
expect(finalFrameText).not.toContain("```")
|
|
2025
|
-
})
|
|
2026
|
-
|
|
2027
|
-
test("CodeRenderable - streaming with drawUnstyledText=false falls back to unstyled text when highlights fail", async () => {
|
|
2028
|
-
const syntaxStyle = SyntaxStyle.fromStyles({
|
|
2029
|
-
default: { fg: RGBA.fromValues(1, 1, 1, 1) },
|
|
2030
|
-
})
|
|
2031
|
-
|
|
2032
|
-
const mockClient = new MockTreeSitterClient({ autoResolveTimeout: 10 })
|
|
2033
|
-
|
|
2034
|
-
const codeRenderable = new CodeRenderable(currentRenderer, {
|
|
2035
|
-
id: "test-code",
|
|
2036
|
-
content: "const initial = 'hello';",
|
|
2037
|
-
filetype: "javascript",
|
|
2038
|
-
syntaxStyle,
|
|
2039
|
-
treeSitterClient: mockClient,
|
|
2040
|
-
streaming: true,
|
|
2041
|
-
drawUnstyledText: false,
|
|
2042
|
-
left: 0,
|
|
2043
|
-
top: 0,
|
|
2044
|
-
})
|
|
2045
|
-
|
|
2046
|
-
currentRenderer.root.add(codeRenderable)
|
|
2047
|
-
currentRenderer.start()
|
|
2048
|
-
|
|
2049
|
-
await Bun.sleep(30)
|
|
2050
|
-
|
|
2051
|
-
mockClient.highlightOnce = async () => {
|
|
2052
|
-
throw new Error("Highlighting failed")
|
|
2053
|
-
}
|
|
2054
|
-
|
|
2055
|
-
codeRenderable.content = "const updated = 'world';"
|
|
2056
|
-
|
|
2057
|
-
await Bun.sleep(30)
|
|
2058
|
-
|
|
2059
|
-
expect(codeRenderable.plainText).toBe("const updated = 'world';")
|
|
2060
|
-
|
|
2061
|
-
currentRenderer.stop()
|
|
2062
|
-
})
|