@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,1109 @@
|
|
|
1
|
+
const std = @import("std");
|
|
2
|
+
const buffer_mod = @import("../buffer.zig");
|
|
3
|
+
const buffer_effects = @import("../buffer-methods.zig");
|
|
4
|
+
const gp = @import("../grapheme.zig");
|
|
5
|
+
|
|
6
|
+
const OptimizedBuffer = buffer_mod.OptimizedBuffer;
|
|
7
|
+
const RGBA = buffer_mod.RGBA;
|
|
8
|
+
const ColorTarget = buffer_effects.ColorTarget;
|
|
9
|
+
|
|
10
|
+
fn expectRGBAApprox(expected: RGBA, actual: RGBA, epsilon: f32) !void {
|
|
11
|
+
const diff_r = @abs(expected[0] - actual[0]);
|
|
12
|
+
const diff_g = @abs(expected[1] - actual[1]);
|
|
13
|
+
const diff_b = @abs(expected[2] - actual[2]);
|
|
14
|
+
const diff_a = @abs(expected[3] - actual[3]);
|
|
15
|
+
|
|
16
|
+
if (diff_r > epsilon or diff_g > epsilon or diff_b > epsilon or diff_a > epsilon) {
|
|
17
|
+
std.debug.print("RGBA mismatch: expected {any}, got {any}\n", .{ expected, actual });
|
|
18
|
+
return error.TestExpectedApprox;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
fn expectVec4fApprox(expected: @Vector(4, f32), actual: @Vector(4, f32), epsilon: f32) !void {
|
|
23
|
+
const diff = @abs(expected - actual);
|
|
24
|
+
if (@reduce(.Or, diff > @as(@Vector(4, f32), @splat(epsilon)))) {
|
|
25
|
+
std.debug.print("Vec4 mismatch: expected {any}, got {any}\n", .{ expected, actual });
|
|
26
|
+
return error.TestExpectedApprox;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
// Identity matrix (no change)
|
|
31
|
+
const IDENTITY_MATRIX = [16]f32{
|
|
32
|
+
1.0, 0.0, 0.0, 0.0, // Red output
|
|
33
|
+
0.0, 1.0, 0.0, 0.0, // Green output
|
|
34
|
+
0.0, 0.0, 1.0, 0.0, // Blue output
|
|
35
|
+
0.0, 0.0, 0.0, 1.0, // Alpha output
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
// Sepia matrix
|
|
39
|
+
const SEPIA_MATRIX = [16]f32{
|
|
40
|
+
0.393, 0.769, 0.189, 0.0, // Red output
|
|
41
|
+
0.349, 0.686, 0.168, 0.0, // Green output
|
|
42
|
+
0.272, 0.534, 0.131, 0.0, // Blue output
|
|
43
|
+
0.0, 0.0, 0.0, 1.0, // Alpha output
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
// Grayscale matrix (luminance)
|
|
47
|
+
const GRAYSCALE_MATRIX = [16]f32{
|
|
48
|
+
0.299, 0.587, 0.114, 0.0, // Red output
|
|
49
|
+
0.299, 0.587, 0.114, 0.0, // Green output
|
|
50
|
+
0.299, 0.587, 0.114, 0.0, // Blue output
|
|
51
|
+
0.0, 0.0, 0.0, 1.0, // Alpha output
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
// Invert matrix
|
|
55
|
+
const INVERT_MATRIX = [16]f32{
|
|
56
|
+
-1.0, 0.0, 0.0, 0.0, // Red output
|
|
57
|
+
0.0, -1.0, 0.0, 0.0, // Green output
|
|
58
|
+
0.0, 0.0, -1.0, 0.0, // Blue output
|
|
59
|
+
0.0, 0.0, 0.0, 1.0, // Alpha output
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
test "colorMatrix - identity matrix leaves colors unchanged" {
|
|
63
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
64
|
+
defer gp.deinitGlobalPool();
|
|
65
|
+
|
|
66
|
+
var buf = try OptimizedBuffer.init(
|
|
67
|
+
std.testing.allocator,
|
|
68
|
+
4,
|
|
69
|
+
4,
|
|
70
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
71
|
+
);
|
|
72
|
+
defer buf.deinit();
|
|
73
|
+
|
|
74
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
75
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
76
|
+
|
|
77
|
+
try buf.clear(bg, null);
|
|
78
|
+
buf.buffer.fg[0] = red; // (0, 0)
|
|
79
|
+
buf.buffer.fg[5] = red; // (1, 1)
|
|
80
|
+
|
|
81
|
+
// Apply identity to specific cells: (0, 0) and (1, 1) with strength 1.0
|
|
82
|
+
// cellMask format: [x, y, strength, x, y, strength, ...]
|
|
83
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0, 1.0, 1.0, 1.0 };
|
|
84
|
+
buffer_effects.colorMatrix(buf, &IDENTITY_MATRIX, &cell_mask, 1.0, ColorTarget.FG); // target=1 (FG)
|
|
85
|
+
|
|
86
|
+
// Colors should be unchanged
|
|
87
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
88
|
+
try expectRGBAApprox(red, buf.buffer.fg[5], 0.0001);
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
test "colorMatrix - applies transformation to specified cells only" {
|
|
92
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
93
|
+
defer gp.deinitGlobalPool();
|
|
94
|
+
|
|
95
|
+
var buf = try OptimizedBuffer.init(
|
|
96
|
+
std.testing.allocator,
|
|
97
|
+
3,
|
|
98
|
+
3,
|
|
99
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
100
|
+
);
|
|
101
|
+
defer buf.deinit();
|
|
102
|
+
|
|
103
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
104
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
105
|
+
|
|
106
|
+
try buf.clear(bg, null);
|
|
107
|
+
|
|
108
|
+
// Set all FG to red
|
|
109
|
+
@memset(buf.buffer.fg, red);
|
|
110
|
+
|
|
111
|
+
// Apply sepia only to cell (1, 1) with full strength
|
|
112
|
+
const cell_mask = [_]f32{ 1.0, 1.0, 1.0 };
|
|
113
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
114
|
+
|
|
115
|
+
// Cell (1, 1) should be transformed (index = y * width + x = 1 * 3 + 1 = 4)
|
|
116
|
+
const expected_r = 0.393;
|
|
117
|
+
const expected_g = 0.349;
|
|
118
|
+
const expected_b = 0.272;
|
|
119
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[4], 0.001);
|
|
120
|
+
|
|
121
|
+
// Other cells should remain red
|
|
122
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001); // (0, 0)
|
|
123
|
+
try expectRGBAApprox(red, buf.buffer.fg[8], 0.0001); // (2, 2)
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
test "colorMatrix - globalStrength scales individual cell strengths" {
|
|
127
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
128
|
+
defer gp.deinitGlobalPool();
|
|
129
|
+
|
|
130
|
+
var buf = try OptimizedBuffer.init(
|
|
131
|
+
std.testing.allocator,
|
|
132
|
+
2,
|
|
133
|
+
1,
|
|
134
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
135
|
+
);
|
|
136
|
+
defer buf.deinit();
|
|
137
|
+
|
|
138
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
139
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
140
|
+
|
|
141
|
+
try buf.clear(bg, null);
|
|
142
|
+
buf.buffer.fg[0] = red;
|
|
143
|
+
|
|
144
|
+
// Apply sepia with cell strength 1.0 but globalStrength 0.5
|
|
145
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0 };
|
|
146
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 0.5, ColorTarget.FG);
|
|
147
|
+
|
|
148
|
+
// Expected: blend(original, sepia, 0.5)
|
|
149
|
+
const sepia_r = 0.393;
|
|
150
|
+
const expected_r = 1.0 + (sepia_r - 1.0) * 0.5;
|
|
151
|
+
const sepia_g = 0.349;
|
|
152
|
+
const expected_g = 0.0 + (sepia_g - 0.0) * 0.5;
|
|
153
|
+
const sepia_b = 0.272;
|
|
154
|
+
const expected_b = 0.0 + (sepia_b - 0.0) * 0.5;
|
|
155
|
+
|
|
156
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
test "colorMatrix - respects target parameter" {
|
|
160
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
161
|
+
defer gp.deinitGlobalPool();
|
|
162
|
+
|
|
163
|
+
var buf = try OptimizedBuffer.init(
|
|
164
|
+
std.testing.allocator,
|
|
165
|
+
2,
|
|
166
|
+
1,
|
|
167
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
168
|
+
);
|
|
169
|
+
defer buf.deinit();
|
|
170
|
+
|
|
171
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
172
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
173
|
+
const blue = RGBA{ 0.0, 0.0, 1.0, 1.0 };
|
|
174
|
+
|
|
175
|
+
try buf.clear(bg, null);
|
|
176
|
+
buf.buffer.fg[0] = red;
|
|
177
|
+
buf.buffer.bg[0] = blue;
|
|
178
|
+
buf.buffer.fg[1] = red;
|
|
179
|
+
buf.buffer.bg[1] = blue;
|
|
180
|
+
|
|
181
|
+
// Apply to FG only (target = 1)
|
|
182
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0 };
|
|
183
|
+
buffer_effects.colorMatrix(buf, &GRAYSCALE_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
184
|
+
|
|
185
|
+
// FG should be grayscale, BG should remain blue
|
|
186
|
+
const gray_red = 0.299 * 1.0;
|
|
187
|
+
try expectRGBAApprox(.{ gray_red, gray_red, gray_red, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
188
|
+
try expectRGBAApprox(blue, buf.buffer.bg[0], 0.0001);
|
|
189
|
+
|
|
190
|
+
// Reset for BG test
|
|
191
|
+
buf.buffer.fg[0] = red;
|
|
192
|
+
buf.buffer.bg[0] = blue;
|
|
193
|
+
buf.buffer.fg[1] = red;
|
|
194
|
+
buf.buffer.bg[1] = blue;
|
|
195
|
+
|
|
196
|
+
buffer_effects.colorMatrix(buf, &GRAYSCALE_MATRIX, &cell_mask, 1.0, ColorTarget.BG); // target=2 (BG)
|
|
197
|
+
|
|
198
|
+
// BG should be grayscale, FG should remain red
|
|
199
|
+
const gray_blue = 0.114 * 1.0;
|
|
200
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
201
|
+
try expectRGBAApprox(.{ gray_blue, gray_blue, gray_blue, 1.0 }, buf.buffer.bg[0], 0.001);
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
test "colorMatrix - skips out-of-bounds coordinates" {
|
|
205
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
206
|
+
defer gp.deinitGlobalPool();
|
|
207
|
+
|
|
208
|
+
var buf = try OptimizedBuffer.init(
|
|
209
|
+
std.testing.allocator,
|
|
210
|
+
3,
|
|
211
|
+
3,
|
|
212
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
213
|
+
);
|
|
214
|
+
defer buf.deinit();
|
|
215
|
+
|
|
216
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
217
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
218
|
+
|
|
219
|
+
try buf.clear(bg, null);
|
|
220
|
+
buf.buffer.fg[4] = red; // (1, 1)
|
|
221
|
+
|
|
222
|
+
// Apply to out-of-bounds and valid cell
|
|
223
|
+
const cell_mask = [_]f32{ 10.0, 10.0, 1.0, 1.0, 1.0, 1.0 }; // (10, 10) is OOB
|
|
224
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
225
|
+
|
|
226
|
+
// Valid cell should be transformed
|
|
227
|
+
const expected_r = 0.393;
|
|
228
|
+
const expected_g = 0.349;
|
|
229
|
+
const expected_b = 0.272;
|
|
230
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[4], 0.001);
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
test "colorMatrix - skips NaN and Inf coordinates" {
|
|
234
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
235
|
+
defer gp.deinitGlobalPool();
|
|
236
|
+
|
|
237
|
+
var buf = try OptimizedBuffer.init(
|
|
238
|
+
std.testing.allocator,
|
|
239
|
+
3,
|
|
240
|
+
3,
|
|
241
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
242
|
+
);
|
|
243
|
+
defer buf.deinit();
|
|
244
|
+
|
|
245
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
246
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
247
|
+
|
|
248
|
+
try buf.clear(bg, null);
|
|
249
|
+
buf.buffer.fg[4] = red; // (1, 1)
|
|
250
|
+
|
|
251
|
+
// Apply with NaN and valid coordinates
|
|
252
|
+
const nan = std.math.nan(f32);
|
|
253
|
+
const cell_mask = [_]f32{ nan, 1.0, 1.0, 1.0, 1.0, 1.0 };
|
|
254
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
255
|
+
|
|
256
|
+
// Valid cell should be transformed
|
|
257
|
+
const expected_r = 0.393;
|
|
258
|
+
const expected_g = 0.349;
|
|
259
|
+
const expected_b = 0.272;
|
|
260
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[4], 0.001);
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
test "colorMatrix - skips zero strength cells" {
|
|
264
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
265
|
+
defer gp.deinitGlobalPool();
|
|
266
|
+
|
|
267
|
+
var buf = try OptimizedBuffer.init(
|
|
268
|
+
std.testing.allocator,
|
|
269
|
+
2,
|
|
270
|
+
1,
|
|
271
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
272
|
+
);
|
|
273
|
+
defer buf.deinit();
|
|
274
|
+
|
|
275
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
276
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
277
|
+
|
|
278
|
+
try buf.clear(bg, null);
|
|
279
|
+
buf.buffer.fg[0] = red;
|
|
280
|
+
|
|
281
|
+
// Apply with zero strength
|
|
282
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 0.0 };
|
|
283
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
284
|
+
|
|
285
|
+
// Color should be unchanged
|
|
286
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
test "colorMatrix - handles multiple cells in mask" {
|
|
290
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
291
|
+
defer gp.deinitGlobalPool();
|
|
292
|
+
|
|
293
|
+
var buf = try OptimizedBuffer.init(
|
|
294
|
+
std.testing.allocator,
|
|
295
|
+
4,
|
|
296
|
+
4,
|
|
297
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
298
|
+
);
|
|
299
|
+
defer buf.deinit();
|
|
300
|
+
|
|
301
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
302
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
303
|
+
const green = RGBA{ 0.0, 1.0, 0.0, 1.0 };
|
|
304
|
+
const blue = RGBA{ 0.0, 0.0, 1.0, 1.0 };
|
|
305
|
+
const white = RGBA{ 1.0, 1.0, 1.0, 1.0 };
|
|
306
|
+
|
|
307
|
+
try buf.clear(bg, null);
|
|
308
|
+
|
|
309
|
+
// Set different colors at different positions
|
|
310
|
+
buf.buffer.fg[0] = red; // (0, 0)
|
|
311
|
+
buf.buffer.fg[5] = green; // (1, 1)
|
|
312
|
+
buf.buffer.fg[10] = blue; // (2, 2)
|
|
313
|
+
buf.buffer.fg[15] = white; // (3, 3)
|
|
314
|
+
|
|
315
|
+
// Apply sepia to all four cells with varying strengths
|
|
316
|
+
const cell_mask = [_]f32{
|
|
317
|
+
0.0, 0.0, 1.0, // (0, 0) - full
|
|
318
|
+
1.0, 1.0, 0.5, // (1, 1) - half
|
|
319
|
+
2.0, 2.0, 0.0, // (2, 2) - none (skipped)
|
|
320
|
+
3.0, 3.0, 1.0, // (3, 3) - full
|
|
321
|
+
};
|
|
322
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
323
|
+
|
|
324
|
+
// (0, 0) should be fully sepia
|
|
325
|
+
const sepia_r = 0.393;
|
|
326
|
+
const sepia_g = 0.349;
|
|
327
|
+
const sepia_b = 0.272;
|
|
328
|
+
try expectRGBAApprox(.{ sepia_r, sepia_g, sepia_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
329
|
+
|
|
330
|
+
// (1, 1) should be half sepia
|
|
331
|
+
const green_sepia_r = 0.0 + (0.769 - 0.0) * 0.5; // Matrix row 0, col 1 = 0.769
|
|
332
|
+
const green_sepia_g = 1.0 + (0.686 - 1.0) * 0.5;
|
|
333
|
+
const green_sepia_b = 0.0 + (0.534 - 0.0) * 0.5;
|
|
334
|
+
try expectRGBAApprox(.{ green_sepia_r, green_sepia_g, green_sepia_b, 1.0 }, buf.buffer.fg[5], 0.001);
|
|
335
|
+
|
|
336
|
+
// (2, 2) should be unchanged (zero strength)
|
|
337
|
+
try expectRGBAApprox(blue, buf.buffer.fg[10], 0.0001);
|
|
338
|
+
|
|
339
|
+
// (3, 3) should be fully sepia of white
|
|
340
|
+
// White * sepia matrix = sum of first 3 columns of each row
|
|
341
|
+
const white_sepia_r = 0.393 + 0.769 + 0.189; // ~1.351
|
|
342
|
+
const white_sepia_g = 0.349 + 0.686 + 0.168; // ~1.203
|
|
343
|
+
const white_sepia_b = 0.272 + 0.534 + 0.131; // ~0.937
|
|
344
|
+
try expectRGBAApprox(.{ white_sepia_r, white_sepia_g, white_sepia_b, 1.0 }, buf.buffer.fg[15], 0.001);
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
test "colorMatrix - truncates incomplete mask triplets" {
|
|
348
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
349
|
+
defer gp.deinitGlobalPool();
|
|
350
|
+
|
|
351
|
+
var buf = try OptimizedBuffer.init(
|
|
352
|
+
std.testing.allocator,
|
|
353
|
+
3,
|
|
354
|
+
1,
|
|
355
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
356
|
+
);
|
|
357
|
+
defer buf.deinit();
|
|
358
|
+
|
|
359
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
360
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
361
|
+
|
|
362
|
+
try buf.clear(bg, null);
|
|
363
|
+
buf.buffer.fg[0] = red;
|
|
364
|
+
buf.buffer.fg[1] = red;
|
|
365
|
+
|
|
366
|
+
// Mask with 5 elements (1 complete triplet + 2 incomplete)
|
|
367
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0, 1.0, 1.0 };
|
|
368
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
369
|
+
|
|
370
|
+
// Only first cell should be transformed
|
|
371
|
+
const sepia_r = 0.393;
|
|
372
|
+
const sepia_g = 0.349;
|
|
373
|
+
const sepia_b = 0.272;
|
|
374
|
+
try expectRGBAApprox(.{ sepia_r, sepia_g, sepia_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
375
|
+
|
|
376
|
+
// Second cell should be unchanged (incomplete triplet ignored)
|
|
377
|
+
try expectRGBAApprox(red, buf.buffer.fg[1], 0.0001);
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
test "colorMatrix - empty mask returns early" {
|
|
381
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
382
|
+
defer gp.deinitGlobalPool();
|
|
383
|
+
|
|
384
|
+
var buf = try OptimizedBuffer.init(
|
|
385
|
+
std.testing.allocator,
|
|
386
|
+
2,
|
|
387
|
+
1,
|
|
388
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
389
|
+
);
|
|
390
|
+
defer buf.deinit();
|
|
391
|
+
|
|
392
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
393
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
394
|
+
|
|
395
|
+
try buf.clear(bg, null);
|
|
396
|
+
buf.buffer.fg[0] = red;
|
|
397
|
+
|
|
398
|
+
// Empty mask - should return early
|
|
399
|
+
const empty_mask = [0]f32{};
|
|
400
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &empty_mask, 1.0, ColorTarget.FG);
|
|
401
|
+
|
|
402
|
+
// Color should be unchanged
|
|
403
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
test "colorMatrix - empty matrix returns early" {
|
|
407
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
408
|
+
defer gp.deinitGlobalPool();
|
|
409
|
+
|
|
410
|
+
var buf = try OptimizedBuffer.init(
|
|
411
|
+
std.testing.allocator,
|
|
412
|
+
2,
|
|
413
|
+
1,
|
|
414
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
415
|
+
);
|
|
416
|
+
defer buf.deinit();
|
|
417
|
+
|
|
418
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
419
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
420
|
+
|
|
421
|
+
try buf.clear(bg, null);
|
|
422
|
+
buf.buffer.fg[0] = red;
|
|
423
|
+
|
|
424
|
+
// Empty matrix - should return early
|
|
425
|
+
const empty_matrix = [0]f32{};
|
|
426
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0 };
|
|
427
|
+
buffer_effects.colorMatrix(buf, &empty_matrix, &cell_mask, 1.0, ColorTarget.FG);
|
|
428
|
+
|
|
429
|
+
// Color should be unchanged
|
|
430
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
// Test matrix that modifies alpha channel
|
|
434
|
+
const ALPHA_MODIFY_MATRIX = [16]f32{
|
|
435
|
+
1.0, 0.0, 0.0, 0.0, // Red output
|
|
436
|
+
0.0, 1.0, 0.0, 0.0, // Green output
|
|
437
|
+
0.0, 0.0, 1.0, 0.0, // Blue output
|
|
438
|
+
0.0, 0.0, 0.0, 0.5, // Alpha output (multiply by 0.5)
|
|
439
|
+
};
|
|
440
|
+
|
|
441
|
+
test "colorMatrix - alpha channel transformation" {
|
|
442
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
443
|
+
defer gp.deinitGlobalPool();
|
|
444
|
+
|
|
445
|
+
var buf = try OptimizedBuffer.init(
|
|
446
|
+
std.testing.allocator,
|
|
447
|
+
2,
|
|
448
|
+
1,
|
|
449
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
450
|
+
);
|
|
451
|
+
defer buf.deinit();
|
|
452
|
+
|
|
453
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
454
|
+
const opaque_color = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
455
|
+
|
|
456
|
+
try buf.clear(bg, null);
|
|
457
|
+
buf.buffer.fg[0] = opaque_color;
|
|
458
|
+
|
|
459
|
+
// Apply matrix that halves alpha
|
|
460
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0 };
|
|
461
|
+
buffer_effects.colorMatrix(buf, &ALPHA_MODIFY_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
462
|
+
|
|
463
|
+
// Alpha should be halved
|
|
464
|
+
try expectRGBAApprox(.{ 1.0, 0.0, 0.0, 0.5 }, buf.buffer.fg[0], 0.0001);
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
test "colorMatrix - mask with only 1 element" {
|
|
468
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
469
|
+
defer gp.deinitGlobalPool();
|
|
470
|
+
|
|
471
|
+
var buf = try OptimizedBuffer.init(
|
|
472
|
+
std.testing.allocator,
|
|
473
|
+
2,
|
|
474
|
+
1,
|
|
475
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
476
|
+
);
|
|
477
|
+
defer buf.deinit();
|
|
478
|
+
|
|
479
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
480
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
481
|
+
|
|
482
|
+
try buf.clear(bg, null);
|
|
483
|
+
buf.buffer.fg[0] = red;
|
|
484
|
+
|
|
485
|
+
// Mask with only 1 element (incomplete triplet)
|
|
486
|
+
const cell_mask = [_]f32{0.0};
|
|
487
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
488
|
+
|
|
489
|
+
// Color should be unchanged (no complete triplets to process)
|
|
490
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
491
|
+
}
|
|
492
|
+
|
|
493
|
+
test "colorMatrix - mask with only 2 elements" {
|
|
494
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
495
|
+
defer gp.deinitGlobalPool();
|
|
496
|
+
|
|
497
|
+
var buf = try OptimizedBuffer.init(
|
|
498
|
+
std.testing.allocator,
|
|
499
|
+
2,
|
|
500
|
+
1,
|
|
501
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
502
|
+
);
|
|
503
|
+
defer buf.deinit();
|
|
504
|
+
|
|
505
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
506
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
507
|
+
|
|
508
|
+
try buf.clear(bg, null);
|
|
509
|
+
buf.buffer.fg[0] = red;
|
|
510
|
+
buf.buffer.fg[1] = red;
|
|
511
|
+
|
|
512
|
+
// Mask with only 2 elements (incomplete triplet)
|
|
513
|
+
const cell_mask = [_]f32{ 0.0, 0.0 };
|
|
514
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
515
|
+
|
|
516
|
+
// Colors should be unchanged (no complete triplets to process)
|
|
517
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
518
|
+
try expectRGBAApprox(red, buf.buffer.fg[1], 0.0001);
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
test "colorMatrix - infinity strength is skipped" {
|
|
522
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
523
|
+
defer gp.deinitGlobalPool();
|
|
524
|
+
|
|
525
|
+
var buf = try OptimizedBuffer.init(
|
|
526
|
+
std.testing.allocator,
|
|
527
|
+
2,
|
|
528
|
+
1,
|
|
529
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
530
|
+
);
|
|
531
|
+
defer buf.deinit();
|
|
532
|
+
|
|
533
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
534
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
535
|
+
|
|
536
|
+
try buf.clear(bg, null);
|
|
537
|
+
buf.buffer.fg[0] = red;
|
|
538
|
+
|
|
539
|
+
// Apply with infinity strength (should be skipped)
|
|
540
|
+
const inf = std.math.inf(f32);
|
|
541
|
+
const cell_mask = [_]f32{ 0.0, 0.0, inf };
|
|
542
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
543
|
+
|
|
544
|
+
// Color should be unchanged
|
|
545
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
546
|
+
}
|
|
547
|
+
|
|
548
|
+
test "colorMatrix - non-finite global strength is skipped" {
|
|
549
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
550
|
+
defer gp.deinitGlobalPool();
|
|
551
|
+
|
|
552
|
+
var buf = try OptimizedBuffer.init(
|
|
553
|
+
std.testing.allocator,
|
|
554
|
+
2,
|
|
555
|
+
1,
|
|
556
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
557
|
+
);
|
|
558
|
+
defer buf.deinit();
|
|
559
|
+
|
|
560
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
561
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
562
|
+
|
|
563
|
+
try buf.clear(bg, null);
|
|
564
|
+
buf.buffer.fg[0] = red;
|
|
565
|
+
|
|
566
|
+
const inf = std.math.inf(f32);
|
|
567
|
+
const cell_mask = [_]f32{ 0.0, 0.0, 1.0 };
|
|
568
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, inf, ColorTarget.FG);
|
|
569
|
+
|
|
570
|
+
// Color should be unchanged
|
|
571
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
test "colorMatrix - large buffer with SIMD and scalar mix" {
|
|
575
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
576
|
+
defer gp.deinitGlobalPool();
|
|
577
|
+
|
|
578
|
+
// 100 pixels = 25 SIMD batches of 4
|
|
579
|
+
var buf = try OptimizedBuffer.init(
|
|
580
|
+
std.testing.allocator,
|
|
581
|
+
100,
|
|
582
|
+
1,
|
|
583
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
584
|
+
);
|
|
585
|
+
defer buf.deinit();
|
|
586
|
+
|
|
587
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
588
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
589
|
+
|
|
590
|
+
try buf.clear(bg, null);
|
|
591
|
+
|
|
592
|
+
// Set all to red
|
|
593
|
+
@memset(buf.buffer.fg, red);
|
|
594
|
+
|
|
595
|
+
// Apply sepia at full strength
|
|
596
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 1.0, ColorTarget.FG);
|
|
597
|
+
|
|
598
|
+
// All pixels should be transformed
|
|
599
|
+
const expected_r = 0.393;
|
|
600
|
+
const expected_g = 0.349;
|
|
601
|
+
const expected_b = 0.272;
|
|
602
|
+
|
|
603
|
+
for (0..100) |i| {
|
|
604
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[i], 0.001);
|
|
605
|
+
}
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
test "colorMatrix - negative coordinates are skipped" {
|
|
609
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
610
|
+
defer gp.deinitGlobalPool();
|
|
611
|
+
|
|
612
|
+
var buf = try OptimizedBuffer.init(
|
|
613
|
+
std.testing.allocator,
|
|
614
|
+
3,
|
|
615
|
+
3,
|
|
616
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
617
|
+
);
|
|
618
|
+
defer buf.deinit();
|
|
619
|
+
|
|
620
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
621
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
622
|
+
|
|
623
|
+
try buf.clear(bg, null);
|
|
624
|
+
buf.buffer.fg[4] = red; // (1, 1)
|
|
625
|
+
|
|
626
|
+
// Apply with negative coordinates followed by valid
|
|
627
|
+
const cell_mask = [_]f32{ -1.0, -1.0, 1.0, 1.0, 1.0, 1.0 };
|
|
628
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
629
|
+
|
|
630
|
+
// Valid cell should be transformed
|
|
631
|
+
const expected_r = 0.393;
|
|
632
|
+
const expected_g = 0.349;
|
|
633
|
+
const expected_b = 0.272;
|
|
634
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[4], 0.001);
|
|
635
|
+
}
|
|
636
|
+
|
|
637
|
+
test "colorMatrix - finite coordinates larger than u32 max are skipped" {
|
|
638
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
639
|
+
defer gp.deinitGlobalPool();
|
|
640
|
+
|
|
641
|
+
var buf = try OptimizedBuffer.init(
|
|
642
|
+
std.testing.allocator,
|
|
643
|
+
3,
|
|
644
|
+
3,
|
|
645
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
646
|
+
);
|
|
647
|
+
defer buf.deinit();
|
|
648
|
+
|
|
649
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
650
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
651
|
+
|
|
652
|
+
try buf.clear(bg, null);
|
|
653
|
+
buf.buffer.fg[4] = red; // (1, 1)
|
|
654
|
+
|
|
655
|
+
// First triplet uses finite but out-of-range coordinates for u32 conversion.
|
|
656
|
+
// Second triplet is valid and should still be processed.
|
|
657
|
+
const huge = std.math.floatMax(f32);
|
|
658
|
+
const cell_mask = [_]f32{ huge, huge, 1.0, 1.0, 1.0, 1.0 };
|
|
659
|
+
buffer_effects.colorMatrix(buf, &SEPIA_MATRIX, &cell_mask, 1.0, ColorTarget.FG);
|
|
660
|
+
|
|
661
|
+
const expected_r = 0.393;
|
|
662
|
+
const expected_g = 0.349;
|
|
663
|
+
const expected_b = 0.272;
|
|
664
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[4], 0.001);
|
|
665
|
+
}
|
|
666
|
+
|
|
667
|
+
// ==================== colorMatrixUniform Tests ====================
|
|
668
|
+
|
|
669
|
+
test "colorMatrixUniform - identity matrix leaves colors unchanged" {
|
|
670
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
671
|
+
defer gp.deinitGlobalPool();
|
|
672
|
+
|
|
673
|
+
var buf = try OptimizedBuffer.init(
|
|
674
|
+
std.testing.allocator,
|
|
675
|
+
4,
|
|
676
|
+
1,
|
|
677
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
678
|
+
);
|
|
679
|
+
defer buf.deinit();
|
|
680
|
+
|
|
681
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
682
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
683
|
+
const green = RGBA{ 0.0, 1.0, 0.0, 1.0 };
|
|
684
|
+
const blue = RGBA{ 0.0, 0.0, 1.0, 1.0 };
|
|
685
|
+
const white = RGBA{ 1.0, 1.0, 1.0, 1.0 };
|
|
686
|
+
|
|
687
|
+
try buf.clear(bg, null);
|
|
688
|
+
|
|
689
|
+
// Set specific colors at different positions
|
|
690
|
+
buf.buffer.fg[0] = red;
|
|
691
|
+
buf.buffer.fg[1] = green;
|
|
692
|
+
buf.buffer.fg[2] = blue;
|
|
693
|
+
buf.buffer.fg[3] = white;
|
|
694
|
+
|
|
695
|
+
// Apply identity matrix at full strength to foreground
|
|
696
|
+
buffer_effects.colorMatrixUniform(buf, &IDENTITY_MATRIX, 1.0, ColorTarget.FG);
|
|
697
|
+
|
|
698
|
+
// Colors should be unchanged
|
|
699
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
700
|
+
try expectRGBAApprox(green, buf.buffer.fg[1], 0.0001);
|
|
701
|
+
try expectRGBAApprox(blue, buf.buffer.fg[2], 0.0001);
|
|
702
|
+
try expectRGBAApprox(white, buf.buffer.fg[3], 0.0001);
|
|
703
|
+
}
|
|
704
|
+
|
|
705
|
+
test "colorMatrixUniform - zero strength has no effect" {
|
|
706
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
707
|
+
defer gp.deinitGlobalPool();
|
|
708
|
+
|
|
709
|
+
var buf = try OptimizedBuffer.init(
|
|
710
|
+
std.testing.allocator,
|
|
711
|
+
2,
|
|
712
|
+
2,
|
|
713
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
714
|
+
);
|
|
715
|
+
defer buf.deinit();
|
|
716
|
+
|
|
717
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
718
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
719
|
+
|
|
720
|
+
try buf.clear(bg, null);
|
|
721
|
+
|
|
722
|
+
@memset(buf.buffer.fg, red);
|
|
723
|
+
|
|
724
|
+
// Apply sepia matrix with zero strength
|
|
725
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 0.0, ColorTarget.FG);
|
|
726
|
+
|
|
727
|
+
// Colors should be unchanged
|
|
728
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
729
|
+
try expectRGBAApprox(red, buf.buffer.fg[3], 0.0001);
|
|
730
|
+
}
|
|
731
|
+
|
|
732
|
+
test "colorMatrixUniform - non-finite strength has no effect" {
|
|
733
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
734
|
+
defer gp.deinitGlobalPool();
|
|
735
|
+
|
|
736
|
+
var buf = try OptimizedBuffer.init(
|
|
737
|
+
std.testing.allocator,
|
|
738
|
+
2,
|
|
739
|
+
1,
|
|
740
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
741
|
+
);
|
|
742
|
+
defer buf.deinit();
|
|
743
|
+
|
|
744
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
745
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
746
|
+
|
|
747
|
+
try buf.clear(bg, null);
|
|
748
|
+
buf.buffer.fg[0] = red;
|
|
749
|
+
buf.buffer.fg[1] = red;
|
|
750
|
+
|
|
751
|
+
const nan = std.math.nan(f32);
|
|
752
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, nan, ColorTarget.FG);
|
|
753
|
+
|
|
754
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
755
|
+
try expectRGBAApprox(red, buf.buffer.fg[1], 0.0001);
|
|
756
|
+
}
|
|
757
|
+
|
|
758
|
+
test "colorMatrixUniform - grayscale transformation" {
|
|
759
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
760
|
+
defer gp.deinitGlobalPool();
|
|
761
|
+
|
|
762
|
+
var buf = try OptimizedBuffer.init(
|
|
763
|
+
std.testing.allocator,
|
|
764
|
+
3,
|
|
765
|
+
1,
|
|
766
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
767
|
+
);
|
|
768
|
+
defer buf.deinit();
|
|
769
|
+
|
|
770
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
771
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
772
|
+
const green = RGBA{ 0.0, 1.0, 0.0, 1.0 };
|
|
773
|
+
const blue = RGBA{ 0.0, 0.0, 1.0, 1.0 };
|
|
774
|
+
|
|
775
|
+
try buf.clear(bg, null);
|
|
776
|
+
|
|
777
|
+
buf.buffer.fg[0] = red;
|
|
778
|
+
buf.buffer.fg[1] = green;
|
|
779
|
+
buf.buffer.fg[2] = blue;
|
|
780
|
+
|
|
781
|
+
// Apply grayscale matrix at full strength to foreground
|
|
782
|
+
buffer_effects.colorMatrixUniform(buf, &GRAYSCALE_MATRIX, 1.0, ColorTarget.FG);
|
|
783
|
+
|
|
784
|
+
// Calculate expected grayscale values
|
|
785
|
+
// Luminance = 0.299*R + 0.587*G + 0.114*B
|
|
786
|
+
const gray_red = 0.299 * 1.0 + 0.587 * 0.0 + 0.114 * 0.0; // ~0.299
|
|
787
|
+
const gray_green = 0.299 * 0.0 + 0.587 * 1.0 + 0.114 * 0.0; // ~0.587
|
|
788
|
+
const gray_blue = 0.299 * 0.0 + 0.587 * 0.0 + 0.114 * 1.0; // ~0.114
|
|
789
|
+
|
|
790
|
+
// All channels should equal the luminance value
|
|
791
|
+
try expectRGBAApprox(.{ gray_red, gray_red, gray_red, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
792
|
+
try expectRGBAApprox(.{ gray_green, gray_green, gray_green, 1.0 }, buf.buffer.fg[1], 0.001);
|
|
793
|
+
try expectRGBAApprox(.{ gray_blue, gray_blue, gray_blue, 1.0 }, buf.buffer.fg[2], 0.001);
|
|
794
|
+
}
|
|
795
|
+
|
|
796
|
+
test "colorMatrixUniform - partial strength blends with original" {
|
|
797
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
798
|
+
defer gp.deinitGlobalPool();
|
|
799
|
+
|
|
800
|
+
var buf = try OptimizedBuffer.init(
|
|
801
|
+
std.testing.allocator,
|
|
802
|
+
2,
|
|
803
|
+
1,
|
|
804
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
805
|
+
);
|
|
806
|
+
defer buf.deinit();
|
|
807
|
+
|
|
808
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
809
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
810
|
+
|
|
811
|
+
try buf.clear(bg, null);
|
|
812
|
+
buf.buffer.fg[0] = red;
|
|
813
|
+
buf.buffer.fg[1] = red;
|
|
814
|
+
|
|
815
|
+
// Apply sepia at 50% strength
|
|
816
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 0.5, ColorTarget.FG);
|
|
817
|
+
|
|
818
|
+
// Expected: blend(original, sepia_result, 0.5)
|
|
819
|
+
// Sepia of pure red: R=0.393, G=0.349, B=0.272
|
|
820
|
+
// Blend: original + (sepia - original) * 0.5
|
|
821
|
+
const expected_r = 1.0 + (0.393 - 1.0) * 0.5;
|
|
822
|
+
const expected_g = 0.0 + (0.349 - 0.0) * 0.5;
|
|
823
|
+
const expected_b = 0.0 + (0.272 - 0.0) * 0.5;
|
|
824
|
+
|
|
825
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
826
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[1], 0.001);
|
|
827
|
+
}
|
|
828
|
+
|
|
829
|
+
test "colorMatrixUniform - target affects correct buffers" {
|
|
830
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
831
|
+
defer gp.deinitGlobalPool();
|
|
832
|
+
|
|
833
|
+
var buf = try OptimizedBuffer.init(
|
|
834
|
+
std.testing.allocator,
|
|
835
|
+
2,
|
|
836
|
+
1,
|
|
837
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
838
|
+
);
|
|
839
|
+
defer buf.deinit();
|
|
840
|
+
|
|
841
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
842
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
843
|
+
const blue = RGBA{ 0.0, 0.0, 1.0, 1.0 };
|
|
844
|
+
|
|
845
|
+
try buf.clear(bg, null);
|
|
846
|
+
|
|
847
|
+
buf.buffer.fg[0] = red;
|
|
848
|
+
buf.buffer.bg[0] = blue;
|
|
849
|
+
buf.buffer.fg[1] = red;
|
|
850
|
+
buf.buffer.bg[1] = blue;
|
|
851
|
+
|
|
852
|
+
// Apply to FG only (target = 1)
|
|
853
|
+
buffer_effects.colorMatrixUniform(buf, &GRAYSCALE_MATRIX, 1.0, ColorTarget.FG);
|
|
854
|
+
|
|
855
|
+
// FG should be grayscale, BG should remain blue
|
|
856
|
+
const gray_red = 0.299 * 1.0;
|
|
857
|
+
try expectRGBAApprox(.{ gray_red, gray_red, gray_red, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
858
|
+
try expectRGBAApprox(blue, buf.buffer.bg[0], 0.0001);
|
|
859
|
+
|
|
860
|
+
// Reset and test BG only (target = 2)
|
|
861
|
+
buf.buffer.fg[0] = red;
|
|
862
|
+
buf.buffer.bg[0] = blue;
|
|
863
|
+
buf.buffer.fg[1] = red;
|
|
864
|
+
buf.buffer.bg[1] = blue;
|
|
865
|
+
|
|
866
|
+
buffer_effects.colorMatrixUniform(buf, &GRAYSCALE_MATRIX, 1.0, ColorTarget.BG);
|
|
867
|
+
|
|
868
|
+
// BG should be grayscale, FG should remain red
|
|
869
|
+
const gray_blue = 0.114 * 1.0;
|
|
870
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
871
|
+
try expectRGBAApprox(.{ gray_blue, gray_blue, gray_blue, 1.0 }, buf.buffer.bg[0], 0.001);
|
|
872
|
+
|
|
873
|
+
// Reset and test Both (target = 3)
|
|
874
|
+
buf.buffer.fg[0] = red;
|
|
875
|
+
buf.buffer.bg[0] = blue;
|
|
876
|
+
buf.buffer.fg[1] = red;
|
|
877
|
+
buf.buffer.bg[1] = blue;
|
|
878
|
+
|
|
879
|
+
buffer_effects.colorMatrixUniform(buf, &GRAYSCALE_MATRIX, 1.0, ColorTarget.Both);
|
|
880
|
+
|
|
881
|
+
// Both should be grayscale
|
|
882
|
+
try expectRGBAApprox(.{ gray_red, gray_red, gray_red, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
883
|
+
try expectRGBAApprox(.{ gray_blue, gray_blue, gray_blue, 1.0 }, buf.buffer.bg[0], 0.001);
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
test "colorMatrixUniform - handles buffer sizes not divisible by 4" {
|
|
887
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
888
|
+
defer gp.deinitGlobalPool();
|
|
889
|
+
|
|
890
|
+
// Test with 5 pixels (1 SIMD batch of 4 + 1 scalar remainder)
|
|
891
|
+
var buf = try OptimizedBuffer.init(
|
|
892
|
+
std.testing.allocator,
|
|
893
|
+
5,
|
|
894
|
+
1,
|
|
895
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
896
|
+
);
|
|
897
|
+
defer buf.deinit();
|
|
898
|
+
|
|
899
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
900
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
901
|
+
|
|
902
|
+
try buf.clear(bg, null);
|
|
903
|
+
|
|
904
|
+
// Set all FG to red
|
|
905
|
+
for (0..5) |i| {
|
|
906
|
+
buf.buffer.fg[i] = red;
|
|
907
|
+
}
|
|
908
|
+
|
|
909
|
+
// Apply sepia at full strength
|
|
910
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 1.0, ColorTarget.FG);
|
|
911
|
+
|
|
912
|
+
// All pixels should be transformed (including the scalar fallback)
|
|
913
|
+
const expected_r = 0.393;
|
|
914
|
+
const expected_g = 0.349;
|
|
915
|
+
const expected_b = 0.272;
|
|
916
|
+
|
|
917
|
+
for (0..5) |i| {
|
|
918
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[i], 0.001);
|
|
919
|
+
}
|
|
920
|
+
}
|
|
921
|
+
|
|
922
|
+
test "colorMatrixUniform - empty matrix returns early" {
|
|
923
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
924
|
+
defer gp.deinitGlobalPool();
|
|
925
|
+
|
|
926
|
+
var buf = try OptimizedBuffer.init(
|
|
927
|
+
std.testing.allocator,
|
|
928
|
+
2,
|
|
929
|
+
1,
|
|
930
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
931
|
+
);
|
|
932
|
+
defer buf.deinit();
|
|
933
|
+
|
|
934
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
935
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
936
|
+
|
|
937
|
+
try buf.clear(bg, null);
|
|
938
|
+
buf.buffer.fg[0] = red;
|
|
939
|
+
|
|
940
|
+
// Empty matrix - should return early without changes
|
|
941
|
+
const empty_matrix = [0]f32{};
|
|
942
|
+
buffer_effects.colorMatrixUniform(buf, &empty_matrix, 1.0, ColorTarget.FG);
|
|
943
|
+
|
|
944
|
+
// Color should be unchanged
|
|
945
|
+
try expectRGBAApprox(red, buf.buffer.fg[0], 0.0001);
|
|
946
|
+
}
|
|
947
|
+
|
|
948
|
+
test "colorMatrixUniform - alpha channel transformation" {
|
|
949
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
950
|
+
defer gp.deinitGlobalPool();
|
|
951
|
+
|
|
952
|
+
var buf = try OptimizedBuffer.init(
|
|
953
|
+
std.testing.allocator,
|
|
954
|
+
2,
|
|
955
|
+
1,
|
|
956
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
957
|
+
);
|
|
958
|
+
defer buf.deinit();
|
|
959
|
+
|
|
960
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
961
|
+
const opaque_color = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
962
|
+
const transparent_color = RGBA{ 0.0, 1.0, 0.0, 0.5 };
|
|
963
|
+
|
|
964
|
+
try buf.clear(bg, null);
|
|
965
|
+
|
|
966
|
+
buf.buffer.fg[0] = opaque_color;
|
|
967
|
+
buf.buffer.fg[1] = transparent_color;
|
|
968
|
+
|
|
969
|
+
// Apply matrix that halves alpha at full strength
|
|
970
|
+
buffer_effects.colorMatrixUniform(buf, &ALPHA_MODIFY_MATRIX, 1.0, ColorTarget.FG);
|
|
971
|
+
|
|
972
|
+
// Opaque should become semi-transparent (alpha = 1.0 * 0.5 = 0.5)
|
|
973
|
+
try expectRGBAApprox(.{ 1.0, 0.0, 0.0, 0.5 }, buf.buffer.fg[0], 0.0001);
|
|
974
|
+
// Semi-transparent should become more transparent (alpha = 0.5 * 0.5 = 0.25)
|
|
975
|
+
try expectRGBAApprox(.{ 0.0, 1.0, 0.0, 0.25 }, buf.buffer.fg[1], 0.0001);
|
|
976
|
+
}
|
|
977
|
+
|
|
978
|
+
test "colorMatrixUniform - very small buffer (less than 4 pixels)" {
|
|
979
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
980
|
+
defer gp.deinitGlobalPool();
|
|
981
|
+
|
|
982
|
+
// Test with 2 pixels (all scalar, no SIMD)
|
|
983
|
+
var buf = try OptimizedBuffer.init(
|
|
984
|
+
std.testing.allocator,
|
|
985
|
+
2,
|
|
986
|
+
1,
|
|
987
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
988
|
+
);
|
|
989
|
+
defer buf.deinit();
|
|
990
|
+
|
|
991
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
992
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
993
|
+
const green = RGBA{ 0.0, 1.0, 0.0, 1.0 };
|
|
994
|
+
|
|
995
|
+
try buf.clear(bg, null);
|
|
996
|
+
buf.buffer.fg[0] = red;
|
|
997
|
+
buf.buffer.fg[1] = green;
|
|
998
|
+
|
|
999
|
+
// Apply sepia at full strength
|
|
1000
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 1.0, ColorTarget.FG);
|
|
1001
|
+
|
|
1002
|
+
// Both pixels should be transformed correctly using scalar path
|
|
1003
|
+
const expected_red_r = 0.393;
|
|
1004
|
+
const expected_red_g = 0.349;
|
|
1005
|
+
const expected_red_b = 0.272;
|
|
1006
|
+
try expectRGBAApprox(.{ expected_red_r, expected_red_g, expected_red_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
1007
|
+
|
|
1008
|
+
// Green transformed: R=0.769, G=0.686, B=0.534
|
|
1009
|
+
const expected_green_r = 0.769;
|
|
1010
|
+
const expected_green_g = 0.686;
|
|
1011
|
+
const expected_green_b = 0.534;
|
|
1012
|
+
try expectRGBAApprox(.{ expected_green_r, expected_green_g, expected_green_b, 1.0 }, buf.buffer.fg[1], 0.001);
|
|
1013
|
+
}
|
|
1014
|
+
|
|
1015
|
+
test "colorMatrixUniform - single pixel buffer" {
|
|
1016
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
1017
|
+
defer gp.deinitGlobalPool();
|
|
1018
|
+
|
|
1019
|
+
// Test with 1 pixel (edge case)
|
|
1020
|
+
var buf = try OptimizedBuffer.init(
|
|
1021
|
+
std.testing.allocator,
|
|
1022
|
+
1,
|
|
1023
|
+
1,
|
|
1024
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
1025
|
+
);
|
|
1026
|
+
defer buf.deinit();
|
|
1027
|
+
|
|
1028
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
1029
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
1030
|
+
|
|
1031
|
+
try buf.clear(bg, null);
|
|
1032
|
+
buf.buffer.fg[0] = red;
|
|
1033
|
+
|
|
1034
|
+
// Apply sepia at full strength
|
|
1035
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 1.0, ColorTarget.FG);
|
|
1036
|
+
|
|
1037
|
+
// Pixel should be transformed correctly
|
|
1038
|
+
const expected_r = 0.393;
|
|
1039
|
+
const expected_g = 0.349;
|
|
1040
|
+
const expected_b = 0.272;
|
|
1041
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[0], 0.001);
|
|
1042
|
+
}
|
|
1043
|
+
|
|
1044
|
+
test "colorMatrixUniform - values can exceed 1.0 (no clamping)" {
|
|
1045
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
1046
|
+
defer gp.deinitGlobalPool();
|
|
1047
|
+
|
|
1048
|
+
var buf = try OptimizedBuffer.init(
|
|
1049
|
+
std.testing.allocator,
|
|
1050
|
+
2,
|
|
1051
|
+
1,
|
|
1052
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
1053
|
+
);
|
|
1054
|
+
defer buf.deinit();
|
|
1055
|
+
|
|
1056
|
+
// Matrix that amplifies colors beyond 1.0
|
|
1057
|
+
const amplify_matrix = [16]f32{
|
|
1058
|
+
2.0, 0.0, 0.0, 0.0, // Red output (2x)
|
|
1059
|
+
0.0, 2.0, 0.0, 0.0, // Green output (2x)
|
|
1060
|
+
0.0, 0.0, 2.0, 0.0, // Blue output (2x)
|
|
1061
|
+
0.0, 0.0, 0.0, 1.0, // Alpha output
|
|
1062
|
+
};
|
|
1063
|
+
|
|
1064
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
1065
|
+
const gray = RGBA{ 0.5, 0.5, 0.5, 1.0 };
|
|
1066
|
+
|
|
1067
|
+
try buf.clear(bg, null);
|
|
1068
|
+
buf.buffer.fg[0] = gray;
|
|
1069
|
+
|
|
1070
|
+
// Apply amplification at full strength
|
|
1071
|
+
buffer_effects.colorMatrixUniform(buf, &lify_matrix, 1.0, ColorTarget.FG);
|
|
1072
|
+
|
|
1073
|
+
// Values should exceed 1.0 (no clamping)
|
|
1074
|
+
try expectRGBAApprox(.{ 1.0, 1.0, 1.0, 1.0 }, buf.buffer.fg[0], 0.0001);
|
|
1075
|
+
}
|
|
1076
|
+
|
|
1077
|
+
test "colorMatrixUniform - 3 pixel buffer (simd_end = 0, all scalar)" {
|
|
1078
|
+
const pool = gp.initGlobalPool(std.testing.allocator);
|
|
1079
|
+
defer gp.deinitGlobalPool();
|
|
1080
|
+
|
|
1081
|
+
// 3 pixels - simd_end will be 0, so all processed via scalar
|
|
1082
|
+
var buf = try OptimizedBuffer.init(
|
|
1083
|
+
std.testing.allocator,
|
|
1084
|
+
3,
|
|
1085
|
+
1,
|
|
1086
|
+
.{ .pool = pool, .id = "test-buffer" },
|
|
1087
|
+
);
|
|
1088
|
+
defer buf.deinit();
|
|
1089
|
+
|
|
1090
|
+
const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
|
|
1091
|
+
const red = RGBA{ 1.0, 0.0, 0.0, 1.0 };
|
|
1092
|
+
|
|
1093
|
+
try buf.clear(bg, null);
|
|
1094
|
+
for (0..3) |i| {
|
|
1095
|
+
buf.buffer.fg[i] = red;
|
|
1096
|
+
}
|
|
1097
|
+
|
|
1098
|
+
// Apply sepia
|
|
1099
|
+
buffer_effects.colorMatrixUniform(buf, &SEPIA_MATRIX, 1.0, ColorTarget.FG);
|
|
1100
|
+
|
|
1101
|
+
// All 3 should be transformed via scalar path
|
|
1102
|
+
const expected_r = 0.393;
|
|
1103
|
+
const expected_g = 0.349;
|
|
1104
|
+
const expected_b = 0.272;
|
|
1105
|
+
|
|
1106
|
+
for (0..3) |i| {
|
|
1107
|
+
try expectRGBAApprox(.{ expected_r, expected_g, expected_b, 1.0 }, buf.buffer.fg[i], 0.001);
|
|
1108
|
+
}
|
|
1109
|
+
}
|