@fairyhunter13/opentui-core 0.1.112 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +63 -51
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-8fks7yv1.js +0 -411
  472. package/index-8fks7yv1.js.map +0 -10
  473. package/index-egy5e2rs.js +0 -12267
  474. package/index-egy5e2rs.js.map +0 -42
  475. package/index-tse8gzh0.js +0 -20614
  476. package/index-tse8gzh0.js.map +0 -67
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,1304 @@
1
+ const std = @import("std");
2
+ const gp = @import("../grapheme.zig");
3
+
4
+ const GraphemePool = gp.GraphemePool;
5
+ const GraphemeTracker = gp.GraphemeTracker;
6
+
7
+ test "GraphemePool - can initialize and cleanup" {
8
+ // Just verify init/deinit don't crash
9
+ var pool = GraphemePool.init(std.testing.allocator);
10
+ pool.deinit();
11
+ }
12
+
13
+ test "GraphemePool - alloc and get small grapheme" {
14
+ var pool = GraphemePool.init(std.testing.allocator);
15
+ defer pool.deinit();
16
+
17
+ const text = "a";
18
+ const id = try pool.alloc(text);
19
+ try pool.incref(id);
20
+ defer pool.decref(id) catch {};
21
+
22
+ const retrieved = try pool.get(id);
23
+ try std.testing.expectEqualSlices(u8, text, retrieved);
24
+ }
25
+
26
+ test "GraphemePool - alloc and get emoji" {
27
+ var pool = GraphemePool.init(std.testing.allocator);
28
+ defer pool.deinit();
29
+
30
+ const emoji = "🌟";
31
+ const id = try pool.alloc(emoji);
32
+ try pool.incref(id);
33
+ defer pool.decref(id) catch {};
34
+
35
+ const retrieved = try pool.get(id);
36
+ try std.testing.expectEqualSlices(u8, emoji, retrieved);
37
+ }
38
+
39
+ test "GraphemePool - alloc and get multi-byte grapheme" {
40
+ var pool = GraphemePool.init(std.testing.allocator);
41
+ defer pool.deinit();
42
+
43
+ const grapheme = "é";
44
+ const id = try pool.alloc(grapheme);
45
+ try pool.incref(id);
46
+ defer pool.decref(id) catch {};
47
+
48
+ const retrieved = try pool.get(id);
49
+ try std.testing.expectEqualSlices(u8, grapheme, retrieved);
50
+ }
51
+
52
+ test "GraphemePool - alloc and get combining character grapheme" {
53
+ var pool = GraphemePool.init(std.testing.allocator);
54
+ defer pool.deinit();
55
+
56
+ const grapheme = "e\u{0301}"; // e with combining acute accent
57
+ const id = try pool.alloc(grapheme);
58
+ try pool.incref(id);
59
+ defer pool.decref(id) catch {};
60
+
61
+ const retrieved = try pool.get(id);
62
+ try std.testing.expectEqualSlices(u8, grapheme, retrieved);
63
+ }
64
+
65
+ test "GraphemePool - multiple allocations" {
66
+ var pool = GraphemePool.init(std.testing.allocator);
67
+ defer pool.deinit();
68
+
69
+ const text1 = "a";
70
+ const text2 = "b";
71
+ const text3 = "🌟";
72
+
73
+ const id1 = try pool.alloc(text1);
74
+ const id2 = try pool.alloc(text2);
75
+ const id3 = try pool.alloc(text3);
76
+ try pool.incref(id1);
77
+ try pool.incref(id2);
78
+ try pool.incref(id3);
79
+ defer pool.decref(id1) catch {};
80
+ defer pool.decref(id2) catch {};
81
+ defer pool.decref(id3) catch {};
82
+
83
+ try std.testing.expect(id1 != id2);
84
+ try std.testing.expect(id2 != id3);
85
+ try std.testing.expect(id1 != id3);
86
+
87
+ try std.testing.expectEqualSlices(u8, text1, try pool.get(id1));
88
+ try std.testing.expectEqualSlices(u8, text2, try pool.get(id2));
89
+ try std.testing.expectEqualSlices(u8, text3, try pool.get(id3));
90
+ }
91
+
92
+ test "GraphemePool - handles various size graphemes" {
93
+ var pool = GraphemePool.init(std.testing.allocator);
94
+ defer pool.deinit();
95
+
96
+ const small = "a";
97
+ const medium = "0123456789";
98
+ const large = "012345678901234567890123456789";
99
+
100
+ const id_small = try pool.alloc(small);
101
+ const id_medium = try pool.alloc(medium);
102
+ const id_large = try pool.alloc(large);
103
+ try pool.incref(id_small);
104
+ try pool.incref(id_medium);
105
+ try pool.incref(id_large);
106
+ defer pool.decref(id_small) catch {};
107
+ defer pool.decref(id_medium) catch {};
108
+ defer pool.decref(id_large) catch {};
109
+
110
+ try std.testing.expectEqualSlices(u8, small, try pool.get(id_small));
111
+ try std.testing.expectEqualSlices(u8, medium, try pool.get(id_medium));
112
+ try std.testing.expectEqualSlices(u8, large, try pool.get(id_large));
113
+ }
114
+
115
+ test "GraphemePool - large allocation (128 bytes)" {
116
+ var pool = GraphemePool.init(std.testing.allocator);
117
+ defer pool.deinit();
118
+
119
+ var buffer: [128]u8 = undefined;
120
+ @memset(&buffer, 'X');
121
+
122
+ const id = try pool.alloc(&buffer);
123
+ try pool.incref(id);
124
+ defer pool.decref(id) catch {};
125
+
126
+ const retrieved = try pool.get(id);
127
+
128
+ try std.testing.expectEqual(@as(usize, 128), retrieved.len);
129
+ try std.testing.expectEqualSlices(u8, &buffer, retrieved);
130
+ }
131
+
132
+ test "GraphemePool - incref increases refcount" {
133
+ var pool = GraphemePool.init(std.testing.allocator);
134
+ defer pool.deinit();
135
+
136
+ const text = "a";
137
+ const id = try pool.alloc(text);
138
+
139
+ // Initial refcount is 0, increment it
140
+ try pool.incref(id);
141
+ defer pool.decref(id) catch {};
142
+
143
+ const retrieved = try pool.get(id);
144
+ try std.testing.expectEqualSlices(u8, text, retrieved);
145
+ }
146
+
147
+ test "GraphemePool - decref once keeps data alive" {
148
+ var pool = GraphemePool.init(std.testing.allocator);
149
+ defer pool.deinit();
150
+
151
+ const text = "a";
152
+ const id = try pool.alloc(text);
153
+
154
+ // Initial refcount is 0, incref to 1, incref to 2
155
+ try pool.incref(id);
156
+ try pool.incref(id);
157
+ defer pool.decref(id) catch {};
158
+
159
+ // Decref from 2 to 1
160
+ try pool.decref(id);
161
+
162
+ // Should still be accessible (refcount is 1)
163
+ const retrieved = try pool.get(id);
164
+ try std.testing.expectEqualSlices(u8, text, retrieved);
165
+ }
166
+
167
+ test "GraphemePool - decref to zero allows slot reuse" {
168
+ var pool = GraphemePool.init(std.testing.allocator);
169
+ defer pool.deinit();
170
+
171
+ const text1 = "a";
172
+ const id1 = try pool.alloc(text1);
173
+ try pool.incref(id1);
174
+
175
+ // Decref to zero makes slot available for reuse
176
+ try pool.decref(id1);
177
+
178
+ // Allocate again - should reuse the freed slot with new generation
179
+ const text2 = "b";
180
+ const id2 = try pool.alloc(text2);
181
+
182
+ // Old ID should fail due to generation mismatch
183
+ const result1 = pool.get(id1);
184
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result1);
185
+
186
+ try pool.incref(id2);
187
+ const retrieved = try pool.get(id2);
188
+ try std.testing.expectEqualSlices(u8, text2, retrieved);
189
+
190
+ try pool.decref(id2);
191
+ }
192
+
193
+ test "GraphemePool - multiple incref and decref" {
194
+ var pool = GraphemePool.init(std.testing.allocator);
195
+ defer pool.deinit();
196
+
197
+ const text = "test";
198
+ const id = try pool.alloc(text);
199
+
200
+ // Increment refcount multiple times (starting from 0)
201
+ try pool.incref(id);
202
+ try pool.incref(id);
203
+ try pool.incref(id);
204
+
205
+ try pool.decref(id);
206
+ try pool.decref(id);
207
+
208
+ // Should still be accessible (refcount is 1)
209
+ const retrieved = try pool.get(id);
210
+ try std.testing.expectEqualSlices(u8, text, retrieved);
211
+
212
+ // Decrement to zero
213
+ try pool.decref(id);
214
+
215
+ // Allocate something else to trigger reuse with new generation
216
+ _ = try pool.alloc("x");
217
+
218
+ // Old ID should now fail due to generation mismatch
219
+ const result = pool.get(id);
220
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
221
+
222
+ // Cleanup not needed since allocated IDs have refcount 0
223
+ }
224
+
225
+ test "GraphemePool - freed IDs become invalid after reuse" {
226
+ var pool = GraphemePool.init(std.testing.allocator);
227
+ defer pool.deinit();
228
+
229
+ const text1 = "a";
230
+ const text2 = "b";
231
+
232
+ const id1 = try pool.alloc(text1);
233
+ try pool.incref(id1);
234
+
235
+ // Decref to free the slot
236
+ try pool.decref(id1);
237
+
238
+ // Allocate again (pool may reuse internal storage)
239
+ const id2 = try pool.alloc(text2);
240
+
241
+ // Old ID should be invalid due to generation mismatch
242
+ const result = pool.get(id1);
243
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
244
+
245
+ try pool.incref(id2);
246
+ const retrieved = try pool.get(id2);
247
+ try std.testing.expectEqualSlices(u8, text2, retrieved);
248
+ try pool.decref(id2);
249
+ }
250
+
251
+ test "GraphemePool - stale ID with wrong generation fails" {
252
+ var pool = GraphemePool.init(std.testing.allocator);
253
+ defer pool.deinit();
254
+
255
+ const text = "test";
256
+ const id = try pool.alloc(text);
257
+ try pool.incref(id);
258
+ defer pool.decref(id) catch {};
259
+
260
+ // Manually create a stale ID by modifying generation
261
+ const stale_id = id ^ (1 << gp.SLOT_BITS); // XOR generation bits
262
+
263
+ const result = pool.get(stale_id);
264
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
265
+ }
266
+
267
+ test "GraphemePool - decref on zero refcount fails" {
268
+ var pool = GraphemePool.init(std.testing.allocator);
269
+ defer pool.deinit();
270
+
271
+ const text = "a";
272
+ const id = try pool.alloc(text);
273
+
274
+ // Refcount starts at 0, so decref should fail immediately
275
+ const result = pool.decref(id);
276
+ try std.testing.expectError(gp.GraphemePoolError.InvalidId, result);
277
+ }
278
+
279
+ test "GraphemePool - many allocations" {
280
+ var pool = GraphemePool.init(std.testing.allocator);
281
+ defer pool.deinit();
282
+
283
+ const count = 1000;
284
+ var ids: [count]u32 = undefined;
285
+
286
+ for (0..count) |i| {
287
+ var buffer: [8]u8 = undefined;
288
+ const slice = std.fmt.bufPrint(&buffer, "{d}", .{i}) catch unreachable;
289
+ ids[i] = try pool.alloc(slice);
290
+ try pool.incref(ids[i]);
291
+ }
292
+
293
+ for (ids, 0..count) |id, i| {
294
+ const retrieved = try pool.get(id);
295
+ var buffer: [8]u8 = undefined;
296
+ const slice = std.fmt.bufPrint(&buffer, "{d}", .{i}) catch unreachable;
297
+ try std.testing.expectEqualSlices(u8, slice, retrieved);
298
+ }
299
+
300
+ for (ids) |id| {
301
+ try pool.decref(id);
302
+ }
303
+ }
304
+
305
+ test "GraphemePool - allocations with varying sizes" {
306
+ var pool = GraphemePool.init(std.testing.allocator);
307
+ defer pool.deinit();
308
+
309
+ var ids: std.ArrayListUnmanaged(u32) = .{};
310
+ defer ids.deinit(std.testing.allocator);
311
+
312
+ for (0..50) |i| {
313
+ const size = (i % 5) * 16 + 5; // Vary sizes: 5, 21, 37, 53, 69...
314
+ var buffer: [128]u8 = undefined;
315
+ @memset(buffer[0..size], @intCast(i % 256));
316
+ const id = try pool.alloc(buffer[0..size]);
317
+ try pool.incref(id);
318
+ try ids.append(std.testing.allocator, id);
319
+ }
320
+
321
+ for (ids.items, 0..50) |id, i| {
322
+ const size = (i % 5) * 16 + 5;
323
+ const retrieved = try pool.get(id);
324
+ try std.testing.expectEqual(size, retrieved.len);
325
+ for (retrieved) |byte| {
326
+ try std.testing.expectEqual(@as(u8, @intCast(i % 256)), byte);
327
+ }
328
+ }
329
+
330
+ for (ids.items) |id| {
331
+ try pool.decref(id);
332
+ }
333
+ }
334
+
335
+ test "GraphemePool - reuse many slots" {
336
+ var pool = GraphemePool.init(std.testing.allocator);
337
+ defer pool.deinit();
338
+
339
+ for (0..100) |i| {
340
+ var buffer: [8]u8 = undefined;
341
+ const slice = std.fmt.bufPrint(&buffer, "{d}", .{i}) catch unreachable;
342
+ const id = try pool.alloc(slice);
343
+ try pool.incref(id);
344
+
345
+ const retrieved = try pool.get(id);
346
+ try std.testing.expectEqualSlices(u8, slice, retrieved);
347
+
348
+ try pool.decref(id);
349
+ }
350
+ }
351
+
352
+ test "GraphemePool - invalid ID returns error" {
353
+ var pool = GraphemePool.init(std.testing.allocator);
354
+ defer pool.deinit();
355
+
356
+ const text = "test";
357
+ const id = try pool.alloc(text);
358
+ try pool.incref(id);
359
+
360
+ // Decref to free the slot
361
+ try pool.decref(id);
362
+
363
+ // Now allocate again to change generation
364
+ const text2 = "test2";
365
+ _ = try pool.alloc(text2);
366
+
367
+ // Original ID should now be invalid due to generation mismatch
368
+ const result = pool.get(id);
369
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
370
+ }
371
+
372
+ test "GraphemePool - IDs from different pools don't interfere" {
373
+ var pool1 = GraphemePool.init(std.testing.allocator);
374
+ defer pool1.deinit();
375
+
376
+ var pool2 = GraphemePool.init(std.testing.allocator);
377
+ defer pool2.deinit();
378
+
379
+ const text1 = "pool1_data";
380
+ const text2 = "pool2_data";
381
+
382
+ const id1 = try pool1.alloc(text1);
383
+ const id2 = try pool2.alloc(text2);
384
+ try pool1.incref(id1);
385
+ try pool2.incref(id2);
386
+ defer pool1.decref(id1) catch {};
387
+ defer pool2.decref(id2) catch {};
388
+
389
+ try std.testing.expectEqualSlices(u8, text1, try pool1.get(id1));
390
+ try std.testing.expectEqualSlices(u8, text2, try pool2.get(id2));
391
+
392
+ // Using ID from pool1 in pool2 may succeed or fail depending on internal state,
393
+ // but should not return pool1's data or crash
394
+ _ = pool2.get(id1) catch |err| {
395
+ try std.testing.expectEqual(gp.GraphemePoolError.InvalidId, err);
396
+ };
397
+ }
398
+
399
+ test "GraphemePool - use-after-free returns error not garbage" {
400
+ var pool = GraphemePool.init(std.testing.allocator);
401
+ defer pool.deinit();
402
+
403
+ const text1 = "first";
404
+ const id1 = try pool.alloc(text1);
405
+ try pool.incref(id1);
406
+ try pool.decref(id1);
407
+
408
+ // Allocate something else to potentially reuse the slot
409
+ const text2 = "second";
410
+ const id2 = try pool.alloc(text2);
411
+
412
+ // Old ID should fail due to generation mismatch, not return text2 or garbage
413
+ const result = pool.get(id1);
414
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
415
+
416
+ try pool.incref(id2);
417
+ try std.testing.expectEqualSlices(u8, text2, try pool.get(id2));
418
+ try pool.decref(id2);
419
+ }
420
+
421
+ test "GraphemePool - IDs remain unique across many allocations" {
422
+ var pool = GraphemePool.init(std.testing.allocator);
423
+ defer pool.deinit();
424
+
425
+ const count = 100;
426
+ var ids: [count]u32 = undefined;
427
+
428
+ for (0..count) |i| {
429
+ var buffer: [8]u8 = undefined;
430
+ const slice = std.fmt.bufPrint(&buffer, "{d}", .{i}) catch unreachable;
431
+ ids[i] = try pool.alloc(slice);
432
+ try pool.incref(ids[i]);
433
+ }
434
+
435
+ for (ids, 0..count) |id1, i| {
436
+ for (ids[i + 1 ..]) |id2| {
437
+ try std.testing.expect(id1 != id2);
438
+ }
439
+ }
440
+
441
+ for (ids) |id| {
442
+ try pool.decref(id);
443
+ }
444
+ }
445
+
446
+ test "GraphemePool - concurrent incref/decref maintains consistency" {
447
+ var pool = GraphemePool.init(std.testing.allocator);
448
+ defer pool.deinit();
449
+
450
+ const text = "test";
451
+ const id = try pool.alloc(text);
452
+
453
+ // Multiple incref/decref operations (starting from refcount 0)
454
+ try pool.incref(id);
455
+ try pool.incref(id);
456
+ try pool.incref(id);
457
+
458
+ // Should still be accessible (refcount is 3)
459
+ try std.testing.expectEqualSlices(u8, text, try pool.get(id));
460
+
461
+ try pool.decref(id);
462
+ try std.testing.expectEqualSlices(u8, text, try pool.get(id));
463
+
464
+ try pool.decref(id);
465
+ try std.testing.expectEqualSlices(u8, text, try pool.get(id));
466
+
467
+ // Final decref brings refcount to 0
468
+ try pool.decref(id);
469
+ }
470
+
471
+ test "GraphemePool - zero-length grapheme" {
472
+ var pool = GraphemePool.init(std.testing.allocator);
473
+ defer pool.deinit();
474
+
475
+ const empty: []const u8 = "";
476
+ const id = try pool.alloc(empty);
477
+ try pool.incref(id);
478
+
479
+ const retrieved = try pool.get(id);
480
+ try std.testing.expectEqual(@as(usize, 0), retrieved.len);
481
+
482
+ try pool.decref(id);
483
+ }
484
+
485
+ test "GraphemePool - incref on stale ID fails" {
486
+ var pool = GraphemePool.init(std.testing.allocator);
487
+ defer pool.deinit();
488
+
489
+ const text = "test";
490
+ const id = try pool.alloc(text);
491
+ try pool.incref(id);
492
+ try pool.decref(id);
493
+
494
+ // Allocate again to invalidate old ID
495
+ _ = try pool.alloc("new");
496
+
497
+ const result = pool.incref(id); // Old ID should fail due to wrong generation
498
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
499
+ }
500
+
501
+ test "GraphemePool - decref on stale ID fails" {
502
+ var pool = GraphemePool.init(std.testing.allocator);
503
+ defer pool.deinit();
504
+
505
+ const text = "test";
506
+ const id = try pool.alloc(text);
507
+
508
+ // Already at refcount 0, decref should fail
509
+ const result = pool.decref(id);
510
+ try std.testing.expectError(gp.GraphemePoolError.InvalidId, result);
511
+ }
512
+
513
+ test "GraphemePool - bit manipulation functions" {
514
+ const grapheme_char = gp.CHAR_FLAG_GRAPHEME | 0x1234;
515
+ try std.testing.expect(gp.isGraphemeChar(grapheme_char));
516
+ try std.testing.expect(!gp.isGraphemeChar(0x41)); // Plain 'A'
517
+
518
+ const cont_char = gp.CHAR_FLAG_CONTINUATION | 0x1234;
519
+ try std.testing.expect(gp.isContinuationChar(cont_char));
520
+ try std.testing.expect(!gp.isContinuationChar(0x41));
521
+
522
+ try std.testing.expect(gp.isClusterChar(grapheme_char));
523
+ try std.testing.expect(gp.isClusterChar(cont_char));
524
+ try std.testing.expect(!gp.isClusterChar(0x41));
525
+
526
+ const id: u32 = 0x12345;
527
+ const packed_char = gp.CHAR_FLAG_GRAPHEME | id;
528
+ try std.testing.expectEqual(id, gp.graphemeIdFromChar(packed_char));
529
+ }
530
+
531
+ test "GraphemePool - extent encoding and decoding" {
532
+ const right: u32 = 2;
533
+ const char_with_right = (right << gp.CHAR_EXT_RIGHT_SHIFT) | gp.CHAR_FLAG_GRAPHEME;
534
+ try std.testing.expectEqual(right, gp.charRightExtent(char_with_right));
535
+
536
+ const left: u32 = 1;
537
+ const char_with_left = (left << gp.CHAR_EXT_LEFT_SHIFT) | gp.CHAR_FLAG_GRAPHEME;
538
+ try std.testing.expectEqual(left, gp.charLeftExtent(char_with_left));
539
+ }
540
+
541
+ test "GraphemePool - packGraphemeStart" {
542
+ const gid: u32 = 0x1234;
543
+ const width: u32 = 2;
544
+
545
+ const packed_char = gp.packGraphemeStart(gid, width);
546
+
547
+ try std.testing.expect(gp.isGraphemeChar(packed_char));
548
+
549
+ try std.testing.expectEqual(gid, gp.graphemeIdFromChar(packed_char));
550
+
551
+ try std.testing.expectEqual(width - 1, gp.charRightExtent(packed_char));
552
+
553
+ try std.testing.expectEqual(@as(u32, 0), gp.charLeftExtent(packed_char));
554
+ }
555
+
556
+ test "GraphemePool - packContinuation" {
557
+ const gid: u32 = 0x1234;
558
+ const left: u32 = 1;
559
+ const right: u32 = 2;
560
+
561
+ const packed_char = gp.packContinuation(left, right, gid);
562
+
563
+ try std.testing.expect(gp.isContinuationChar(packed_char));
564
+
565
+ try std.testing.expectEqual(gid, gp.graphemeIdFromChar(packed_char));
566
+
567
+ try std.testing.expectEqual(left, gp.charLeftExtent(packed_char));
568
+ try std.testing.expectEqual(right, gp.charRightExtent(packed_char));
569
+ }
570
+
571
+ test "GraphemePool - encodedCharWidth" {
572
+ const single = @as(u32, 'A');
573
+ try std.testing.expectEqual(@as(u32, 1), gp.encodedCharWidth(single));
574
+
575
+ const grapheme_2 = gp.packGraphemeStart(0x1234, 2);
576
+ try std.testing.expectEqual(@as(u32, 2), gp.encodedCharWidth(grapheme_2));
577
+
578
+ const cont = gp.packContinuation(1, 1, 0x1234);
579
+ try std.testing.expectEqual(@as(u32, 3), gp.encodedCharWidth(cont));
580
+ }
581
+
582
+ test "GraphemeTracker - init and deinit" {
583
+ var pool = GraphemePool.init(std.testing.allocator);
584
+ defer pool.deinit();
585
+
586
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
587
+ defer tracker.deinit();
588
+
589
+ try std.testing.expect(!tracker.hasAny());
590
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
591
+ }
592
+
593
+ test "GraphemeTracker - add single grapheme" {
594
+ var pool = GraphemePool.init(std.testing.allocator);
595
+ defer pool.deinit();
596
+
597
+ const text = "a";
598
+ const id = try pool.alloc(text);
599
+
600
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
601
+ defer tracker.deinit();
602
+
603
+ tracker.add(id);
604
+
605
+ try std.testing.expect(tracker.hasAny());
606
+ try std.testing.expect(tracker.contains(id));
607
+ try std.testing.expectEqual(@as(u32, 1), tracker.getGraphemeCount());
608
+ }
609
+
610
+ test "GraphemeTracker - add multiple graphemes" {
611
+ var pool = GraphemePool.init(std.testing.allocator);
612
+ defer pool.deinit();
613
+
614
+ const text1 = "a";
615
+ const text2 = "b";
616
+ const text3 = "🌟";
617
+
618
+ const id1 = try pool.alloc(text1);
619
+ const id2 = try pool.alloc(text2);
620
+ const id3 = try pool.alloc(text3);
621
+
622
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
623
+ defer tracker.deinit();
624
+
625
+ tracker.add(id1);
626
+ tracker.add(id2);
627
+ tracker.add(id3);
628
+
629
+ try std.testing.expectEqual(@as(u32, 3), tracker.getGraphemeCount());
630
+ try std.testing.expect(tracker.contains(id1));
631
+ try std.testing.expect(tracker.contains(id2));
632
+ try std.testing.expect(tracker.contains(id3));
633
+ }
634
+
635
+ test "GraphemeTracker - add same grapheme twice increfs once" {
636
+ var pool = GraphemePool.init(std.testing.allocator);
637
+ defer pool.deinit();
638
+
639
+ const text = "a";
640
+ const id = try pool.alloc(text);
641
+
642
+ {
643
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
644
+ defer tracker.deinit();
645
+
646
+ tracker.add(id);
647
+ tracker.add(id); // Should not incref again
648
+
649
+ try std.testing.expectEqual(@as(u32, 1), tracker.getGraphemeCount());
650
+ try std.testing.expectEqual(@as(u32, 2), tracker.getGraphemeCellCount());
651
+ try std.testing.expectEqual(@as(u32, 2), tracker.getTotalGraphemeBytes());
652
+
653
+ tracker.remove(id);
654
+ try std.testing.expect(tracker.contains(id));
655
+ try std.testing.expectEqual(@as(u32, 1), tracker.getGraphemeCellCount());
656
+
657
+ // After deinit (via defer), tracker decrefs once, bringing refcount to 0
658
+ }
659
+
660
+ // Allocate new item to trigger slot reuse
661
+ const text2 = "b";
662
+ _ = try pool.alloc(text2);
663
+
664
+ // Old ID should now be invalid due to generation change
665
+ const result = pool.get(id);
666
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
667
+ }
668
+
669
+ test "GraphemeTracker - remove grapheme" {
670
+ var pool = GraphemePool.init(std.testing.allocator);
671
+ defer pool.deinit();
672
+
673
+ const text = "a";
674
+ const id = try pool.alloc(text);
675
+
676
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
677
+ defer tracker.deinit();
678
+
679
+ tracker.add(id);
680
+ try std.testing.expect(tracker.contains(id));
681
+
682
+ tracker.remove(id);
683
+ try std.testing.expect(!tracker.contains(id));
684
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
685
+ }
686
+
687
+ test "GraphemeTracker - remove non-existent grapheme is safe" {
688
+ var pool = GraphemePool.init(std.testing.allocator);
689
+ defer pool.deinit();
690
+
691
+ const text = "a";
692
+ const id = try pool.alloc(text);
693
+
694
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
695
+ defer tracker.deinit();
696
+
697
+ // Remove without adding - should be safe
698
+ tracker.remove(id);
699
+
700
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
701
+ }
702
+
703
+ test "GraphemeTracker - clear removes all graphemes" {
704
+ var pool = GraphemePool.init(std.testing.allocator);
705
+ defer pool.deinit();
706
+
707
+ const text1 = "a";
708
+ const text2 = "b";
709
+ const id1 = try pool.alloc(text1);
710
+ const id2 = try pool.alloc(text2);
711
+
712
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
713
+ defer tracker.deinit();
714
+
715
+ tracker.add(id1);
716
+ tracker.add(id2);
717
+ try std.testing.expectEqual(@as(u32, 2), tracker.getGraphemeCount());
718
+
719
+ tracker.clear();
720
+
721
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
722
+ try std.testing.expect(!tracker.contains(id1));
723
+ try std.testing.expect(!tracker.contains(id2));
724
+ try std.testing.expect(!tracker.hasAny());
725
+ }
726
+
727
+ test "GraphemeTracker - getTotalGraphemeBytes" {
728
+ var pool = GraphemePool.init(std.testing.allocator);
729
+ defer pool.deinit();
730
+
731
+ const text1 = "a"; // 1 byte
732
+ const text2 = "🌟"; // 4 bytes
733
+ const text3 = "test"; // 4 bytes
734
+
735
+ const id1 = try pool.alloc(text1);
736
+ const id2 = try pool.alloc(text2);
737
+ const id3 = try pool.alloc(text3);
738
+
739
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
740
+ defer tracker.deinit();
741
+
742
+ tracker.add(id1);
743
+ tracker.add(id2);
744
+ tracker.add(id3);
745
+
746
+ const total_bytes = tracker.getTotalGraphemeBytes();
747
+ try std.testing.expectEqual(@as(u32, 1 + 4 + 4), total_bytes);
748
+ }
749
+
750
+ test "GraphemeTracker - tracker keeps graphemes alive" {
751
+ var pool = GraphemePool.init(std.testing.allocator);
752
+ defer pool.deinit();
753
+
754
+ const text = "test";
755
+ const id = try pool.alloc(text);
756
+
757
+ {
758
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
759
+ defer tracker.deinit();
760
+
761
+ tracker.add(id);
762
+
763
+ // Should be accessible because tracker holds a reference (refcount is 1)
764
+ const retrieved = try pool.get(id);
765
+ try std.testing.expectEqualSlices(u8, text, retrieved);
766
+
767
+ // After tracker deinit (via defer), refcount will be 0
768
+ }
769
+
770
+ // Allocate new item to trigger slot reuse with new generation
771
+ const text2 = "x";
772
+ _ = try pool.alloc(text2);
773
+
774
+ // Old ID should fail due to generation mismatch
775
+ const result = pool.get(id);
776
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
777
+ }
778
+
779
+ test "GraphemeTracker - multiple trackers share same grapheme" {
780
+ var pool = GraphemePool.init(std.testing.allocator);
781
+ defer pool.deinit();
782
+
783
+ const text = "shared";
784
+ const id = try pool.alloc(text);
785
+
786
+ {
787
+ var tracker1 = GraphemeTracker.init(std.testing.allocator, &pool);
788
+ defer tracker1.deinit();
789
+
790
+ {
791
+ var tracker2 = GraphemeTracker.init(std.testing.allocator, &pool);
792
+ defer tracker2.deinit();
793
+
794
+ tracker1.add(id);
795
+ tracker2.add(id);
796
+
797
+ try std.testing.expect(tracker1.contains(id));
798
+ try std.testing.expect(tracker2.contains(id));
799
+
800
+ // Should be accessible (ref count is 2 from both trackers)
801
+ const retrieved = try pool.get(id);
802
+ try std.testing.expectEqualSlices(u8, text, retrieved);
803
+
804
+ // tracker2 deinit via defer here (decrefs to 1)
805
+ }
806
+
807
+ // Should still be accessible (ref count is 1)
808
+ const retrieved2 = try pool.get(id);
809
+ try std.testing.expectEqualSlices(u8, text, retrieved2);
810
+
811
+ // tracker1 deinit via defer here (decrefs to 0)
812
+ }
813
+
814
+ // Allocate new item to trigger slot reuse with new generation
815
+ const text2 = "y";
816
+ _ = try pool.alloc(text2);
817
+
818
+ // Old ID should fail due to generation mismatch
819
+ const result = pool.get(id);
820
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
821
+ }
822
+
823
+ test "GraphemeTracker - stress test many graphemes" {
824
+ var pool = GraphemePool.init(std.testing.allocator);
825
+ defer pool.deinit();
826
+
827
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
828
+ defer tracker.deinit();
829
+
830
+ const count = 500;
831
+ var ids: [count]u32 = undefined;
832
+
833
+ // Add many graphemes
834
+ for (0..count) |i| {
835
+ var buffer: [8]u8 = undefined;
836
+ const slice = std.fmt.bufPrint(&buffer, "{d}", .{i}) catch unreachable;
837
+ ids[i] = try pool.alloc(slice);
838
+ tracker.add(ids[i]);
839
+ }
840
+
841
+ try std.testing.expectEqual(@as(u32, count), tracker.getGraphemeCount());
842
+
843
+ // Verify all are tracked
844
+ for (ids) |id| {
845
+ try std.testing.expect(tracker.contains(id));
846
+ }
847
+
848
+ // Clear should remove all
849
+ tracker.clear();
850
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
851
+
852
+ for (ids) |id| {
853
+ try std.testing.expect(!tracker.contains(id));
854
+ }
855
+ }
856
+
857
+ test "GraphemePool - global pool init and deinit" {
858
+ const pool = gp.initGlobalPool(std.testing.allocator);
859
+ defer gp.deinitGlobalPool();
860
+
861
+ const text = "test";
862
+ const id = try pool.alloc(text);
863
+ try pool.incref(id);
864
+
865
+ const retrieved = try pool.get(id);
866
+ try std.testing.expectEqualSlices(u8, text, retrieved);
867
+
868
+ try pool.decref(id);
869
+ }
870
+
871
+ test "GraphemePool - global pool reinitialization returns same instance" {
872
+ const pool1 = gp.initGlobalPool(std.testing.allocator);
873
+ const pool2 = gp.initGlobalPool(std.testing.allocator);
874
+
875
+ try std.testing.expectEqual(pool1, pool2);
876
+
877
+ gp.deinitGlobalPool();
878
+ }
879
+
880
+ test "GraphemePool - global unicode data init" {
881
+
882
+ // Pointers should not be null (just verify they're returned)
883
+ // We can't easily test their validity without using them
884
+ }
885
+
886
+ test "GraphemePool - allocUnowned basic" {
887
+ var pool = GraphemePool.init(std.testing.allocator);
888
+ defer pool.deinit();
889
+
890
+ // External memory that we manage
891
+ const external_text = "external";
892
+ const id = try pool.allocUnowned(external_text);
893
+ try pool.incref(id);
894
+
895
+ const retrieved = try pool.get(id);
896
+ try std.testing.expectEqualSlices(u8, external_text, retrieved);
897
+
898
+ // Verify it's actually pointing to the same memory location
899
+ try std.testing.expectEqual(@intFromPtr(external_text.ptr), @intFromPtr(retrieved.ptr));
900
+
901
+ try pool.decref(id);
902
+ }
903
+
904
+ test "GraphemePool - allocUnowned multiple references" {
905
+ var pool = GraphemePool.init(std.testing.allocator);
906
+ defer pool.deinit();
907
+
908
+ const external_text1 = "external1";
909
+ const external_text2 = "external2";
910
+ const external_text3 = "external3";
911
+
912
+ const id1 = try pool.allocUnowned(external_text1);
913
+ const id2 = try pool.allocUnowned(external_text2);
914
+ const id3 = try pool.allocUnowned(external_text3);
915
+ try pool.incref(id1);
916
+ try pool.incref(id2);
917
+ try pool.incref(id3);
918
+
919
+ try std.testing.expectEqualSlices(u8, external_text1, try pool.get(id1));
920
+ try std.testing.expectEqualSlices(u8, external_text2, try pool.get(id2));
921
+ try std.testing.expectEqualSlices(u8, external_text3, try pool.get(id3));
922
+
923
+ // Verify they point to original memory
924
+ try std.testing.expectEqual(@intFromPtr(external_text1.ptr), @intFromPtr((try pool.get(id1)).ptr));
925
+ try std.testing.expectEqual(@intFromPtr(external_text2.ptr), @intFromPtr((try pool.get(id2)).ptr));
926
+ try std.testing.expectEqual(@intFromPtr(external_text3.ptr), @intFromPtr((try pool.get(id3)).ptr));
927
+
928
+ try pool.decref(id1);
929
+ try pool.decref(id2);
930
+ try pool.decref(id3);
931
+ }
932
+
933
+ test "GraphemePool - allocUnowned with emoji" {
934
+ var pool = GraphemePool.init(std.testing.allocator);
935
+ defer pool.deinit();
936
+
937
+ const external_emoji = "🌟🎉🚀";
938
+ const id = try pool.allocUnowned(external_emoji);
939
+ try pool.incref(id);
940
+
941
+ const retrieved = try pool.get(id);
942
+ try std.testing.expectEqualSlices(u8, external_emoji, retrieved);
943
+ try std.testing.expectEqual(@intFromPtr(external_emoji.ptr), @intFromPtr(retrieved.ptr));
944
+
945
+ try pool.decref(id);
946
+ }
947
+
948
+ test "GraphemePool - allocUnowned refcounting" {
949
+ var pool = GraphemePool.init(std.testing.allocator);
950
+ defer pool.deinit();
951
+
952
+ const external_text = "refcount_test";
953
+ const id = try pool.allocUnowned(external_text);
954
+
955
+ // Increment refcount (starting from 0)
956
+ try pool.incref(id);
957
+ try pool.incref(id);
958
+ try pool.incref(id);
959
+
960
+ // Should still be accessible (refcount is 3)
961
+ try std.testing.expectEqualSlices(u8, external_text, try pool.get(id));
962
+
963
+ // Decrement
964
+ try pool.decref(id);
965
+ try std.testing.expectEqualSlices(u8, external_text, try pool.get(id));
966
+
967
+ try pool.decref(id);
968
+ try std.testing.expectEqualSlices(u8, external_text, try pool.get(id));
969
+
970
+ // Final decref
971
+ try pool.decref(id);
972
+ }
973
+
974
+ test "GraphemePool - mix owned and unowned allocations" {
975
+ var pool = GraphemePool.init(std.testing.allocator);
976
+ defer pool.deinit();
977
+
978
+ const owned_text = "owned";
979
+ const external_text = "unowned";
980
+
981
+ const owned_id = try pool.alloc(owned_text);
982
+ const unowned_id = try pool.allocUnowned(external_text);
983
+ try pool.incref(owned_id);
984
+ try pool.incref(unowned_id);
985
+
986
+ const retrieved_owned = try pool.get(owned_id);
987
+ const retrieved_unowned = try pool.get(unowned_id);
988
+
989
+ try std.testing.expectEqualSlices(u8, owned_text, retrieved_owned);
990
+ try std.testing.expectEqualSlices(u8, external_text, retrieved_unowned);
991
+
992
+ // Owned should be different memory location (copy)
993
+ try std.testing.expect(@intFromPtr(owned_text.ptr) != @intFromPtr(retrieved_owned.ptr));
994
+
995
+ // Unowned should be same memory location (reference)
996
+ try std.testing.expectEqual(@intFromPtr(external_text.ptr), @intFromPtr(retrieved_unowned.ptr));
997
+
998
+ try pool.decref(owned_id);
999
+ try pool.decref(unowned_id);
1000
+ }
1001
+
1002
+ test "GraphemePool - allocUnowned slot reuse" {
1003
+ var pool = GraphemePool.init(std.testing.allocator);
1004
+ defer pool.deinit();
1005
+
1006
+ const text1 = "first";
1007
+ const id1 = try pool.allocUnowned(text1);
1008
+ try pool.incref(id1);
1009
+ try pool.decref(id1);
1010
+
1011
+ // Allocate again - should reuse slot
1012
+ const text2 = "second";
1013
+ const id2 = try pool.allocUnowned(text2);
1014
+
1015
+ const result = pool.get(id1);
1016
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
1017
+
1018
+ try pool.incref(id2);
1019
+ const retrieved = try pool.get(id2);
1020
+ try std.testing.expectEqualSlices(u8, text2, retrieved);
1021
+ try std.testing.expectEqual(@intFromPtr(text2.ptr), @intFromPtr(retrieved.ptr));
1022
+
1023
+ try pool.decref(id2);
1024
+ }
1025
+
1026
+ test "GraphemePool - allocUnowned large text" {
1027
+ var pool = GraphemePool.init(std.testing.allocator);
1028
+ defer pool.deinit();
1029
+
1030
+ // Large external buffer
1031
+ var large_buffer: [1000]u8 = undefined;
1032
+ @memset(&large_buffer, 'X');
1033
+ const large_slice: []const u8 = &large_buffer;
1034
+
1035
+ const id = try pool.allocUnowned(large_slice);
1036
+ try pool.incref(id);
1037
+
1038
+ const retrieved = try pool.get(id);
1039
+ try std.testing.expectEqual(@as(usize, 1000), retrieved.len);
1040
+ try std.testing.expectEqualSlices(u8, large_slice, retrieved);
1041
+ try std.testing.expectEqual(@intFromPtr(large_slice.ptr), @intFromPtr(retrieved.ptr));
1042
+
1043
+ try pool.decref(id);
1044
+ }
1045
+
1046
+ test "GraphemePool - alloc does not reuse unowned IDs" {
1047
+ var pool = GraphemePool.init(std.testing.allocator);
1048
+ defer pool.deinit();
1049
+
1050
+ const external_text = "shared";
1051
+
1052
+ const unowned_id = try pool.allocUnowned(external_text);
1053
+ try pool.incref(unowned_id);
1054
+ defer pool.decref(unowned_id) catch {};
1055
+
1056
+ const owned_id = try pool.alloc(external_text);
1057
+ try pool.incref(owned_id);
1058
+ defer pool.decref(owned_id) catch {};
1059
+
1060
+ try std.testing.expect(owned_id != unowned_id);
1061
+
1062
+ const owned_bytes = try pool.get(owned_id);
1063
+ try std.testing.expectEqualSlices(u8, external_text, owned_bytes);
1064
+ try std.testing.expect(@intFromPtr(owned_bytes.ptr) != @intFromPtr(external_text.ptr));
1065
+ }
1066
+
1067
+ test "GraphemeTracker - with unowned allocations" {
1068
+ var pool = GraphemePool.init(std.testing.allocator);
1069
+ defer pool.deinit();
1070
+
1071
+ const text1 = "external1";
1072
+ const text2 = "external2";
1073
+
1074
+ const id1 = try pool.allocUnowned(text1);
1075
+ const id2 = try pool.allocUnowned(text2);
1076
+
1077
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
1078
+ defer tracker.deinit();
1079
+
1080
+ tracker.add(id1);
1081
+ tracker.add(id2);
1082
+
1083
+ try std.testing.expectEqual(@as(u32, 2), tracker.getGraphemeCount());
1084
+ try std.testing.expect(tracker.contains(id1));
1085
+ try std.testing.expect(tracker.contains(id2));
1086
+
1087
+ // Should still get correct bytes
1088
+ try std.testing.expectEqualSlices(u8, text1, try pool.get(id1));
1089
+ try std.testing.expectEqualSlices(u8, text2, try pool.get(id2));
1090
+ }
1091
+
1092
+ test "GraphemeTracker - mix owned and unowned" {
1093
+ var pool = GraphemePool.init(std.testing.allocator);
1094
+ defer pool.deinit();
1095
+
1096
+ const owned_text = "owned_data";
1097
+ const external_text = "external_data";
1098
+
1099
+ const owned_id = try pool.alloc(owned_text);
1100
+ const unowned_id = try pool.allocUnowned(external_text);
1101
+
1102
+ var tracker = GraphemeTracker.init(std.testing.allocator, &pool);
1103
+ defer tracker.deinit();
1104
+
1105
+ tracker.add(owned_id);
1106
+ tracker.add(unowned_id);
1107
+
1108
+ try std.testing.expectEqual(@as(u32, 2), tracker.getGraphemeCount());
1109
+
1110
+ const total_bytes = tracker.getTotalGraphemeBytes();
1111
+ try std.testing.expectEqual(@as(u32, owned_text.len + external_text.len), total_bytes);
1112
+
1113
+ try std.testing.expectEqualSlices(u8, owned_text, try pool.get(owned_id));
1114
+ try std.testing.expectEqualSlices(u8, external_text, try pool.get(unowned_id));
1115
+ }
1116
+
1117
+ test "GraphemePool - allocUnowned with stack memory" {
1118
+ var pool = GraphemePool.init(std.testing.allocator);
1119
+ defer pool.deinit();
1120
+
1121
+ // Simulate stack-allocated buffer
1122
+ var stack_buffer: [50]u8 = undefined;
1123
+ @memcpy(stack_buffer[0..11], "stack_based");
1124
+ const stack_slice = stack_buffer[0..11];
1125
+
1126
+ const id = try pool.allocUnowned(stack_slice);
1127
+ try pool.incref(id);
1128
+
1129
+ const retrieved = try pool.get(id);
1130
+ try std.testing.expectEqualSlices(u8, "stack_based", retrieved);
1131
+ try std.testing.expectEqual(@intFromPtr(stack_slice.ptr), @intFromPtr(retrieved.ptr));
1132
+
1133
+ try pool.decref(id);
1134
+ // Note: In real usage, caller must ensure stack_buffer stays valid while ID is in use
1135
+ }
1136
+
1137
+ test "GraphemePool - allocUnowned zero-length slice" {
1138
+ var pool = GraphemePool.init(std.testing.allocator);
1139
+ defer pool.deinit();
1140
+
1141
+ const empty: []const u8 = "";
1142
+ const id = try pool.allocUnowned(empty);
1143
+ try pool.incref(id);
1144
+
1145
+ const retrieved = try pool.get(id);
1146
+ try std.testing.expectEqual(@as(usize, 0), retrieved.len);
1147
+
1148
+ try pool.decref(id);
1149
+ }
1150
+
1151
+ test "GraphemePool - initWithOptions with small slots_per_page" {
1152
+ // Create a pool with very small slots_per_page to test exhaustion
1153
+ const small_slots = [_]u32{ 2, 2, 2, 2, 2 }; // Only 2 slots per page for each class
1154
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1155
+ .slots_per_page = small_slots,
1156
+ });
1157
+ defer pool.deinit();
1158
+
1159
+ const id1 = try pool.alloc("abc");
1160
+ const id2 = try pool.alloc("def");
1161
+ try pool.incref(id1);
1162
+ try pool.incref(id2);
1163
+
1164
+ try std.testing.expectEqualSlices(u8, "abc", try pool.get(id1));
1165
+ try std.testing.expectEqualSlices(u8, "def", try pool.get(id2));
1166
+
1167
+ try pool.decref(id1);
1168
+ try pool.decref(id2);
1169
+ }
1170
+
1171
+ test "GraphemePool - alloc reuses live ID for same bytes" {
1172
+ const tiny_slots = [_]u32{ 1, 1, 1, 1, 1 };
1173
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1174
+ .slots_per_page = tiny_slots,
1175
+ });
1176
+ defer pool.deinit();
1177
+
1178
+ const grapheme = "👋";
1179
+
1180
+ const id1 = try pool.alloc(grapheme);
1181
+ try pool.incref(id1);
1182
+
1183
+ const id2 = try pool.alloc(grapheme);
1184
+ try std.testing.expectEqual(id1, id2);
1185
+ try std.testing.expectEqual(@as(u32, 1), try pool.getRefcount(id1));
1186
+
1187
+ try pool.decref(id1);
1188
+
1189
+ const id3 = try pool.alloc(grapheme);
1190
+ try pool.incref(id3);
1191
+ defer pool.decref(id3) catch @panic("Failed to decref id3");
1192
+
1193
+ try std.testing.expect(id3 != id1);
1194
+ try std.testing.expectEqualSlices(u8, grapheme, try pool.get(id3));
1195
+
1196
+ const id4 = try pool.alloc(grapheme);
1197
+ try std.testing.expectEqual(id3, id4);
1198
+ }
1199
+
1200
+ test "GraphemePool - small pool exhaustion and growth" {
1201
+ // Create a tiny pool that will need to grow
1202
+ const tiny_slots = [_]u32{ 1, 1, 1, 1, 1 }; // Only 1 slot per page initially
1203
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1204
+ .slots_per_page = tiny_slots,
1205
+ });
1206
+ defer pool.deinit();
1207
+
1208
+ // Allocate first item - uses initial page
1209
+ const id1 = try pool.alloc("a");
1210
+
1211
+ // Allocate second item - should trigger growth (new page)
1212
+ const id2 = try pool.alloc("b");
1213
+ try pool.incref(id1);
1214
+ try pool.incref(id2);
1215
+
1216
+ try std.testing.expectEqualSlices(u8, "a", try pool.get(id1));
1217
+ try std.testing.expectEqualSlices(u8, "b", try pool.get(id2));
1218
+
1219
+ try pool.decref(id1);
1220
+ try pool.decref(id2);
1221
+ }
1222
+
1223
+ test "GraphemePool - small pool with refcount prevents exhaustion" {
1224
+ const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
1225
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1226
+ .slots_per_page = tiny_slots,
1227
+ });
1228
+ defer pool.deinit();
1229
+
1230
+ // Allocate 2 items (fills the first page)
1231
+ const id1 = try pool.alloc("aa");
1232
+ const id2 = try pool.alloc("bb");
1233
+ try pool.incref(id1);
1234
+ try pool.incref(id2);
1235
+
1236
+ // Free one
1237
+ try pool.decref(id1);
1238
+
1239
+ const id3 = try pool.alloc("cc");
1240
+ try pool.incref(id3);
1241
+
1242
+ try std.testing.expectEqualSlices(u8, "bb", try pool.get(id2));
1243
+ try std.testing.expectEqualSlices(u8, "cc", try pool.get(id3));
1244
+
1245
+ // Old id1 should be invalid due to generation change
1246
+ const result = pool.get(id1);
1247
+ try std.testing.expectError(gp.GraphemePoolError.WrongGeneration, result);
1248
+
1249
+ try pool.decref(id2);
1250
+ try pool.decref(id3);
1251
+ }
1252
+
1253
+ test "GraphemePool - different size classes with small limits" {
1254
+ const tiny_slots = [_]u32{ 2, 2, 2, 2, 2 };
1255
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1256
+ .slots_per_page = tiny_slots,
1257
+ });
1258
+ defer pool.deinit();
1259
+
1260
+ // Allocate different sizes (should use different classes)
1261
+ const id_small = try pool.alloc("ab"); // 2 bytes -> class 0 (8-byte slots)
1262
+ const id_medium = try pool.alloc("0123456789abc"); // 13 bytes -> class 1 (16-byte slots)
1263
+ const id_large = try pool.alloc("012345678901234567890"); // 21 bytes -> class 2 (32-byte slots)
1264
+ try pool.incref(id_small);
1265
+ try pool.incref(id_medium);
1266
+ try pool.incref(id_large);
1267
+
1268
+ try std.testing.expectEqualSlices(u8, "ab", try pool.get(id_small));
1269
+ try std.testing.expectEqualSlices(u8, "0123456789abc", try pool.get(id_medium));
1270
+ try std.testing.expectEqualSlices(u8, "012345678901234567890", try pool.get(id_large));
1271
+
1272
+ try pool.decref(id_small);
1273
+ try pool.decref(id_medium);
1274
+ try pool.decref(id_large);
1275
+ }
1276
+
1277
+ test "GraphemePool - tracker with small pool" {
1278
+ const tiny_slots = [_]u32{ 3, 3, 3, 3, 3 };
1279
+ var pool = gp.GraphemePool.initWithOptions(std.testing.allocator, .{
1280
+ .slots_per_page = tiny_slots,
1281
+ });
1282
+ defer pool.deinit();
1283
+
1284
+ var tracker = gp.GraphemeTracker.init(std.testing.allocator, &pool);
1285
+ defer tracker.deinit();
1286
+
1287
+ // Add multiple graphemes
1288
+ const id1 = try pool.alloc("🌟");
1289
+ const id2 = try pool.alloc("🎨");
1290
+ const id3 = try pool.alloc("🚀");
1291
+
1292
+ tracker.add(id1);
1293
+ tracker.add(id2);
1294
+ tracker.add(id3);
1295
+
1296
+ try std.testing.expectEqual(@as(u32, 3), tracker.getGraphemeCount());
1297
+
1298
+ // Clear tracker should free all refs
1299
+ tracker.clear();
1300
+ try std.testing.expectEqual(@as(u32, 0), tracker.getGraphemeCount());
1301
+
1302
+ // After tracker.clear(), the graphemes have been decref'd by tracker
1303
+ // Since alloc() starts with refcount 0, after tracker decrefs, they're freed
1304
+ }