@fairyhunter13/opentui-core 0.1.112 → 0.1.114
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dev/keypress-debug-renderer.ts +148 -0
- package/dev/keypress-debug.ts +43 -0
- package/dev/print-env-vars.ts +32 -0
- package/dev/test-tmux-graphics-334.sh +68 -0
- package/dev/thai-debug-test.ts +68 -0
- package/docs/development.md +144 -0
- package/package.json +63 -51
- package/scripts/build.ts +400 -0
- package/scripts/publish.ts +60 -0
- package/src/3d/SpriteResourceManager.ts +286 -0
- package/src/3d/SpriteUtils.ts +70 -0
- package/src/3d/TextureUtils.ts +196 -0
- package/src/3d/ThreeRenderable.ts +197 -0
- package/src/3d/WGPURenderer.ts +294 -0
- package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
- package/src/3d/animation/SpriteAnimator.ts +633 -0
- package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
- package/src/3d/canvas.ts +464 -0
- package/src/3d/index.ts +12 -0
- package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
- package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
- package/src/3d/physics/physics-interface.ts +31 -0
- package/src/3d/shaders/supersampling.wgsl +201 -0
- package/src/3d.ts +3 -0
- package/src/NativeSpanFeed.ts +300 -0
- package/src/Renderable.ts +1704 -0
- package/src/__snapshots__/buffer.test.ts.snap +28 -0
- package/src/animation/Timeline.test.ts +2709 -0
- package/src/animation/Timeline.ts +598 -0
- package/src/ansi.ts +18 -0
- package/src/benchmark/attenuation-benchmark.ts +81 -0
- package/src/benchmark/colormatrix-benchmark.ts +128 -0
- package/src/benchmark/gain-benchmark.ts +80 -0
- package/src/benchmark/latest-all-bench-run.json +707 -0
- package/src/benchmark/latest-async-bench-run.json +336 -0
- package/src/benchmark/latest-default-bench-run.json +657 -0
- package/src/benchmark/latest-large-bench-run.json +707 -0
- package/src/benchmark/latest-quick-bench-run.json +207 -0
- package/src/benchmark/markdown-benchmark.ts +1796 -0
- package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
- package/src/benchmark/native-span-feed-benchmark.md +56 -0
- package/src/benchmark/native-span-feed-benchmark.ts +596 -0
- package/src/benchmark/native-span-feed-compare.ts +280 -0
- package/src/benchmark/renderer-benchmark.ts +754 -0
- package/src/benchmark/text-table-benchmark.ts +948 -0
- package/src/buffer.test.ts +291 -0
- package/src/buffer.ts +554 -0
- package/src/console.test.ts +612 -0
- package/src/console.ts +1254 -0
- package/src/edit-buffer.test.ts +1769 -0
- package/src/edit-buffer.ts +411 -0
- package/src/editor-view.test.ts +1032 -0
- package/src/editor-view.ts +284 -0
- package/src/examples/ascii-font-selection-demo.ts +245 -0
- package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
- package/src/examples/assets/concrete.png +0 -0
- package/src/examples/assets/crate.png +0 -0
- package/src/examples/assets/crate_emissive.png +0 -0
- package/src/examples/assets/forrest_background.png +0 -0
- package/src/examples/assets/hast-example.json +1018 -0
- package/src/examples/assets/heart.png +0 -0
- package/src/examples/assets/main_char_heavy_attack.png +0 -0
- package/src/examples/assets/main_char_idle.png +0 -0
- package/src/examples/assets/main_char_jump_end.png +0 -0
- package/src/examples/assets/main_char_jump_landing.png +0 -0
- package/src/examples/assets/main_char_jump_start.png +0 -0
- package/src/examples/assets/main_char_run_loop.png +0 -0
- package/src/examples/assets/roughness_map.jpg +0 -0
- package/src/examples/build.ts +115 -0
- package/src/examples/code-demo.ts +924 -0
- package/src/examples/console-demo.ts +358 -0
- package/src/examples/core-plugin-slots-demo.ts +759 -0
- package/src/examples/diff-demo.ts +701 -0
- package/src/examples/draggable-three-demo.ts +259 -0
- package/src/examples/editor-demo.ts +322 -0
- package/src/examples/extmarks-demo.ts +196 -0
- package/src/examples/focus-restore-demo.ts +310 -0
- package/src/examples/fonts.ts +245 -0
- package/src/examples/fractal-shader-demo.ts +268 -0
- package/src/examples/framebuffer-demo.ts +674 -0
- package/src/examples/full-unicode-demo.ts +241 -0
- package/src/examples/golden-star-demo.ts +933 -0
- package/src/examples/grayscale-buffer-demo.ts +249 -0
- package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
- package/src/examples/index.ts +926 -0
- package/src/examples/input-demo.ts +377 -0
- package/src/examples/input-select-layout-demo.ts +425 -0
- package/src/examples/install.sh +143 -0
- package/src/examples/keypress-debug-demo.ts +452 -0
- package/src/examples/lib/HexList.ts +122 -0
- package/src/examples/lib/PaletteGrid.ts +125 -0
- package/src/examples/lib/standalone-keys.ts +25 -0
- package/src/examples/lib/tab-controller.ts +243 -0
- package/src/examples/lights-phong-demo.ts +290 -0
- package/src/examples/link-demo.ts +220 -0
- package/src/examples/live-state-demo.ts +480 -0
- package/src/examples/markdown-demo.ts +725 -0
- package/src/examples/mouse-interaction-demo.ts +428 -0
- package/src/examples/nested-zindex-demo.ts +357 -0
- package/src/examples/opacity-example.ts +235 -0
- package/src/examples/opentui-demo.ts +1057 -0
- package/src/examples/physx-planck-2d-demo.ts +623 -0
- package/src/examples/physx-rapier-2d-demo.ts +655 -0
- package/src/examples/relative-positioning-demo.ts +323 -0
- package/src/examples/scroll-example.ts +214 -0
- package/src/examples/scrollbox-mouse-test.ts +112 -0
- package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
- package/src/examples/select-demo.ts +237 -0
- package/src/examples/shader-cube-demo.ts +1015 -0
- package/src/examples/simple-layout-example.ts +591 -0
- package/src/examples/slider-demo.ts +617 -0
- package/src/examples/split-mode-demo.ts +453 -0
- package/src/examples/sprite-animation-demo.ts +443 -0
- package/src/examples/sprite-particle-generator-demo.ts +486 -0
- package/src/examples/static-sprite-demo.ts +193 -0
- package/src/examples/sticky-scroll-example.ts +308 -0
- package/src/examples/styled-text-demo.ts +282 -0
- package/src/examples/tab-select-demo.ts +219 -0
- package/src/examples/terminal-title.ts +29 -0
- package/src/examples/terminal.ts +305 -0
- package/src/examples/text-node-demo.ts +416 -0
- package/src/examples/text-selection-demo.ts +377 -0
- package/src/examples/text-table-demo.ts +503 -0
- package/src/examples/text-truncation-demo.ts +481 -0
- package/src/examples/text-wrap.ts +757 -0
- package/src/examples/texture-loading-demo.ts +259 -0
- package/src/examples/timeline-example.ts +670 -0
- package/src/examples/transparency-demo.ts +400 -0
- package/src/examples/vnode-composition-demo.ts +404 -0
- package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
- package/src/index.ts +24 -0
- package/src/lib/KeyHandler.integration.test.ts +292 -0
- package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
- package/src/lib/KeyHandler.test.ts +662 -0
- package/src/lib/KeyHandler.ts +222 -0
- package/src/lib/RGBA.test.ts +984 -0
- package/src/lib/RGBA.ts +204 -0
- package/src/lib/ascii.font.ts +330 -0
- package/src/lib/border.test.ts +83 -0
- package/src/lib/border.ts +170 -0
- package/src/lib/bunfs.test.ts +27 -0
- package/src/lib/bunfs.ts +18 -0
- package/src/lib/clipboard.test.ts +41 -0
- package/src/lib/clipboard.ts +47 -0
- package/src/lib/clock.ts +35 -0
- package/src/lib/data-paths.test.ts +133 -0
- package/src/lib/data-paths.ts +109 -0
- package/src/lib/debounce.ts +106 -0
- package/src/lib/detect-links.test.ts +98 -0
- package/src/lib/detect-links.ts +56 -0
- package/src/lib/env.test.ts +228 -0
- package/src/lib/env.ts +209 -0
- package/src/lib/extmarks-history.ts +51 -0
- package/src/lib/extmarks-multiwidth.test.ts +322 -0
- package/src/lib/extmarks.test.ts +3457 -0
- package/src/lib/extmarks.ts +843 -0
- package/src/lib/fonts/block.json +405 -0
- package/src/lib/fonts/grid.json +265 -0
- package/src/lib/fonts/huge.json +741 -0
- package/src/lib/fonts/pallet.json +314 -0
- package/src/lib/fonts/shade.json +591 -0
- package/src/lib/fonts/slick.json +321 -0
- package/src/lib/fonts/tiny.json +69 -0
- package/src/lib/hast-styled-text.ts +59 -0
- package/src/lib/index.ts +21 -0
- package/src/lib/keymapping.test.ts +317 -0
- package/src/lib/keymapping.ts +115 -0
- package/src/lib/objects-in-viewport.test.ts +787 -0
- package/src/lib/objects-in-viewport.ts +153 -0
- package/src/lib/output.capture.ts +58 -0
- package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
- package/src/lib/parse.keypress-kitty.test.ts +663 -0
- package/src/lib/parse.keypress-kitty.ts +439 -0
- package/src/lib/parse.keypress.test.ts +1849 -0
- package/src/lib/parse.keypress.ts +397 -0
- package/src/lib/parse.mouse.test.ts +552 -0
- package/src/lib/parse.mouse.ts +232 -0
- package/src/lib/paste.ts +16 -0
- package/src/lib/queue.ts +65 -0
- package/src/lib/renderable.validations.test.ts +87 -0
- package/src/lib/renderable.validations.ts +83 -0
- package/src/lib/scroll-acceleration.ts +98 -0
- package/src/lib/selection.ts +240 -0
- package/src/lib/singleton.ts +28 -0
- package/src/lib/stdin-parser.test.ts +2290 -0
- package/src/lib/stdin-parser.ts +1810 -0
- package/src/lib/styled-text.ts +178 -0
- package/src/lib/terminal-capability-detection.test.ts +202 -0
- package/src/lib/terminal-capability-detection.ts +79 -0
- package/src/lib/terminal-palette.test.ts +878 -0
- package/src/lib/terminal-palette.ts +383 -0
- package/src/lib/tree-sitter/assets/README.md +118 -0
- package/src/lib/tree-sitter/assets/update.ts +334 -0
- package/src/lib/tree-sitter/assets.d.ts +9 -0
- package/src/lib/tree-sitter/cache.test.ts +273 -0
- package/src/lib/tree-sitter/client.test.ts +1165 -0
- package/src/lib/tree-sitter/client.ts +607 -0
- package/src/lib/tree-sitter/default-parsers.ts +86 -0
- package/src/lib/tree-sitter/download-utils.ts +148 -0
- package/src/lib/tree-sitter/index.ts +28 -0
- package/src/lib/tree-sitter/parser.worker.ts +1042 -0
- package/src/lib/tree-sitter/parsers-config.ts +81 -0
- package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
- package/src/lib/tree-sitter/resolve-ft.ts +189 -0
- package/src/lib/tree-sitter/types.ts +82 -0
- package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
- package/src/lib/tree-sitter-styled-text.ts +306 -0
- package/src/lib/validate-dir-name.ts +55 -0
- package/src/lib/yoga.options.test.ts +628 -0
- package/src/lib/yoga.options.ts +346 -0
- package/src/plugins/core-slot.ts +579 -0
- package/src/plugins/registry.ts +402 -0
- package/src/plugins/types.ts +46 -0
- package/src/post/effects.ts +930 -0
- package/src/post/filters.ts +489 -0
- package/src/post/matrices.ts +288 -0
- package/src/renderables/ASCIIFont.ts +219 -0
- package/src/renderables/Box.test.ts +205 -0
- package/src/renderables/Box.ts +326 -0
- package/src/renderables/Code.test.ts +2062 -0
- package/src/renderables/Code.ts +357 -0
- package/src/renderables/Diff.regression.test.ts +226 -0
- package/src/renderables/Diff.test.ts +3101 -0
- package/src/renderables/Diff.ts +1211 -0
- package/src/renderables/EditBufferRenderable.test.ts +288 -0
- package/src/renderables/EditBufferRenderable.ts +1166 -0
- package/src/renderables/FrameBuffer.ts +47 -0
- package/src/renderables/Input.test.ts +1228 -0
- package/src/renderables/Input.ts +247 -0
- package/src/renderables/LineNumberRenderable.ts +724 -0
- package/src/renderables/Markdown.ts +1393 -0
- package/src/renderables/ScrollBar.ts +422 -0
- package/src/renderables/ScrollBox.ts +883 -0
- package/src/renderables/Select.test.ts +1033 -0
- package/src/renderables/Select.ts +524 -0
- package/src/renderables/Slider.test.ts +456 -0
- package/src/renderables/Slider.ts +342 -0
- package/src/renderables/TabSelect.test.ts +197 -0
- package/src/renderables/TabSelect.ts +455 -0
- package/src/renderables/Text.selection-buffer.test.ts +123 -0
- package/src/renderables/Text.test.ts +2660 -0
- package/src/renderables/Text.ts +147 -0
- package/src/renderables/TextBufferRenderable.ts +518 -0
- package/src/renderables/TextNode.test.ts +1058 -0
- package/src/renderables/TextNode.ts +325 -0
- package/src/renderables/TextTable.test.ts +1421 -0
- package/src/renderables/TextTable.ts +1344 -0
- package/src/renderables/Textarea.ts +430 -0
- package/src/renderables/TimeToFirstDraw.ts +89 -0
- package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
- package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
- package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
- package/src/renderables/__tests__/Markdown.test.ts +2518 -0
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
- package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
- package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
- package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
- package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
- package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
- package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
- package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
- package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
- package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
- package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
- package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
- package/src/renderables/composition/README.md +8 -0
- package/src/renderables/composition/VRenderable.ts +32 -0
- package/src/renderables/composition/constructs.ts +127 -0
- package/src/renderables/composition/vnode.ts +289 -0
- package/src/renderables/index.ts +23 -0
- package/src/renderables/markdown-parser.ts +66 -0
- package/src/renderer.ts +2681 -0
- package/src/runtime-plugin-support.ts +39 -0
- package/src/runtime-plugin.ts +615 -0
- package/src/syntax-style.test.ts +841 -0
- package/src/syntax-style.ts +257 -0
- package/src/testing/README.md +210 -0
- package/src/testing/capture-spans.test.ts +194 -0
- package/src/testing/integration.test.ts +276 -0
- package/src/testing/manual-clock.ts +117 -0
- package/src/testing/mock-keys.test.ts +1378 -0
- package/src/testing/mock-keys.ts +457 -0
- package/src/testing/mock-mouse.test.ts +218 -0
- package/src/testing/mock-mouse.ts +247 -0
- package/src/testing/mock-tree-sitter-client.ts +73 -0
- package/src/testing/spy.ts +13 -0
- package/src/testing/test-recorder.test.ts +415 -0
- package/src/testing/test-recorder.ts +145 -0
- package/src/testing/test-renderer.ts +132 -0
- package/src/testing.ts +7 -0
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
- package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
- package/src/tests/allocator-stats.test.ts +38 -0
- package/src/tests/destroy-during-render.test.ts +200 -0
- package/src/tests/destroy-on-exit.fixture.ts +36 -0
- package/src/tests/destroy-on-exit.test.ts +41 -0
- package/src/tests/hover-cursor.test.ts +98 -0
- package/src/tests/native-span-feed-async.test.ts +173 -0
- package/src/tests/native-span-feed-close.test.ts +120 -0
- package/src/tests/native-span-feed-coverage.test.ts +227 -0
- package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
- package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
- package/src/tests/opacity.test.ts +123 -0
- package/src/tests/renderable.snapshot.test.ts +524 -0
- package/src/tests/renderable.test.ts +1281 -0
- package/src/tests/renderer.clock.test.ts +158 -0
- package/src/tests/renderer.console-startup.test.ts +185 -0
- package/src/tests/renderer.control.test.ts +425 -0
- package/src/tests/renderer.core-slot-binding.test.ts +952 -0
- package/src/tests/renderer.cursor.test.ts +26 -0
- package/src/tests/renderer.destroy-during-render.test.ts +147 -0
- package/src/tests/renderer.focus-restore.test.ts +257 -0
- package/src/tests/renderer.focus.test.ts +294 -0
- package/src/tests/renderer.idle.test.ts +219 -0
- package/src/tests/renderer.input.test.ts +2237 -0
- package/src/tests/renderer.kitty-flags.test.ts +195 -0
- package/src/tests/renderer.mouse.test.ts +1274 -0
- package/src/tests/renderer.palette.test.ts +629 -0
- package/src/tests/renderer.selection.test.ts +49 -0
- package/src/tests/renderer.slot-registry.test.ts +684 -0
- package/src/tests/renderer.useMouse.test.ts +47 -0
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
- package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
- package/src/tests/runtime-plugin-support.fixture.ts +11 -0
- package/src/tests/runtime-plugin-support.test.ts +19 -0
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
- package/src/tests/runtime-plugin.fixture.ts +40 -0
- package/src/tests/runtime-plugin.test.ts +354 -0
- package/src/tests/scrollbox-culling-bug.test.ts +114 -0
- package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
- package/src/tests/scrollbox-hitgrid.test.ts +909 -0
- package/src/tests/scrollbox.test.ts +1530 -0
- package/src/tests/wrap-resize-perf.test.ts +276 -0
- package/src/tests/yoga-setters.test.ts +921 -0
- package/src/text-buffer-view.test.ts +705 -0
- package/src/text-buffer-view.ts +189 -0
- package/src/text-buffer.test.ts +347 -0
- package/src/text-buffer.ts +250 -0
- package/src/types.ts +161 -0
- package/src/utils.ts +88 -0
- package/src/zig/ansi.zig +268 -0
- package/src/zig/bench/README.md +50 -0
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
- package/src/zig/bench/edit-buffer_bench.zig +476 -0
- package/src/zig/bench/native-span-feed_bench.zig +100 -0
- package/src/zig/bench/rope-markers_bench.zig +713 -0
- package/src/zig/bench/rope_bench.zig +514 -0
- package/src/zig/bench/styled-text_bench.zig +470 -0
- package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
- package/src/zig/bench/text-buffer-view_bench.zig +459 -0
- package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
- package/src/zig/bench/utf8_bench.zig +799 -0
- package/src/zig/bench-utils.zig +431 -0
- package/src/zig/bench.zig +217 -0
- package/src/zig/buffer-methods.zig +211 -0
- package/src/zig/buffer.zig +2281 -0
- package/src/zig/build.zig +289 -0
- package/src/zig/build.zig.zon +16 -0
- package/src/zig/edit-buffer.zig +825 -0
- package/src/zig/editor-view.zig +802 -0
- package/src/zig/event-bus.zig +13 -0
- package/src/zig/event-emitter.zig +65 -0
- package/src/zig/file-logger.zig +92 -0
- package/src/zig/grapheme.zig +599 -0
- package/src/zig/lib.zig +1854 -0
- package/src/zig/link.zig +333 -0
- package/src/zig/logger.zig +43 -0
- package/src/zig/mem-registry.zig +125 -0
- package/src/zig/native-span-feed-bench-lib.zig +7 -0
- package/src/zig/native-span-feed.zig +708 -0
- package/src/zig/renderer.zig +1393 -0
- package/src/zig/rope.zig +1220 -0
- package/src/zig/syntax-style.zig +161 -0
- package/src/zig/terminal.zig +987 -0
- package/src/zig/test.zig +72 -0
- package/src/zig/tests/README.md +18 -0
- package/src/zig/tests/buffer-methods_test.zig +1109 -0
- package/src/zig/tests/buffer_test.zig +2557 -0
- package/src/zig/tests/edit-buffer-history_test.zig +271 -0
- package/src/zig/tests/edit-buffer_test.zig +1689 -0
- package/src/zig/tests/editor-view_test.zig +3299 -0
- package/src/zig/tests/event-emitter_test.zig +249 -0
- package/src/zig/tests/grapheme_test.zig +1304 -0
- package/src/zig/tests/link_test.zig +190 -0
- package/src/zig/tests/mem-registry_test.zig +473 -0
- package/src/zig/tests/memory_leak_regression_test.zig +159 -0
- package/src/zig/tests/native-span-feed_test.zig +1264 -0
- package/src/zig/tests/renderer_test.zig +1017 -0
- package/src/zig/tests/rope-nested_test.zig +712 -0
- package/src/zig/tests/rope_fuzz_test.zig +238 -0
- package/src/zig/tests/rope_test.zig +2362 -0
- package/src/zig/tests/segment-merge.test.zig +148 -0
- package/src/zig/tests/syntax-style_test.zig +557 -0
- package/src/zig/tests/terminal_test.zig +754 -0
- package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
- package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
- package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
- package/src/zig/tests/text-buffer-segment_test.zig +320 -0
- package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
- package/src/zig/tests/text-buffer-view_test.zig +3649 -0
- package/src/zig/tests/text-buffer_test.zig +2191 -0
- package/src/zig/tests/unicode-width-map.zon +3909 -0
- package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
- package/src/zig/tests/utf8_test.zig +4057 -0
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
- package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
- package/src/zig/tests/word-wrap-editing_test.zig +498 -0
- package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
- package/src/zig/text-buffer-iterators.zig +499 -0
- package/src/zig/text-buffer-segment.zig +404 -0
- package/src/zig/text-buffer-view.zig +1371 -0
- package/src/zig/text-buffer.zig +1180 -0
- package/src/zig/utf8.zig +1948 -0
- package/src/zig/utils.zig +9 -0
- package/src/zig-structs.ts +261 -0
- package/src/zig.ts +3884 -0
- package/tsconfig.build.json +24 -0
- package/tsconfig.json +27 -0
- package/3d/SpriteResourceManager.d.ts +0 -74
- package/3d/SpriteUtils.d.ts +0 -13
- package/3d/TextureUtils.d.ts +0 -24
- package/3d/ThreeRenderable.d.ts +0 -40
- package/3d/WGPURenderer.d.ts +0 -61
- package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
- package/3d/animation/SpriteAnimator.d.ts +0 -124
- package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
- package/3d/canvas.d.ts +0 -44
- package/3d/index.d.ts +0 -12
- package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
- package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
- package/3d/physics/physics-interface.d.ts +0 -27
- package/3d.d.ts +0 -2
- package/3d.js +0 -34041
- package/3d.js.map +0 -155
- package/LICENSE +0 -21
- package/NativeSpanFeed.d.ts +0 -41
- package/Renderable.d.ts +0 -334
- package/animation/Timeline.d.ts +0 -126
- package/ansi.d.ts +0 -13
- package/buffer.d.ts +0 -111
- package/console.d.ts +0 -144
- package/edit-buffer.d.ts +0 -98
- package/editor-view.d.ts +0 -73
- package/index-8fks7yv1.js +0 -411
- package/index-8fks7yv1.js.map +0 -10
- package/index-egy5e2rs.js +0 -12267
- package/index-egy5e2rs.js.map +0 -42
- package/index-tse8gzh0.js +0 -20614
- package/index-tse8gzh0.js.map +0 -67
- package/index.d.ts +0 -23
- package/index.js +0 -478
- package/index.js.map +0 -9
- package/lib/KeyHandler.d.ts +0 -61
- package/lib/RGBA.d.ts +0 -25
- package/lib/ascii.font.d.ts +0 -508
- package/lib/border.d.ts +0 -51
- package/lib/bunfs.d.ts +0 -7
- package/lib/clipboard.d.ts +0 -17
- package/lib/clock.d.ts +0 -15
- package/lib/data-paths.d.ts +0 -26
- package/lib/debounce.d.ts +0 -42
- package/lib/detect-links.d.ts +0 -6
- package/lib/env.d.ts +0 -42
- package/lib/extmarks-history.d.ts +0 -17
- package/lib/extmarks.d.ts +0 -89
- package/lib/hast-styled-text.d.ts +0 -17
- package/lib/index.d.ts +0 -21
- package/lib/keymapping.d.ts +0 -25
- package/lib/objects-in-viewport.d.ts +0 -24
- package/lib/output.capture.d.ts +0 -24
- package/lib/parse.keypress-kitty.d.ts +0 -2
- package/lib/parse.keypress.d.ts +0 -26
- package/lib/parse.mouse.d.ts +0 -30
- package/lib/paste.d.ts +0 -7
- package/lib/queue.d.ts +0 -15
- package/lib/renderable.validations.d.ts +0 -12
- package/lib/scroll-acceleration.d.ts +0 -43
- package/lib/selection.d.ts +0 -63
- package/lib/singleton.d.ts +0 -7
- package/lib/stdin-parser.d.ts +0 -87
- package/lib/styled-text.d.ts +0 -63
- package/lib/terminal-capability-detection.d.ts +0 -30
- package/lib/terminal-palette.d.ts +0 -50
- package/lib/tree-sitter/assets/update.d.ts +0 -11
- package/lib/tree-sitter/client.d.ts +0 -47
- package/lib/tree-sitter/default-parsers.d.ts +0 -2
- package/lib/tree-sitter/download-utils.d.ts +0 -21
- package/lib/tree-sitter/index.d.ts +0 -8
- package/lib/tree-sitter/parser.worker.d.ts +0 -1
- package/lib/tree-sitter/parsers-config.d.ts +0 -53
- package/lib/tree-sitter/resolve-ft.d.ts +0 -5
- package/lib/tree-sitter/types.d.ts +0 -82
- package/lib/tree-sitter-styled-text.d.ts +0 -14
- package/lib/validate-dir-name.d.ts +0 -1
- package/lib/yoga.options.d.ts +0 -32
- package/parser.worker.js +0 -899
- package/parser.worker.js.map +0 -12
- package/plugins/core-slot.d.ts +0 -72
- package/plugins/registry.d.ts +0 -42
- package/plugins/types.d.ts +0 -34
- package/post/effects.d.ts +0 -147
- package/post/filters.d.ts +0 -65
- package/post/matrices.d.ts +0 -20
- package/renderables/ASCIIFont.d.ts +0 -52
- package/renderables/Box.d.ts +0 -81
- package/renderables/Code.d.ts +0 -78
- package/renderables/Diff.d.ts +0 -142
- package/renderables/EditBufferRenderable.d.ts +0 -237
- package/renderables/FrameBuffer.d.ts +0 -16
- package/renderables/Input.d.ts +0 -67
- package/renderables/LineNumberRenderable.d.ts +0 -78
- package/renderables/Markdown.d.ts +0 -185
- package/renderables/ScrollBar.d.ts +0 -77
- package/renderables/ScrollBox.d.ts +0 -124
- package/renderables/Select.d.ts +0 -115
- package/renderables/Slider.d.ts +0 -47
- package/renderables/TabSelect.d.ts +0 -96
- package/renderables/Text.d.ts +0 -36
- package/renderables/TextBufferRenderable.d.ts +0 -105
- package/renderables/TextNode.d.ts +0 -91
- package/renderables/TextTable.d.ts +0 -140
- package/renderables/Textarea.d.ts +0 -63
- package/renderables/TimeToFirstDraw.d.ts +0 -24
- package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
- package/renderables/composition/VRenderable.d.ts +0 -16
- package/renderables/composition/constructs.d.ts +0 -35
- package/renderables/composition/vnode.d.ts +0 -46
- package/renderables/index.d.ts +0 -23
- package/renderables/markdown-parser.d.ts +0 -10
- package/renderer.d.ts +0 -419
- package/runtime-plugin-support.d.ts +0 -3
- package/runtime-plugin-support.js +0 -29
- package/runtime-plugin-support.js.map +0 -10
- package/runtime-plugin.d.ts +0 -16
- package/runtime-plugin.js +0 -16
- package/runtime-plugin.js.map +0 -9
- package/syntax-style.d.ts +0 -54
- package/testing/manual-clock.d.ts +0 -17
- package/testing/mock-keys.d.ts +0 -81
- package/testing/mock-mouse.d.ts +0 -38
- package/testing/mock-tree-sitter-client.d.ts +0 -23
- package/testing/spy.d.ts +0 -7
- package/testing/test-recorder.d.ts +0 -61
- package/testing/test-renderer.d.ts +0 -23
- package/testing.d.ts +0 -6
- package/testing.js +0 -697
- package/testing.js.map +0 -15
- package/text-buffer-view.d.ts +0 -42
- package/text-buffer.d.ts +0 -67
- package/types.d.ts +0 -139
- package/utils.d.ts +0 -14
- package/zig-structs.d.ts +0 -155
- package/zig.d.ts +0 -353
- /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -0,0 +1,1393 @@
|
|
|
1
|
+
const std = @import("std");
|
|
2
|
+
const Allocator = std.mem.Allocator;
|
|
3
|
+
const ansi = @import("ansi.zig");
|
|
4
|
+
const buf = @import("buffer.zig");
|
|
5
|
+
const gp = @import("grapheme.zig");
|
|
6
|
+
const link = @import("link.zig");
|
|
7
|
+
const Terminal = @import("terminal.zig");
|
|
8
|
+
const logger = @import("logger.zig");
|
|
9
|
+
|
|
10
|
+
pub const RGBA = ansi.RGBA;
|
|
11
|
+
pub const OptimizedBuffer = buf.OptimizedBuffer;
|
|
12
|
+
pub const TextAttributes = ansi.TextAttributes;
|
|
13
|
+
pub const CursorStyle = Terminal.CursorStyle;
|
|
14
|
+
|
|
15
|
+
const CLEAR_CHAR = '\u{0a00}';
|
|
16
|
+
const MAX_STAT_SAMPLES = 30;
|
|
17
|
+
const STAT_SAMPLE_CAPACITY = 30;
|
|
18
|
+
|
|
19
|
+
const COLOR_EPSILON_DEFAULT: f32 = 0.00001;
|
|
20
|
+
const OUTPUT_BUFFER_SIZE = 1024 * 1024 * 2; // 2MB
|
|
21
|
+
|
|
22
|
+
pub const RendererError = error{
|
|
23
|
+
OutOfMemory,
|
|
24
|
+
InvalidDimensions,
|
|
25
|
+
ThreadingFailed,
|
|
26
|
+
WriteFailed,
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
fn rgbaComponentToU8(component: f32) u8 {
|
|
30
|
+
if (!std.math.isFinite(component)) return 0;
|
|
31
|
+
|
|
32
|
+
const clamped = std.math.clamp(component, 0.0, 1.0);
|
|
33
|
+
return @intFromFloat(@round(clamped * 255.0));
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
pub const DebugOverlayCorner = enum {
|
|
37
|
+
topLeft,
|
|
38
|
+
topRight,
|
|
39
|
+
bottomLeft,
|
|
40
|
+
bottomRight,
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
pub const CliRenderer = struct {
|
|
44
|
+
width: u32,
|
|
45
|
+
height: u32,
|
|
46
|
+
currentRenderBuffer: *OptimizedBuffer,
|
|
47
|
+
nextRenderBuffer: *OptimizedBuffer,
|
|
48
|
+
pool: *gp.GraphemePool,
|
|
49
|
+
backgroundColor: RGBA,
|
|
50
|
+
renderOffset: u32,
|
|
51
|
+
terminal: Terminal,
|
|
52
|
+
testing: bool = false,
|
|
53
|
+
useAlternateScreen: bool = true,
|
|
54
|
+
terminalSetup: bool = false,
|
|
55
|
+
|
|
56
|
+
renderStats: struct {
|
|
57
|
+
lastFrameTime: f64,
|
|
58
|
+
averageFrameTime: f64,
|
|
59
|
+
frameCount: u64,
|
|
60
|
+
fps: u32,
|
|
61
|
+
cellsUpdated: u32,
|
|
62
|
+
renderTime: ?f64,
|
|
63
|
+
overallFrameTime: ?f64,
|
|
64
|
+
bufferResetTime: ?f64,
|
|
65
|
+
stdoutWriteTime: ?f64,
|
|
66
|
+
heapUsed: u32,
|
|
67
|
+
heapTotal: u32,
|
|
68
|
+
arrayBuffers: u32,
|
|
69
|
+
frameCallbackTime: ?f64,
|
|
70
|
+
},
|
|
71
|
+
statSamples: struct {
|
|
72
|
+
lastFrameTime: std.ArrayListUnmanaged(f64),
|
|
73
|
+
renderTime: std.ArrayListUnmanaged(f64),
|
|
74
|
+
overallFrameTime: std.ArrayListUnmanaged(f64),
|
|
75
|
+
bufferResetTime: std.ArrayListUnmanaged(f64),
|
|
76
|
+
stdoutWriteTime: std.ArrayListUnmanaged(f64),
|
|
77
|
+
cellsUpdated: std.ArrayListUnmanaged(u32),
|
|
78
|
+
frameCallbackTime: std.ArrayListUnmanaged(f64),
|
|
79
|
+
},
|
|
80
|
+
lastRenderTime: i64,
|
|
81
|
+
allocator: Allocator,
|
|
82
|
+
renderThread: ?std.Thread = null,
|
|
83
|
+
stdoutBuffer: [4096]u8,
|
|
84
|
+
writeOutBuf: [1024]u8 = undefined,
|
|
85
|
+
debugOverlay: struct {
|
|
86
|
+
enabled: bool,
|
|
87
|
+
corner: DebugOverlayCorner,
|
|
88
|
+
} = .{
|
|
89
|
+
.enabled = false,
|
|
90
|
+
.corner = .bottomRight,
|
|
91
|
+
},
|
|
92
|
+
// Threading
|
|
93
|
+
useThread: bool = false,
|
|
94
|
+
renderMutex: std.Thread.Mutex = .{},
|
|
95
|
+
renderCondition: std.Thread.Condition = .{},
|
|
96
|
+
renderRequested: bool = false,
|
|
97
|
+
shouldTerminate: bool = false,
|
|
98
|
+
renderInProgress: bool = false,
|
|
99
|
+
currentOutputBuffer: []u8 = &[_]u8{},
|
|
100
|
+
currentOutputLen: usize = 0,
|
|
101
|
+
|
|
102
|
+
// Hit grid for mouse event dispatch.
|
|
103
|
+
//
|
|
104
|
+
// The hit grid is a screen-sized array where each cell stores the renderable ID
|
|
105
|
+
// at that position. Mouse events query checkHit(x, y) to find which element to
|
|
106
|
+
// dispatch to.
|
|
107
|
+
//
|
|
108
|
+
// Double buffering: During render, addToHitGrid writes to nextHitGrid. After
|
|
109
|
+
// render completes, the buffers swap. This keeps hit testing consistent during
|
|
110
|
+
// a frame. Queries see the previous frame's state, not a half-built grid.
|
|
111
|
+
//
|
|
112
|
+
// On-demand sync: When scroll/translate changes between renders, the TypeScript
|
|
113
|
+
// layer can rebuild currentHitGrid directly via addToCurrentHitGridClipped. This
|
|
114
|
+
// updates hover states immediately rather than waiting for the next render.
|
|
115
|
+
//
|
|
116
|
+
// Scissor clipping: The hitScissorStack mirrors overflow:hidden regions. Elements
|
|
117
|
+
// outside their parent's visible area are excluded from hit testing. The stack
|
|
118
|
+
// uses screen coordinates. Buffered renderables need getHitGridScissorRect() to
|
|
119
|
+
// convert from buffer-local (0,0) to their actual screen position.
|
|
120
|
+
currentHitGrid: []u32,
|
|
121
|
+
nextHitGrid: []u32,
|
|
122
|
+
hitGridWidth: u32,
|
|
123
|
+
hitGridHeight: u32,
|
|
124
|
+
hitScissorStack: std.ArrayListUnmanaged(buf.ClipRect),
|
|
125
|
+
hitGridDirty: bool = false,
|
|
126
|
+
|
|
127
|
+
lastCursorStyleTag: ?u8 = null,
|
|
128
|
+
lastCursorBlinking: ?bool = null,
|
|
129
|
+
lastCursorColorRGB: ?[3]u8 = null,
|
|
130
|
+
lastMousePointerStyle: Terminal.MousePointerStyle = .default,
|
|
131
|
+
|
|
132
|
+
// Preallocated output buffer
|
|
133
|
+
var outputBuffer: [OUTPUT_BUFFER_SIZE]u8 = undefined;
|
|
134
|
+
var outputBufferLen: usize = 0;
|
|
135
|
+
var outputBufferB: [OUTPUT_BUFFER_SIZE]u8 = undefined;
|
|
136
|
+
var outputBufferBLen: usize = 0;
|
|
137
|
+
var activeBuffer: enum { A, B } = .A;
|
|
138
|
+
|
|
139
|
+
const OutputBufferWriter = struct {
|
|
140
|
+
pub fn write(_: void, data: []const u8) !usize {
|
|
141
|
+
const bufferLen = if (activeBuffer == .A) &outputBufferLen else &outputBufferBLen;
|
|
142
|
+
const buffer = if (activeBuffer == .A) &outputBuffer else &outputBufferB;
|
|
143
|
+
|
|
144
|
+
if (bufferLen.* + data.len > buffer.len) {
|
|
145
|
+
// TODO: Resize buffer when necessary
|
|
146
|
+
return error.BufferFull;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
@memcpy(buffer.*[bufferLen.*..][0..data.len], data);
|
|
150
|
+
bufferLen.* += data.len;
|
|
151
|
+
|
|
152
|
+
return data.len;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
// TODO: std.io.GenericWriter is deprecated, however the "correct" option seems to be much more involved
|
|
156
|
+
// So I have simply used GenericWriter here, and then the proper migration can be done later
|
|
157
|
+
pub fn writer() std.io.GenericWriter(void, error{BufferFull}, write) {
|
|
158
|
+
return .{ .context = {} };
|
|
159
|
+
}
|
|
160
|
+
};
|
|
161
|
+
|
|
162
|
+
pub fn create(allocator: Allocator, width: u32, height: u32, pool: *gp.GraphemePool, testing: bool) !*CliRenderer {
|
|
163
|
+
return createWithOptions(allocator, width, height, pool, testing, false);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
pub fn createWithOptions(
|
|
167
|
+
allocator: Allocator,
|
|
168
|
+
width: u32,
|
|
169
|
+
height: u32,
|
|
170
|
+
pool: *gp.GraphemePool,
|
|
171
|
+
testing: bool,
|
|
172
|
+
remote: bool,
|
|
173
|
+
) !*CliRenderer {
|
|
174
|
+
const self = try allocator.create(CliRenderer);
|
|
175
|
+
errdefer allocator.destroy(self);
|
|
176
|
+
|
|
177
|
+
const currentBuffer = try OptimizedBuffer.init(allocator, width, height, .{ .pool = pool, .width_method = .unicode, .id = "current buffer" });
|
|
178
|
+
errdefer currentBuffer.deinit();
|
|
179
|
+
const nextBuffer = try OptimizedBuffer.init(allocator, width, height, .{ .pool = pool, .width_method = .unicode, .id = "next buffer" });
|
|
180
|
+
errdefer nextBuffer.deinit();
|
|
181
|
+
|
|
182
|
+
// stat sample arrays
|
|
183
|
+
var lastFrameTime: std.ArrayListUnmanaged(f64) = .{};
|
|
184
|
+
errdefer lastFrameTime.deinit(allocator);
|
|
185
|
+
var renderTime: std.ArrayListUnmanaged(f64) = .{};
|
|
186
|
+
errdefer renderTime.deinit(allocator);
|
|
187
|
+
var overallFrameTime: std.ArrayListUnmanaged(f64) = .{};
|
|
188
|
+
errdefer overallFrameTime.deinit(allocator);
|
|
189
|
+
var bufferResetTime: std.ArrayListUnmanaged(f64) = .{};
|
|
190
|
+
errdefer bufferResetTime.deinit(allocator);
|
|
191
|
+
var stdoutWriteTime: std.ArrayListUnmanaged(f64) = .{};
|
|
192
|
+
errdefer stdoutWriteTime.deinit(allocator);
|
|
193
|
+
var cellsUpdated: std.ArrayListUnmanaged(u32) = .{};
|
|
194
|
+
errdefer cellsUpdated.deinit(allocator);
|
|
195
|
+
var frameCallbackTimes: std.ArrayListUnmanaged(f64) = .{};
|
|
196
|
+
errdefer frameCallbackTimes.deinit(allocator);
|
|
197
|
+
|
|
198
|
+
try lastFrameTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
199
|
+
try renderTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
200
|
+
try overallFrameTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
201
|
+
try bufferResetTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
202
|
+
try stdoutWriteTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
203
|
+
try cellsUpdated.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
204
|
+
try frameCallbackTimes.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
|
|
205
|
+
|
|
206
|
+
const hitGridSize = width * height;
|
|
207
|
+
const currentHitGrid = try allocator.alloc(u32, hitGridSize);
|
|
208
|
+
errdefer allocator.free(currentHitGrid);
|
|
209
|
+
const nextHitGrid = try allocator.alloc(u32, hitGridSize);
|
|
210
|
+
errdefer allocator.free(nextHitGrid);
|
|
211
|
+
@memset(currentHitGrid, 0); // Initialize with 0 (no renderable)
|
|
212
|
+
@memset(nextHitGrid, 0);
|
|
213
|
+
const hitScissorStack: std.ArrayListUnmanaged(buf.ClipRect) = .{};
|
|
214
|
+
|
|
215
|
+
self.* = .{
|
|
216
|
+
.width = width,
|
|
217
|
+
.height = height,
|
|
218
|
+
.currentRenderBuffer = currentBuffer,
|
|
219
|
+
.nextRenderBuffer = nextBuffer,
|
|
220
|
+
.pool = pool,
|
|
221
|
+
.backgroundColor = .{ 0.0, 0.0, 0.0, 0.0 },
|
|
222
|
+
.renderOffset = 0,
|
|
223
|
+
.terminal = Terminal.init(.{ .remote = remote }),
|
|
224
|
+
.testing = testing,
|
|
225
|
+
.lastCursorStyleTag = null,
|
|
226
|
+
.lastCursorBlinking = null,
|
|
227
|
+
.lastCursorColorRGB = null,
|
|
228
|
+
|
|
229
|
+
.renderStats = .{
|
|
230
|
+
.lastFrameTime = 0,
|
|
231
|
+
.averageFrameTime = 0,
|
|
232
|
+
.frameCount = 0,
|
|
233
|
+
.fps = 0,
|
|
234
|
+
.cellsUpdated = 0,
|
|
235
|
+
.renderTime = null,
|
|
236
|
+
.overallFrameTime = null,
|
|
237
|
+
.bufferResetTime = null,
|
|
238
|
+
.stdoutWriteTime = null,
|
|
239
|
+
.heapUsed = 0,
|
|
240
|
+
.heapTotal = 0,
|
|
241
|
+
.arrayBuffers = 0,
|
|
242
|
+
.frameCallbackTime = null,
|
|
243
|
+
},
|
|
244
|
+
.statSamples = .{
|
|
245
|
+
.lastFrameTime = lastFrameTime,
|
|
246
|
+
.renderTime = renderTime,
|
|
247
|
+
.overallFrameTime = overallFrameTime,
|
|
248
|
+
.bufferResetTime = bufferResetTime,
|
|
249
|
+
.stdoutWriteTime = stdoutWriteTime,
|
|
250
|
+
.cellsUpdated = cellsUpdated,
|
|
251
|
+
.frameCallbackTime = frameCallbackTimes,
|
|
252
|
+
},
|
|
253
|
+
.lastRenderTime = std.time.microTimestamp(),
|
|
254
|
+
.allocator = allocator,
|
|
255
|
+
.stdoutBuffer = undefined,
|
|
256
|
+
.currentHitGrid = currentHitGrid,
|
|
257
|
+
.nextHitGrid = nextHitGrid,
|
|
258
|
+
.hitGridWidth = width,
|
|
259
|
+
.hitGridHeight = height,
|
|
260
|
+
.hitScissorStack = hitScissorStack,
|
|
261
|
+
};
|
|
262
|
+
|
|
263
|
+
nextBuffer.setBlendBackdropColor(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], 1.0 });
|
|
264
|
+
|
|
265
|
+
try currentBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, CLEAR_CHAR);
|
|
266
|
+
try nextBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null);
|
|
267
|
+
|
|
268
|
+
return self;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
pub fn destroy(self: *CliRenderer) void {
|
|
272
|
+
self.renderMutex.lock();
|
|
273
|
+
while (self.renderInProgress) {
|
|
274
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
self.shouldTerminate = true;
|
|
278
|
+
self.renderRequested = true;
|
|
279
|
+
self.renderCondition.signal();
|
|
280
|
+
self.renderMutex.unlock();
|
|
281
|
+
|
|
282
|
+
if (self.renderThread) |thread| {
|
|
283
|
+
thread.join();
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
self.performShutdownSequence();
|
|
287
|
+
self.terminal.deinit();
|
|
288
|
+
|
|
289
|
+
self.currentRenderBuffer.deinit();
|
|
290
|
+
self.nextRenderBuffer.deinit();
|
|
291
|
+
|
|
292
|
+
// Free stat sample arrays
|
|
293
|
+
self.statSamples.lastFrameTime.deinit(self.allocator);
|
|
294
|
+
self.statSamples.renderTime.deinit(self.allocator);
|
|
295
|
+
self.statSamples.overallFrameTime.deinit(self.allocator);
|
|
296
|
+
self.statSamples.bufferResetTime.deinit(self.allocator);
|
|
297
|
+
self.statSamples.stdoutWriteTime.deinit(self.allocator);
|
|
298
|
+
self.statSamples.cellsUpdated.deinit(self.allocator);
|
|
299
|
+
self.statSamples.frameCallbackTime.deinit(self.allocator);
|
|
300
|
+
|
|
301
|
+
self.allocator.free(self.currentHitGrid);
|
|
302
|
+
self.allocator.free(self.nextHitGrid);
|
|
303
|
+
self.hitScissorStack.deinit(self.allocator);
|
|
304
|
+
|
|
305
|
+
self.allocator.destroy(self);
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
pub fn setupTerminal(self: *CliRenderer, useAlternateScreen: bool) void {
|
|
309
|
+
self.useAlternateScreen = useAlternateScreen;
|
|
310
|
+
self.terminalSetup = true;
|
|
311
|
+
|
|
312
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
313
|
+
const writer = &stdoutWriter.interface;
|
|
314
|
+
|
|
315
|
+
self.terminal.queryTerminalSend(writer) catch {
|
|
316
|
+
logger.warn("Failed to query terminal capabilities", .{});
|
|
317
|
+
};
|
|
318
|
+
writer.flush() catch {};
|
|
319
|
+
|
|
320
|
+
self.setupTerminalWithoutDetection(useAlternateScreen);
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
fn setupTerminalWithoutDetection(self: *CliRenderer, useAlternateScreen: bool) void {
|
|
324
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
325
|
+
const writer = &stdoutWriter.interface;
|
|
326
|
+
|
|
327
|
+
writer.writeAll(ansi.ANSI.saveCursorState) catch {};
|
|
328
|
+
|
|
329
|
+
if (useAlternateScreen) {
|
|
330
|
+
self.terminal.enterAltScreen(writer) catch {};
|
|
331
|
+
} else {
|
|
332
|
+
ansi.ANSI.makeRoomForRendererOutput(writer, @max(self.height, 1)) catch {};
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
self.terminal.setCursorPosition(1, 1, false);
|
|
336
|
+
const useKitty = self.terminal.opts.kitty_keyboard_flags > 0;
|
|
337
|
+
self.terminal.enableDetectedFeatures(writer, useKitty) catch {};
|
|
338
|
+
|
|
339
|
+
writer.flush() catch {};
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
pub fn suspendRenderer(self: *CliRenderer) void {
|
|
343
|
+
if (!self.terminalSetup) return;
|
|
344
|
+
self.performShutdownSequence();
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
pub fn resumeRenderer(self: *CliRenderer) void {
|
|
348
|
+
if (!self.terminalSetup) return;
|
|
349
|
+
self.setupTerminalWithoutDetection(self.useAlternateScreen);
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
pub fn performShutdownSequence(self: *CliRenderer) void {
|
|
353
|
+
if (!self.terminalSetup) return;
|
|
354
|
+
|
|
355
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
356
|
+
const direct = &stdoutWriter.interface;
|
|
357
|
+
self.terminal.resetState(direct) catch {
|
|
358
|
+
logger.warn("Failed to reset terminal state", .{});
|
|
359
|
+
};
|
|
360
|
+
|
|
361
|
+
if (self.useAlternateScreen) {
|
|
362
|
+
direct.flush() catch {};
|
|
363
|
+
} else if (self.renderOffset == 0) {
|
|
364
|
+
direct.writeAll("\x1b[H\x1b[J") catch {};
|
|
365
|
+
direct.flush() catch {};
|
|
366
|
+
} else if (self.renderOffset > 0) {
|
|
367
|
+
// Currently still handled in typescript
|
|
368
|
+
// const consoleEndLine = self.height - self.renderOffset;
|
|
369
|
+
// ansi.ANSI.moveToOutput(direct, 1, consoleEndLine) catch {};
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
// NOTE: This messes up state after shutdown, but might be necessary for windows?
|
|
373
|
+
// direct.writeAll(ansi.ANSI.restoreCursorState) catch {};
|
|
374
|
+
|
|
375
|
+
direct.writeAll(ansi.ANSI.resetCursorColorFallback) catch {};
|
|
376
|
+
direct.writeAll(ansi.ANSI.resetCursorColor) catch {};
|
|
377
|
+
direct.writeAll(ansi.ANSI.defaultCursorStyle) catch {};
|
|
378
|
+
// Workaround for Ghostty not showing the cursor after shutdown for some reason
|
|
379
|
+
direct.writeAll(ansi.ANSI.showCursor) catch {};
|
|
380
|
+
direct.flush() catch {};
|
|
381
|
+
std.Thread.sleep(10 * std.time.ns_per_ms);
|
|
382
|
+
direct.writeAll(ansi.ANSI.showCursor) catch {};
|
|
383
|
+
direct.flush() catch {};
|
|
384
|
+
std.Thread.sleep(10 * std.time.ns_per_ms);
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
fn addStatSample(self: *CliRenderer, comptime T: type, samples: *std.ArrayListUnmanaged(T), value: T) void {
|
|
388
|
+
samples.append(self.allocator, value) catch return;
|
|
389
|
+
|
|
390
|
+
if (samples.items.len > MAX_STAT_SAMPLES) {
|
|
391
|
+
_ = samples.orderedRemove(0);
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
|
|
395
|
+
fn getStatAverage(comptime T: type, samples: *const std.ArrayListUnmanaged(T)) T {
|
|
396
|
+
if (samples.items.len == 0) {
|
|
397
|
+
return 0;
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
var sum: T = 0;
|
|
401
|
+
for (samples.items) |value| {
|
|
402
|
+
sum += value;
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
if (@typeInfo(T) == .float) {
|
|
406
|
+
return sum / @as(T, @floatFromInt(samples.items.len));
|
|
407
|
+
} else {
|
|
408
|
+
return sum / @as(T, @intCast(samples.items.len));
|
|
409
|
+
}
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
pub fn setUseThread(self: *CliRenderer, useThread: bool) void {
|
|
413
|
+
if (self.useThread == useThread) return;
|
|
414
|
+
|
|
415
|
+
if (useThread) {
|
|
416
|
+
if (self.renderThread == null) {
|
|
417
|
+
self.renderThread = std.Thread.spawn(.{}, renderThreadFn, .{self}) catch |err| {
|
|
418
|
+
std.log.warn("Failed to spawn render thread: {}, falling back to non-threaded mode", .{err});
|
|
419
|
+
self.useThread = false;
|
|
420
|
+
return;
|
|
421
|
+
};
|
|
422
|
+
}
|
|
423
|
+
} else {
|
|
424
|
+
if (self.renderThread) |thread| {
|
|
425
|
+
// Signal the render thread to terminate (same pattern as destroy)
|
|
426
|
+
self.renderMutex.lock();
|
|
427
|
+
while (self.renderInProgress) {
|
|
428
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
429
|
+
}
|
|
430
|
+
self.shouldTerminate = true;
|
|
431
|
+
self.renderRequested = true;
|
|
432
|
+
self.renderCondition.signal();
|
|
433
|
+
self.renderMutex.unlock();
|
|
434
|
+
|
|
435
|
+
thread.join();
|
|
436
|
+
self.renderThread = null;
|
|
437
|
+
|
|
438
|
+
// Reset termination flag so thread can be re-enabled later
|
|
439
|
+
self.shouldTerminate = false;
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
self.useThread = useThread;
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
pub fn updateStats(self: *CliRenderer, time: f64, fps: u32, frameCallbackTime: f64) void {
|
|
447
|
+
self.renderStats.overallFrameTime = time;
|
|
448
|
+
self.renderStats.fps = fps;
|
|
449
|
+
self.renderStats.frameCallbackTime = frameCallbackTime;
|
|
450
|
+
|
|
451
|
+
self.addStatSample(f64, &self.statSamples.overallFrameTime, time);
|
|
452
|
+
self.addStatSample(f64, &self.statSamples.frameCallbackTime, frameCallbackTime);
|
|
453
|
+
}
|
|
454
|
+
|
|
455
|
+
pub fn updateMemoryStats(self: *CliRenderer, heapUsed: u32, heapTotal: u32, arrayBuffers: u32) void {
|
|
456
|
+
self.renderStats.heapUsed = heapUsed;
|
|
457
|
+
self.renderStats.heapTotal = heapTotal;
|
|
458
|
+
self.renderStats.arrayBuffers = arrayBuffers;
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
pub fn resize(self: *CliRenderer, width: u32, height: u32) !void {
|
|
462
|
+
if (self.width == width and self.height == height) return;
|
|
463
|
+
|
|
464
|
+
self.width = width;
|
|
465
|
+
self.height = height;
|
|
466
|
+
|
|
467
|
+
try self.currentRenderBuffer.resize(width, height);
|
|
468
|
+
try self.nextRenderBuffer.resize(width, height);
|
|
469
|
+
self.nextRenderBuffer.setBlendBackdropColor(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], 1.0 });
|
|
470
|
+
|
|
471
|
+
try self.currentRenderBuffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, CLEAR_CHAR);
|
|
472
|
+
try self.nextRenderBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null);
|
|
473
|
+
|
|
474
|
+
const newHitGridSize = width * height;
|
|
475
|
+
const currentHitGridSize = self.hitGridWidth * self.hitGridHeight;
|
|
476
|
+
if (newHitGridSize > currentHitGridSize) {
|
|
477
|
+
const newCurrentHitGrid = try self.allocator.alloc(u32, newHitGridSize);
|
|
478
|
+
errdefer self.allocator.free(newCurrentHitGrid);
|
|
479
|
+
const newNextHitGrid = try self.allocator.alloc(u32, newHitGridSize);
|
|
480
|
+
@memset(newCurrentHitGrid, 0);
|
|
481
|
+
@memset(newNextHitGrid, 0);
|
|
482
|
+
|
|
483
|
+
self.allocator.free(self.currentHitGrid);
|
|
484
|
+
self.allocator.free(self.nextHitGrid);
|
|
485
|
+
self.currentHitGrid = newCurrentHitGrid;
|
|
486
|
+
self.nextHitGrid = newNextHitGrid;
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
// Always update dimensions. The backing buffer is at least as large as
|
|
490
|
+
// width*height, so this is safe even when the terminal shrinks. Without
|
|
491
|
+
// this, checkHit keeps using stale dimensions after a shrink and returns
|
|
492
|
+
// 0 for any coordinate beyond the old bounds.
|
|
493
|
+
self.hitGridWidth = width;
|
|
494
|
+
self.hitGridHeight = height;
|
|
495
|
+
|
|
496
|
+
const cursor = self.terminal.getCursorPosition();
|
|
497
|
+
self.terminal.setCursorPosition(@min(cursor.x, width), @min(cursor.y, height), cursor.visible);
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
pub fn setBackgroundColor(self: *CliRenderer, rgba: RGBA) void {
|
|
501
|
+
self.backgroundColor = rgba;
|
|
502
|
+
self.nextRenderBuffer.setBlendBackdropColor(.{ rgba[0], rgba[1], rgba[2], 1.0 });
|
|
503
|
+
}
|
|
504
|
+
|
|
505
|
+
pub fn setRenderOffset(self: *CliRenderer, offset: u32) void {
|
|
506
|
+
self.renderOffset = offset;
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
fn renderThreadFn(self: *CliRenderer) void {
|
|
510
|
+
while (true) {
|
|
511
|
+
self.renderMutex.lock();
|
|
512
|
+
while (!self.renderRequested and !self.shouldTerminate) {
|
|
513
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
514
|
+
}
|
|
515
|
+
|
|
516
|
+
if (self.shouldTerminate and !self.renderRequested) {
|
|
517
|
+
self.renderMutex.unlock();
|
|
518
|
+
break;
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
self.renderRequested = false;
|
|
522
|
+
|
|
523
|
+
const outputData = self.currentOutputBuffer;
|
|
524
|
+
const outputLen = self.currentOutputLen;
|
|
525
|
+
|
|
526
|
+
const writeStart = std.time.microTimestamp();
|
|
527
|
+
|
|
528
|
+
if (outputLen > 0 and !self.testing) {
|
|
529
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
530
|
+
const w = &stdoutWriter.interface;
|
|
531
|
+
w.writeAll(outputData[0..outputLen]) catch {};
|
|
532
|
+
w.flush() catch {};
|
|
533
|
+
}
|
|
534
|
+
|
|
535
|
+
// Signal that rendering is complete
|
|
536
|
+
self.renderStats.stdoutWriteTime = @as(f64, @floatFromInt(std.time.microTimestamp() - writeStart));
|
|
537
|
+
self.renderInProgress = false;
|
|
538
|
+
self.renderCondition.signal();
|
|
539
|
+
self.renderMutex.unlock();
|
|
540
|
+
}
|
|
541
|
+
}
|
|
542
|
+
|
|
543
|
+
// Render once with current state
|
|
544
|
+
pub fn render(self: *CliRenderer, force: bool) void {
|
|
545
|
+
const now = std.time.microTimestamp();
|
|
546
|
+
const deltaTimeMs = @as(f64, @floatFromInt(now - self.lastRenderTime));
|
|
547
|
+
const deltaTime = deltaTimeMs / 1000.0; // Convert to seconds
|
|
548
|
+
|
|
549
|
+
self.lastRenderTime = now;
|
|
550
|
+
self.renderDebugOverlay();
|
|
551
|
+
|
|
552
|
+
self.prepareRenderFrame(force);
|
|
553
|
+
|
|
554
|
+
if (self.useThread) {
|
|
555
|
+
self.renderMutex.lock();
|
|
556
|
+
while (self.renderInProgress) {
|
|
557
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
558
|
+
}
|
|
559
|
+
|
|
560
|
+
if (activeBuffer == .A) {
|
|
561
|
+
activeBuffer = .B;
|
|
562
|
+
self.currentOutputBuffer = &outputBuffer;
|
|
563
|
+
self.currentOutputLen = outputBufferLen;
|
|
564
|
+
} else {
|
|
565
|
+
activeBuffer = .A;
|
|
566
|
+
self.currentOutputBuffer = &outputBufferB;
|
|
567
|
+
self.currentOutputLen = outputBufferBLen;
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
self.renderRequested = true;
|
|
571
|
+
self.renderInProgress = true;
|
|
572
|
+
self.renderCondition.signal();
|
|
573
|
+
self.renderMutex.unlock();
|
|
574
|
+
} else {
|
|
575
|
+
const writeStart = std.time.microTimestamp();
|
|
576
|
+
if (!self.testing) {
|
|
577
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
578
|
+
const w = &stdoutWriter.interface;
|
|
579
|
+
w.writeAll(outputBuffer[0..outputBufferLen]) catch {};
|
|
580
|
+
w.flush() catch {};
|
|
581
|
+
}
|
|
582
|
+
self.renderStats.stdoutWriteTime = @as(f64, @floatFromInt(std.time.microTimestamp() - writeStart));
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
self.renderStats.lastFrameTime = deltaTime * 1000.0;
|
|
586
|
+
self.renderStats.frameCount += 1;
|
|
587
|
+
|
|
588
|
+
self.addStatSample(f64, &self.statSamples.lastFrameTime, deltaTime * 1000.0);
|
|
589
|
+
if (self.renderStats.renderTime) |rt| {
|
|
590
|
+
self.addStatSample(f64, &self.statSamples.renderTime, rt);
|
|
591
|
+
}
|
|
592
|
+
if (self.renderStats.bufferResetTime) |brt| {
|
|
593
|
+
self.addStatSample(f64, &self.statSamples.bufferResetTime, brt);
|
|
594
|
+
}
|
|
595
|
+
if (self.renderStats.stdoutWriteTime) |swt| {
|
|
596
|
+
self.addStatSample(f64, &self.statSamples.stdoutWriteTime, swt);
|
|
597
|
+
}
|
|
598
|
+
self.addStatSample(u32, &self.statSamples.cellsUpdated, self.renderStats.cellsUpdated);
|
|
599
|
+
}
|
|
600
|
+
|
|
601
|
+
pub fn getNextBuffer(self: *CliRenderer) *OptimizedBuffer {
|
|
602
|
+
return self.nextRenderBuffer;
|
|
603
|
+
}
|
|
604
|
+
|
|
605
|
+
pub fn getCurrentBuffer(self: *CliRenderer) *OptimizedBuffer {
|
|
606
|
+
return self.currentRenderBuffer;
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
fn prepareRenderFrame(self: *CliRenderer, force: bool) void {
|
|
610
|
+
const renderStartTime = std.time.microTimestamp();
|
|
611
|
+
var cellsUpdated: u32 = 0;
|
|
612
|
+
|
|
613
|
+
if (activeBuffer == .A) {
|
|
614
|
+
outputBufferLen = 0;
|
|
615
|
+
} else {
|
|
616
|
+
outputBufferBLen = 0;
|
|
617
|
+
}
|
|
618
|
+
|
|
619
|
+
var writer = OutputBufferWriter.writer();
|
|
620
|
+
|
|
621
|
+
writer.writeAll(ansi.ANSI.syncSet) catch {};
|
|
622
|
+
writer.writeAll(ansi.ANSI.hideCursor) catch {};
|
|
623
|
+
|
|
624
|
+
var currentFg: ?RGBA = null;
|
|
625
|
+
var currentBg: ?RGBA = null;
|
|
626
|
+
var currentAttributes: i32 = -1;
|
|
627
|
+
var currentLinkId: u32 = 0;
|
|
628
|
+
var utf8Buf: [4]u8 = undefined;
|
|
629
|
+
|
|
630
|
+
const colorEpsilon: f32 = COLOR_EPSILON_DEFAULT;
|
|
631
|
+
const hyperlinksEnabled = self.terminal.getCapabilities().hyperlinks;
|
|
632
|
+
|
|
633
|
+
for (0..self.height) |uy| {
|
|
634
|
+
const y = @as(u32, @intCast(uy));
|
|
635
|
+
|
|
636
|
+
var runStart: i64 = -1;
|
|
637
|
+
var runLength: u32 = 0;
|
|
638
|
+
|
|
639
|
+
for (0..self.width) |ux| {
|
|
640
|
+
const x = @as(u32, @intCast(ux));
|
|
641
|
+
const currentCell = self.currentRenderBuffer.get(x, y);
|
|
642
|
+
const nextCell = self.nextRenderBuffer.get(x, y);
|
|
643
|
+
|
|
644
|
+
if (currentCell == null or nextCell == null) continue;
|
|
645
|
+
|
|
646
|
+
if (!force) {
|
|
647
|
+
const charEqual = currentCell.?.char == nextCell.?.char;
|
|
648
|
+
const attrEqual = currentCell.?.attributes == nextCell.?.attributes;
|
|
649
|
+
|
|
650
|
+
if (charEqual and attrEqual and
|
|
651
|
+
buf.rgbaEqual(currentCell.?.fg, nextCell.?.fg, colorEpsilon) and
|
|
652
|
+
buf.rgbaEqual(currentCell.?.bg, nextCell.?.bg, colorEpsilon))
|
|
653
|
+
{
|
|
654
|
+
if (runLength > 0) {
|
|
655
|
+
writer.writeAll(ansi.ANSI.reset) catch {};
|
|
656
|
+
runStart = -1;
|
|
657
|
+
runLength = 0;
|
|
658
|
+
}
|
|
659
|
+
continue;
|
|
660
|
+
}
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
const cell = nextCell.?;
|
|
664
|
+
|
|
665
|
+
const fgMatch = currentFg != null and buf.rgbaEqual(currentFg.?, cell.fg, colorEpsilon);
|
|
666
|
+
const bgMatch = currentBg != null and buf.rgbaEqual(currentBg.?, cell.bg, colorEpsilon);
|
|
667
|
+
const sameAttributes = fgMatch and bgMatch and @as(i32, @intCast(cell.attributes)) == currentAttributes;
|
|
668
|
+
|
|
669
|
+
const linkId = if (hyperlinksEnabled) ansi.TextAttributes.getLinkId(cell.attributes) else 0;
|
|
670
|
+
|
|
671
|
+
if (hyperlinksEnabled and linkId != currentLinkId) {
|
|
672
|
+
if (currentLinkId != 0) {
|
|
673
|
+
writer.writeAll("\x1b]8;;\x1b\\") catch {};
|
|
674
|
+
}
|
|
675
|
+
currentLinkId = linkId;
|
|
676
|
+
if (currentLinkId != 0) {
|
|
677
|
+
const lp = link.initGlobalLinkPool(self.allocator);
|
|
678
|
+
if (lp.get(currentLinkId)) |url_bytes| {
|
|
679
|
+
writer.print("\x1b]8;id={d};{s}\x1b\\", .{ currentLinkId, url_bytes }) catch {};
|
|
680
|
+
} else |_| {
|
|
681
|
+
// Link not found, treat as no link
|
|
682
|
+
currentLinkId = 0;
|
|
683
|
+
}
|
|
684
|
+
}
|
|
685
|
+
}
|
|
686
|
+
|
|
687
|
+
if (!sameAttributes or runStart == -1) {
|
|
688
|
+
if (runLength > 0) {
|
|
689
|
+
writer.writeAll(ansi.ANSI.reset) catch {};
|
|
690
|
+
}
|
|
691
|
+
|
|
692
|
+
runStart = @intCast(x);
|
|
693
|
+
runLength = 0;
|
|
694
|
+
|
|
695
|
+
currentFg = cell.fg;
|
|
696
|
+
currentBg = cell.bg;
|
|
697
|
+
currentAttributes = @as(i32, @intCast(cell.attributes));
|
|
698
|
+
|
|
699
|
+
ansi.ANSI.moveToOutput(writer, x + 1, y + 1 + self.renderOffset) catch {};
|
|
700
|
+
|
|
701
|
+
const fgR = rgbaComponentToU8(cell.fg[0]);
|
|
702
|
+
const fgG = rgbaComponentToU8(cell.fg[1]);
|
|
703
|
+
const fgB = rgbaComponentToU8(cell.fg[2]);
|
|
704
|
+
|
|
705
|
+
const bgR = rgbaComponentToU8(cell.bg[0]);
|
|
706
|
+
const bgG = rgbaComponentToU8(cell.bg[1]);
|
|
707
|
+
const bgB = rgbaComponentToU8(cell.bg[2]);
|
|
708
|
+
const bgA = cell.bg[3];
|
|
709
|
+
|
|
710
|
+
ansi.ANSI.fgColorOutput(writer, fgR, fgG, fgB) catch {};
|
|
711
|
+
|
|
712
|
+
// If alpha is 0 (transparent), use terminal default background instead of black
|
|
713
|
+
if (bgA < 0.001) {
|
|
714
|
+
writer.writeAll("\x1b[49m") catch {};
|
|
715
|
+
} else {
|
|
716
|
+
ansi.ANSI.bgColorOutput(writer, bgR, bgG, bgB) catch {};
|
|
717
|
+
}
|
|
718
|
+
|
|
719
|
+
ansi.TextAttributes.applyAttributesOutputWriter(writer, cell.attributes) catch {};
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
// Handle grapheme characters
|
|
723
|
+
if (gp.isGraphemeChar(cell.char)) {
|
|
724
|
+
const gid: u32 = gp.graphemeIdFromChar(cell.char);
|
|
725
|
+
const bytes = self.pool.get(gid) catch |err| {
|
|
726
|
+
self.performShutdownSequence();
|
|
727
|
+
std.debug.panic("Fatal: no grapheme bytes in pool for gid {d}: {}", .{ gid, err });
|
|
728
|
+
};
|
|
729
|
+
if (bytes.len > 0) {
|
|
730
|
+
const capabilities = self.terminal.getCapabilities();
|
|
731
|
+
const graphemeWidth = gp.charRightExtent(cell.char) + 1;
|
|
732
|
+
if (capabilities.explicit_width) {
|
|
733
|
+
ansi.ANSI.explicitWidthOutput(writer, graphemeWidth, bytes) catch {};
|
|
734
|
+
} else {
|
|
735
|
+
writer.writeAll(bytes) catch {};
|
|
736
|
+
if (capabilities.explicit_cursor_positioning) {
|
|
737
|
+
const nextX = x + graphemeWidth;
|
|
738
|
+
if (nextX < self.width) {
|
|
739
|
+
ansi.ANSI.moveToOutput(writer, nextX + 1, y + 1 + self.renderOffset) catch {};
|
|
740
|
+
}
|
|
741
|
+
}
|
|
742
|
+
}
|
|
743
|
+
}
|
|
744
|
+
} else if (gp.isContinuationChar(cell.char)) {
|
|
745
|
+
// Intentionally do not write a space for continuation cells.
|
|
746
|
+
// NOTE: disabled to fix 2-cell emoji rendering when the two
|
|
747
|
+
// cells have distinct colors (space overwrite can break glyph output)
|
|
748
|
+
|
|
749
|
+
// writer.writeByte(' ') catch {};
|
|
750
|
+
} else {
|
|
751
|
+
const len = std.unicode.utf8Encode(@intCast(cell.char), &utf8Buf) catch 1;
|
|
752
|
+
writer.writeAll(utf8Buf[0..len]) catch {};
|
|
753
|
+
}
|
|
754
|
+
runLength += 1;
|
|
755
|
+
|
|
756
|
+
// Sync this cell to the current buffer so the next frame's diff
|
|
757
|
+
// is correct. Use syncCell (set without span cleanup) because
|
|
758
|
+
// span cleanup would destroy continuation cells written by an
|
|
759
|
+
// earlier iteration of this same left-to-right pass (#723).
|
|
760
|
+
self.currentRenderBuffer.syncCell(x, y, nextCell.?);
|
|
761
|
+
|
|
762
|
+
cellsUpdated += 1;
|
|
763
|
+
}
|
|
764
|
+
}
|
|
765
|
+
|
|
766
|
+
if (hyperlinksEnabled and currentLinkId != 0) {
|
|
767
|
+
writer.writeAll("\x1b]8;;\x1b\\") catch {};
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
writer.writeAll(ansi.ANSI.reset) catch {};
|
|
771
|
+
|
|
772
|
+
const cursorPos = self.terminal.getCursorPosition();
|
|
773
|
+
const cursorStyle = self.terminal.getCursorStyle();
|
|
774
|
+
const cursorColor = self.terminal.getCursorColor();
|
|
775
|
+
|
|
776
|
+
if (cursorPos.visible) {
|
|
777
|
+
var cursorStyleCode: []const u8 = undefined;
|
|
778
|
+
|
|
779
|
+
switch (cursorStyle.style) {
|
|
780
|
+
.block => {
|
|
781
|
+
cursorStyleCode = if (cursorStyle.blinking)
|
|
782
|
+
ansi.ANSI.cursorBlockBlink
|
|
783
|
+
else
|
|
784
|
+
ansi.ANSI.cursorBlock;
|
|
785
|
+
},
|
|
786
|
+
.line => {
|
|
787
|
+
cursorStyleCode = if (cursorStyle.blinking)
|
|
788
|
+
ansi.ANSI.cursorLineBlink
|
|
789
|
+
else
|
|
790
|
+
ansi.ANSI.cursorLine;
|
|
791
|
+
},
|
|
792
|
+
.underline => {
|
|
793
|
+
cursorStyleCode = if (cursorStyle.blinking)
|
|
794
|
+
ansi.ANSI.cursorUnderlineBlink
|
|
795
|
+
else
|
|
796
|
+
ansi.ANSI.cursorUnderline;
|
|
797
|
+
},
|
|
798
|
+
.default => {
|
|
799
|
+
cursorStyleCode = ansi.ANSI.defaultCursorStyle;
|
|
800
|
+
},
|
|
801
|
+
}
|
|
802
|
+
|
|
803
|
+
const cursorR = rgbaComponentToU8(cursorColor[0]);
|
|
804
|
+
const cursorG = rgbaComponentToU8(cursorColor[1]);
|
|
805
|
+
const cursorB = rgbaComponentToU8(cursorColor[2]);
|
|
806
|
+
|
|
807
|
+
const styleTag: u8 = @intFromEnum(cursorStyle.style);
|
|
808
|
+
const styleChanged = (self.lastCursorStyleTag == null or self.lastCursorStyleTag.? != styleTag) or
|
|
809
|
+
(self.lastCursorBlinking == null or self.lastCursorBlinking.? != cursorStyle.blinking);
|
|
810
|
+
const colorChanged = (self.lastCursorColorRGB == null or self.lastCursorColorRGB.?[0] != cursorR or self.lastCursorColorRGB.?[1] != cursorG or self.lastCursorColorRGB.?[2] != cursorB);
|
|
811
|
+
|
|
812
|
+
if (colorChanged) {
|
|
813
|
+
ansi.ANSI.cursorColorOutputWriter(writer, cursorR, cursorG, cursorB) catch {};
|
|
814
|
+
self.lastCursorColorRGB = .{ cursorR, cursorG, cursorB };
|
|
815
|
+
}
|
|
816
|
+
if (styleChanged) {
|
|
817
|
+
writer.writeAll(cursorStyleCode) catch {};
|
|
818
|
+
self.lastCursorStyleTag = styleTag;
|
|
819
|
+
self.lastCursorBlinking = cursorStyle.blinking;
|
|
820
|
+
}
|
|
821
|
+
ansi.ANSI.moveToOutput(writer, cursorPos.x, cursorPos.y + self.renderOffset) catch {};
|
|
822
|
+
writer.writeAll(ansi.ANSI.showCursor) catch {};
|
|
823
|
+
} else {
|
|
824
|
+
writer.writeAll(ansi.ANSI.hideCursor) catch {};
|
|
825
|
+
self.lastCursorStyleTag = null;
|
|
826
|
+
self.lastCursorBlinking = null;
|
|
827
|
+
self.lastCursorColorRGB = null;
|
|
828
|
+
}
|
|
829
|
+
|
|
830
|
+
const mousePointer = self.terminal.getMousePointer();
|
|
831
|
+
if (mousePointer != self.lastMousePointerStyle) {
|
|
832
|
+
ansi.ANSI.setMousePointerOutput(writer, mousePointer.toName()) catch {};
|
|
833
|
+
self.lastMousePointerStyle = mousePointer;
|
|
834
|
+
}
|
|
835
|
+
|
|
836
|
+
writer.writeAll(ansi.ANSI.syncReset) catch {};
|
|
837
|
+
|
|
838
|
+
const renderEndTime = std.time.microTimestamp();
|
|
839
|
+
const renderTime = @as(f64, @floatFromInt(renderEndTime - renderStartTime));
|
|
840
|
+
|
|
841
|
+
self.renderStats.cellsUpdated = cellsUpdated;
|
|
842
|
+
self.renderStats.renderTime = renderTime;
|
|
843
|
+
|
|
844
|
+
self.nextRenderBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null) catch {};
|
|
845
|
+
|
|
846
|
+
// Compare hit grids before swap to detect changes. This allows TypeScript to
|
|
847
|
+
// know if hover state needs rechecking without manually tracking dirty state.
|
|
848
|
+
self.hitGridDirty = !std.mem.eql(u32, self.currentHitGrid, self.nextHitGrid);
|
|
849
|
+
|
|
850
|
+
// Swap hit grids: nextHitGrid (built this frame) becomes the active grid for
|
|
851
|
+
// hit testing. The old currentHitGrid becomes nextHitGrid and is cleared for
|
|
852
|
+
// the next frame.
|
|
853
|
+
const temp = self.currentHitGrid;
|
|
854
|
+
self.currentHitGrid = self.nextHitGrid;
|
|
855
|
+
self.nextHitGrid = temp;
|
|
856
|
+
@memset(self.nextHitGrid, 0);
|
|
857
|
+
}
|
|
858
|
+
|
|
859
|
+
pub fn setDebugOverlay(self: *CliRenderer, enabled: bool, corner: DebugOverlayCorner) void {
|
|
860
|
+
self.debugOverlay.enabled = enabled;
|
|
861
|
+
self.debugOverlay.corner = corner;
|
|
862
|
+
}
|
|
863
|
+
|
|
864
|
+
pub fn clearTerminal(self: *CliRenderer) void {
|
|
865
|
+
self.writeOut(ansi.ANSI.clearAndHome);
|
|
866
|
+
}
|
|
867
|
+
|
|
868
|
+
pub fn writeOut(self: *CliRenderer, data: []const u8) void {
|
|
869
|
+
if (data.len == 0) return;
|
|
870
|
+
if (self.testing) return;
|
|
871
|
+
|
|
872
|
+
if (self.useThread) {
|
|
873
|
+
self.renderMutex.lock();
|
|
874
|
+
while (self.renderInProgress) {
|
|
875
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
876
|
+
}
|
|
877
|
+
self.renderMutex.unlock();
|
|
878
|
+
}
|
|
879
|
+
|
|
880
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
881
|
+
const w = &stdoutWriter.interface;
|
|
882
|
+
w.writeAll(data) catch {};
|
|
883
|
+
w.flush() catch {};
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
pub fn writeOutMultiple(self: *CliRenderer, data_slices: []const []const u8) void {
|
|
887
|
+
if (self.testing) return;
|
|
888
|
+
|
|
889
|
+
if (self.useThread) {
|
|
890
|
+
self.renderMutex.lock();
|
|
891
|
+
while (self.renderInProgress) {
|
|
892
|
+
self.renderCondition.wait(&self.renderMutex);
|
|
893
|
+
}
|
|
894
|
+
self.renderMutex.unlock();
|
|
895
|
+
}
|
|
896
|
+
|
|
897
|
+
var totalLen: usize = 0;
|
|
898
|
+
for (data_slices) |slice| {
|
|
899
|
+
totalLen += slice.len;
|
|
900
|
+
}
|
|
901
|
+
|
|
902
|
+
if (totalLen == 0) return;
|
|
903
|
+
|
|
904
|
+
var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
|
|
905
|
+
const w = &stdoutWriter.interface;
|
|
906
|
+
for (data_slices) |slice| {
|
|
907
|
+
w.writeAll(slice) catch {};
|
|
908
|
+
}
|
|
909
|
+
w.flush() catch {};
|
|
910
|
+
}
|
|
911
|
+
|
|
912
|
+
/// Write a renderable's bounds to nextHitGrid for the upcoming frame.
|
|
913
|
+
///
|
|
914
|
+
/// Called during render for each visible renderable. The rect is clipped to
|
|
915
|
+
/// the current hit scissor stack, so elements inside overflow:hidden parents
|
|
916
|
+
/// only register hits within the visible region. Later renderables overwrite
|
|
917
|
+
/// earlier ones. Z-order is determined by render order.
|
|
918
|
+
pub fn addToHitGrid(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
|
|
919
|
+
const clipped = self.clipRectToHitScissor(x, y, width, height) orelse return;
|
|
920
|
+
const startX = @max(0, clipped.x);
|
|
921
|
+
const startY = @max(0, clipped.y);
|
|
922
|
+
const endX = @min(
|
|
923
|
+
@as(i32, @intCast(self.hitGridWidth)),
|
|
924
|
+
clipped.x + @as(i32, @intCast(clipped.width)),
|
|
925
|
+
);
|
|
926
|
+
const endY = @min(
|
|
927
|
+
@as(i32, @intCast(self.hitGridHeight)),
|
|
928
|
+
clipped.y + @as(i32, @intCast(clipped.height)),
|
|
929
|
+
);
|
|
930
|
+
|
|
931
|
+
if (startX >= endX or startY >= endY) return;
|
|
932
|
+
|
|
933
|
+
const uStartX: u32 = @intCast(startX);
|
|
934
|
+
const uStartY: u32 = @intCast(startY);
|
|
935
|
+
const uEndX: u32 = @intCast(endX);
|
|
936
|
+
const uEndY: u32 = @intCast(endY);
|
|
937
|
+
|
|
938
|
+
for (uStartY..uEndY) |row| {
|
|
939
|
+
const rowStart = row * self.hitGridWidth;
|
|
940
|
+
const startIdx = rowStart + uStartX;
|
|
941
|
+
const endIdx = rowStart + uEndX;
|
|
942
|
+
|
|
943
|
+
@memset(self.nextHitGrid[startIdx..endIdx], id);
|
|
944
|
+
}
|
|
945
|
+
}
|
|
946
|
+
|
|
947
|
+
/// Clear currentHitGrid before an immediate rebuild.
|
|
948
|
+
///
|
|
949
|
+
/// Used by syncHitGridIfNeeded in TypeScript when scroll/translate changes
|
|
950
|
+
/// require updating hit targets without waiting for the next render.
|
|
951
|
+
pub fn clearCurrentHitGrid(self: *CliRenderer) void {
|
|
952
|
+
@memset(self.currentHitGrid, 0);
|
|
953
|
+
}
|
|
954
|
+
|
|
955
|
+
/// Return whether the hit grid changed during the last render.
|
|
956
|
+
/// This is set by comparing the previous and current hit grids after render.
|
|
957
|
+
/// TypeScript can use this to decide if hover state needs rechecking.
|
|
958
|
+
pub fn getHitGridDirty(self: *CliRenderer) bool {
|
|
959
|
+
return self.hitGridDirty;
|
|
960
|
+
}
|
|
961
|
+
|
|
962
|
+
/// Return the renderable ID at screen position (x, y), or 0 if none.
|
|
963
|
+
pub fn checkHit(self: *CliRenderer, x: u32, y: u32) u32 {
|
|
964
|
+
if (x >= self.hitGridWidth or y >= self.hitGridHeight) {
|
|
965
|
+
return 0;
|
|
966
|
+
}
|
|
967
|
+
|
|
968
|
+
const index = y * self.hitGridWidth + x;
|
|
969
|
+
return self.currentHitGrid[index];
|
|
970
|
+
}
|
|
971
|
+
|
|
972
|
+
/// Return the current (topmost) hit scissor rect, or null if the stack is empty.
|
|
973
|
+
fn getCurrentHitScissorRect(self: *const CliRenderer) ?buf.ClipRect {
|
|
974
|
+
if (self.hitScissorStack.items.len == 0) return null;
|
|
975
|
+
return self.hitScissorStack.items[self.hitScissorStack.items.len - 1];
|
|
976
|
+
}
|
|
977
|
+
|
|
978
|
+
/// Intersect a rect with the current hit scissor. Returns null if fully clipped.
|
|
979
|
+
fn clipRectToHitScissor(self: *const CliRenderer, x: i32, y: i32, width: u32, height: u32) ?buf.ClipRect {
|
|
980
|
+
const scissor = self.getCurrentHitScissorRect() orelse return buf.ClipRect{
|
|
981
|
+
.x = x,
|
|
982
|
+
.y = y,
|
|
983
|
+
.width = width,
|
|
984
|
+
.height = height,
|
|
985
|
+
};
|
|
986
|
+
|
|
987
|
+
const rect_end_x = x + @as(i32, @intCast(width));
|
|
988
|
+
const rect_end_y = y + @as(i32, @intCast(height));
|
|
989
|
+
const scissor_end_x = scissor.x + @as(i32, @intCast(scissor.width));
|
|
990
|
+
const scissor_end_y = scissor.y + @as(i32, @intCast(scissor.height));
|
|
991
|
+
|
|
992
|
+
const intersect_x = @max(x, scissor.x);
|
|
993
|
+
const intersect_y = @max(y, scissor.y);
|
|
994
|
+
const intersect_end_x = @min(rect_end_x, scissor_end_x);
|
|
995
|
+
const intersect_end_y = @min(rect_end_y, scissor_end_y);
|
|
996
|
+
|
|
997
|
+
if (intersect_x >= intersect_end_x or intersect_y >= intersect_end_y) {
|
|
998
|
+
return null;
|
|
999
|
+
}
|
|
1000
|
+
|
|
1001
|
+
return buf.ClipRect{
|
|
1002
|
+
.x = intersect_x,
|
|
1003
|
+
.y = intersect_y,
|
|
1004
|
+
.width = @intCast(intersect_end_x - intersect_x),
|
|
1005
|
+
.height = @intCast(intersect_end_y - intersect_y),
|
|
1006
|
+
};
|
|
1007
|
+
}
|
|
1008
|
+
|
|
1009
|
+
/// Push a scissor rect for hit grid clipping.
|
|
1010
|
+
///
|
|
1011
|
+
/// The rect is intersected with any existing scissor, so nested overflow:hidden
|
|
1012
|
+
/// containers compound correctly. All coordinates are in screen space.
|
|
1013
|
+
pub fn hitGridPushScissorRect(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32) void {
|
|
1014
|
+
var rect = buf.ClipRect{
|
|
1015
|
+
.x = x,
|
|
1016
|
+
.y = y,
|
|
1017
|
+
.width = width,
|
|
1018
|
+
.height = height,
|
|
1019
|
+
};
|
|
1020
|
+
|
|
1021
|
+
if (self.getCurrentHitScissorRect() != null) {
|
|
1022
|
+
const intersect = self.clipRectToHitScissor(rect.x, rect.y, rect.width, rect.height);
|
|
1023
|
+
if (intersect) |clipped| {
|
|
1024
|
+
rect = clipped;
|
|
1025
|
+
} else {
|
|
1026
|
+
rect = buf.ClipRect{ .x = 0, .y = 0, .width = 0, .height = 0 };
|
|
1027
|
+
}
|
|
1028
|
+
}
|
|
1029
|
+
|
|
1030
|
+
self.hitScissorStack.append(self.allocator, rect) catch |err| {
|
|
1031
|
+
logger.warn("Failed to push hit-grid scissor rect: {}", .{err});
|
|
1032
|
+
};
|
|
1033
|
+
}
|
|
1034
|
+
|
|
1035
|
+
pub fn hitGridPopScissorRect(self: *CliRenderer) void {
|
|
1036
|
+
if (self.hitScissorStack.items.len > 0) {
|
|
1037
|
+
_ = self.hitScissorStack.pop();
|
|
1038
|
+
}
|
|
1039
|
+
}
|
|
1040
|
+
|
|
1041
|
+
/// Clear all hit grid scissors. Called at start of render to reset state.
|
|
1042
|
+
pub fn hitGridClearScissorRects(self: *CliRenderer) void {
|
|
1043
|
+
self.hitScissorStack.clearRetainingCapacity();
|
|
1044
|
+
}
|
|
1045
|
+
|
|
1046
|
+
/// Write directly to currentHitGrid with scissor clipping.
|
|
1047
|
+
///
|
|
1048
|
+
/// Used for immediate hit grid sync when scroll/translate changes. Unlike
|
|
1049
|
+
/// addToHitGrid (which writes to nextHitGrid for the upcoming frame), this
|
|
1050
|
+
/// updates the grid that checkHit reads right now. Lets hover states update
|
|
1051
|
+
/// without waiting for the next render.
|
|
1052
|
+
pub fn addToCurrentHitGridClipped(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
|
|
1053
|
+
const clipped = self.clipRectToHitScissor(x, y, width, height) orelse return;
|
|
1054
|
+
|
|
1055
|
+
const startX = @max(0, clipped.x);
|
|
1056
|
+
const startY = @max(0, clipped.y);
|
|
1057
|
+
const endX = @min(@as(i32, @intCast(self.hitGridWidth)), clipped.x + @as(i32, @intCast(clipped.width)));
|
|
1058
|
+
const endY = @min(@as(i32, @intCast(self.hitGridHeight)), clipped.y + @as(i32, @intCast(clipped.height)));
|
|
1059
|
+
|
|
1060
|
+
if (startX >= endX or startY >= endY) return;
|
|
1061
|
+
|
|
1062
|
+
const uStartX: u32 = @intCast(startX);
|
|
1063
|
+
const uStartY: u32 = @intCast(startY);
|
|
1064
|
+
const uEndX: u32 = @intCast(endX);
|
|
1065
|
+
const uEndY: u32 = @intCast(endY);
|
|
1066
|
+
|
|
1067
|
+
for (uStartY..uEndY) |row| {
|
|
1068
|
+
const rowStart = row * self.hitGridWidth;
|
|
1069
|
+
const startIdx = rowStart + uStartX;
|
|
1070
|
+
const endIdx = rowStart + uEndX;
|
|
1071
|
+
|
|
1072
|
+
@memset(self.currentHitGrid[startIdx..endIdx], id);
|
|
1073
|
+
}
|
|
1074
|
+
}
|
|
1075
|
+
|
|
1076
|
+
pub fn dumpHitGrid(self: *CliRenderer) void {
|
|
1077
|
+
const timestamp = std.time.timestamp();
|
|
1078
|
+
var filename_buf: [64]u8 = undefined;
|
|
1079
|
+
const filename = std.fmt.bufPrint(&filename_buf, "hitgrid_{d}.txt", .{timestamp}) catch return;
|
|
1080
|
+
|
|
1081
|
+
const file = std.fs.cwd().createFile(filename, .{}) catch return;
|
|
1082
|
+
defer file.close();
|
|
1083
|
+
|
|
1084
|
+
var fileBuffer: [4096]u8 = undefined;
|
|
1085
|
+
var fileWriter = file.writer(&fileBuffer);
|
|
1086
|
+
const writer = &fileWriter.interface;
|
|
1087
|
+
|
|
1088
|
+
for (0..self.hitGridHeight) |y| {
|
|
1089
|
+
for (0..self.hitGridWidth) |x| {
|
|
1090
|
+
const index = y * self.hitGridWidth + x;
|
|
1091
|
+
const id = self.currentHitGrid[index];
|
|
1092
|
+
|
|
1093
|
+
const char = if (id == 0) '.' else ('0' + @as(u8, @intCast(id % 10)));
|
|
1094
|
+
writer.writeByte(char) catch return;
|
|
1095
|
+
}
|
|
1096
|
+
writer.writeByte('\n') catch return;
|
|
1097
|
+
}
|
|
1098
|
+
writer.flush() catch {};
|
|
1099
|
+
}
|
|
1100
|
+
|
|
1101
|
+
fn dumpSingleBuffer(self: *CliRenderer, buffer: *OptimizedBuffer, buffer_name: []const u8, timestamp: i64) void {
|
|
1102
|
+
std.fs.cwd().makeDir("buffer_dump") catch |err| switch (err) {
|
|
1103
|
+
error.PathAlreadyExists => {},
|
|
1104
|
+
else => return,
|
|
1105
|
+
};
|
|
1106
|
+
|
|
1107
|
+
var filename_buf: [128]u8 = undefined;
|
|
1108
|
+
const filename = std.fmt.bufPrint(&filename_buf, "buffer_dump/{s}_buffer_{d}.txt", .{ buffer_name, timestamp }) catch return;
|
|
1109
|
+
|
|
1110
|
+
const file = std.fs.cwd().createFile(filename, .{}) catch return;
|
|
1111
|
+
defer file.close();
|
|
1112
|
+
|
|
1113
|
+
var fileBuffer: [4096]u8 = undefined;
|
|
1114
|
+
var fileWriter = file.writer(&fileBuffer);
|
|
1115
|
+
const writer = &fileWriter.interface;
|
|
1116
|
+
|
|
1117
|
+
writer.print("{s} Buffer ({d}x{d}):\n", .{ buffer_name, self.width, self.height }) catch return;
|
|
1118
|
+
writer.writeAll("Characters:\n") catch return;
|
|
1119
|
+
|
|
1120
|
+
for (0..self.height) |y| {
|
|
1121
|
+
for (0..self.width) |x| {
|
|
1122
|
+
const cell = buffer.get(@intCast(x), @intCast(y));
|
|
1123
|
+
if (cell) |c| {
|
|
1124
|
+
if (gp.isContinuationChar(c.char)) {
|
|
1125
|
+
// skip
|
|
1126
|
+
} else if (gp.isGraphemeChar(c.char)) {
|
|
1127
|
+
const gid: u32 = gp.graphemeIdFromChar(c.char);
|
|
1128
|
+
const bytes = self.pool.get(gid) catch &[_]u8{};
|
|
1129
|
+
if (bytes.len > 0) writer.writeAll(bytes) catch return;
|
|
1130
|
+
} else {
|
|
1131
|
+
var utf8Buf: [4]u8 = undefined;
|
|
1132
|
+
const len = std.unicode.utf8Encode(@intCast(c.char), &utf8Buf) catch 1;
|
|
1133
|
+
writer.writeAll(utf8Buf[0..len]) catch return;
|
|
1134
|
+
}
|
|
1135
|
+
} else {
|
|
1136
|
+
writer.writeByte(' ') catch return;
|
|
1137
|
+
}
|
|
1138
|
+
}
|
|
1139
|
+
writer.writeByte('\n') catch return;
|
|
1140
|
+
}
|
|
1141
|
+
writer.flush() catch {};
|
|
1142
|
+
}
|
|
1143
|
+
|
|
1144
|
+
pub fn getLastOutputForTest(_: *CliRenderer) []const u8 {
|
|
1145
|
+
// In non-threaded mode, we want the current active buffer
|
|
1146
|
+
// In threaded mode, we want the previously rendered buffer
|
|
1147
|
+
const currentBuffer = if (activeBuffer == .A) &outputBuffer else &outputBufferB;
|
|
1148
|
+
const currentLen = if (activeBuffer == .A) outputBufferLen else outputBufferBLen;
|
|
1149
|
+
return currentBuffer.*[0..currentLen];
|
|
1150
|
+
}
|
|
1151
|
+
|
|
1152
|
+
pub fn dumpStdoutBuffer(self: *CliRenderer, timestamp: i64) void {
|
|
1153
|
+
_ = self;
|
|
1154
|
+
std.fs.cwd().makeDir("buffer_dump") catch |err| switch (err) {
|
|
1155
|
+
error.PathAlreadyExists => {},
|
|
1156
|
+
else => return,
|
|
1157
|
+
};
|
|
1158
|
+
|
|
1159
|
+
var filename_buf: [128]u8 = undefined;
|
|
1160
|
+
const filename = std.fmt.bufPrint(&filename_buf, "buffer_dump/stdout_buffer_{d}.txt", .{timestamp}) catch return;
|
|
1161
|
+
|
|
1162
|
+
const file = std.fs.cwd().createFile(filename, .{}) catch return;
|
|
1163
|
+
defer file.close();
|
|
1164
|
+
|
|
1165
|
+
var fileBuffer: [4096]u8 = undefined;
|
|
1166
|
+
var fileWriter = file.writer(&fileBuffer);
|
|
1167
|
+
const writer = &fileWriter.interface;
|
|
1168
|
+
|
|
1169
|
+
writer.print("Stdout Buffer Output (timestamp: {d}):\n", .{timestamp}) catch return;
|
|
1170
|
+
writer.writeAll("Last Rendered ANSI Output:\n") catch return;
|
|
1171
|
+
writer.writeAll("================\n") catch return;
|
|
1172
|
+
|
|
1173
|
+
const lastBuffer = if (activeBuffer == .A) &outputBufferB else &outputBuffer;
|
|
1174
|
+
const lastLen = if (activeBuffer == .A) outputBufferBLen else outputBufferLen;
|
|
1175
|
+
|
|
1176
|
+
if (lastLen > 0) {
|
|
1177
|
+
writer.writeAll(lastBuffer.*[0..lastLen]) catch return;
|
|
1178
|
+
} else {
|
|
1179
|
+
writer.writeAll("(no output rendered yet)\n") catch return;
|
|
1180
|
+
}
|
|
1181
|
+
|
|
1182
|
+
writer.writeAll("\n================\n") catch return;
|
|
1183
|
+
writer.print("Buffer size: {d} bytes\n", .{lastLen}) catch return;
|
|
1184
|
+
writer.print("Active buffer: {s}\n", .{if (activeBuffer == .A) "A" else "B"}) catch return;
|
|
1185
|
+
writer.flush() catch {};
|
|
1186
|
+
}
|
|
1187
|
+
|
|
1188
|
+
pub fn dumpBuffers(self: *CliRenderer, timestamp: i64) void {
|
|
1189
|
+
self.dumpSingleBuffer(self.currentRenderBuffer, "current", timestamp);
|
|
1190
|
+
self.dumpSingleBuffer(self.nextRenderBuffer, "next", timestamp);
|
|
1191
|
+
self.dumpStdoutBuffer(timestamp);
|
|
1192
|
+
}
|
|
1193
|
+
|
|
1194
|
+
pub fn restoreTerminalModes(self: *CliRenderer) void {
|
|
1195
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1196
|
+
self.terminal.restoreTerminalModes(stream.writer()) catch {};
|
|
1197
|
+
self.writeOut(stream.getWritten());
|
|
1198
|
+
}
|
|
1199
|
+
|
|
1200
|
+
pub fn enableMouse(self: *CliRenderer, enableMovement: bool) void {
|
|
1201
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1202
|
+
self.terminal.setMouseMode(stream.writer(), true, enableMovement) catch {};
|
|
1203
|
+
self.writeOut(stream.getWritten());
|
|
1204
|
+
}
|
|
1205
|
+
|
|
1206
|
+
pub fn queryPixelResolution(self: *CliRenderer) void {
|
|
1207
|
+
self.writeOut(ansi.ANSI.queryPixelSize);
|
|
1208
|
+
}
|
|
1209
|
+
|
|
1210
|
+
pub fn disableMouse(self: *CliRenderer) void {
|
|
1211
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1212
|
+
self.terminal.setMouseMode(stream.writer(), false, self.terminal.state.mouse_movement) catch {};
|
|
1213
|
+
self.writeOut(stream.getWritten());
|
|
1214
|
+
}
|
|
1215
|
+
|
|
1216
|
+
pub fn enableKittyKeyboard(self: *CliRenderer, flags: u8) void {
|
|
1217
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1218
|
+
self.terminal.setKittyKeyboard(stream.writer(), true, flags) catch {};
|
|
1219
|
+
self.writeOut(stream.getWritten());
|
|
1220
|
+
}
|
|
1221
|
+
|
|
1222
|
+
pub fn disableKittyKeyboard(self: *CliRenderer) void {
|
|
1223
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1224
|
+
self.terminal.setKittyKeyboard(stream.writer(), false, 0) catch {};
|
|
1225
|
+
self.writeOut(stream.getWritten());
|
|
1226
|
+
}
|
|
1227
|
+
|
|
1228
|
+
pub fn getTerminalCapabilities(self: *CliRenderer) Terminal.Capabilities {
|
|
1229
|
+
return self.terminal.getCapabilities();
|
|
1230
|
+
}
|
|
1231
|
+
|
|
1232
|
+
pub fn setTerminalEnvVar(self: *CliRenderer, key: []const u8, value: []const u8) bool {
|
|
1233
|
+
self.terminal.setHostEnvVar(self.allocator, key, value) catch return false;
|
|
1234
|
+
return true;
|
|
1235
|
+
}
|
|
1236
|
+
|
|
1237
|
+
pub fn processCapabilityResponse(self: *CliRenderer, response: []const u8) void {
|
|
1238
|
+
self.terminal.processCapabilityResponse(response);
|
|
1239
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1240
|
+
_ = self.terminal.sendPendingQueries(stream.writer()) catch |err| blk: {
|
|
1241
|
+
logger.warn("Failed to send pending queries: {}", .{err});
|
|
1242
|
+
break :blk false;
|
|
1243
|
+
};
|
|
1244
|
+
const useKitty = self.terminal.opts.kitty_keyboard_flags > 0;
|
|
1245
|
+
self.terminal.enableDetectedFeatures(stream.writer(), useKitty) catch {};
|
|
1246
|
+
self.writeOut(stream.getWritten());
|
|
1247
|
+
}
|
|
1248
|
+
|
|
1249
|
+
pub fn setKittyKeyboardFlags(self: *CliRenderer, flags: u8) void {
|
|
1250
|
+
self.terminal.setKittyKeyboardFlags(flags);
|
|
1251
|
+
}
|
|
1252
|
+
|
|
1253
|
+
pub fn getKittyKeyboardFlags(self: *CliRenderer) u8 {
|
|
1254
|
+
return self.terminal.opts.kitty_keyboard_flags;
|
|
1255
|
+
}
|
|
1256
|
+
|
|
1257
|
+
pub fn setTerminalTitle(self: *CliRenderer, title: []const u8) void {
|
|
1258
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1259
|
+
self.terminal.setTerminalTitle(stream.writer(), title);
|
|
1260
|
+
self.writeOut(stream.getWritten());
|
|
1261
|
+
}
|
|
1262
|
+
|
|
1263
|
+
pub fn copyToClipboardOSC52(self: *CliRenderer, target: Terminal.ClipboardTarget, payload: []const u8) bool {
|
|
1264
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1265
|
+
self.terminal.writeClipboard(stream.writer(), target, payload) catch return false;
|
|
1266
|
+
self.writeOut(stream.getWritten());
|
|
1267
|
+
return true;
|
|
1268
|
+
}
|
|
1269
|
+
|
|
1270
|
+
pub fn clearClipboardOSC52(self: *CliRenderer, target: Terminal.ClipboardTarget) bool {
|
|
1271
|
+
var stream = std.io.fixedBufferStream(&self.writeOutBuf);
|
|
1272
|
+
self.terminal.writeClipboard(stream.writer(), target, "") catch return false;
|
|
1273
|
+
self.writeOut(stream.getWritten());
|
|
1274
|
+
return true;
|
|
1275
|
+
}
|
|
1276
|
+
|
|
1277
|
+
fn renderDebugOverlay(self: *CliRenderer) void {
|
|
1278
|
+
if (!self.debugOverlay.enabled) return;
|
|
1279
|
+
|
|
1280
|
+
const width: u32 = 40;
|
|
1281
|
+
const height: u32 = 11;
|
|
1282
|
+
var x: u32 = 0;
|
|
1283
|
+
var y: u32 = 0;
|
|
1284
|
+
|
|
1285
|
+
if (self.width < width + 2 or self.height < height + 2) return;
|
|
1286
|
+
|
|
1287
|
+
switch (self.debugOverlay.corner) {
|
|
1288
|
+
.topLeft => {
|
|
1289
|
+
x = 1;
|
|
1290
|
+
y = 1;
|
|
1291
|
+
},
|
|
1292
|
+
.topRight => {
|
|
1293
|
+
x = self.width - width - 1;
|
|
1294
|
+
y = 1;
|
|
1295
|
+
},
|
|
1296
|
+
.bottomLeft => {
|
|
1297
|
+
x = 1;
|
|
1298
|
+
y = self.height - height - 1;
|
|
1299
|
+
},
|
|
1300
|
+
.bottomRight => {
|
|
1301
|
+
x = self.width - width - 1;
|
|
1302
|
+
y = self.height - height - 1;
|
|
1303
|
+
},
|
|
1304
|
+
}
|
|
1305
|
+
|
|
1306
|
+
self.nextRenderBuffer.fillRect(x, y, width, height, .{ 20.0 / 255.0, 20.0 / 255.0, 40.0 / 255.0, 1.0 }) catch {};
|
|
1307
|
+
self.nextRenderBuffer.drawText("Debug Information", x + 1, y + 1, .{ 1.0, 1.0, 100.0 / 255.0, 1.0 }, .{ 0.0, 0.0, 0.0, 0.0 }, ansi.TextAttributes.BOLD) catch {};
|
|
1308
|
+
|
|
1309
|
+
var row: u32 = 2;
|
|
1310
|
+
const bg: RGBA = .{ 0.0, 0.0, 0.0, 0.0 };
|
|
1311
|
+
const fg: RGBA = .{ 200.0 / 255.0, 200.0 / 255.0, 200.0 / 255.0, 1.0 };
|
|
1312
|
+
|
|
1313
|
+
// Calculate averages
|
|
1314
|
+
const lastFrameTimeAvg = getStatAverage(f64, &self.statSamples.lastFrameTime);
|
|
1315
|
+
const renderTimeAvg = getStatAverage(f64, &self.statSamples.renderTime);
|
|
1316
|
+
const overallFrameTimeAvg = getStatAverage(f64, &self.statSamples.overallFrameTime);
|
|
1317
|
+
const bufferResetTimeAvg = getStatAverage(f64, &self.statSamples.bufferResetTime);
|
|
1318
|
+
const stdoutWriteTimeAvg = getStatAverage(f64, &self.statSamples.stdoutWriteTime);
|
|
1319
|
+
const cellsUpdatedAvg = getStatAverage(u32, &self.statSamples.cellsUpdated);
|
|
1320
|
+
const frameCallbackTimeAvg = getStatAverage(f64, &self.statSamples.frameCallbackTime);
|
|
1321
|
+
|
|
1322
|
+
// FPS
|
|
1323
|
+
var fpsText: [32]u8 = undefined;
|
|
1324
|
+
const fpsLen = std.fmt.bufPrint(&fpsText, "FPS: {d}", .{self.renderStats.fps}) catch return;
|
|
1325
|
+
self.nextRenderBuffer.drawText(fpsLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1326
|
+
row += 1;
|
|
1327
|
+
|
|
1328
|
+
// Frame Time
|
|
1329
|
+
var frameTimeText: [64]u8 = undefined;
|
|
1330
|
+
const frameTimeLen = std.fmt.bufPrint(&frameTimeText, "Frame: {d:.3}ms (avg: {d:.3}ms)", .{ self.renderStats.lastFrameTime / 1000.0, lastFrameTimeAvg / 1000.0 }) catch return;
|
|
1331
|
+
self.nextRenderBuffer.drawText(frameTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1332
|
+
row += 1;
|
|
1333
|
+
|
|
1334
|
+
// Frame Callback Time
|
|
1335
|
+
if (self.renderStats.frameCallbackTime) |frameCallbackTime| {
|
|
1336
|
+
var frameCallbackTimeText: [64]u8 = undefined;
|
|
1337
|
+
const frameCallbackTimeLen = std.fmt.bufPrint(&frameCallbackTimeText, "Frame Callback: {d:.3}ms (avg: {d:.3}ms)", .{ frameCallbackTime, frameCallbackTimeAvg }) catch return;
|
|
1338
|
+
self.nextRenderBuffer.drawText(frameCallbackTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1339
|
+
row += 1;
|
|
1340
|
+
}
|
|
1341
|
+
|
|
1342
|
+
// Overall Time
|
|
1343
|
+
if (self.renderStats.overallFrameTime) |overallTime| {
|
|
1344
|
+
var overallTimeText: [64]u8 = undefined;
|
|
1345
|
+
const overallTimeLen = std.fmt.bufPrint(&overallTimeText, "Overall: {d:.3}ms (avg: {d:.3}ms)", .{ overallTime, overallFrameTimeAvg }) catch return;
|
|
1346
|
+
self.nextRenderBuffer.drawText(overallTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1347
|
+
row += 1;
|
|
1348
|
+
}
|
|
1349
|
+
|
|
1350
|
+
// Render Time
|
|
1351
|
+
if (self.renderStats.renderTime) |renderTime| {
|
|
1352
|
+
var renderTimeText: [64]u8 = undefined;
|
|
1353
|
+
const renderTimeLen = std.fmt.bufPrint(&renderTimeText, "Render: {d:.3}ms (avg: {d:.3}ms)", .{ renderTime / 1000.0, renderTimeAvg / 1000.0 }) catch return;
|
|
1354
|
+
self.nextRenderBuffer.drawText(renderTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1355
|
+
row += 1;
|
|
1356
|
+
}
|
|
1357
|
+
|
|
1358
|
+
// Buffer Reset Time
|
|
1359
|
+
if (self.renderStats.bufferResetTime) |resetTime| {
|
|
1360
|
+
var resetTimeText: [64]u8 = undefined;
|
|
1361
|
+
const resetTimeLen = std.fmt.bufPrint(&resetTimeText, "Reset: {d:.3}ms (avg: {d:.3}ms)", .{ resetTime / 1000.0, bufferResetTimeAvg / 1000.0 }) catch return;
|
|
1362
|
+
self.nextRenderBuffer.drawText(resetTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1363
|
+
row += 1;
|
|
1364
|
+
}
|
|
1365
|
+
|
|
1366
|
+
// Stdout Write Time
|
|
1367
|
+
if (self.renderStats.stdoutWriteTime) |writeTime| {
|
|
1368
|
+
var writeTimeText: [64]u8 = undefined;
|
|
1369
|
+
const writeTimeLen = std.fmt.bufPrint(&writeTimeText, "Stdout: {d:.3}ms (avg: {d:.3}ms)", .{ writeTime / 1000.0, stdoutWriteTimeAvg / 1000.0 }) catch return;
|
|
1370
|
+
self.nextRenderBuffer.drawText(writeTimeLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1371
|
+
row += 1;
|
|
1372
|
+
}
|
|
1373
|
+
|
|
1374
|
+
// Cells Updated
|
|
1375
|
+
var cellsText: [64]u8 = undefined;
|
|
1376
|
+
const cellsLen = std.fmt.bufPrint(&cellsText, "Cells: {d} (avg: {d})", .{ self.renderStats.cellsUpdated, cellsUpdatedAvg }) catch return;
|
|
1377
|
+
self.nextRenderBuffer.drawText(cellsLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1378
|
+
row += 1;
|
|
1379
|
+
|
|
1380
|
+
if (self.renderStats.heapUsed > 0 or self.renderStats.heapTotal > 0) {
|
|
1381
|
+
var memoryText: [64]u8 = undefined;
|
|
1382
|
+
const memoryLen = std.fmt.bufPrint(&memoryText, "Memory: {d:.2}MB / {d:.2}MB / {d:.2}MB", .{ @as(f64, @floatFromInt(self.renderStats.heapUsed)) / 1024.0 / 1024.0, @as(f64, @floatFromInt(self.renderStats.heapTotal)) / 1024.0 / 1024.0, @as(f64, @floatFromInt(self.renderStats.arrayBuffers)) / 1024.0 / 1024.0 }) catch return;
|
|
1383
|
+
self.nextRenderBuffer.drawText(memoryLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1384
|
+
row += 1;
|
|
1385
|
+
}
|
|
1386
|
+
|
|
1387
|
+
// Is threaded?
|
|
1388
|
+
var isThreadedText: [64]u8 = undefined;
|
|
1389
|
+
const isThreadedLen = std.fmt.bufPrint(&isThreadedText, "Threaded: {s}", .{if (self.useThread) "Yes" else "No"}) catch return;
|
|
1390
|
+
self.nextRenderBuffer.drawText(isThreadedLen, x + 1, y + row, fg, bg, 0) catch {};
|
|
1391
|
+
row += 1;
|
|
1392
|
+
}
|
|
1393
|
+
};
|