@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.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +63 -51
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-8fks7yv1.js +0 -411
  472. package/index-8fks7yv1.js.map +0 -10
  473. package/index-egy5e2rs.js +0 -12267
  474. package/index-egy5e2rs.js.map +0 -42
  475. package/index-tse8gzh0.js +0 -20614
  476. package/index-tse8gzh0.js.map +0 -67
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /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, &amplify_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
+ }