@fairyhunter13/opentui-core 0.1.91 → 0.1.94

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 (570) hide show
  1. package/3d/SpriteResourceManager.d.ts +74 -0
  2. package/3d/SpriteUtils.d.ts +13 -0
  3. package/3d/TextureUtils.d.ts +24 -0
  4. package/3d/ThreeRenderable.d.ts +40 -0
  5. package/3d/WGPURenderer.d.ts +61 -0
  6. package/3d/animation/ExplodingSpriteEffect.d.ts +71 -0
  7. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +76 -0
  8. package/3d/animation/SpriteAnimator.d.ts +124 -0
  9. package/3d/animation/SpriteParticleGenerator.d.ts +62 -0
  10. package/3d/canvas.d.ts +44 -0
  11. package/3d/index.d.ts +12 -0
  12. package/3d/physics/PlanckPhysicsAdapter.d.ts +19 -0
  13. package/3d/physics/RapierPhysicsAdapter.d.ts +19 -0
  14. package/3d/physics/physics-interface.d.ts +27 -0
  15. package/3d.d.ts +2 -0
  16. package/3d.js +34042 -0
  17. package/3d.js.map +155 -0
  18. package/LICENSE +21 -0
  19. package/NativeSpanFeed.d.ts +41 -0
  20. package/Renderable.d.ts +334 -0
  21. package/animation/Timeline.d.ts +126 -0
  22. package/ansi.d.ts +13 -0
  23. package/buffer.d.ts +107 -0
  24. package/console.d.ts +143 -0
  25. package/edit-buffer.d.ts +98 -0
  26. package/editor-view.d.ts +73 -0
  27. package/index-e6ec7apq.js +18415 -0
  28. package/index-e6ec7apq.js.map +64 -0
  29. package/index-h066zmrb.js +12619 -0
  30. package/index-h066zmrb.js.map +43 -0
  31. package/index-ynzawt3n.js +113 -0
  32. package/index-ynzawt3n.js.map +10 -0
  33. package/index.d.ts +21 -0
  34. package/index.js +430 -0
  35. package/index.js.map +9 -0
  36. package/lib/KeyHandler.d.ts +61 -0
  37. package/lib/RGBA.d.ts +25 -0
  38. package/lib/ascii.font.d.ts +508 -0
  39. package/lib/border.d.ts +49 -0
  40. package/lib/bunfs.d.ts +7 -0
  41. package/lib/clipboard.d.ts +17 -0
  42. package/lib/clock.d.ts +15 -0
  43. package/lib/data-paths.d.ts +26 -0
  44. package/lib/debounce.d.ts +42 -0
  45. package/lib/detect-links.d.ts +6 -0
  46. package/lib/env.d.ts +42 -0
  47. package/lib/extmarks-history.d.ts +17 -0
  48. package/lib/extmarks.d.ts +89 -0
  49. package/lib/hast-styled-text.d.ts +17 -0
  50. package/lib/index.d.ts +21 -0
  51. package/lib/keymapping.d.ts +25 -0
  52. package/lib/objects-in-viewport.d.ts +24 -0
  53. package/lib/output.capture.d.ts +24 -0
  54. package/lib/parse.keypress-kitty.d.ts +2 -0
  55. package/lib/parse.keypress.d.ts +26 -0
  56. package/lib/parse.mouse.d.ts +30 -0
  57. package/lib/paste.d.ts +7 -0
  58. package/lib/queue.d.ts +15 -0
  59. package/lib/renderable.validations.d.ts +12 -0
  60. package/lib/scroll-acceleration.d.ts +43 -0
  61. package/lib/selection.d.ts +63 -0
  62. package/lib/singleton.d.ts +7 -0
  63. package/lib/stdin-parser.d.ts +76 -0
  64. package/lib/styled-text.d.ts +63 -0
  65. package/lib/terminal-capability-detection.d.ts +30 -0
  66. package/lib/terminal-palette.d.ts +50 -0
  67. package/lib/tree-sitter/assets/update.d.ts +11 -0
  68. package/lib/tree-sitter/client.d.ts +47 -0
  69. package/lib/tree-sitter/default-parsers.d.ts +2 -0
  70. package/lib/tree-sitter/download-utils.d.ts +21 -0
  71. package/lib/tree-sitter/index.d.ts +8 -0
  72. package/lib/tree-sitter/parser.worker.d.ts +1 -0
  73. package/lib/tree-sitter/parsers-config.d.ts +38 -0
  74. package/lib/tree-sitter/resolve-ft.d.ts +2 -0
  75. package/lib/tree-sitter/types.d.ts +81 -0
  76. package/lib/tree-sitter-styled-text.d.ts +14 -0
  77. package/lib/validate-dir-name.d.ts +1 -0
  78. package/lib/yoga.options.d.ts +32 -0
  79. package/package.json +51 -63
  80. package/parser.worker.js +869 -0
  81. package/parser.worker.js.map +12 -0
  82. package/plugins/core-slot.d.ts +72 -0
  83. package/plugins/registry.d.ts +38 -0
  84. package/plugins/types.d.ts +34 -0
  85. package/post/filters.d.ts +105 -0
  86. package/renderables/ASCIIFont.d.ts +52 -0
  87. package/renderables/Box.d.ts +72 -0
  88. package/renderables/Code.d.ts +78 -0
  89. package/renderables/Diff.d.ts +142 -0
  90. package/renderables/EditBufferRenderable.d.ts +162 -0
  91. package/renderables/FrameBuffer.d.ts +16 -0
  92. package/renderables/Input.d.ts +67 -0
  93. package/renderables/LineNumberRenderable.d.ts +74 -0
  94. package/renderables/Markdown.d.ts +173 -0
  95. package/renderables/ScrollBar.d.ts +77 -0
  96. package/renderables/ScrollBox.d.ts +124 -0
  97. package/renderables/Select.d.ts +115 -0
  98. package/renderables/Slider.d.ts +44 -0
  99. package/renderables/TabSelect.d.ts +96 -0
  100. package/renderables/Text.d.ts +36 -0
  101. package/renderables/TextBufferRenderable.d.ts +105 -0
  102. package/renderables/TextNode.d.ts +91 -0
  103. package/renderables/TextTable.d.ts +140 -0
  104. package/renderables/Textarea.d.ts +114 -0
  105. package/renderables/TimeToFirstDraw.d.ts +24 -0
  106. package/renderables/__tests__/renderable-test-utils.d.ts +12 -0
  107. package/renderables/composition/VRenderable.d.ts +16 -0
  108. package/renderables/composition/constructs.d.ts +35 -0
  109. package/renderables/composition/vnode.d.ts +46 -0
  110. package/renderables/index.d.ts +22 -0
  111. package/renderables/markdown-parser.d.ts +10 -0
  112. package/renderer.d.ts +388 -0
  113. package/runtime-plugin-support.d.ts +3 -0
  114. package/runtime-plugin-support.js +29 -0
  115. package/runtime-plugin-support.js.map +10 -0
  116. package/runtime-plugin.d.ts +11 -0
  117. package/runtime-plugin.js +16 -0
  118. package/runtime-plugin.js.map +9 -0
  119. package/syntax-style.d.ts +54 -0
  120. package/testing/manual-clock.d.ts +16 -0
  121. package/testing/mock-keys.d.ts +81 -0
  122. package/testing/mock-mouse.d.ts +38 -0
  123. package/testing/mock-tree-sitter-client.d.ts +23 -0
  124. package/testing/spy.d.ts +7 -0
  125. package/testing/test-recorder.d.ts +61 -0
  126. package/testing/test-renderer.d.ts +23 -0
  127. package/testing.d.ts +6 -0
  128. package/testing.js +675 -0
  129. package/testing.js.map +15 -0
  130. package/text-buffer-view.d.ts +42 -0
  131. package/text-buffer.d.ts +67 -0
  132. package/types.d.ts +131 -0
  133. package/utils.d.ts +14 -0
  134. package/zig-structs.d.ts +155 -0
  135. package/zig.d.ts +351 -0
  136. package/dev/keypress-debug-renderer.ts +0 -148
  137. package/dev/keypress-debug.ts +0 -43
  138. package/dev/print-env-vars.ts +0 -32
  139. package/dev/test-tmux-graphics-334.sh +0 -68
  140. package/dev/thai-debug-test.ts +0 -68
  141. package/docs/development.md +0 -141
  142. package/docs/env-vars.md +0 -140
  143. package/docs/getting-started.md +0 -353
  144. package/docs/renderables-vs-constructs.md +0 -159
  145. package/docs/tree-sitter.md +0 -311
  146. package/scripts/build.ts +0 -400
  147. package/scripts/publish.ts +0 -60
  148. package/src/3d/SpriteResourceManager.ts +0 -286
  149. package/src/3d/SpriteUtils.ts +0 -71
  150. package/src/3d/TextureUtils.ts +0 -196
  151. package/src/3d/ThreeRenderable.ts +0 -197
  152. package/src/3d/WGPURenderer.ts +0 -294
  153. package/src/3d/animation/ExplodingSpriteEffect.ts +0 -513
  154. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +0 -429
  155. package/src/3d/animation/SpriteAnimator.ts +0 -633
  156. package/src/3d/animation/SpriteParticleGenerator.ts +0 -435
  157. package/src/3d/canvas.ts +0 -464
  158. package/src/3d/index.ts +0 -12
  159. package/src/3d/physics/PlanckPhysicsAdapter.ts +0 -72
  160. package/src/3d/physics/RapierPhysicsAdapter.ts +0 -66
  161. package/src/3d/physics/physics-interface.ts +0 -31
  162. package/src/3d/shaders/supersampling.wgsl +0 -201
  163. package/src/3d.ts +0 -3
  164. package/src/NativeSpanFeed.ts +0 -300
  165. package/src/Renderable.ts +0 -1698
  166. package/src/__snapshots__/buffer.test.ts.snap +0 -28
  167. package/src/animation/Timeline.test.ts +0 -2709
  168. package/src/animation/Timeline.ts +0 -598
  169. package/src/ansi.ts +0 -18
  170. package/src/benchmark/latest-all-bench-run.json +0 -707
  171. package/src/benchmark/latest-async-bench-run.json +0 -336
  172. package/src/benchmark/latest-default-bench-run.json +0 -657
  173. package/src/benchmark/latest-large-bench-run.json +0 -707
  174. package/src/benchmark/latest-quick-bench-run.json +0 -207
  175. package/src/benchmark/markdown-benchmark.ts +0 -1804
  176. package/src/benchmark/native-span-feed-async-benchmark.ts +0 -355
  177. package/src/benchmark/native-span-feed-benchmark.md +0 -56
  178. package/src/benchmark/native-span-feed-benchmark.ts +0 -596
  179. package/src/benchmark/native-span-feed-compare.ts +0 -280
  180. package/src/benchmark/renderer-benchmark.ts +0 -754
  181. package/src/benchmark/text-table-benchmark.ts +0 -947
  182. package/src/buffer.test.ts +0 -291
  183. package/src/buffer.ts +0 -519
  184. package/src/console.test.ts +0 -612
  185. package/src/console.ts +0 -1255
  186. package/src/edit-buffer.test.ts +0 -1769
  187. package/src/edit-buffer.ts +0 -411
  188. package/src/editor-view.test.ts +0 -1032
  189. package/src/editor-view.ts +0 -284
  190. package/src/examples/ascii-font-selection-demo.ts +0 -245
  191. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  192. package/src/examples/assets/concrete.png +0 -0
  193. package/src/examples/assets/crate.png +0 -0
  194. package/src/examples/assets/crate_emissive.png +0 -0
  195. package/src/examples/assets/forrest_background.png +0 -0
  196. package/src/examples/assets/hast-example.json +0 -1018
  197. package/src/examples/assets/heart.png +0 -0
  198. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  199. package/src/examples/assets/main_char_idle.png +0 -0
  200. package/src/examples/assets/main_char_jump_end.png +0 -0
  201. package/src/examples/assets/main_char_jump_landing.png +0 -0
  202. package/src/examples/assets/main_char_jump_start.png +0 -0
  203. package/src/examples/assets/main_char_run_loop.png +0 -0
  204. package/src/examples/assets/roughness_map.jpg +0 -0
  205. package/src/examples/build.ts +0 -115
  206. package/src/examples/code-demo.ts +0 -584
  207. package/src/examples/console-demo.ts +0 -358
  208. package/src/examples/core-plugin-slots-demo.ts +0 -759
  209. package/src/examples/diff-demo.ts +0 -699
  210. package/src/examples/draggable-three-demo.ts +0 -259
  211. package/src/examples/editor-demo.ts +0 -322
  212. package/src/examples/extmarks-demo.ts +0 -204
  213. package/src/examples/focus-restore-demo.ts +0 -310
  214. package/src/examples/fonts.ts +0 -245
  215. package/src/examples/fractal-shader-demo.ts +0 -268
  216. package/src/examples/framebuffer-demo.ts +0 -674
  217. package/src/examples/full-unicode-demo.ts +0 -181
  218. package/src/examples/golden-star-demo.ts +0 -933
  219. package/src/examples/grayscale-buffer-demo.ts +0 -249
  220. package/src/examples/hast-syntax-highlighting-demo.ts +0 -129
  221. package/src/examples/index.ts +0 -925
  222. package/src/examples/input-demo.ts +0 -377
  223. package/src/examples/input-select-layout-demo.ts +0 -425
  224. package/src/examples/install.sh +0 -143
  225. package/src/examples/keypress-debug-demo.ts +0 -452
  226. package/src/examples/lib/HexList.ts +0 -122
  227. package/src/examples/lib/PaletteGrid.ts +0 -125
  228. package/src/examples/lib/standalone-keys.ts +0 -25
  229. package/src/examples/lib/tab-controller.ts +0 -243
  230. package/src/examples/lights-phong-demo.ts +0 -290
  231. package/src/examples/link-demo.ts +0 -220
  232. package/src/examples/live-state-demo.ts +0 -480
  233. package/src/examples/markdown-demo.ts +0 -620
  234. package/src/examples/mouse-interaction-demo.ts +0 -428
  235. package/src/examples/nested-zindex-demo.ts +0 -357
  236. package/src/examples/opacity-example.ts +0 -235
  237. package/src/examples/opentui-demo.ts +0 -1057
  238. package/src/examples/physx-planck-2d-demo.ts +0 -507
  239. package/src/examples/physx-rapier-2d-demo.ts +0 -526
  240. package/src/examples/relative-positioning-demo.ts +0 -323
  241. package/src/examples/scroll-example.ts +0 -214
  242. package/src/examples/scrollbox-mouse-test.ts +0 -112
  243. package/src/examples/scrollbox-overlay-hit-test.ts +0 -206
  244. package/src/examples/select-demo.ts +0 -237
  245. package/src/examples/shader-cube-demo.ts +0 -772
  246. package/src/examples/simple-layout-example.ts +0 -591
  247. package/src/examples/slider-demo.ts +0 -617
  248. package/src/examples/split-mode-demo.ts +0 -445
  249. package/src/examples/sprite-animation-demo.ts +0 -443
  250. package/src/examples/sprite-particle-generator-demo.ts +0 -486
  251. package/src/examples/static-sprite-demo.ts +0 -193
  252. package/src/examples/sticky-scroll-example.ts +0 -308
  253. package/src/examples/styled-text-demo.ts +0 -282
  254. package/src/examples/tab-select-demo.ts +0 -219
  255. package/src/examples/terminal-title.ts +0 -29
  256. package/src/examples/terminal.ts +0 -305
  257. package/src/examples/text-node-demo.ts +0 -416
  258. package/src/examples/text-selection-demo.ts +0 -377
  259. package/src/examples/text-table-demo.ts +0 -503
  260. package/src/examples/text-truncation-demo.ts +0 -481
  261. package/src/examples/text-wrap.ts +0 -757
  262. package/src/examples/texture-loading-demo.ts +0 -259
  263. package/src/examples/timeline-example.ts +0 -670
  264. package/src/examples/transparency-demo.ts +0 -241
  265. package/src/examples/vnode-composition-demo.ts +0 -404
  266. package/src/index.ts +0 -22
  267. package/src/lib/KeyHandler.integration.test.ts +0 -292
  268. package/src/lib/KeyHandler.stopPropagation.test.ts +0 -289
  269. package/src/lib/KeyHandler.test.ts +0 -662
  270. package/src/lib/KeyHandler.ts +0 -222
  271. package/src/lib/RGBA.test.ts +0 -984
  272. package/src/lib/RGBA.ts +0 -204
  273. package/src/lib/ascii.font.ts +0 -330
  274. package/src/lib/border.test.ts +0 -83
  275. package/src/lib/border.ts +0 -168
  276. package/src/lib/bunfs.test.ts +0 -27
  277. package/src/lib/bunfs.ts +0 -18
  278. package/src/lib/clipboard.test.ts +0 -41
  279. package/src/lib/clipboard.ts +0 -47
  280. package/src/lib/clock.ts +0 -31
  281. package/src/lib/data-paths.test.ts +0 -133
  282. package/src/lib/data-paths.ts +0 -109
  283. package/src/lib/debounce.ts +0 -106
  284. package/src/lib/detect-links.test.ts +0 -98
  285. package/src/lib/detect-links.ts +0 -56
  286. package/src/lib/env.test.ts +0 -228
  287. package/src/lib/env.ts +0 -209
  288. package/src/lib/extmarks-history.ts +0 -51
  289. package/src/lib/extmarks-multiwidth.test.ts +0 -322
  290. package/src/lib/extmarks.test.ts +0 -3457
  291. package/src/lib/extmarks.ts +0 -843
  292. package/src/lib/fonts/block.json +0 -405
  293. package/src/lib/fonts/grid.json +0 -265
  294. package/src/lib/fonts/huge.json +0 -741
  295. package/src/lib/fonts/pallet.json +0 -314
  296. package/src/lib/fonts/shade.json +0 -591
  297. package/src/lib/fonts/slick.json +0 -321
  298. package/src/lib/fonts/tiny.json +0 -69
  299. package/src/lib/hast-styled-text.ts +0 -59
  300. package/src/lib/index.ts +0 -21
  301. package/src/lib/keymapping.test.ts +0 -280
  302. package/src/lib/keymapping.ts +0 -87
  303. package/src/lib/objects-in-viewport.test.ts +0 -787
  304. package/src/lib/objects-in-viewport.ts +0 -153
  305. package/src/lib/output.capture.ts +0 -58
  306. package/src/lib/parse.keypress-kitty.protocol.test.ts +0 -340
  307. package/src/lib/parse.keypress-kitty.test.ts +0 -663
  308. package/src/lib/parse.keypress-kitty.ts +0 -439
  309. package/src/lib/parse.keypress.test.ts +0 -1849
  310. package/src/lib/parse.keypress.ts +0 -397
  311. package/src/lib/parse.mouse.test.ts +0 -552
  312. package/src/lib/parse.mouse.ts +0 -232
  313. package/src/lib/paste.ts +0 -16
  314. package/src/lib/queue.ts +0 -65
  315. package/src/lib/renderable.validations.test.ts +0 -87
  316. package/src/lib/renderable.validations.ts +0 -83
  317. package/src/lib/scroll-acceleration.ts +0 -98
  318. package/src/lib/selection.ts +0 -240
  319. package/src/lib/singleton.ts +0 -28
  320. package/src/lib/stdin-parser.test.ts +0 -1676
  321. package/src/lib/stdin-parser.ts +0 -1248
  322. package/src/lib/styled-text.ts +0 -178
  323. package/src/lib/terminal-capability-detection.test.ts +0 -202
  324. package/src/lib/terminal-capability-detection.ts +0 -79
  325. package/src/lib/terminal-palette.test.ts +0 -878
  326. package/src/lib/terminal-palette.ts +0 -383
  327. package/src/lib/tree-sitter/assets/README.md +0 -118
  328. package/src/lib/tree-sitter/assets/update.ts +0 -331
  329. package/src/lib/tree-sitter/assets.d.ts +0 -9
  330. package/src/lib/tree-sitter/cache.test.ts +0 -270
  331. package/src/lib/tree-sitter/client.test.ts +0 -1061
  332. package/src/lib/tree-sitter/client.ts +0 -615
  333. package/src/lib/tree-sitter/default-parsers.ts +0 -80
  334. package/src/lib/tree-sitter/download-utils.ts +0 -148
  335. package/src/lib/tree-sitter/index.ts +0 -28
  336. package/src/lib/tree-sitter/parser.worker.ts +0 -1001
  337. package/src/lib/tree-sitter/parsers-config.ts +0 -75
  338. package/src/lib/tree-sitter/resolve-ft.ts +0 -62
  339. package/src/lib/tree-sitter/types.ts +0 -81
  340. package/src/lib/tree-sitter-styled-text.test.ts +0 -1253
  341. package/src/lib/tree-sitter-styled-text.ts +0 -306
  342. package/src/lib/validate-dir-name.ts +0 -55
  343. package/src/lib/yoga.options.test.ts +0 -628
  344. package/src/lib/yoga.options.ts +0 -346
  345. package/src/plugins/core-slot.ts +0 -579
  346. package/src/plugins/registry.ts +0 -377
  347. package/src/plugins/types.ts +0 -46
  348. package/src/post/filters.ts +0 -888
  349. package/src/renderables/ASCIIFont.ts +0 -219
  350. package/src/renderables/Box.test.ts +0 -160
  351. package/src/renderables/Box.ts +0 -295
  352. package/src/renderables/Code.test.ts +0 -2062
  353. package/src/renderables/Code.ts +0 -357
  354. package/src/renderables/Diff.regression.test.ts +0 -226
  355. package/src/renderables/Diff.test.ts +0 -3027
  356. package/src/renderables/Diff.ts +0 -1209
  357. package/src/renderables/EditBufferRenderable.ts +0 -764
  358. package/src/renderables/FrameBuffer.ts +0 -47
  359. package/src/renderables/Input.test.ts +0 -1228
  360. package/src/renderables/Input.ts +0 -245
  361. package/src/renderables/LineNumberRenderable.ts +0 -675
  362. package/src/renderables/Markdown.ts +0 -1106
  363. package/src/renderables/ScrollBar.ts +0 -422
  364. package/src/renderables/ScrollBox.ts +0 -883
  365. package/src/renderables/Select.test.ts +0 -1010
  366. package/src/renderables/Select.ts +0 -523
  367. package/src/renderables/Slider.test.ts +0 -456
  368. package/src/renderables/Slider.ts +0 -347
  369. package/src/renderables/TabSelect.test.ts +0 -197
  370. package/src/renderables/TabSelect.ts +0 -455
  371. package/src/renderables/Text.selection-buffer.test.ts +0 -123
  372. package/src/renderables/Text.test.ts +0 -2660
  373. package/src/renderables/Text.ts +0 -147
  374. package/src/renderables/TextBufferRenderable.ts +0 -518
  375. package/src/renderables/TextNode.test.ts +0 -1058
  376. package/src/renderables/TextNode.ts +0 -325
  377. package/src/renderables/TextTable.test.ts +0 -1421
  378. package/src/renderables/TextTable.ts +0 -1344
  379. package/src/renderables/Textarea.ts +0 -732
  380. package/src/renderables/TimeToFirstDraw.ts +0 -89
  381. package/src/renderables/__snapshots__/Code.test.ts.snap +0 -13
  382. package/src/renderables/__snapshots__/Diff.test.ts.snap +0 -785
  383. package/src/renderables/__snapshots__/Text.test.ts.snap +0 -421
  384. package/src/renderables/__snapshots__/TextTable.test.ts.snap +0 -215
  385. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +0 -144
  386. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +0 -816
  387. package/src/renderables/__tests__/LineNumberRenderable.test.ts +0 -1787
  388. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +0 -85
  389. package/src/renderables/__tests__/Markdown.test.ts +0 -2287
  390. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +0 -87
  391. package/src/renderables/__tests__/Textarea.buffer.test.ts +0 -682
  392. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +0 -675
  393. package/src/renderables/__tests__/Textarea.editing.test.ts +0 -2041
  394. package/src/renderables/__tests__/Textarea.error-handling.test.ts +0 -35
  395. package/src/renderables/__tests__/Textarea.events.test.ts +0 -738
  396. package/src/renderables/__tests__/Textarea.highlights.test.ts +0 -590
  397. package/src/renderables/__tests__/Textarea.keybinding.test.ts +0 -3149
  398. package/src/renderables/__tests__/Textarea.paste.test.ts +0 -357
  399. package/src/renderables/__tests__/Textarea.rendering.test.ts +0 -1864
  400. package/src/renderables/__tests__/Textarea.scroll.test.ts +0 -733
  401. package/src/renderables/__tests__/Textarea.selection.test.ts +0 -1590
  402. package/src/renderables/__tests__/Textarea.stress.test.ts +0 -670
  403. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +0 -383
  404. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +0 -310
  405. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +0 -221
  406. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +0 -89
  407. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +0 -457
  408. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +0 -158
  409. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +0 -387
  410. package/src/renderables/__tests__/markdown-parser.test.ts +0 -217
  411. package/src/renderables/__tests__/renderable-test-utils.ts +0 -60
  412. package/src/renderables/composition/README.md +0 -8
  413. package/src/renderables/composition/VRenderable.ts +0 -32
  414. package/src/renderables/composition/constructs.ts +0 -127
  415. package/src/renderables/composition/vnode.ts +0 -289
  416. package/src/renderables/index.ts +0 -22
  417. package/src/renderables/markdown-parser.ts +0 -66
  418. package/src/renderer.ts +0 -2363
  419. package/src/runtime-plugin-support.ts +0 -39
  420. package/src/runtime-plugin.ts +0 -144
  421. package/src/syntax-style.test.ts +0 -841
  422. package/src/syntax-style.ts +0 -264
  423. package/src/testing/README.md +0 -210
  424. package/src/testing/capture-spans.test.ts +0 -194
  425. package/src/testing/integration.test.ts +0 -276
  426. package/src/testing/manual-clock.ts +0 -106
  427. package/src/testing/mock-keys.test.ts +0 -1356
  428. package/src/testing/mock-keys.ts +0 -449
  429. package/src/testing/mock-mouse.test.ts +0 -218
  430. package/src/testing/mock-mouse.ts +0 -247
  431. package/src/testing/mock-tree-sitter-client.ts +0 -73
  432. package/src/testing/spy.ts +0 -13
  433. package/src/testing/test-recorder.test.ts +0 -415
  434. package/src/testing/test-recorder.ts +0 -145
  435. package/src/testing/test-renderer.ts +0 -116
  436. package/src/testing.ts +0 -7
  437. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +0 -481
  438. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +0 -19
  439. package/src/tests/__snapshots__/scrollbox.test.ts.snap +0 -29
  440. package/src/tests/absolute-positioning.snapshot.test.ts +0 -638
  441. package/src/tests/allocator-stats.test.ts +0 -38
  442. package/src/tests/destroy-during-render.test.ts +0 -200
  443. package/src/tests/hover-cursor.test.ts +0 -98
  444. package/src/tests/native-span-feed-async.test.ts +0 -173
  445. package/src/tests/native-span-feed-close.test.ts +0 -120
  446. package/src/tests/native-span-feed-coverage.test.ts +0 -227
  447. package/src/tests/native-span-feed-edge-cases.test.ts +0 -352
  448. package/src/tests/native-span-feed-use-after-free.test.ts +0 -45
  449. package/src/tests/opacity.test.ts +0 -123
  450. package/src/tests/renderable.snapshot.test.ts +0 -524
  451. package/src/tests/renderable.test.ts +0 -1281
  452. package/src/tests/renderer.console-startup.test.ts +0 -65
  453. package/src/tests/renderer.control.test.ts +0 -364
  454. package/src/tests/renderer.core-slot-binding.test.ts +0 -952
  455. package/src/tests/renderer.cursor.test.ts +0 -26
  456. package/src/tests/renderer.destroy-during-render.test.ts +0 -110
  457. package/src/tests/renderer.focus-restore.test.ts +0 -228
  458. package/src/tests/renderer.focus.test.ts +0 -251
  459. package/src/tests/renderer.idle.test.ts +0 -219
  460. package/src/tests/renderer.input.test.ts +0 -2145
  461. package/src/tests/renderer.kitty-flags.test.ts +0 -195
  462. package/src/tests/renderer.mouse.test.ts +0 -1269
  463. package/src/tests/renderer.palette.test.ts +0 -629
  464. package/src/tests/renderer.selection.test.ts +0 -49
  465. package/src/tests/renderer.slot-registry.test.ts +0 -649
  466. package/src/tests/renderer.useMouse.test.ts +0 -50
  467. package/src/tests/runtime-plugin-support.fixture.ts +0 -11
  468. package/src/tests/runtime-plugin-support.test.ts +0 -28
  469. package/src/tests/runtime-plugin.fixture.ts +0 -40
  470. package/src/tests/runtime-plugin.test.ts +0 -190
  471. package/src/tests/scrollbox-culling-bug.test.ts +0 -114
  472. package/src/tests/scrollbox-hitgrid-resize.test.ts +0 -136
  473. package/src/tests/scrollbox-hitgrid.test.ts +0 -909
  474. package/src/tests/scrollbox.test.ts +0 -1530
  475. package/src/tests/wrap-resize-perf.test.ts +0 -229
  476. package/src/tests/yoga-setters.test.ts +0 -921
  477. package/src/text-buffer-view.test.ts +0 -705
  478. package/src/text-buffer-view.ts +0 -189
  479. package/src/text-buffer.test.ts +0 -347
  480. package/src/text-buffer.ts +0 -250
  481. package/src/types.ts +0 -152
  482. package/src/utils.ts +0 -88
  483. package/src/zig/ansi.zig +0 -268
  484. package/src/zig/bench/README.md +0 -50
  485. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +0 -887
  486. package/src/zig/bench/edit-buffer_bench.zig +0 -476
  487. package/src/zig/bench/native-span-feed_bench.zig +0 -100
  488. package/src/zig/bench/rope-markers_bench.zig +0 -713
  489. package/src/zig/bench/rope_bench.zig +0 -514
  490. package/src/zig/bench/styled-text_bench.zig +0 -470
  491. package/src/zig/bench/text-buffer-coords_bench.zig +0 -362
  492. package/src/zig/bench/text-buffer-view_bench.zig +0 -459
  493. package/src/zig/bench/text-chunk-graphemes_bench.zig +0 -273
  494. package/src/zig/bench/utf8_bench.zig +0 -799
  495. package/src/zig/bench-utils.zig +0 -431
  496. package/src/zig/bench.zig +0 -217
  497. package/src/zig/buffer.zig +0 -2223
  498. package/src/zig/build.zig +0 -289
  499. package/src/zig/build.zig.zon +0 -16
  500. package/src/zig/edit-buffer.zig +0 -825
  501. package/src/zig/editor-view.zig +0 -802
  502. package/src/zig/event-bus.zig +0 -13
  503. package/src/zig/event-emitter.zig +0 -65
  504. package/src/zig/file-logger.zig +0 -92
  505. package/src/zig/grapheme.zig +0 -599
  506. package/src/zig/lib.zig +0 -1834
  507. package/src/zig/link.zig +0 -333
  508. package/src/zig/logger.zig +0 -43
  509. package/src/zig/mem-registry.zig +0 -125
  510. package/src/zig/native-span-feed-bench-lib.zig +0 -7
  511. package/src/zig/native-span-feed.zig +0 -708
  512. package/src/zig/renderer.zig +0 -1386
  513. package/src/zig/rope.zig +0 -1220
  514. package/src/zig/syntax-style.zig +0 -161
  515. package/src/zig/terminal.zig +0 -975
  516. package/src/zig/test.zig +0 -70
  517. package/src/zig/tests/README.md +0 -18
  518. package/src/zig/tests/buffer_test.zig +0 -2526
  519. package/src/zig/tests/edit-buffer-history_test.zig +0 -271
  520. package/src/zig/tests/edit-buffer_test.zig +0 -1689
  521. package/src/zig/tests/editor-view_test.zig +0 -3299
  522. package/src/zig/tests/event-emitter_test.zig +0 -249
  523. package/src/zig/tests/grapheme_test.zig +0 -1304
  524. package/src/zig/tests/link_test.zig +0 -190
  525. package/src/zig/tests/mem-registry_test.zig +0 -473
  526. package/src/zig/tests/memory_leak_regression_test.zig +0 -159
  527. package/src/zig/tests/native-span-feed_test.zig +0 -1264
  528. package/src/zig/tests/renderer_test.zig +0 -1010
  529. package/src/zig/tests/rope-nested_test.zig +0 -712
  530. package/src/zig/tests/rope_fuzz_test.zig +0 -238
  531. package/src/zig/tests/rope_test.zig +0 -2362
  532. package/src/zig/tests/segment-merge.test.zig +0 -148
  533. package/src/zig/tests/syntax-style_test.zig +0 -557
  534. package/src/zig/tests/terminal_test.zig +0 -719
  535. package/src/zig/tests/text-buffer-drawing_test.zig +0 -3237
  536. package/src/zig/tests/text-buffer-highlights_test.zig +0 -666
  537. package/src/zig/tests/text-buffer-iterators_test.zig +0 -776
  538. package/src/zig/tests/text-buffer-segment_test.zig +0 -320
  539. package/src/zig/tests/text-buffer-selection_test.zig +0 -1035
  540. package/src/zig/tests/text-buffer-selection_viewport_test.zig +0 -358
  541. package/src/zig/tests/text-buffer-view_test.zig +0 -3649
  542. package/src/zig/tests/text-buffer_test.zig +0 -2191
  543. package/src/zig/tests/unicode-width-map.zon +0 -3909
  544. package/src/zig/tests/utf8_no_zwj_test.zig +0 -260
  545. package/src/zig/tests/utf8_test.zig +0 -4057
  546. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +0 -267
  547. package/src/zig/tests/utf8_wcwidth_test.zig +0 -357
  548. package/src/zig/tests/word-wrap-editing_test.zig +0 -498
  549. package/src/zig/tests/wrap-cache-perf_test.zig +0 -113
  550. package/src/zig/text-buffer-iterators.zig +0 -499
  551. package/src/zig/text-buffer-segment.zig +0 -404
  552. package/src/zig/text-buffer-view.zig +0 -1371
  553. package/src/zig/text-buffer.zig +0 -1180
  554. package/src/zig/utf8.zig +0 -1948
  555. package/src/zig/utils.zig +0 -9
  556. package/src/zig-structs.ts +0 -261
  557. package/src/zig.ts +0 -3843
  558. package/tsconfig.build.json +0 -22
  559. package/tsconfig.json +0 -28
  560. /package/{src/lib/tree-sitter/assets → assets}/javascript/highlights.scm +0 -0
  561. /package/{src/lib/tree-sitter/assets → assets}/javascript/tree-sitter-javascript.wasm +0 -0
  562. /package/{src/lib/tree-sitter/assets → assets}/markdown/highlights.scm +0 -0
  563. /package/{src/lib/tree-sitter/assets → assets}/markdown/injections.scm +0 -0
  564. /package/{src/lib/tree-sitter/assets → assets}/markdown/tree-sitter-markdown.wasm +0 -0
  565. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/highlights.scm +0 -0
  566. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  567. /package/{src/lib/tree-sitter/assets → assets}/typescript/highlights.scm +0 -0
  568. /package/{src/lib/tree-sitter/assets → assets}/typescript/tree-sitter-typescript.wasm +0 -0
  569. /package/{src/lib/tree-sitter/assets → assets}/zig/highlights.scm +0 -0
  570. /package/{src/lib/tree-sitter/assets → assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -1,2526 +0,0 @@
1
- const std = @import("std");
2
- const buffer_mod = @import("../buffer.zig");
3
- const text_buffer = @import("../text-buffer.zig");
4
- const text_buffer_view = @import("../text-buffer-view.zig");
5
- const gp = @import("../grapheme.zig");
6
- const link = @import("../link.zig");
7
- const ansi = @import("../ansi.zig");
8
-
9
- const OptimizedBuffer = buffer_mod.OptimizedBuffer;
10
- const TextBuffer = text_buffer.UnifiedTextBuffer;
11
- const TextBufferView = text_buffer_view.UnifiedTextBufferView;
12
- const RGBA = buffer_mod.RGBA;
13
-
14
- fn initBufferForOomRegression(allocator: std.mem.Allocator) !void {
15
- var local_pool = gp.GraphemePool.initWithOptions(allocator, .{});
16
- defer local_pool.deinit();
17
-
18
- var local_link_pool = link.LinkPool.init(allocator);
19
- defer local_link_pool.deinit();
20
-
21
- var buf = try OptimizedBuffer.init(
22
- allocator,
23
- 1,
24
- 1,
25
- .{ .pool = &local_pool, .id = "oom-regression", .link_pool = &local_link_pool },
26
- );
27
- defer buf.deinit();
28
- }
29
-
30
- test "OptimizedBuffer - init frees allocations on OOM" {
31
- try std.testing.checkAllAllocationFailures(std.testing.allocator, initBufferForOomRegression, .{});
32
- }
33
-
34
- test "OptimizedBuffer - init and deinit" {
35
- const pool = gp.initGlobalPool(std.testing.allocator);
36
- defer gp.deinitGlobalPool();
37
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
38
- defer local_link_pool.deinit();
39
-
40
- var buf = try OptimizedBuffer.init(
41
- std.testing.allocator,
42
- 10,
43
- 10,
44
- .{ .pool = pool, .id = "test-buffer" },
45
- );
46
- defer buf.deinit();
47
-
48
- try std.testing.expectEqual(@as(u32, 10), buf.getWidth());
49
- try std.testing.expectEqual(@as(u32, 10), buf.getHeight());
50
- }
51
-
52
- test "OptimizedBuffer - clear fills with default char" {
53
- const pool = gp.initGlobalPool(std.testing.allocator);
54
- defer gp.deinitGlobalPool();
55
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
56
- defer local_link_pool.deinit();
57
-
58
- var buf = try OptimizedBuffer.init(
59
- std.testing.allocator,
60
- 5,
61
- 5,
62
- .{ .pool = pool, .id = "test-buffer" },
63
- );
64
- defer buf.deinit();
65
-
66
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
67
- try buf.clear(bg, null);
68
-
69
- var y: u32 = 0;
70
- while (y < 5) : (y += 1) {
71
- var x: u32 = 0;
72
- while (x < 5) : (x += 1) {
73
- const cell = buf.get(x, y).?;
74
- try std.testing.expectEqual(@as(u32, 32), cell.char);
75
- }
76
- }
77
- }
78
-
79
- test "OptimizedBuffer - drawText with ASCII" {
80
- const pool = gp.initGlobalPool(std.testing.allocator);
81
- defer gp.deinitGlobalPool();
82
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
83
- defer local_link_pool.deinit();
84
-
85
- var buf = try OptimizedBuffer.init(
86
- std.testing.allocator,
87
- 20,
88
- 5,
89
- .{ .pool = pool, .id = "test-buffer" },
90
- );
91
- defer buf.deinit();
92
-
93
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
94
- try buf.clear(bg, null);
95
-
96
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
97
- try buf.drawText("Hello", 0, 0, fg, bg, 0);
98
-
99
- const cell_h = buf.get(0, 0).?;
100
- try std.testing.expectEqual(@as(u32, 'H'), cell_h.char);
101
-
102
- const cell_e = buf.get(1, 0).?;
103
- try std.testing.expectEqual(@as(u32, 'e'), cell_e.char);
104
- }
105
-
106
- test "OptimizedBuffer - repeated emoji rendering should not exhaust pool" {
107
- const pool = gp.initGlobalPool(std.testing.allocator);
108
- defer gp.deinitGlobalPool();
109
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
110
- defer local_link_pool.deinit();
111
-
112
- var buf = try OptimizedBuffer.init(
113
- std.testing.allocator,
114
- 20,
115
- 5,
116
- .{ .pool = pool, .id = "test-buffer" },
117
- );
118
- defer buf.deinit();
119
-
120
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
121
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
122
-
123
- var i: u32 = 0;
124
- while (i < 1000) : (i += 1) {
125
- try buf.clear(bg, null);
126
- try buf.drawText("🌟🎨🚀", 0, 0, fg, bg, 0);
127
- }
128
-
129
- const cell = buf.get(0, 0).?;
130
- try std.testing.expect(gp.isGraphemeChar(cell.char));
131
- }
132
-
133
- test "OptimizedBuffer - repeated CJK rendering should not exhaust pool" {
134
- const pool = gp.initGlobalPool(std.testing.allocator);
135
- defer gp.deinitGlobalPool();
136
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
137
- defer local_link_pool.deinit();
138
-
139
- var buf = try OptimizedBuffer.init(
140
- std.testing.allocator,
141
- 20,
142
- 5,
143
- .{ .pool = pool, .id = "test-buffer" },
144
- );
145
- defer buf.deinit();
146
-
147
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
148
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
149
-
150
- var i: u32 = 0;
151
- while (i < 1000) : (i += 1) {
152
- try buf.clear(bg, null);
153
- try buf.drawText("测试文字", 0, 0, fg, bg, 0);
154
- }
155
-
156
- const cell = buf.get(0, 0).?;
157
- try std.testing.expect(gp.isGraphemeChar(cell.char));
158
- }
159
-
160
- test "OptimizedBuffer - drawTextBuffer repeatedly should not exhaust pool" {
161
- const pool = gp.initGlobalPool(std.testing.allocator);
162
- defer gp.deinitGlobalPool();
163
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
164
- defer local_link_pool.deinit();
165
-
166
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
167
- defer tb.deinit();
168
-
169
- var view = try TextBufferView.init(std.testing.allocator, tb);
170
- defer view.deinit();
171
-
172
- try tb.setText("Hello 🌟 World\n测试 🎨 Test\n🚀 Rocket");
173
-
174
- var buf = try OptimizedBuffer.init(
175
- std.testing.allocator,
176
- 80,
177
- 25,
178
- .{ .pool = pool, .id = "test-buffer" },
179
- );
180
- defer buf.deinit();
181
-
182
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
183
-
184
- var i: u32 = 0;
185
- while (i < 1000) : (i += 1) {
186
- try buf.clear(bg, null);
187
- try buf.drawTextBuffer(view, 0, 0);
188
- }
189
- }
190
-
191
- test "OptimizedBuffer - mixed ASCII and emoji repeated rendering" {
192
- const pool = gp.initGlobalPool(std.testing.allocator);
193
- defer gp.deinitGlobalPool();
194
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
195
- defer local_link_pool.deinit();
196
-
197
- var buf = try OptimizedBuffer.init(
198
- std.testing.allocator,
199
- 40,
200
- 5,
201
- .{ .pool = pool, .id = "test-buffer" },
202
- );
203
- defer buf.deinit();
204
-
205
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
206
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
207
-
208
- var i: u32 = 0;
209
- while (i < 500) : (i += 1) {
210
- try buf.clear(bg, null);
211
- try buf.drawText("A🌟B🎨C🚀D", 0, 0, fg, bg, 0);
212
- try buf.drawText("测试文字处理", 0, 1, fg, bg, 0);
213
- try buf.drawText("Hello World!", 0, 2, fg, bg, 0);
214
- }
215
-
216
- const cell = buf.get(0, 0).?;
217
- try std.testing.expectEqual(@as(u32, 'A'), cell.char);
218
- }
219
-
220
- test "OptimizedBuffer - overwriting graphemes repeatedly" {
221
- const pool = gp.initGlobalPool(std.testing.allocator);
222
- defer gp.deinitGlobalPool();
223
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
224
- defer local_link_pool.deinit();
225
-
226
- var buf = try OptimizedBuffer.init(
227
- std.testing.allocator,
228
- 20,
229
- 5,
230
- .{ .pool = pool, .id = "test-buffer" },
231
- );
232
- defer buf.deinit();
233
-
234
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
235
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
236
-
237
- var i: u32 = 0;
238
- while (i < 1000) : (i += 1) {
239
- try buf.drawText("🌟", 0, 0, fg, bg, 0);
240
- try buf.drawText("🎨", 0, 0, fg, bg, 0);
241
- try buf.drawText("🚀", 0, 0, fg, bg, 0);
242
- }
243
-
244
- const cell = buf.get(0, 0).?;
245
- try std.testing.expect(gp.isGraphemeChar(cell.char));
246
- }
247
-
248
- test "OptimizedBuffer - rendering to different positions" {
249
- const pool = gp.initGlobalPool(std.testing.allocator);
250
- defer gp.deinitGlobalPool();
251
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
252
- defer local_link_pool.deinit();
253
-
254
- var buf = try OptimizedBuffer.init(
255
- std.testing.allocator,
256
- 80,
257
- 25,
258
- .{ .pool = pool, .id = "test-buffer" },
259
- );
260
- defer buf.deinit();
261
-
262
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
263
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
264
-
265
- var i: u32 = 0;
266
- while (i < 100) : (i += 1) {
267
- try buf.clear(bg, null);
268
-
269
- var y: u32 = 0;
270
- while (y < 20) : (y += 1) {
271
- var x: u32 = 0;
272
- while (x < 60) : (x += 10) {
273
- try buf.drawText("🌟", x, y, fg, bg, 0);
274
- }
275
- }
276
- }
277
-
278
- const cell = buf.get(0, 0).?;
279
- try std.testing.expect(gp.isGraphemeChar(cell.char));
280
- }
281
-
282
- test "OptimizedBuffer - large text buffer with wrapping repeated render" {
283
- const pool = gp.initGlobalPool(std.testing.allocator);
284
- defer gp.deinitGlobalPool();
285
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
286
- defer local_link_pool.deinit();
287
-
288
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
289
- defer tb.deinit();
290
-
291
- var view = try TextBufferView.init(std.testing.allocator, tb);
292
- defer view.deinit();
293
-
294
- var text_builder: std.ArrayListUnmanaged(u8) = .{};
295
- defer text_builder.deinit(std.testing.allocator);
296
-
297
- var line: u32 = 0;
298
- while (line < 20) : (line += 1) {
299
- try text_builder.appendSlice(std.testing.allocator, "Line ");
300
- try text_builder.writer(std.testing.allocator).print("{d}", .{line});
301
- try text_builder.appendSlice(std.testing.allocator, ": 🌟 测试 🎨 Test 🚀\n");
302
- }
303
-
304
- try tb.setText(text_builder.items);
305
-
306
- view.setWrapMode(.char);
307
- view.setWrapWidth(40);
308
-
309
- var buf = try OptimizedBuffer.init(
310
- std.testing.allocator,
311
- 80,
312
- 50,
313
- .{ .pool = pool, .id = "test-buffer" },
314
- );
315
- defer buf.deinit();
316
-
317
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
318
-
319
- var i: u32 = 0;
320
- while (i < 200) : (i += 1) {
321
- try buf.clear(bg, null);
322
- try buf.drawTextBuffer(view, 0, 0);
323
- }
324
- }
325
-
326
- test "OptimizedBuffer - grapheme tracker counts" {
327
- const pool = gp.initGlobalPool(std.testing.allocator);
328
- defer gp.deinitGlobalPool();
329
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
330
- defer local_link_pool.deinit();
331
-
332
- var buf = try OptimizedBuffer.init(
333
- std.testing.allocator,
334
- 20,
335
- 5,
336
- .{ .pool = pool, .id = "test-buffer" },
337
- );
338
- defer buf.deinit();
339
-
340
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
341
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
342
-
343
- try buf.drawText("🌟🎨🚀", 0, 0, fg, bg, 0);
344
-
345
- const count_after_draw = buf.grapheme_tracker.getGraphemeCount();
346
- try std.testing.expect(count_after_draw > 0);
347
- try std.testing.expect(count_after_draw <= 10);
348
-
349
- var i: u32 = 0;
350
- while (i < 100) : (i += 1) {
351
- try buf.clear(bg, null);
352
- try buf.drawText("🌟🎨🚀", 0, 0, fg, bg, 0);
353
- }
354
-
355
- const count_after_repeated = buf.grapheme_tracker.getGraphemeCount();
356
- try std.testing.expect(count_after_repeated <= 20);
357
- }
358
-
359
- test "OptimizedBuffer - alternating emojis should not leak" {
360
- const pool = gp.initGlobalPool(std.testing.allocator);
361
- defer gp.deinitGlobalPool();
362
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
363
- defer local_link_pool.deinit();
364
-
365
- var buf = try OptimizedBuffer.init(
366
- std.testing.allocator,
367
- 20,
368
- 5,
369
- .{ .pool = pool, .id = "test-buffer" },
370
- );
371
- defer buf.deinit();
372
-
373
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
374
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
375
-
376
- var i: u32 = 0;
377
- while (i < 500) : (i += 1) {
378
- if (i % 2 == 0) {
379
- try buf.drawText("🌟🎨🚀", 0, 0, fg, bg, 0);
380
- } else {
381
- try buf.drawText("🍕🍔🍟", 0, 0, fg, bg, 0);
382
- }
383
- }
384
-
385
- const count = buf.grapheme_tracker.getGraphemeCount();
386
- try std.testing.expect(count <= 20);
387
- }
388
-
389
- test "OptimizedBuffer - drawTextBuffer without clear should not exhaust pool" {
390
- const pool = gp.initGlobalPool(std.testing.allocator);
391
- defer gp.deinitGlobalPool();
392
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
393
- defer local_link_pool.deinit();
394
-
395
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
396
- defer tb.deinit();
397
-
398
- var view = try TextBufferView.init(std.testing.allocator, tb);
399
- defer view.deinit();
400
-
401
- try tb.setText("🌟🎨🚀");
402
-
403
- var buf = try OptimizedBuffer.init(
404
- std.testing.allocator,
405
- 80,
406
- 25,
407
- .{ .pool = pool, .id = "test-buffer" },
408
- );
409
- defer buf.deinit();
410
-
411
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
412
- try buf.clear(bg, null);
413
-
414
- var i: u32 = 0;
415
- while (i < 2000) : (i += 1) {
416
- try buf.drawTextBuffer(view, 0, 0);
417
- }
418
-
419
- const count = buf.grapheme_tracker.getGraphemeCount();
420
- try std.testing.expect(count < 100);
421
- }
422
-
423
- test "OptimizedBuffer - many small graphemes without clear" {
424
- const pool = gp.initGlobalPool(std.testing.allocator);
425
- defer gp.deinitGlobalPool();
426
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
427
- defer local_link_pool.deinit();
428
-
429
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
430
- defer tb.deinit();
431
-
432
- var view = try TextBufferView.init(std.testing.allocator, tb);
433
- defer view.deinit();
434
-
435
- try tb.setText("• • • •");
436
-
437
- var buf = try OptimizedBuffer.init(
438
- std.testing.allocator,
439
- 80,
440
- 25,
441
- .{ .pool = pool, .id = "test-buffer" },
442
- );
443
- defer buf.deinit();
444
-
445
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
446
- try buf.clear(bg, null);
447
-
448
- var i: u32 = 0;
449
- while (i < 5000) : (i += 1) {
450
- try buf.drawTextBuffer(view, 0, 0);
451
- }
452
-
453
- const count = buf.grapheme_tracker.getGraphemeCount();
454
- try std.testing.expect(count < 200);
455
- }
456
-
457
- test "OptimizedBuffer - stress test with many graphemes" {
458
- const pool = gp.initGlobalPool(std.testing.allocator);
459
- defer gp.deinitGlobalPool();
460
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
461
- defer local_link_pool.deinit();
462
-
463
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
464
- defer tb.deinit();
465
-
466
- var view = try TextBufferView.init(std.testing.allocator, tb);
467
- defer view.deinit();
468
-
469
- var text_builder: std.ArrayListUnmanaged(u8) = .{};
470
- defer text_builder.deinit(std.testing.allocator);
471
-
472
- var line: u32 = 0;
473
- while (line < 10) : (line += 1) {
474
- try text_builder.appendSlice(std.testing.allocator, "🌟🎨🚀🍕🍔🍟🌈🎭🎪🎨🎬🎤🎧🎼🎹🎺🎸🎻\n");
475
- }
476
-
477
- try tb.setText(text_builder.items);
478
-
479
- var buf = try OptimizedBuffer.init(
480
- std.testing.allocator,
481
- 80,
482
- 25,
483
- .{ .pool = pool, .id = "test-buffer" },
484
- );
485
- defer buf.deinit();
486
-
487
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
488
- try buf.clear(bg, null);
489
-
490
- var i: u32 = 0;
491
- while (i < 1000) : (i += 1) {
492
- try buf.drawTextBuffer(view, 0, 0);
493
- }
494
-
495
- const count = buf.grapheme_tracker.getGraphemeCount();
496
- try std.testing.expect(count > 0);
497
- try std.testing.expect(count < 1000);
498
-
499
- const first_cell = buf.get(0, 0).?;
500
- try std.testing.expect(gp.isGraphemeChar(first_cell.char));
501
- }
502
-
503
- test "OptimizedBuffer - pool slot exhaustion test" {
504
- const pool = gp.initGlobalPool(std.testing.allocator);
505
- defer gp.deinitGlobalPool();
506
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
507
- defer local_link_pool.deinit();
508
-
509
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
510
- defer tb.deinit();
511
-
512
- var view = try TextBufferView.init(std.testing.allocator, tb);
513
- defer view.deinit();
514
-
515
- try tb.setText("• • • • • • • • • •");
516
-
517
- var buf = try OptimizedBuffer.init(
518
- std.testing.allocator,
519
- 80,
520
- 25,
521
- .{ .pool = pool, .id = "test-buffer" },
522
- );
523
- defer buf.deinit();
524
-
525
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
526
-
527
- var i: u32 = 0;
528
- while (i < 10000) : (i += 1) {
529
- if (i % 100 == 0) {
530
- try buf.clear(bg, null);
531
- }
532
- try buf.drawTextBuffer(view, 0, 0);
533
- }
534
-
535
- const cell = buf.get(0, 0).?;
536
- try std.testing.expect(gp.isGraphemeChar(cell.char));
537
-
538
- const count = buf.grapheme_tracker.getGraphemeCount();
539
- try std.testing.expect(count > 0);
540
- try std.testing.expect(count < 500);
541
- }
542
-
543
- test "OptimizedBuffer - many unique graphemes with small pool" {
544
- const tiny_slots = [_]u32{ 4, 4, 4, 4, 4 };
545
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
546
- .slots_per_page = tiny_slots,
547
- });
548
- defer local_pool.deinit();
549
-
550
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
551
- defer tb.deinit();
552
-
553
- var view = try TextBufferView.init(std.testing.allocator, tb);
554
- defer view.deinit();
555
-
556
- var buf = try OptimizedBuffer.init(
557
- std.testing.allocator,
558
- 80,
559
- 25,
560
- .{ .pool = &local_pool, .id = "test-buffer" },
561
- );
562
- defer buf.deinit();
563
-
564
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
565
-
566
- var render_count: u32 = 0;
567
- var failure_count: u32 = 0;
568
-
569
- while (render_count < 1000) : (render_count += 1) {
570
- var text_builder: std.ArrayListUnmanaged(u8) = .{};
571
- defer text_builder.deinit(std.testing.allocator);
572
-
573
- const base_codepoint: u21 = 0x2600 + @as(u21, @intCast(render_count % 500));
574
- const char_bytes = [_]u8{
575
- @intCast(0xE0 | (base_codepoint >> 12)),
576
- @intCast(0x80 | ((base_codepoint >> 6) & 0x3F)),
577
- @intCast(0x80 | (base_codepoint & 0x3F)),
578
- };
579
- try text_builder.appendSlice(std.testing.allocator, &char_bytes);
580
- try text_builder.appendSlice(std.testing.allocator, " ");
581
- try text_builder.appendSlice(std.testing.allocator, &char_bytes);
582
-
583
- tb.setText(text_builder.items) catch {
584
- failure_count += 1;
585
- continue;
586
- };
587
-
588
- if (render_count % 50 == 0) {
589
- try buf.clear(bg, null);
590
- tb.reset();
591
- }
592
-
593
- buf.drawTextBuffer(view, 0, 0) catch {
594
- failure_count += 1;
595
- continue;
596
- };
597
- }
598
-
599
- try std.testing.expect(failure_count == 0);
600
- }
601
-
602
- test "OptimizedBuffer - continuous rendering without buffer recreation" {
603
- const pool = gp.initGlobalPool(std.testing.allocator);
604
- defer gp.deinitGlobalPool();
605
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
606
- defer local_link_pool.deinit();
607
-
608
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
609
- defer tb.deinit();
610
-
611
- var view = try TextBufferView.init(std.testing.allocator, tb);
612
- defer view.deinit();
613
-
614
- try tb.setText("• Hello World •\n• Test Line •\n• Another Line •");
615
-
616
- var buf = try OptimizedBuffer.init(
617
- std.testing.allocator,
618
- 80,
619
- 25,
620
- .{ .pool = pool, .id = "test-buffer" },
621
- );
622
- defer buf.deinit();
623
-
624
- var i: u32 = 0;
625
- while (i < 50000) : (i += 1) {
626
- try buf.drawTextBuffer(view, 0, 0);
627
- }
628
- }
629
-
630
- test "OptimizedBuffer - multiple buffers rendering same TextBuffer" {
631
- const pool = gp.initGlobalPool(std.testing.allocator);
632
- defer gp.deinitGlobalPool();
633
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
634
- defer local_link_pool.deinit();
635
-
636
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
637
- defer tb.deinit();
638
-
639
- var view = try TextBufferView.init(std.testing.allocator, tb);
640
- defer view.deinit();
641
-
642
- try tb.setText("🌟 • 测试 • 🎨");
643
-
644
- var buf1 = try OptimizedBuffer.init(
645
- std.testing.allocator,
646
- 40,
647
- 10,
648
- .{ .pool = pool, .id = "buffer-1" },
649
- );
650
- defer buf1.deinit();
651
-
652
- var buf2 = try OptimizedBuffer.init(
653
- std.testing.allocator,
654
- 40,
655
- 10,
656
- .{ .pool = pool, .id = "buffer-2" },
657
- );
658
- defer buf2.deinit();
659
-
660
- var buf3 = try OptimizedBuffer.init(
661
- std.testing.allocator,
662
- 40,
663
- 10,
664
- .{ .pool = pool, .id = "buffer-3" },
665
- );
666
- defer buf3.deinit();
667
-
668
- var i: u32 = 0;
669
- while (i < 5000) : (i += 1) {
670
- try buf1.drawTextBuffer(view, 0, 0);
671
- try buf2.drawTextBuffer(view, 0, 0);
672
- try buf3.drawTextBuffer(view, 0, 0);
673
- }
674
- }
675
-
676
- test "OptimizedBuffer - continuous render without clear with small pool" {
677
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
678
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
679
- .slots_per_page = tiny_slots,
680
- });
681
- defer local_pool.deinit();
682
-
683
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
684
- defer tb.deinit();
685
-
686
- var view = try TextBufferView.init(std.testing.allocator, tb);
687
- defer view.deinit();
688
-
689
- try tb.setText("• Test •");
690
-
691
- var buf = try OptimizedBuffer.init(
692
- std.testing.allocator,
693
- 80,
694
- 25,
695
- .{ .pool = &local_pool, .id = "test-buffer" },
696
- );
697
- defer buf.deinit();
698
-
699
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
700
- try buf.clear(bg, null);
701
-
702
- var i: u32 = 0;
703
- while (i < 100) : (i += 1) {
704
- try buf.drawTextBuffer(view, 0, 0);
705
- }
706
- }
707
-
708
- test "OptimizedBuffer - graphemes with scissor clipping and small pool" {
709
- const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
710
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
711
- .slots_per_page = tiny_slots,
712
- });
713
- defer local_pool.deinit();
714
-
715
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
716
- defer tb.deinit();
717
-
718
- var view = try TextBufferView.init(std.testing.allocator, tb);
719
- defer view.deinit();
720
-
721
- try tb.setText("• • • • •");
722
-
723
- var buf = try OptimizedBuffer.init(
724
- std.testing.allocator,
725
- 80,
726
- 25,
727
- .{ .pool = &local_pool, .id = "test-buffer" },
728
- );
729
- defer buf.deinit();
730
-
731
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
732
- try buf.clear(bg, null);
733
-
734
- try buf.pushScissorRect(0, 0, 5, 5);
735
-
736
- var i: u32 = 0;
737
- while (i < 100) : (i += 1) {
738
- try buf.drawTextBuffer(view, 20, 20);
739
- }
740
- }
741
-
742
- test "OptimizedBuffer - drawText with alpha blending and scissor" {
743
- const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
744
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
745
- .slots_per_page = tiny_slots,
746
- });
747
- defer local_pool.deinit();
748
-
749
- var buf = try OptimizedBuffer.init(
750
- std.testing.allocator,
751
- 80,
752
- 25,
753
- .{ .pool = &local_pool, .id = "test-buffer" },
754
- );
755
- defer buf.deinit();
756
-
757
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
758
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
759
- const bg_alpha = RGBA{ 0.0, 0.0, 0.0, 0.5 };
760
-
761
- try buf.clear(bg, null);
762
-
763
- try buf.pushScissorRect(0, 0, 10, 10);
764
-
765
- var i: u32 = 0;
766
- while (i < 200) : (i += 1) {
767
- try buf.drawText("• • • •", 50, 0, fg, bg_alpha, 0);
768
- }
769
- }
770
-
771
- test "OptimizedBuffer - many unique graphemes with alpha and small pool" {
772
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
773
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
774
- .slots_per_page = tiny_slots,
775
- });
776
- defer local_pool.deinit();
777
-
778
- var buf = try OptimizedBuffer.init(
779
- std.testing.allocator,
780
- 80,
781
- 25,
782
- .{ .pool = &local_pool, .id = "test-buffer" },
783
- );
784
- defer buf.deinit();
785
-
786
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
787
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
788
- const bg_alpha = RGBA{ 0.0, 0.0, 0.0, 0.5 };
789
-
790
- try buf.clear(bg, null);
791
-
792
- var i: u32 = 0;
793
- while (i < 50) : (i += 1) {
794
- const base_codepoint: u21 = 0x2600 + @as(u21, @intCast(i));
795
- const char_bytes = [_]u8{
796
- @intCast(0xE0 | (base_codepoint >> 12)),
797
- @intCast(0x80 | ((base_codepoint >> 6) & 0x3F)),
798
- @intCast(0x80 | (base_codepoint & 0x3F)),
799
- };
800
-
801
- var text: [4]u8 = undefined;
802
- @memcpy(text[0..3], &char_bytes);
803
- text[3] = ' ';
804
-
805
- try buf.drawText(&text, @intCast(i % 70), @intCast(i / 70), fg, bg_alpha, 0);
806
- }
807
- }
808
-
809
- test "OptimizedBuffer - fill buffer with many unique graphemes" {
810
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
811
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
812
- .slots_per_page = tiny_slots,
813
- });
814
- defer local_pool.deinit();
815
-
816
- var buf = try OptimizedBuffer.init(
817
- std.testing.allocator,
818
- 40,
819
- 20,
820
- .{ .pool = &local_pool, .id = "test-buffer" },
821
- );
822
- defer buf.deinit();
823
-
824
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
825
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
826
-
827
- try buf.clear(bg, null);
828
-
829
- var char_idx: u32 = 0;
830
- var y: u32 = 0;
831
- while (y < 15) : (y += 1) {
832
- var x: u32 = 0;
833
- while (x < 35) : (x += 2) {
834
- const base_codepoint: u21 = 0x2600 + @as(u21, @intCast(char_idx % 200));
835
- const char_bytes = [_]u8{
836
- @intCast(0xE0 | (base_codepoint >> 12)),
837
- @intCast(0x80 | ((base_codepoint >> 6) & 0x3F)),
838
- @intCast(0x80 | (base_codepoint & 0x3F)),
839
- };
840
-
841
- try buf.drawText(&char_bytes, x, y, fg, bg, 0);
842
-
843
- char_idx += 1;
844
- }
845
- }
846
- }
847
-
848
- test "OptimizedBuffer - verify pool growth works correctly" {
849
- const one_slot = [_]u32{ 1, 1, 1, 1, 1 };
850
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
851
- .slots_per_page = one_slot,
852
- });
853
- defer local_pool.deinit();
854
-
855
- var buf = try OptimizedBuffer.init(
856
- std.testing.allocator,
857
- 80,
858
- 25,
859
- .{ .pool = &local_pool, .id = "test-buffer" },
860
- );
861
- defer buf.deinit();
862
-
863
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
864
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
865
-
866
- try buf.clear(bg, null);
867
-
868
- var char_idx: u32 = 0;
869
- while (char_idx < 150) : (char_idx += 1) {
870
- const base_codepoint: u21 = 0x2600 + @as(u21, @intCast(char_idx));
871
- const char_bytes = [_]u8{
872
- @intCast(0xE0 | (base_codepoint >> 12)),
873
- @intCast(0x80 | ((base_codepoint >> 6) & 0x3F)),
874
- @intCast(0x80 | (base_codepoint & 0x3F)),
875
- };
876
-
877
- const x = @as(u32, @intCast((char_idx * 2) % 70));
878
- const y = @as(u32, @intCast((char_idx * 2) / 70));
879
-
880
- try buf.drawText(&char_bytes, x, y, fg, bg, 0);
881
- }
882
- }
883
-
884
- test "OptimizedBuffer - repeated overwriting of same grapheme" {
885
- const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
886
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
887
- .slots_per_page = tiny_slots,
888
- });
889
- defer local_pool.deinit();
890
-
891
- var buf = try OptimizedBuffer.init(
892
- std.testing.allocator,
893
- 10,
894
- 5,
895
- .{ .pool = &local_pool, .id = "test-buffer" },
896
- );
897
- defer buf.deinit();
898
-
899
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
900
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
901
-
902
- try buf.drawText("•", 0, 0, fg, bg, 0);
903
-
904
- var i: u32 = 0;
905
- while (i < 500) : (i += 1) {
906
- try buf.drawText("•", 0, 0, fg, bg, 0);
907
- }
908
-
909
- try std.testing.expect(buf.grapheme_tracker.getGraphemeCount() <= 2);
910
- }
911
-
912
- test "OptimizedBuffer - two-buffer pattern should not leak" {
913
- const tiny_slots = [_]u32{ 4, 4, 4, 4, 4 };
914
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
915
- .slots_per_page = tiny_slots,
916
- });
917
- defer local_pool.deinit();
918
-
919
- var nextBuffer = try OptimizedBuffer.init(
920
- std.testing.allocator,
921
- 10,
922
- 5,
923
- .{ .pool = &local_pool, .id = "next-buffer" },
924
- );
925
- defer nextBuffer.deinit();
926
-
927
- var currentBuffer = try OptimizedBuffer.init(
928
- std.testing.allocator,
929
- 10,
930
- 5,
931
- .{ .pool = &local_pool, .id = "current-buffer" },
932
- );
933
- defer currentBuffer.deinit();
934
-
935
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
936
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
937
-
938
- var frame: u32 = 0;
939
- while (frame < 100) : (frame += 1) {
940
- try nextBuffer.drawText("• Test •", 0, 0, fg, bg, 0);
941
-
942
- const cell = nextBuffer.get(0, 0).?;
943
- currentBuffer.setRaw(0, 0, cell);
944
-
945
- try nextBuffer.clear(bg, null);
946
- }
947
- }
948
-
949
- test "OptimizedBuffer - set and clear cycle should not leak" {
950
- const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
951
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
952
- .slots_per_page = tiny_slots,
953
- });
954
- defer local_pool.deinit();
955
-
956
- var buf = try OptimizedBuffer.init(
957
- std.testing.allocator,
958
- 10,
959
- 5,
960
- .{ .pool = &local_pool, .id = "test-buffer" },
961
- );
962
- defer buf.deinit();
963
-
964
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
965
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
966
-
967
- var frame: u32 = 0;
968
- while (frame < 200) : (frame += 1) {
969
- try buf.drawText("•", 0, 0, fg, bg, 0);
970
- try buf.clear(bg, null);
971
- }
972
- }
973
-
974
- test "OptimizedBuffer - repeated drawTextBuffer without clear should not leak" {
975
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
976
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
977
- .slots_per_page = tiny_slots,
978
- });
979
- defer local_pool.deinit();
980
-
981
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
982
- defer tb.deinit();
983
-
984
- var view = try TextBufferView.init(std.testing.allocator, tb);
985
- defer view.deinit();
986
-
987
- try tb.setText("• Hello • World •");
988
-
989
- var buf = try OptimizedBuffer.init(
990
- std.testing.allocator,
991
- 80,
992
- 25,
993
- .{ .pool = &local_pool, .id = "render-buffer" },
994
- );
995
- defer buf.deinit();
996
-
997
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
998
- try buf.clear(bg, null);
999
-
1000
- var frame: u32 = 0;
1001
- while (frame < 500) : (frame += 1) {
1002
- try buf.drawTextBuffer(view, 0, 0);
1003
- }
1004
- }
1005
-
1006
- test "OptimizedBuffer - renderer two-buffer swap pattern should not leak" {
1007
- const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
1008
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1009
- .slots_per_page = tiny_slots,
1010
- });
1011
- defer local_pool.deinit();
1012
-
1013
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1014
- defer tb.deinit();
1015
-
1016
- var view = try TextBufferView.init(std.testing.allocator, tb);
1017
- defer view.deinit();
1018
-
1019
- try tb.setText("• • •");
1020
-
1021
- var current = try OptimizedBuffer.init(
1022
- std.testing.allocator,
1023
- 20,
1024
- 5,
1025
- .{ .pool = &local_pool, .id = "current" },
1026
- );
1027
- defer current.deinit();
1028
-
1029
- var next = try OptimizedBuffer.init(
1030
- std.testing.allocator,
1031
- 20,
1032
- 5,
1033
- .{ .pool = &local_pool, .id = "next" },
1034
- );
1035
- defer next.deinit();
1036
-
1037
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1038
- try current.clear(bg, null);
1039
-
1040
- var frame: u32 = 0;
1041
- while (frame < 300) : (frame += 1) {
1042
- try next.drawTextBuffer(view, 0, 0);
1043
-
1044
- var x: u32 = 0;
1045
- while (x < 10) : (x += 1) {
1046
- if (next.get(x, 0)) |cell| {
1047
- current.setRaw(x, 0, cell);
1048
- }
1049
- }
1050
-
1051
- try next.clear(bg, null);
1052
- }
1053
- }
1054
-
1055
- test "OptimizedBuffer - set should not clear newly written adjacent grapheme continuation" {
1056
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{});
1057
- defer local_pool.deinit();
1058
-
1059
- var buf = try OptimizedBuffer.init(
1060
- std.testing.allocator,
1061
- 8,
1062
- 1,
1063
- .{ .pool = &local_pool, .id = "set-adjacent-grapheme" },
1064
- );
1065
- defer buf.deinit();
1066
-
1067
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1068
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1069
- try buf.clear(bg, null);
1070
-
1071
- const old_gid = try local_pool.alloc("🌟");
1072
- const old_start = gp.packGraphemeStart(old_gid & gp.GRAPHEME_ID_MASK, 2);
1073
- buf.set(3, 0, .{ .char = old_start, .fg = fg, .bg = bg, .attributes = 0 });
1074
-
1075
- const new_gid = try local_pool.alloc("🔥");
1076
- const new_start = gp.packGraphemeStart(new_gid & gp.GRAPHEME_ID_MASK, 2);
1077
-
1078
- // Simulate renderer's left-to-right in-place update:
1079
- // - x=2 writes a new grapheme (which writes continuation at x=3)
1080
- // - x=3 would be skipped by char-equality
1081
- // - x=4 overwrites an old continuation from the previous frame
1082
- // The overwrite at x=4 must not clear the new continuation at x=3.
1083
- buf.set(2, 0, .{ .char = new_start, .fg = fg, .bg = bg, .attributes = 0 });
1084
- buf.set(4, 0, .{ .char = ' ', .fg = fg, .bg = bg, .attributes = 0 });
1085
-
1086
- const c2 = buf.get(2, 0).?;
1087
- const c3 = buf.get(3, 0).?;
1088
- const c4 = buf.get(4, 0).?;
1089
-
1090
- try std.testing.expect(gp.isGraphemeChar(c2.char));
1091
- try std.testing.expect(gp.graphemeIdFromChar(c2.char) == (new_gid & gp.GRAPHEME_ID_MASK));
1092
-
1093
- try std.testing.expect(gp.isContinuationChar(c3.char));
1094
- try std.testing.expect(gp.graphemeIdFromChar(c3.char) == (new_gid & gp.GRAPHEME_ID_MASK));
1095
-
1096
- try std.testing.expect(c4.char == ' ');
1097
- }
1098
-
1099
- test "OptimizedBuffer - set span cleanup keeps shared link refcounts consistent" {
1100
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{});
1101
- defer local_pool.deinit();
1102
-
1103
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1104
- defer local_link_pool.deinit();
1105
-
1106
- var buf = try OptimizedBuffer.init(
1107
- std.testing.allocator,
1108
- 10,
1109
- 1,
1110
- .{ .pool = &local_pool, .id = "set-span-link-refcount", .link_pool = &local_link_pool },
1111
- );
1112
- defer buf.deinit();
1113
-
1114
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1115
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1116
- try buf.clear(bg, null);
1117
-
1118
- const link_id = try local_link_pool.alloc("https://example.com");
1119
- const linked_attr = ansi.TextAttributes.setLinkId(0, link_id);
1120
-
1121
- const gid = try local_pool.alloc("你");
1122
- const start = gp.packGraphemeStart(gid & gp.GRAPHEME_ID_MASK, 2);
1123
-
1124
- // Create three linked cells total:
1125
- // - a 2-cell grapheme span at x=2..3
1126
- // - one additional linked cell at x=6
1127
- buf.set(2, 0, .{ .char = start, .fg = fg, .bg = bg, .attributes = linked_attr });
1128
- buf.set(6, 0, .{ .char = 'X', .fg = fg, .bg = bg, .attributes = linked_attr });
1129
-
1130
- try std.testing.expectEqual(@as(u32, 3), buf.link_tracker.used_ids.get(link_id).?);
1131
- try std.testing.expectEqual(@as(u32, 1), try local_link_pool.getRefcount(link_id));
1132
-
1133
- // Overwrite the continuation cell at x=3 with a non-grapheme char.
1134
- // set() will run span cleanup and clear x=2..3. The independent linked
1135
- // cell at x=6 must remain tracked.
1136
- buf.set(3, 0, .{ .char = ' ', .fg = fg, .bg = bg, .attributes = 0 });
1137
-
1138
- try std.testing.expectEqual(@as(u32, 1), buf.link_tracker.getLinkCount());
1139
- try std.testing.expectEqual(@as(u32, 1), buf.link_tracker.used_ids.get(link_id).?);
1140
- try std.testing.expectEqual(@as(u32, 1), try local_link_pool.getRefcount(link_id));
1141
- }
1142
-
1143
- test "OptimizedBuffer - syncCell updates grapheme tracker for start transitions" {
1144
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{});
1145
- defer local_pool.deinit();
1146
-
1147
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1148
- defer local_link_pool.deinit();
1149
-
1150
- var buf = try OptimizedBuffer.init(
1151
- std.testing.allocator,
1152
- 10,
1153
- 1,
1154
- .{ .pool = &local_pool, .id = "sync-cell-grapheme-tracker", .link_pool = &local_link_pool },
1155
- );
1156
- defer buf.deinit();
1157
-
1158
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1159
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1160
- try buf.clear(bg, null);
1161
-
1162
- const gid_old = try local_pool.alloc("你");
1163
- const gid_new = try local_pool.alloc("好");
1164
- const old_id = gid_old & gp.GRAPHEME_ID_MASK;
1165
- const new_id = gid_new & gp.GRAPHEME_ID_MASK;
1166
- const start_old = gp.packGraphemeStart(old_id, 2);
1167
- const start_new = gp.packGraphemeStart(new_id, 2);
1168
-
1169
- buf.syncCell(1, 0, .{ .char = start_old, .fg = fg, .bg = bg, .attributes = 0 });
1170
- try std.testing.expectEqual(@as(u32, 1), buf.grapheme_tracker.getGraphemeCount());
1171
- try std.testing.expect(buf.grapheme_tracker.contains(old_id));
1172
-
1173
- buf.syncCell(1, 0, .{ .char = start_new, .fg = fg, .bg = bg, .attributes = 0 });
1174
- try std.testing.expectEqual(@as(u32, 1), buf.grapheme_tracker.getGraphemeCount());
1175
- try std.testing.expect(!buf.grapheme_tracker.contains(old_id));
1176
- try std.testing.expect(buf.grapheme_tracker.contains(new_id));
1177
-
1178
- buf.syncCell(1, 0, .{ .char = ' ', .fg = fg, .bg = bg, .attributes = 0 });
1179
- try std.testing.expectEqual(@as(u32, 0), buf.grapheme_tracker.getGraphemeCount());
1180
- try std.testing.expect(!buf.grapheme_tracker.contains(new_id));
1181
- }
1182
-
1183
- test "OptimizedBuffer - sustained rendering should not leak" {
1184
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
1185
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1186
- .slots_per_page = tiny_slots,
1187
- });
1188
- defer local_pool.deinit();
1189
-
1190
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1191
- defer tb.deinit();
1192
-
1193
- var view = try TextBufferView.init(std.testing.allocator, tb);
1194
- defer view.deinit();
1195
-
1196
- try tb.setText(" • Type any text to insert\n • Arrow keys to move cursor\n • Backspace/Delete to remove text");
1197
-
1198
- var buf = try OptimizedBuffer.init(
1199
- std.testing.allocator,
1200
- 80,
1201
- 25,
1202
- .{ .pool = &local_pool, .id = "render-buffer" },
1203
- );
1204
- defer buf.deinit();
1205
-
1206
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1207
- try buf.clear(bg, null);
1208
-
1209
- var frame: u32 = 0;
1210
- while (frame < 3000) : (frame += 1) {
1211
- try buf.drawTextBuffer(view, 0, 0);
1212
- }
1213
- }
1214
-
1215
- test "OptimizedBuffer - rendering with changing content should not leak" {
1216
- const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
1217
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1218
- .slots_per_page = tiny_slots,
1219
- });
1220
- defer local_pool.deinit();
1221
-
1222
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1223
- defer tb.deinit();
1224
-
1225
- var view = try TextBufferView.init(std.testing.allocator, tb);
1226
- defer view.deinit();
1227
-
1228
- var buf = try OptimizedBuffer.init(
1229
- std.testing.allocator,
1230
- 80,
1231
- 25,
1232
- .{ .pool = &local_pool, .id = "render-buffer" },
1233
- );
1234
- defer buf.deinit();
1235
-
1236
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1237
- try buf.clear(bg, null);
1238
-
1239
- var frame: u32 = 0;
1240
- while (frame < 100) : (frame += 1) {
1241
- const char_idx = frame % 10;
1242
- const base_codepoint: u21 = 0x2600 + @as(u21, @intCast(char_idx));
1243
- const char_bytes = [_]u8{
1244
- @intCast(0xE0 | (base_codepoint >> 12)),
1245
- @intCast(0x80 | ((base_codepoint >> 6) & 0x3F)),
1246
- @intCast(0x80 | (base_codepoint & 0x3F)),
1247
- };
1248
-
1249
- var text: [11]u8 = undefined;
1250
- @memcpy(text[0..3], &char_bytes);
1251
- text[3] = ' ';
1252
- @memcpy(text[4..7], &char_bytes);
1253
- text[7] = ' ';
1254
- @memcpy(text[8..11], &char_bytes);
1255
-
1256
- tb.setText(&text) catch continue;
1257
-
1258
- try buf.drawTextBuffer(view, 0, 0);
1259
- }
1260
- }
1261
-
1262
- test "OptimizedBuffer - multiple TextBuffers rendering simultaneously should not leak" {
1263
- const one_slot = [_]u32{ 1, 1, 1, 1, 1 };
1264
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1265
- .slots_per_page = one_slot,
1266
- });
1267
- defer local_pool.deinit();
1268
-
1269
- var tb1 = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1270
- defer tb1.deinit();
1271
- var view1 = try TextBufferView.init(std.testing.allocator, tb1);
1272
- defer view1.deinit();
1273
-
1274
- var tb2 = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1275
- defer tb2.deinit();
1276
- var view2 = try TextBufferView.init(std.testing.allocator, tb2);
1277
- defer view2.deinit();
1278
-
1279
- var tb3 = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1280
- defer tb3.deinit();
1281
- var view3 = try TextBufferView.init(std.testing.allocator, tb3);
1282
- defer view3.deinit();
1283
-
1284
- try tb1.setText("• First •");
1285
- try tb2.setText("• Second •");
1286
- try tb3.setText("• Third •");
1287
-
1288
- var buf = try OptimizedBuffer.init(
1289
- std.testing.allocator,
1290
- 80,
1291
- 30,
1292
- .{ .pool = &local_pool, .id = "main-buffer" },
1293
- );
1294
- defer buf.deinit();
1295
-
1296
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1297
- try buf.clear(bg, null);
1298
-
1299
- var frame: u32 = 0;
1300
- while (frame < 500) : (frame += 1) {
1301
- try buf.drawTextBuffer(view1, 0, 0);
1302
- try buf.drawTextBuffer(view2, 0, 10);
1303
- try buf.drawTextBuffer(view3, 0, 20);
1304
- }
1305
- }
1306
-
1307
- test "OptimizedBuffer - grapheme refcount management" {
1308
- const two_slots = [_]u32{ 2, 2, 2, 2, 2 };
1309
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1310
- .slots_per_page = two_slots,
1311
- });
1312
- defer local_pool.deinit();
1313
-
1314
- var buf = try OptimizedBuffer.init(
1315
- std.testing.allocator,
1316
- 5,
1317
- 1,
1318
- .{ .pool = &local_pool, .id = "test-buffer" },
1319
- );
1320
- defer buf.deinit();
1321
-
1322
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1323
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1324
-
1325
- try buf.drawText("•", 0, 0, fg, bg, 0);
1326
- const initial_cell = buf.get(0, 0).?;
1327
- const initial_id = gp.graphemeIdFromChar(initial_cell.char);
1328
- const initial_refcount = local_pool.getRefcount(initial_id) catch 0;
1329
-
1330
- try std.testing.expectEqual(@as(u32, 1), initial_refcount);
1331
-
1332
- var i: u32 = 0;
1333
- while (i < 100) : (i += 1) {
1334
- try buf.drawText("•", 0, 0, fg, bg, 0);
1335
-
1336
- const cell = buf.get(0, 0).?;
1337
- const id = gp.graphemeIdFromChar(cell.char);
1338
- const rc = local_pool.getRefcount(id) catch 999;
1339
- const slot = id & 0xFFFF;
1340
-
1341
- try std.testing.expectEqual(@as(u32, 1), rc);
1342
- try std.testing.expect(slot == 0 or slot == 1);
1343
- }
1344
- }
1345
-
1346
- test "OptimizedBuffer - drawTextBuffer with graphemes then clear removes all pool references" {
1347
- const small_slots = [_]u32{ 4, 4, 4, 4, 4 };
1348
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1349
- .slots_per_page = small_slots,
1350
- });
1351
- defer local_pool.deinit();
1352
-
1353
- var tb = try TextBuffer.init(std.testing.allocator, &local_pool, link.initGlobalLinkPool(std.testing.allocator), .wcwidth);
1354
- defer tb.deinit();
1355
-
1356
- var view = try TextBufferView.init(std.testing.allocator, tb);
1357
- defer view.deinit();
1358
-
1359
- try tb.setText("• Test • 🌟 • 🎨 •");
1360
-
1361
- var buf = try OptimizedBuffer.init(
1362
- std.testing.allocator,
1363
- 80,
1364
- 25,
1365
- .{ .pool = &local_pool, .id = "test-buffer" },
1366
- );
1367
- defer buf.deinit();
1368
-
1369
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1370
-
1371
- try buf.drawTextBuffer(view, 0, 0);
1372
-
1373
- const count_after_draw = buf.grapheme_tracker.getGraphemeCount();
1374
- try std.testing.expect(count_after_draw > 0);
1375
-
1376
- var total_allocated_slots: u32 = 0;
1377
- var total_free_slots: u32 = 0;
1378
- for (local_pool.classes) |class| {
1379
- total_allocated_slots += class.num_slots;
1380
- total_free_slots += @intCast(class.free_list.items.len);
1381
- }
1382
- const slots_in_use_after_draw = total_allocated_slots - total_free_slots;
1383
- try std.testing.expect(slots_in_use_after_draw > 0);
1384
-
1385
- try buf.clear(bg, null);
1386
-
1387
- const count_after_clear = buf.grapheme_tracker.getGraphemeCount();
1388
- try std.testing.expectEqual(@as(u32, 0), count_after_clear);
1389
-
1390
- var total_allocated_after_clear: u32 = 0;
1391
- var total_free_after_clear: u32 = 0;
1392
- for (local_pool.classes) |class| {
1393
- total_allocated_after_clear += class.num_slots;
1394
- total_free_after_clear += @intCast(class.free_list.items.len);
1395
- }
1396
- try std.testing.expectEqual(total_allocated_after_clear, total_free_after_clear);
1397
-
1398
- var y: u32 = 0;
1399
- while (y < 5) : (y += 1) {
1400
- var x: u32 = 0;
1401
- while (x < 20) : (x += 1) {
1402
- const cell = buf.get(x, y).?;
1403
- try std.testing.expectEqual(@as(u32, 32), cell.char);
1404
- try std.testing.expect(!gp.isGraphemeChar(cell.char));
1405
- try std.testing.expect(!gp.isContinuationChar(cell.char));
1406
- }
1407
- }
1408
-
1409
- try buf.drawTextBuffer(view, 0, 0);
1410
- const count_after_redraw = buf.grapheme_tracker.getGraphemeCount();
1411
- try std.testing.expect(count_after_redraw > 0);
1412
-
1413
- var allocated_after_redraw: u32 = 0;
1414
- var free_after_redraw: u32 = 0;
1415
- for (local_pool.classes) |class| {
1416
- allocated_after_redraw += class.num_slots;
1417
- free_after_redraw += @intCast(class.free_list.items.len);
1418
- }
1419
- const slots_in_use_after_redraw = allocated_after_redraw - free_after_redraw;
1420
- try std.testing.expect(slots_in_use_after_redraw > 0);
1421
-
1422
- try buf.clear(bg, null);
1423
- const count_after_second_clear = buf.grapheme_tracker.getGraphemeCount();
1424
- try std.testing.expectEqual(@as(u32, 0), count_after_second_clear);
1425
-
1426
- var allocated_after_second_clear: u32 = 0;
1427
- var free_after_second_clear: u32 = 0;
1428
- for (local_pool.classes) |class| {
1429
- allocated_after_second_clear += class.num_slots;
1430
- free_after_second_clear += @intCast(class.free_list.items.len);
1431
- }
1432
- try std.testing.expectEqual(allocated_after_second_clear, free_after_second_clear);
1433
- }
1434
-
1435
- test "OptimizedBuffer - drawTextBuffer with negative y coordinate should not panic" {
1436
- const pool = gp.initGlobalPool(std.testing.allocator);
1437
- defer gp.deinitGlobalPool();
1438
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1439
- defer local_link_pool.deinit();
1440
-
1441
- var tb = try TextBuffer.init(std.testing.allocator, pool, &local_link_pool, .wcwidth);
1442
- defer tb.deinit();
1443
-
1444
- var view = try TextBufferView.init(std.testing.allocator, tb);
1445
- defer view.deinit();
1446
-
1447
- try tb.setText("Line 1\nLine 2\nLine 3\nLine 4\nLine 5");
1448
-
1449
- var buf = try OptimizedBuffer.init(
1450
- std.testing.allocator,
1451
- 80,
1452
- 25,
1453
- .{ .pool = pool, .id = "test-buffer" },
1454
- );
1455
- defer buf.deinit();
1456
-
1457
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1458
- try buf.clear(bg, null);
1459
-
1460
- // Draw text buffer at negative y coordinate (-2)
1461
- // This simulates a scenario where content is scrolled partially off-screen
1462
- // The first 2 lines should be clipped, and lines 3, 4, 5 should be visible
1463
- try buf.drawTextBuffer(view, 0, -2);
1464
-
1465
- // Verify that content is properly clipped when drawn at negative y
1466
- // Lines that are off-screen (negative y) should be skipped
1467
- // Line 3 should appear at y=0, Line 4 at y=1, Line 5 at y=2
1468
-
1469
- // Check that Line 3 is rendered at y=0
1470
- const cell_y0 = buf.get(0, 0).?;
1471
- try std.testing.expectEqual(@as(u32, 'L'), cell_y0.char);
1472
-
1473
- // Check that Line 4 is rendered at y=1
1474
- const cell_y1 = buf.get(0, 1).?;
1475
- try std.testing.expectEqual(@as(u32, 'L'), cell_y1.char);
1476
-
1477
- // Check that Line 5 is rendered at y=2
1478
- const cell_y2 = buf.get(0, 2).?;
1479
- try std.testing.expectEqual(@as(u32, 'L'), cell_y2.char);
1480
-
1481
- // Verify the full content of the first visible line (Line 3)
1482
- try std.testing.expectEqual(@as(u32, 'L'), buf.get(0, 0).?.char);
1483
- try std.testing.expectEqual(@as(u32, 'i'), buf.get(1, 0).?.char);
1484
- try std.testing.expectEqual(@as(u32, 'n'), buf.get(2, 0).?.char);
1485
- try std.testing.expectEqual(@as(u32, 'e'), buf.get(3, 0).?.char);
1486
- try std.testing.expectEqual(@as(u32, ' '), buf.get(4, 0).?.char);
1487
- try std.testing.expectEqual(@as(u32, '3'), buf.get(5, 0).?.char);
1488
- }
1489
-
1490
- test "OptimizedBuffer - cells are initialized after resize grow" {
1491
- const pool = gp.initGlobalPool(std.testing.allocator);
1492
- defer gp.deinitGlobalPool();
1493
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1494
- defer local_link_pool.deinit();
1495
-
1496
- var buf = try OptimizedBuffer.init(
1497
- std.testing.allocator,
1498
- 10,
1499
- 10,
1500
- .{ .pool = pool, .id = "test-buffer" },
1501
- );
1502
- defer buf.deinit();
1503
-
1504
- try buf.resize(20, 20);
1505
-
1506
- // Verify new cells have default values (space = 32), not garbage
1507
- const cell = buf.get(15, 15);
1508
- try std.testing.expect(cell != null);
1509
- try std.testing.expectEqual(@as(u32, 32), cell.?.char);
1510
- }
1511
-
1512
- test "OptimizedBuffer - link encoding round-trip" {
1513
- const pool = gp.initGlobalPool(std.testing.allocator);
1514
- defer gp.deinitGlobalPool();
1515
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1516
- defer local_link_pool.deinit();
1517
-
1518
- var buf = try OptimizedBuffer.init(
1519
- std.testing.allocator,
1520
- 20,
1521
- 5,
1522
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1523
- );
1524
- defer buf.deinit();
1525
-
1526
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1527
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1528
- try buf.clear(bg, null);
1529
-
1530
- // Allocate a link
1531
- const link_id = try local_link_pool.alloc("https://example.com");
1532
- const attributes = ansi.TextAttributes.setLinkId(ansi.TextAttributes.BOLD, link_id);
1533
-
1534
- // Draw text with link
1535
- try buf.drawText("Click", 0, 0, fg, bg, attributes);
1536
-
1537
- // Verify cell has correct char and attributes
1538
- const cell = buf.get(0, 0).?;
1539
- try std.testing.expectEqual(@as(u32, 'C'), cell.char);
1540
- try std.testing.expectEqual(ansi.TextAttributes.BOLD, ansi.TextAttributes.getBaseAttributes(cell.attributes));
1541
- try std.testing.expectEqual(link_id, ansi.TextAttributes.getLinkId(cell.attributes));
1542
-
1543
- // Verify link tracker has the link
1544
- try std.testing.expect(buf.link_tracker.hasAny());
1545
- try std.testing.expectEqual(@as(u32, 1), buf.link_tracker.getLinkCount());
1546
- }
1547
-
1548
- test "OptimizedBuffer - link tracker per-cell counting" {
1549
- const pool = gp.initGlobalPool(std.testing.allocator);
1550
- defer gp.deinitGlobalPool();
1551
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1552
- defer local_link_pool.deinit();
1553
-
1554
- var buf = try OptimizedBuffer.init(
1555
- std.testing.allocator,
1556
- 20,
1557
- 5,
1558
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1559
- );
1560
- defer buf.deinit();
1561
-
1562
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1563
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1564
- try buf.clear(bg, null);
1565
-
1566
- // Allocate a link
1567
- const link_id = try local_link_pool.alloc("https://example.com");
1568
- const attributes = ansi.TextAttributes.setLinkId(0, link_id);
1569
-
1570
- // Draw text covering 3 cells
1571
- try buf.drawText("ABC", 0, 0, fg, bg, attributes);
1572
-
1573
- // Verify link tracker has 1 unique link
1574
- // Pool refcount is 1 (tracker owns one ref, tracks 3 cells internally)
1575
- try std.testing.expectEqual(@as(u32, 1), buf.link_tracker.getLinkCount());
1576
- const pool_refcount = try local_link_pool.getRefcount(link_id);
1577
- try std.testing.expectEqual(@as(u32, 1), pool_refcount);
1578
-
1579
- // Verify tracker knows about 3 cells
1580
- const cell_count = buf.link_tracker.used_ids.get(link_id).?;
1581
- try std.testing.expectEqual(@as(u32, 3), cell_count);
1582
-
1583
- // Overwrite one cell without link
1584
- try buf.drawText("X", 0, 0, fg, bg, 0);
1585
-
1586
- // Tracker cell count should drop to 2, pool refcount stays 1
1587
- const cell_count2 = buf.link_tracker.used_ids.get(link_id).?;
1588
- try std.testing.expectEqual(@as(u32, 2), cell_count2);
1589
- const pool_refcount2 = try local_link_pool.getRefcount(link_id);
1590
- try std.testing.expectEqual(@as(u32, 1), pool_refcount2);
1591
-
1592
- // Clear all - refcount should be 0 and link freed
1593
- try buf.clear(bg, null);
1594
- try std.testing.expectEqual(@as(u32, 0), buf.link_tracker.getLinkCount());
1595
- }
1596
-
1597
- test "OptimizedBuffer - fillRect removes links" {
1598
- const pool = gp.initGlobalPool(std.testing.allocator);
1599
- defer gp.deinitGlobalPool();
1600
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1601
- defer local_link_pool.deinit();
1602
-
1603
- var buf = try OptimizedBuffer.init(
1604
- std.testing.allocator,
1605
- 20,
1606
- 5,
1607
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1608
- );
1609
- defer buf.deinit();
1610
-
1611
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1612
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1613
- try buf.clear(bg, null);
1614
-
1615
- // Allocate a link
1616
- const link_id = try local_link_pool.alloc("https://example.com");
1617
- const attributes = ansi.TextAttributes.setLinkId(0, link_id);
1618
-
1619
- // Draw linked text
1620
- try buf.drawText("Linked", 0, 0, fg, bg, attributes);
1621
- try buf.drawText("Text", 10, 0, fg, bg, attributes);
1622
-
1623
- // Verify links exist
1624
- try std.testing.expect(ansi.TextAttributes.hasLink(buf.get(0, 0).?.attributes));
1625
- try std.testing.expect(ansi.TextAttributes.hasLink(buf.get(10, 0).?.attributes));
1626
-
1627
- // Fill rect over first link
1628
- try buf.fillRect(0, 0, 6, 1, bg);
1629
-
1630
- // Cells in rect should have no link
1631
- try std.testing.expect(!ansi.TextAttributes.hasLink(buf.get(0, 0).?.attributes));
1632
- try std.testing.expect(!ansi.TextAttributes.hasLink(buf.get(5, 0).?.attributes));
1633
-
1634
- // Cells outside rect should preserve link
1635
- try std.testing.expect(ansi.TextAttributes.hasLink(buf.get(10, 0).?.attributes));
1636
- }
1637
-
1638
- test "OptimizedBuffer - link reuse after free" {
1639
- const pool = gp.initGlobalPool(std.testing.allocator);
1640
- defer gp.deinitGlobalPool();
1641
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1642
- defer local_link_pool.deinit();
1643
-
1644
- var buf = try OptimizedBuffer.init(
1645
- std.testing.allocator,
1646
- 20,
1647
- 5,
1648
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1649
- );
1650
- defer buf.deinit();
1651
-
1652
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1653
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1654
-
1655
- // Allocate first link
1656
- const link_id1 = try local_link_pool.alloc("https://first.com");
1657
- const attr1 = ansi.TextAttributes.setLinkId(0, link_id1);
1658
- try buf.drawText("A", 0, 0, fg, bg, attr1);
1659
-
1660
- // Clear - should free the link
1661
- try buf.clear(bg, null);
1662
-
1663
- // Allocate second link - should reuse same slot but different generation
1664
- const link_id2 = try local_link_pool.alloc("https://second.com");
1665
- try std.testing.expect(link_id1 != link_id2); // Different due to generation
1666
-
1667
- const attr2 = ansi.TextAttributes.setLinkId(0, link_id2);
1668
- try buf.drawText("B", 0, 0, fg, bg, attr2);
1669
-
1670
- const url = try local_link_pool.get(link_id2);
1671
- try std.testing.expect(std.mem.eql(u8, url, "https://second.com"));
1672
- }
1673
-
1674
- test "OptimizedBuffer - alpha blending preserves overlay link not dest link" {
1675
- const pool = gp.initGlobalPool(std.testing.allocator);
1676
- defer gp.deinitGlobalPool();
1677
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1678
- defer local_link_pool.deinit();
1679
-
1680
- var buf = try OptimizedBuffer.init(
1681
- std.testing.allocator,
1682
- 20,
1683
- 5,
1684
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1685
- );
1686
- defer buf.deinit();
1687
-
1688
- const bg_opaque = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1689
- const bg_alpha = RGBA{ 0.5, 0.5, 0.5, 0.5 };
1690
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1691
- try buf.clear(bg_opaque, null);
1692
-
1693
- // Draw underlying text with link A
1694
- const link_id_a = try local_link_pool.alloc("https://underlying.com");
1695
- const attr_a = ansi.TextAttributes.setLinkId(ansi.TextAttributes.BOLD, link_id_a);
1696
- try buf.drawText("X", 5, 0, fg, bg_opaque, attr_a);
1697
-
1698
- // Verify dest cell has link A
1699
- const dest_cell = buf.get(5, 0).?;
1700
- try std.testing.expectEqual(link_id_a, ansi.TextAttributes.getLinkId(dest_cell.attributes));
1701
- try std.testing.expectEqual(@as(u32, 'X'), dest_cell.char);
1702
-
1703
- // Draw space with alpha and link B over it (will preserve 'X' but blend colors)
1704
- const link_id_b = try local_link_pool.alloc("https://overlay.com");
1705
- const attr_b = ansi.TextAttributes.setLinkId(0, link_id_b);
1706
- try buf.drawText(" ", 5, 0, fg, bg_alpha, attr_b);
1707
-
1708
- // Result: char should be preserved 'X', but link should be from overlay (B), not dest (A)
1709
- const result_cell = buf.get(5, 0).?;
1710
- try std.testing.expectEqual(@as(u32, 'X'), result_cell.char);
1711
- try std.testing.expectEqual(link_id_b, ansi.TextAttributes.getLinkId(result_cell.attributes));
1712
- try std.testing.expect(ansi.TextAttributes.getLinkId(result_cell.attributes) != link_id_a);
1713
- }
1714
-
1715
- test "OptimizedBuffer - alpha blending with no link clears underlying link" {
1716
- const pool = gp.initGlobalPool(std.testing.allocator);
1717
- defer gp.deinitGlobalPool();
1718
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1719
- defer local_link_pool.deinit();
1720
-
1721
- var buf = try OptimizedBuffer.init(
1722
- std.testing.allocator,
1723
- 20,
1724
- 5,
1725
- .{ .pool = pool, .id = "test-buffer", .link_pool = &local_link_pool },
1726
- );
1727
- defer buf.deinit();
1728
-
1729
- const bg_opaque = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1730
- const bg_alpha = RGBA{ 0.5, 0.5, 0.5, 0.5 };
1731
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
1732
- try buf.clear(bg_opaque, null);
1733
-
1734
- // Draw underlying text with link
1735
- const link_id = try local_link_pool.alloc("https://underlying.com");
1736
- const attr_link = ansi.TextAttributes.setLinkId(ansi.TextAttributes.BOLD, link_id);
1737
- try buf.drawText("X", 5, 0, fg, bg_opaque, attr_link);
1738
-
1739
- // Verify dest cell has link
1740
- const dest_cell = buf.get(5, 0).?;
1741
- try std.testing.expectEqual(link_id, ansi.TextAttributes.getLinkId(dest_cell.attributes));
1742
-
1743
- // Draw space with alpha but NO link over it (will preserve 'X')
1744
- try buf.drawText(" ", 5, 0, fg, bg_alpha, 0);
1745
-
1746
- // Result: char 'X' preserved, but link should be CLEARED (0), not preserved
1747
- const result_cell = buf.get(5, 0).?;
1748
- try std.testing.expectEqual(@as(u32, 'X'), result_cell.char);
1749
- try std.testing.expectEqual(@as(u32, 0), ansi.TextAttributes.getLinkId(result_cell.attributes));
1750
-
1751
- // Link should no longer be tracked
1752
- try std.testing.expect(!ansi.TextAttributes.hasLink(result_cell.attributes));
1753
- }
1754
-
1755
- test "OptimizedBuffer - drawGrayscaleBuffer basic rendering" {
1756
- const pool = gp.initGlobalPool(std.testing.allocator);
1757
- defer gp.deinitGlobalPool();
1758
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1759
- defer local_link_pool.deinit();
1760
-
1761
- var buf = try OptimizedBuffer.init(
1762
- std.testing.allocator,
1763
- 10,
1764
- 5,
1765
- .{ .pool = pool, .id = "test-buffer" },
1766
- );
1767
- defer buf.deinit();
1768
-
1769
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1770
- try buf.clear(bg, null);
1771
-
1772
- // Create a 3x3 intensity buffer with varying values
1773
- const intensities = [_]f32{
1774
- 0.0, 0.5, 1.0,
1775
- 0.25, 0.75, 0.0,
1776
- 1.0, 0.0, 0.5,
1777
- };
1778
-
1779
- buf.drawGrayscaleBuffer(2, 1, &intensities, 3, 3, null, bg);
1780
-
1781
- const cell_0_0 = buf.get(2, 1).?;
1782
- try std.testing.expectEqual(@as(u32, 32), cell_0_0.char);
1783
-
1784
- const cell_1_0 = buf.get(3, 1).?;
1785
- try std.testing.expect(cell_1_0.char != 32);
1786
- try std.testing.expect(cell_1_0.fg[0] > 0.3);
1787
-
1788
- const cell_2_0 = buf.get(4, 1).?;
1789
- try std.testing.expect(cell_2_0.char != 32);
1790
- try std.testing.expect(cell_2_0.fg[0] > 0.9);
1791
- }
1792
-
1793
- test "OptimizedBuffer - drawGrayscaleBuffer negative position clipping" {
1794
- const pool = gp.initGlobalPool(std.testing.allocator);
1795
- defer gp.deinitGlobalPool();
1796
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1797
- defer local_link_pool.deinit();
1798
-
1799
- var buf = try OptimizedBuffer.init(
1800
- std.testing.allocator,
1801
- 10,
1802
- 5,
1803
- .{ .pool = pool, .id = "test-buffer" },
1804
- );
1805
- defer buf.deinit();
1806
-
1807
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1808
- try buf.clear(bg, null);
1809
-
1810
- // Create a 4x4 intensity buffer
1811
- const intensities = [_]f32{
1812
- 0.5, 0.5, 0.5, 0.5,
1813
- 0.5, 0.5, 0.5, 0.5,
1814
- 0.5, 0.5, 0.5, 0.5,
1815
- 0.5, 0.5, 0.5, 0.5,
1816
- };
1817
-
1818
- buf.drawGrayscaleBuffer(-1, -1, &intensities, 4, 4, null, bg);
1819
-
1820
- const cell_0_0 = buf.get(0, 0).?;
1821
- try std.testing.expect(cell_0_0.char != 32);
1822
-
1823
- const cell_2_0 = buf.get(2, 0).?;
1824
- try std.testing.expect(cell_2_0.char != 32);
1825
- }
1826
-
1827
- test "OptimizedBuffer - drawGrayscaleBuffer negative position fully clipped" {
1828
- const pool = gp.initGlobalPool(std.testing.allocator);
1829
- defer gp.deinitGlobalPool();
1830
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1831
- defer local_link_pool.deinit();
1832
-
1833
- var buf = try OptimizedBuffer.init(
1834
- std.testing.allocator,
1835
- 6,
1836
- 3,
1837
- .{ .pool = pool, .id = "test-buffer" },
1838
- );
1839
- defer buf.deinit();
1840
-
1841
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1842
- try buf.clear(bg, null);
1843
-
1844
- const intensities = [_]f32{
1845
- 1.0, 1.0, 1.0, 1.0,
1846
- 1.0, 1.0, 1.0, 1.0,
1847
- 1.0, 1.0, 1.0, 1.0,
1848
- 1.0, 1.0, 1.0, 1.0,
1849
- };
1850
-
1851
- buf.drawGrayscaleBuffer(-10, -10, &intensities, 4, 4, null, bg);
1852
-
1853
- const cell = buf.get(0, 0).?;
1854
- try std.testing.expectEqual(@as(u32, 32), cell.char);
1855
- }
1856
-
1857
- test "OptimizedBuffer - drawGrayscaleBuffer respects scissor rect" {
1858
- const pool = gp.initGlobalPool(std.testing.allocator);
1859
- defer gp.deinitGlobalPool();
1860
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1861
- defer local_link_pool.deinit();
1862
-
1863
- var buf = try OptimizedBuffer.init(
1864
- std.testing.allocator,
1865
- 10,
1866
- 5,
1867
- .{ .pool = pool, .id = "test-buffer" },
1868
- );
1869
- defer buf.deinit();
1870
-
1871
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1872
- try buf.clear(bg, null);
1873
-
1874
- try buf.pushScissorRect(0, 0, 2, 2);
1875
-
1876
- const intensities = [_]f32{
1877
- 1.0, 1.0, 1.0, 1.0,
1878
- 1.0, 1.0, 1.0, 1.0,
1879
- 1.0, 1.0, 1.0, 1.0,
1880
- 1.0, 1.0, 1.0, 1.0,
1881
- };
1882
-
1883
- buf.drawGrayscaleBuffer(0, 0, &intensities, 4, 4, null, bg);
1884
-
1885
- const cell_0_0 = buf.get(0, 0).?;
1886
- const cell_1_1 = buf.get(1, 1).?;
1887
- try std.testing.expect(cell_0_0.char != 32);
1888
- try std.testing.expect(cell_1_1.char != 32);
1889
-
1890
- const cell_3_3 = buf.get(3, 3).?;
1891
- try std.testing.expectEqual(@as(u32, 32), cell_3_3.char);
1892
-
1893
- buf.popScissorRect();
1894
- }
1895
-
1896
- test "OptimizedBuffer - drawGrayscaleBuffer intensity to character mapping" {
1897
- const pool = gp.initGlobalPool(std.testing.allocator);
1898
- defer gp.deinitGlobalPool();
1899
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1900
- defer local_link_pool.deinit();
1901
-
1902
- var buf = try OptimizedBuffer.init(
1903
- std.testing.allocator,
1904
- 10,
1905
- 5,
1906
- .{ .pool = pool, .id = "test-buffer" },
1907
- );
1908
- defer buf.deinit();
1909
-
1910
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
1911
- try buf.clear(bg, null);
1912
-
1913
- const intensities = [_]f32{
1914
- 0.005,
1915
- 0.02,
1916
- 0.5,
1917
- 1.0,
1918
- };
1919
-
1920
- buf.drawGrayscaleBuffer(0, 0, &intensities, 4, 1, null, bg);
1921
-
1922
- const cell_0 = buf.get(0, 0).?;
1923
- try std.testing.expectEqual(@as(u32, 32), cell_0.char);
1924
-
1925
- const cell_1 = buf.get(1, 0).?;
1926
- try std.testing.expect(cell_1.char != 32);
1927
-
1928
- const cell_3 = buf.get(3, 0).?;
1929
- try std.testing.expect(cell_3.fg[0] > 0.9);
1930
- try std.testing.expect(cell_3.fg[1] > 0.9);
1931
- try std.testing.expect(cell_3.fg[2] > 0.9);
1932
- }
1933
-
1934
- test "OptimizedBuffer - drawGrayscaleBuffer alpha blending preserves underlying bg" {
1935
- const pool = gp.initGlobalPool(std.testing.allocator);
1936
- defer gp.deinitGlobalPool();
1937
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1938
- defer local_link_pool.deinit();
1939
-
1940
- var buf = try OptimizedBuffer.init(
1941
- std.testing.allocator,
1942
- 10,
1943
- 5,
1944
- .{ .pool = pool, .id = "test-buffer" },
1945
- );
1946
- defer buf.deinit();
1947
-
1948
- const red_bg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
1949
- try buf.clear(red_bg, null);
1950
-
1951
- const initial_cell = buf.get(1, 1).?;
1952
- try std.testing.expectEqual(@as(f32, 1.0), initial_cell.bg[0]);
1953
- try std.testing.expectEqual(@as(f32, 0.0), initial_cell.bg[1]);
1954
- try std.testing.expectEqual(@as(f32, 0.0), initial_cell.bg[2]);
1955
-
1956
- const semi_transparent_bg = RGBA{ 0.0, 0.0, 1.0, 0.5 };
1957
- const intensities = [_]f32{
1958
- 1.0, 1.0, 1.0,
1959
- 1.0, 1.0, 1.0,
1960
- 1.0, 1.0, 1.0,
1961
- };
1962
-
1963
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, null, semi_transparent_bg);
1964
-
1965
- const cell = buf.get(1, 1).?;
1966
- try std.testing.expect(cell.bg[0] > 0.1);
1967
- try std.testing.expect(cell.bg[2] > 0.1);
1968
-
1969
- try std.testing.expect(cell.fg[0] > 0.9);
1970
- try std.testing.expect(cell.fg[1] > 0.9);
1971
- try std.testing.expect(cell.fg[2] > 0.9);
1972
- }
1973
-
1974
- test "OptimizedBuffer - drawGrayscaleBuffer fully transparent bg preserves underlying" {
1975
- const pool = gp.initGlobalPool(std.testing.allocator);
1976
- defer gp.deinitGlobalPool();
1977
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
1978
- defer local_link_pool.deinit();
1979
-
1980
- var buf = try OptimizedBuffer.init(
1981
- std.testing.allocator,
1982
- 10,
1983
- 5,
1984
- .{ .pool = pool, .id = "test-buffer" },
1985
- );
1986
- defer buf.deinit();
1987
-
1988
- const green_bg = RGBA{ 0.0, 1.0, 0.0, 1.0 };
1989
- try buf.clear(green_bg, null);
1990
-
1991
- const transparent_bg = RGBA{ 0.0, 0.0, 1.0, 0.0 };
1992
- const intensities = [_]f32{
1993
- 1.0, 1.0, 1.0,
1994
- 1.0, 1.0, 1.0,
1995
- 1.0, 1.0, 1.0,
1996
- };
1997
-
1998
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, null, transparent_bg);
1999
-
2000
- const cell = buf.get(1, 1).?;
2001
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[0]);
2002
- try std.testing.expectEqual(@as(f32, 1.0), cell.bg[1]);
2003
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[2]);
2004
-
2005
- try std.testing.expect(cell.fg[0] > 0.9);
2006
- }
2007
-
2008
- test "OptimizedBuffer - drawGrayscaleBuffer opaque bg overwrites underlying" {
2009
- const pool = gp.initGlobalPool(std.testing.allocator);
2010
- defer gp.deinitGlobalPool();
2011
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2012
- defer local_link_pool.deinit();
2013
-
2014
- var buf = try OptimizedBuffer.init(
2015
- std.testing.allocator,
2016
- 10,
2017
- 5,
2018
- .{ .pool = pool, .id = "test-buffer" },
2019
- );
2020
- defer buf.deinit();
2021
-
2022
- const red_bg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
2023
- try buf.clear(red_bg, null);
2024
-
2025
- const blue_bg = RGBA{ 0.0, 0.0, 1.0, 1.0 };
2026
- const intensities = [_]f32{
2027
- 1.0, 1.0, 1.0,
2028
- 1.0, 1.0, 1.0,
2029
- 1.0, 1.0, 1.0,
2030
- };
2031
-
2032
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, null, blue_bg);
2033
-
2034
- const cell = buf.get(1, 1).?;
2035
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[0]);
2036
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[1]);
2037
- try std.testing.expectEqual(@as(f32, 1.0), cell.bg[2]);
2038
- }
2039
-
2040
- test "OptimizedBuffer - drawGrayscaleBuffer with opacity stack" {
2041
- const pool = gp.initGlobalPool(std.testing.allocator);
2042
- defer gp.deinitGlobalPool();
2043
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2044
- defer local_link_pool.deinit();
2045
-
2046
- var buf = try OptimizedBuffer.init(
2047
- std.testing.allocator,
2048
- 10,
2049
- 5,
2050
- .{ .pool = pool, .id = "test-buffer" },
2051
- );
2052
- defer buf.deinit();
2053
-
2054
- const red_bg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
2055
- try buf.clear(red_bg, null);
2056
-
2057
- try buf.pushOpacity(0.5);
2058
-
2059
- const blue_bg = RGBA{ 0.0, 0.0, 1.0, 1.0 };
2060
- const intensities = [_]f32{
2061
- 1.0, 1.0, 1.0,
2062
- 1.0, 1.0, 1.0,
2063
- 1.0, 1.0, 1.0,
2064
- };
2065
-
2066
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, null, blue_bg);
2067
-
2068
- buf.popOpacity();
2069
-
2070
- const cell = buf.get(1, 1).?;
2071
- try std.testing.expect(cell.bg[0] > 0.1);
2072
- try std.testing.expect(cell.bg[2] > 0.1);
2073
- }
2074
-
2075
- test "OptimizedBuffer - drawGrayscaleBufferSupersampled alpha blending" {
2076
- const pool = gp.initGlobalPool(std.testing.allocator);
2077
- defer gp.deinitGlobalPool();
2078
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2079
- defer local_link_pool.deinit();
2080
-
2081
- var buf = try OptimizedBuffer.init(
2082
- std.testing.allocator,
2083
- 10,
2084
- 5,
2085
- .{ .pool = pool, .id = "test-buffer" },
2086
- );
2087
- defer buf.deinit();
2088
-
2089
- const red_bg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
2090
- try buf.clear(red_bg, null);
2091
-
2092
- const intensities = [_]f32{
2093
- 1.0, 1.0, 1.0, 1.0,
2094
- 1.0, 1.0, 1.0, 1.0,
2095
- 1.0, 1.0, 1.0, 1.0,
2096
- 1.0, 1.0, 1.0, 1.0,
2097
- };
2098
-
2099
- const semi_transparent_bg = RGBA{ 0.0, 0.0, 1.0, 0.5 };
2100
- buf.drawGrayscaleBufferSupersampled(0, 0, &intensities, 4, 4, null, semi_transparent_bg);
2101
-
2102
- const cell = buf.get(0, 0).?;
2103
- try std.testing.expect(cell.bg[0] > 0.1);
2104
- try std.testing.expect(cell.bg[2] > 0.1);
2105
- }
2106
-
2107
- test "OptimizedBuffer - drawGrayscaleBufferSupersampled fully transparent preserves bg" {
2108
- const pool = gp.initGlobalPool(std.testing.allocator);
2109
- defer gp.deinitGlobalPool();
2110
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2111
- defer local_link_pool.deinit();
2112
-
2113
- var buf = try OptimizedBuffer.init(
2114
- std.testing.allocator,
2115
- 10,
2116
- 5,
2117
- .{ .pool = pool, .id = "test-buffer" },
2118
- );
2119
- defer buf.deinit();
2120
-
2121
- const green_bg = RGBA{ 0.0, 1.0, 0.0, 1.0 };
2122
- try buf.clear(green_bg, null);
2123
-
2124
- const intensities = [_]f32{
2125
- 1.0, 1.0, 1.0, 1.0,
2126
- 1.0, 1.0, 1.0, 1.0,
2127
- 1.0, 1.0, 1.0, 1.0,
2128
- 1.0, 1.0, 1.0, 1.0,
2129
- };
2130
-
2131
- const transparent_bg = RGBA{ 0.0, 0.0, 1.0, 0.0 };
2132
- buf.drawGrayscaleBufferSupersampled(0, 0, &intensities, 4, 4, null, transparent_bg);
2133
-
2134
- const cell = buf.get(0, 0).?;
2135
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[0]);
2136
- try std.testing.expectEqual(@as(f32, 1.0), cell.bg[1]);
2137
- try std.testing.expectEqual(@as(f32, 0.0), cell.bg[2]);
2138
- }
2139
-
2140
- test "OptimizedBuffer - drawGrayscaleBufferSupersampled respects scissor" {
2141
- const pool = gp.initGlobalPool(std.testing.allocator);
2142
- defer gp.deinitGlobalPool();
2143
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2144
- defer local_link_pool.deinit();
2145
-
2146
- var buf = try OptimizedBuffer.init(
2147
- std.testing.allocator,
2148
- 6,
2149
- 4,
2150
- .{ .pool = pool, .id = "test-buffer" },
2151
- );
2152
- defer buf.deinit();
2153
-
2154
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2155
- try buf.clear(bg, null);
2156
-
2157
- try buf.pushScissorRect(0, 0, 1, 1);
2158
-
2159
- const intensities = [_]f32{
2160
- 1.0, 1.0, 1.0, 1.0,
2161
- 1.0, 1.0, 1.0, 1.0,
2162
- 1.0, 1.0, 1.0, 1.0,
2163
- 1.0, 1.0, 1.0, 1.0,
2164
- };
2165
-
2166
- buf.drawGrayscaleBufferSupersampled(0, 0, &intensities, 4, 4, null, bg);
2167
-
2168
- const inCell = buf.get(0, 0).?;
2169
- const outCell = buf.get(2, 2).?;
2170
- try std.testing.expect(inCell.char != 32);
2171
- try std.testing.expectEqual(@as(u32, 32), outCell.char);
2172
-
2173
- buf.popScissorRect();
2174
- }
2175
-
2176
- test "OptimizedBuffer - drawGrayscaleBufferSupersampled with opacity stack" {
2177
- const pool = gp.initGlobalPool(std.testing.allocator);
2178
- defer gp.deinitGlobalPool();
2179
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2180
- defer local_link_pool.deinit();
2181
-
2182
- var buf = try OptimizedBuffer.init(
2183
- std.testing.allocator,
2184
- 10,
2185
- 5,
2186
- .{ .pool = pool, .id = "test-buffer" },
2187
- );
2188
- defer buf.deinit();
2189
-
2190
- const red_bg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
2191
- try buf.clear(red_bg, null);
2192
-
2193
- try buf.pushOpacity(0.5);
2194
-
2195
- const intensities = [_]f32{
2196
- 1.0, 1.0, 1.0, 1.0,
2197
- 1.0, 1.0, 1.0, 1.0,
2198
- 1.0, 1.0, 1.0, 1.0,
2199
- 1.0, 1.0, 1.0, 1.0,
2200
- };
2201
-
2202
- const blue_bg = RGBA{ 0.0, 0.0, 1.0, 1.0 };
2203
- buf.drawGrayscaleBufferSupersampled(0, 0, &intensities, 4, 4, null, blue_bg);
2204
-
2205
- buf.popOpacity();
2206
-
2207
- const cell = buf.get(0, 0).?;
2208
- try std.testing.expect(cell.bg[0] > 0.1);
2209
- try std.testing.expect(cell.bg[2] > 0.1);
2210
- }
2211
-
2212
- test "OptimizedBuffer - blendColors with transparent destination" {
2213
- const pool = gp.initGlobalPool(std.testing.allocator);
2214
- defer gp.deinitGlobalPool();
2215
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2216
- defer local_link_pool.deinit();
2217
-
2218
- var buf = try OptimizedBuffer.init(
2219
- std.testing.allocator,
2220
- 2,
2221
- 2,
2222
- .{ .pool = pool, .id = "test-buffer" },
2223
- );
2224
- defer buf.deinit();
2225
-
2226
- const transparent_bg = RGBA{ 0.0, 0.0, 0.0, 0.0 };
2227
- try buf.clear(transparent_bg, null);
2228
-
2229
- const semi_white = RGBA{ 1.0, 1.0, 1.0, 0.5 };
2230
- const transparent_fg = RGBA{ 0.0, 0.0, 0.0, 0.0 };
2231
- try buf.setCellWithAlphaBlending(0, 0, 'X', semi_white, transparent_fg, 0);
2232
-
2233
- const cell = buf.get(0, 0).?;
2234
- try std.testing.expect(cell.fg[0] > 0.45);
2235
- try std.testing.expect(cell.fg[0] < 0.55);
2236
- try std.testing.expectEqual(@as(f32, 0.5), cell.fg[3]);
2237
- }
2238
-
2239
- test "OptimizedBuffer - drawGrayscaleBuffer with custom fg color" {
2240
- const pool = gp.initGlobalPool(std.testing.allocator);
2241
- defer gp.deinitGlobalPool();
2242
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2243
- defer local_link_pool.deinit();
2244
-
2245
- var buf = try OptimizedBuffer.init(
2246
- std.testing.allocator,
2247
- 10,
2248
- 5,
2249
- .{ .pool = pool, .id = "test-buffer" },
2250
- );
2251
- defer buf.deinit();
2252
-
2253
- const black_bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2254
- try buf.clear(black_bg, null);
2255
-
2256
- const intensities = [_]f32{
2257
- 1.0, 1.0, 1.0,
2258
- 1.0, 1.0, 1.0,
2259
- 1.0, 1.0, 1.0,
2260
- };
2261
-
2262
- const red_fg = RGBA{ 1.0, 0.0, 0.0, 1.0 };
2263
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, red_fg, black_bg);
2264
-
2265
- const cell = buf.get(1, 1).?;
2266
- try std.testing.expect(cell.fg[0] > 0.9);
2267
- try std.testing.expect(cell.fg[1] < 0.1);
2268
- try std.testing.expect(cell.fg[2] < 0.1);
2269
- }
2270
-
2271
- test "OptimizedBuffer - drawGrayscaleBuffer custom fg with partial intensity" {
2272
- const pool = gp.initGlobalPool(std.testing.allocator);
2273
- defer gp.deinitGlobalPool();
2274
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2275
- defer local_link_pool.deinit();
2276
-
2277
- var buf = try OptimizedBuffer.init(
2278
- std.testing.allocator,
2279
- 10,
2280
- 5,
2281
- .{ .pool = pool, .id = "test-buffer" },
2282
- );
2283
- defer buf.deinit();
2284
-
2285
- const blue_bg = RGBA{ 0.0, 0.0, 1.0, 1.0 };
2286
- try buf.clear(blue_bg, null);
2287
-
2288
- const intensities = [_]f32{
2289
- 0.5, 0.5, 0.5,
2290
- 0.5, 0.5, 0.5,
2291
- 0.5, 0.5, 0.5,
2292
- };
2293
-
2294
- const green_fg = RGBA{ 0.0, 1.0, 0.0, 1.0 };
2295
- const transparent_bg = RGBA{ 0.0, 0.0, 0.0, 0.0 };
2296
- buf.drawGrayscaleBuffer(0, 0, &intensities, 3, 3, green_fg, transparent_bg);
2297
-
2298
- const cell = buf.get(1, 1).?;
2299
- try std.testing.expect(cell.fg[1] > 0.2);
2300
- try std.testing.expect(cell.fg[2] > 0.2);
2301
- }
2302
-
2303
- test "OptimizedBuffer - drawGrayscaleBufferSupersampled with custom fg color" {
2304
- const pool = gp.initGlobalPool(std.testing.allocator);
2305
- defer gp.deinitGlobalPool();
2306
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2307
- defer local_link_pool.deinit();
2308
-
2309
- var buf = try OptimizedBuffer.init(
2310
- std.testing.allocator,
2311
- 10,
2312
- 5,
2313
- .{ .pool = pool, .id = "test-buffer" },
2314
- );
2315
- defer buf.deinit();
2316
-
2317
- const black_bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2318
- try buf.clear(black_bg, null);
2319
-
2320
- const intensities = [_]f32{
2321
- 1.0, 1.0, 1.0, 1.0,
2322
- 1.0, 1.0, 1.0, 1.0,
2323
- 1.0, 1.0, 1.0, 1.0,
2324
- 1.0, 1.0, 1.0, 1.0,
2325
- };
2326
-
2327
- const cyan_fg = RGBA{ 0.0, 1.0, 1.0, 1.0 };
2328
- buf.drawGrayscaleBufferSupersampled(0, 0, &intensities, 4, 4, cyan_fg, black_bg);
2329
-
2330
- const cell = buf.get(0, 0).?;
2331
- try std.testing.expect(cell.fg[0] < 0.1);
2332
- try std.testing.expect(cell.fg[1] > 0.9);
2333
- try std.testing.expect(cell.fg[2] > 0.9);
2334
- }
2335
-
2336
- // Overwriting a grapheme cell with the same ID but different extent bits must
2337
- // not free the pool slot (which would allow reuse and generation bump).
2338
- test "buffer - set same grapheme ID with different extents keeps slot alive" {
2339
- var local_pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
2340
- .slots_per_page = .{ 1, 1, 1, 1, 1 },
2341
- });
2342
- defer local_pool.deinit();
2343
-
2344
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2345
- defer local_link_pool.deinit();
2346
-
2347
- var buf = try OptimizedBuffer.init(std.testing.allocator, 10, 2, .{
2348
- .pool = &local_pool,
2349
- .link_pool = &local_link_pool,
2350
- .width_method = .unicode,
2351
- });
2352
- defer buf.deinit();
2353
-
2354
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
2355
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2356
-
2357
- const emoji = "👋";
2358
-
2359
- const gid = local_pool.alloc(emoji) catch @panic("alloc failed");
2360
- const packed_w2 = gp.packGraphemeStart(gid & gp.GRAPHEME_ID_MASK, 2);
2361
- buf.set(0, 0, buffer_mod.Cell{ .char = packed_w2, .fg = fg, .bg = bg, .attributes = 0 });
2362
-
2363
- const id_from_char = gp.graphemeIdFromChar(packed_w2);
2364
- try std.testing.expect(buf.grapheme_tracker.contains(id_from_char));
2365
-
2366
- // Same grapheme ID, different width → different packed char
2367
- const packed_w1 = gp.packGraphemeStart(gid & gp.GRAPHEME_ID_MASK, 1);
2368
- buf.set(0, 0, buffer_mod.Cell{ .char = packed_w1, .fg = fg, .bg = bg, .attributes = 0 });
2369
-
2370
- try std.testing.expect(buf.grapheme_tracker.contains(id_from_char));
2371
-
2372
- const bytes = local_pool.get(gid) catch @panic("get failed - slot was freed");
2373
- try std.testing.expectEqualSlices(u8, emoji, bytes);
2374
- }
2375
-
2376
- // Exercises grapheme pool slot reuse across multiple render frames with
2377
- // alternating dialog/form content to stress the alloc→set→render cycle.
2378
- test "renderer - grapheme WrongGeneration repro with pool slot reuse" {
2379
- const pool = gp.initGlobalPool(std.testing.allocator);
2380
- defer gp.deinitGlobalPool();
2381
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2382
- defer local_link_pool.deinit();
2383
-
2384
- const renderer_mod = @import("../renderer.zig");
2385
- var cli_renderer = try renderer_mod.CliRenderer.create(
2386
- std.testing.allocator,
2387
- 40,
2388
- 5,
2389
- pool,
2390
- true,
2391
- );
2392
- defer cli_renderer.destroy();
2393
-
2394
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
2395
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2396
-
2397
- {
2398
- const next = cli_renderer.getNextBuffer();
2399
- try next.drawText("╭────────────────────────────────────╮", 0, 0, fg, bg, 0);
2400
- try next.drawText("│ ◇ Select Files │", 0, 1, fg, bg, 0);
2401
- try next.drawText("│ ▫ src/ ▪ file.ts │", 0, 2, fg, bg, 0);
2402
- try next.drawText("│ ↑↓ navigate ⏎ select esc close │", 0, 3, fg, bg, 0);
2403
- try next.drawText("╰────────────────────────────────────╯", 0, 4, fg, bg, 0);
2404
- cli_renderer.render(false);
2405
- }
2406
-
2407
- {
2408
- const next = cli_renderer.getNextBuffer();
2409
- try next.drawText(" Your Name ", 0, 0, fg, bg, 0);
2410
- try next.drawText(" John Doe ", 0, 1, fg, bg, 0);
2411
- try next.drawText(" ", 0, 2, fg, bg, 0);
2412
- try next.drawText(" Select Files ", 0, 3, fg, bg, 0);
2413
- try next.drawText(" Enter file path... ", 0, 4, fg, bg, 0);
2414
- cli_renderer.render(false);
2415
- }
2416
-
2417
- {
2418
- const next = cli_renderer.getNextBuffer();
2419
- try next.drawText("╭────────────────────────────────────╮", 0, 0, fg, bg, 0);
2420
- try next.drawText("│ ◇ Select Files │", 0, 1, fg, bg, 0);
2421
- try next.drawText("│ ▫ src/ ▪ file.ts │", 0, 2, fg, bg, 0);
2422
- try next.drawText("│ ↑↓ navigate ⏎ select esc close │", 0, 3, fg, bg, 0);
2423
- try next.drawText("╰────────────────────────────────────╯", 0, 4, fg, bg, 0);
2424
- cli_renderer.render(false);
2425
- }
2426
-
2427
- {
2428
- const next = cli_renderer.getNextBuffer();
2429
- try next.drawText(" Your Name ", 0, 0, fg, bg, 0);
2430
- try next.drawText(" John Doe ", 0, 1, fg, bg, 0);
2431
- try next.drawText(" ", 0, 2, fg, bg, 0);
2432
- try next.drawText(" Select Files ", 0, 3, fg, bg, 0);
2433
- try next.drawText(" Enter file path... ", 0, 4, fg, bg, 0);
2434
- cli_renderer.render(false);
2435
- }
2436
-
2437
- {
2438
- const next = cli_renderer.getNextBuffer();
2439
- try next.drawText("╭────────────────────────────────────╮", 0, 0, fg, bg, 0);
2440
- try next.drawText("│ Filter: s │", 0, 1, fg, bg, 0);
2441
- try next.drawText("│ ▫ src/ │", 0, 2, fg, bg, 0);
2442
- try next.drawText("│ ↑↓ navigate ⏎/tab select │", 0, 3, fg, bg, 0);
2443
- try next.drawText("╰────────────────────────────────────╯", 0, 4, fg, bg, 0);
2444
- cli_renderer.render(false);
2445
- }
2446
- }
2447
-
2448
- // Issue #723: CJK grapheme continuation cells are destroyed when graphemes
2449
- // shift left (e.g. after backspace). The renderer's diff loop calls
2450
- // currentRenderBuffer.set() left-to-right, and set()'s span cleanup at
2451
- // position N+2 destroys the continuation cell at N+1 that was just written
2452
- // by set() at position N, because both share the same stable grapheme pool ID.
2453
- test "renderer - CJK graphemes shifting left must preserve continuation cells (#723)" {
2454
- const pool = gp.initGlobalPool(std.testing.allocator);
2455
- defer gp.deinitGlobalPool();
2456
- var local_link_pool = link.LinkPool.init(std.testing.allocator);
2457
- defer local_link_pool.deinit();
2458
-
2459
- const renderer_mod = @import("../renderer.zig");
2460
- var cli_renderer = try renderer_mod.CliRenderer.create(
2461
- std.testing.allocator,
2462
- 20,
2463
- 1,
2464
- pool,
2465
- true,
2466
- );
2467
- defer cli_renderer.destroy();
2468
-
2469
- const fg = RGBA{ 1.0, 1.0, 1.0, 1.0 };
2470
- const bg = RGBA{ 0.0, 0.0, 0.0, 1.0 };
2471
-
2472
- // Frame 1: "abcd你好世" — CJK chars start at column 4
2473
- // Layout: a(0) b(1) c(2) d(3) 你(4,5) 好(6,7) 世(8,9) spaces(10..19)
2474
- {
2475
- const next = cli_renderer.getNextBuffer();
2476
- try next.drawText("abcd你好世 ", 0, 0, fg, bg, 0);
2477
- cli_renderer.render(false);
2478
- }
2479
-
2480
- // Frame 2: "abc你好世" — backspace deleted 'd', CJK chars shift left by 1
2481
- // Layout: a(0) b(1) c(2) 你(3,4) 好(5,6) 世(7,8) spaces(9..19)
2482
- {
2483
- const next = cli_renderer.getNextBuffer();
2484
- try next.drawText("abc你好世 ", 0, 0, fg, bg, 0);
2485
- cli_renderer.render(false);
2486
- }
2487
-
2488
- // After frame 2, currentRenderBuffer should match the frame 2 layout exactly.
2489
- // The bug: span cleanup in set() destroys continuation cells (positions 4, 6, 8)
2490
- // leaving spaces instead of proper continuation chars.
2491
- const current = cli_renderer.getCurrentBuffer();
2492
-
2493
- // Check that position 3 is a grapheme start (你)
2494
- const cell3 = current.get(3, 0).?;
2495
- try std.testing.expect(gp.isGraphemeChar(cell3.char));
2496
- try std.testing.expectEqual(@as(u32, 1), gp.charRightExtent(cell3.char));
2497
-
2498
- // Check that position 4 is a continuation cell for the same grapheme (你)
2499
- const cell4 = current.get(4, 0).?;
2500
- try std.testing.expect(gp.isContinuationChar(cell4.char));
2501
- const id3 = gp.graphemeIdFromChar(cell3.char);
2502
- const id4 = gp.graphemeIdFromChar(cell4.char);
2503
- try std.testing.expectEqual(id3, id4);
2504
-
2505
- // Check that position 5 is a grapheme start (好)
2506
- const cell5 = current.get(5, 0).?;
2507
- try std.testing.expect(gp.isGraphemeChar(cell5.char));
2508
-
2509
- // Check that position 6 is a continuation cell for the same grapheme (好)
2510
- const cell6 = current.get(6, 0).?;
2511
- try std.testing.expect(gp.isContinuationChar(cell6.char));
2512
- const id5 = gp.graphemeIdFromChar(cell5.char);
2513
- const id6 = gp.graphemeIdFromChar(cell6.char);
2514
- try std.testing.expectEqual(id5, id6);
2515
-
2516
- // Check that position 7 is a grapheme start (世)
2517
- const cell7 = current.get(7, 0).?;
2518
- try std.testing.expect(gp.isGraphemeChar(cell7.char));
2519
-
2520
- // Check that position 8 is a continuation cell for the same grapheme (世)
2521
- const cell8 = current.get(8, 0).?;
2522
- try std.testing.expect(gp.isContinuationChar(cell8.char));
2523
- const id7 = gp.graphemeIdFromChar(cell7.char);
2524
- const id8 = gp.graphemeIdFromChar(cell8.char);
2525
- try std.testing.expectEqual(id7, id8);
2526
- }