@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,599 @@
1
+ const std = @import("std");
2
+
3
+ pub const GraphemePoolError = error{
4
+ OutOfMemory,
5
+ InvalidId,
6
+ WrongGeneration,
7
+ };
8
+
9
+ // Encoding flags for char buffer entries (u32)
10
+ // Bits 31-30: encoding type
11
+ // 00xxxxxxxx: direct unicode scalar value (30 bits, as-is)
12
+ // 10xxxxxxxx: grapheme start cell with pool ID (26 bits total payload)
13
+ // 11xxxxxxxx: continuation cell marker for wide/grapheme rendering
14
+ pub const CHAR_FLAG_GRAPHEME: u32 = 0x8000_0000;
15
+ pub const CHAR_FLAG_CONTINUATION: u32 = 0xC000_0000;
16
+
17
+ // For grapheme start and continuation cells:
18
+ // Bits 29..28: right extent (u2), Bits 27..26: left extent (u2)
19
+ pub const CHAR_EXT_RIGHT_SHIFT: u5 = 28;
20
+ pub const CHAR_EXT_LEFT_SHIFT: u5 = 26;
21
+ pub const CHAR_EXT_MASK: u32 = 0x3;
22
+
23
+ // Grapheme ID payload layout (26 bits total):
24
+ // [ class (3 bits) | generation (7 bits) | slot_index (16 bits) ]
25
+ pub const GRAPHEME_ID_MASK: u32 = 0x03FF_FFFF;
26
+ pub const CLASS_BITS: u5 = 3;
27
+ pub const GENERATION_BITS: u5 = 7;
28
+ pub const SLOT_BITS: u5 = 16;
29
+ pub const CLASS_MASK: u32 = (@as(u32, 1) << CLASS_BITS) - 1; // 0b111
30
+ pub const GENERATION_MASK: u32 = (@as(u32, 1) << GENERATION_BITS) - 1; // 0b1111111
31
+ pub const SLOT_MASK: u32 = (@as(u32, 1) << SLOT_BITS) - 1; // 0xFFFF
32
+
33
+ /// Global slab-allocated pool for grapheme clusters (byte slices)
34
+ /// This is total overkill probably, but fun
35
+ /// ID layout (26-bit payload):
36
+ /// [ class (3 bits) | generation (7 bits) | slot_index (16 bits) ]
37
+ pub const GraphemePool = struct {
38
+ const MAX_CLASSES: u5 = 5; // 0..4 => 8,16,32,64,128
39
+ const CLASS_SIZES = [_]u32{ 8, 16, 32, 64, 128 };
40
+ const DEFAULT_SLOTS_PER_PAGE = [_]u32{ 256, 128, 64, 16, 8 };
41
+
42
+ pub const IdPayload = u32;
43
+
44
+ pub const InitOptions = struct {
45
+ /// Slots per page for each size class. If null, uses DEFAULT_SLOTS_PER_PAGE.
46
+ /// Used to limit pool size for testing.
47
+ slots_per_page: ?[MAX_CLASSES]u32 = null,
48
+ };
49
+
50
+ allocator: std.mem.Allocator,
51
+ classes: [MAX_CLASSES]ClassPool,
52
+ interned_live_ids: std.StringHashMapUnmanaged(IdPayload),
53
+
54
+ const SlotHeader = extern struct {
55
+ len: u16,
56
+ refcount: u32,
57
+ generation: u32,
58
+ is_owned: u32, // 0 = unowned (external memory), 1 = owned (copied into pool)
59
+ };
60
+
61
+ pub fn init(allocator: std.mem.Allocator) GraphemePool {
62
+ return initWithOptions(allocator, .{});
63
+ }
64
+
65
+ pub fn initWithOptions(allocator: std.mem.Allocator, options: InitOptions) GraphemePool {
66
+ const slots_per_page = options.slots_per_page orelse DEFAULT_SLOTS_PER_PAGE;
67
+ var classes: [MAX_CLASSES]ClassPool = undefined;
68
+ var i: usize = 0;
69
+ while (i < MAX_CLASSES) : (i += 1) {
70
+ classes[i] = ClassPool.init(allocator, CLASS_SIZES[i], slots_per_page[i]);
71
+ }
72
+ return .{ .allocator = allocator, .classes = classes, .interned_live_ids = .{} };
73
+ }
74
+
75
+ pub fn deinit(self: *GraphemePool) void {
76
+ var key_it = self.interned_live_ids.keyIterator();
77
+ while (key_it.next()) |key_ptr| {
78
+ self.allocator.free(@constCast(key_ptr.*));
79
+ }
80
+ self.interned_live_ids.deinit(self.allocator);
81
+
82
+ var i: usize = 0;
83
+ while (i < MAX_CLASSES) : (i += 1) {
84
+ self.classes[i].deinit();
85
+ }
86
+ }
87
+
88
+ /// removeInternedLiveId removes an interned ID from the live set if it
89
+ /// matches the expected ID.
90
+ fn removeInternedLiveId(self: *GraphemePool, bytes: []const u8, expected_id: IdPayload) void {
91
+ const live_id = self.interned_live_ids.get(bytes) orelse return;
92
+ if (live_id != expected_id) return;
93
+ if (self.interned_live_ids.fetchRemove(bytes)) |removed| {
94
+ self.allocator.free(@constCast(removed.key));
95
+ }
96
+ }
97
+
98
+ /// lookupOrInvalidate checks if the given bytes are already interned and live, returning the existing ID if so.
99
+ fn lookupOrInvalidate(self: *GraphemePool, bytes: []const u8) ?IdPayload {
100
+ const live_id = self.interned_live_ids.get(bytes) orelse return null;
101
+
102
+ // Verify that the live ID is still valid and matches the bytes. If get
103
+ // fails, the ID is no longer valid, so remove it from the interned map.
104
+ const live_bytes = self.get(live_id) catch {
105
+ self.removeInternedLiveId(bytes, live_id);
106
+ return null;
107
+ };
108
+
109
+ // If the bytes don't match, this means the ID was recycled and now points
110
+ // to different data. Invalidate the interned ID.
111
+ if (!std.mem.eql(u8, live_bytes, bytes)) {
112
+ self.removeInternedLiveId(bytes, live_id);
113
+ return null;
114
+ }
115
+
116
+ // check refcount > 0 to ensure the ID is still live. If refcount is 0,
117
+ // the slot is free but hasn't been reused yet, so we can treat it as
118
+ // not found.
119
+ const live_refcount = self.getRefcount(live_id) catch {
120
+ self.removeInternedLiveId(bytes, live_id);
121
+ return null;
122
+ };
123
+ if (live_refcount == 0) {
124
+ self.removeInternedLiveId(bytes, live_id);
125
+ return null;
126
+ }
127
+
128
+ return live_id;
129
+ }
130
+
131
+ /// internLiveId interns the grapheme bytes.
132
+ fn internLiveId(self: *GraphemePool, id: IdPayload, bytes: []const u8) GraphemePoolError!void {
133
+ if (self.lookupOrInvalidate(bytes) != null) {
134
+ // Keep existing interned ID if it's still valid.
135
+ return;
136
+ }
137
+
138
+ const owned_key = self.allocator.dupe(u8, bytes) catch return GraphemePoolError.OutOfMemory;
139
+ errdefer self.allocator.free(owned_key);
140
+
141
+ if (self.interned_live_ids.fetchPut(self.allocator, owned_key, id) catch return GraphemePoolError.OutOfMemory) |replaced| {
142
+ // A previous key allocation was replaced.
143
+ self.allocator.free(@constCast(replaced.key));
144
+ }
145
+ }
146
+
147
+ fn classForSize(size: usize) u32 {
148
+ if (size <= 8) return 0;
149
+ if (size <= 16) return 1;
150
+ if (size <= 32) return 2;
151
+ if (size <= 64) return 3;
152
+ return 4; // up to 128
153
+ }
154
+
155
+ fn packId(class_id: u32, slot_index: u32, generation: u32) GraphemePoolError!IdPayload {
156
+ if (slot_index > SLOT_MASK) return GraphemePoolError.OutOfMemory;
157
+ return (class_id << (GENERATION_BITS + SLOT_BITS)) |
158
+ ((generation & GENERATION_MASK) << SLOT_BITS) |
159
+ (slot_index & SLOT_MASK);
160
+ }
161
+
162
+ pub fn alloc(self: *GraphemePool, bytes: []const u8) GraphemePoolError!IdPayload {
163
+ if (self.lookupOrInvalidate(bytes)) |live_id| {
164
+ return live_id;
165
+ }
166
+
167
+ const class_id: u32 = classForSize(bytes.len);
168
+ const slot_index = try self.classes[class_id].allocInternal(bytes, true);
169
+ const generation = self.classes[class_id].getGeneration(slot_index);
170
+ return try packId(class_id, slot_index, generation);
171
+ }
172
+
173
+ /// Allocate an ID for externally managed memory (no copy, just reference)
174
+ /// The caller is responsible for keeping the memory valid while the ID is in use
175
+ pub fn allocUnowned(self: *GraphemePool, bytes: []const u8) GraphemePoolError!IdPayload {
176
+ // For unowned allocations, we need space for a pointer
177
+ const ptr_size = @sizeOf(usize);
178
+ const class_id: u32 = classForSize(ptr_size);
179
+ const slot_index = try self.classes[class_id].allocInternal(bytes, false);
180
+ const generation = self.classes[class_id].getGeneration(slot_index);
181
+ return try packId(class_id, slot_index, generation);
182
+ }
183
+
184
+ pub fn incref(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
185
+ const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
186
+ if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
187
+ const slot_index: u32 = id & SLOT_MASK;
188
+ const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
189
+ const old_refcount = try self.classes[class_id].getRefcount(slot_index, generation);
190
+ try self.classes[class_id].incref(slot_index, generation);
191
+
192
+ if (old_refcount == 0) {
193
+ const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
194
+ if (is_owned) {
195
+ // This is a transition from 0 to 1 for owned bytes, so intern it.
196
+ const bytes = try self.classes[class_id].get(slot_index, generation);
197
+ try self.internLiveId(id, bytes);
198
+ }
199
+ }
200
+ }
201
+
202
+ pub fn decref(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
203
+ const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
204
+ if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
205
+ const slot_index: u32 = id & SLOT_MASK;
206
+ const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
207
+
208
+ const old_refcount = try self.classes[class_id].getRefcount(slot_index, generation);
209
+ if (old_refcount == 1) {
210
+ const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
211
+ if (is_owned) {
212
+ // This is a transition from 1 to 0 for owned bytes, remove map entry.
213
+ const bytes = try self.classes[class_id].get(slot_index, generation);
214
+ self.removeInternedLiveId(bytes, id);
215
+ }
216
+ }
217
+
218
+ try self.classes[class_id].decref(slot_index, generation);
219
+ }
220
+
221
+ /// Free a freshly allocated slot that was never incref'd (refcount=0).
222
+ /// Use this for cleanup when allocation succeeded but the slot was never used.
223
+ /// This prevents slot leaks when an error occurs between alloc and incref.
224
+ pub fn freeUnreferenced(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
225
+ const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
226
+ if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
227
+ const slot_index: u32 = id & SLOT_MASK;
228
+ const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
229
+
230
+ const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
231
+ if (is_owned) {
232
+ const bytes = try self.classes[class_id].get(slot_index, generation);
233
+ self.removeInternedLiveId(bytes, id);
234
+ }
235
+
236
+ try self.classes[class_id].freeUnreferenced(slot_index, generation);
237
+ }
238
+
239
+ pub fn get(self: *GraphemePool, id: IdPayload) GraphemePoolError![]const u8 {
240
+ const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
241
+ if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
242
+ const slot_index: u32 = id & SLOT_MASK;
243
+ const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
244
+ return self.classes[class_id].get(slot_index, generation);
245
+ }
246
+
247
+ pub fn getRefcount(self: *GraphemePool, id: IdPayload) GraphemePoolError!u32 {
248
+ const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
249
+ if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
250
+ const slot_index: u32 = id & SLOT_MASK;
251
+ const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
252
+ return self.classes[class_id].getRefcount(slot_index, generation);
253
+ }
254
+
255
+ const ClassPool = struct {
256
+ allocator: std.mem.Allocator,
257
+ slot_capacity: u32,
258
+ slots_per_page: u32,
259
+ slot_size_bytes: usize,
260
+ slots: std.ArrayListUnmanaged(u8),
261
+ free_list: std.ArrayListUnmanaged(u32),
262
+ num_slots: u32,
263
+
264
+ pub fn init(allocator: std.mem.Allocator, slot_capacity: u32, slots_per_page: u32) ClassPool {
265
+ // Align slot size to SlotHeader alignment to prevent UB from misaligned access
266
+ const raw_slot_size = @sizeOf(SlotHeader) + slot_capacity;
267
+ const slot_size_bytes = std.mem.alignForward(usize, raw_slot_size, @alignOf(SlotHeader));
268
+ return .{
269
+ .allocator = allocator,
270
+ .slot_capacity = slot_capacity,
271
+ .slots_per_page = slots_per_page,
272
+ .slot_size_bytes = slot_size_bytes,
273
+ .slots = .{},
274
+ .free_list = .{},
275
+ .num_slots = 0,
276
+ };
277
+ }
278
+
279
+ pub fn deinit(self: *ClassPool) void {
280
+ self.slots.deinit(self.allocator);
281
+ self.free_list.deinit(self.allocator);
282
+ }
283
+
284
+ fn grow(self: *ClassPool) GraphemePoolError!void {
285
+ const add_bytes = self.slot_size_bytes * self.slots_per_page;
286
+
287
+ try self.slots.ensureTotalCapacity(self.allocator, self.slots.items.len + add_bytes);
288
+ try self.slots.appendNTimes(self.allocator, 0, add_bytes);
289
+
290
+ var i: u32 = 0;
291
+ while (i < self.slots_per_page) : (i += 1) {
292
+ try self.free_list.append(self.allocator, self.num_slots + i);
293
+ }
294
+ self.num_slots += self.slots_per_page;
295
+ }
296
+
297
+ fn slotPtr(self: *ClassPool, slot_index: u32) *u8 {
298
+ const offset: usize = @as(usize, slot_index) * self.slot_size_bytes;
299
+ return &self.slots.items[offset];
300
+ }
301
+
302
+ pub fn allocInternal(self: *ClassPool, bytes: []const u8, is_owned: bool) GraphemePoolError!u32 {
303
+ // Validate size for owned allocations
304
+ if (is_owned and bytes.len > self.slot_capacity) {
305
+ @panic("ClassPool.allocInternal: bytes.len > slot_capacity");
306
+ }
307
+
308
+ if (self.free_list.items.len == 0) try self.grow();
309
+
310
+ const slot_index = self.free_list.pop().?;
311
+ const p = self.slotPtr(slot_index);
312
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
313
+
314
+ // Increment generation when reusing a slot, wrapping at 7 bits (128 values)
315
+ const new_generation = (header_ptr.generation + 1) & GENERATION_MASK;
316
+
317
+ // Calculate length based on ownership
318
+ const len: u16 = if (is_owned) @intCast(@min(bytes.len, self.slot_capacity)) else @intCast(bytes.len);
319
+
320
+ header_ptr.* = .{
321
+ .len = len,
322
+ .refcount = 0,
323
+ .generation = new_generation,
324
+ .is_owned = if (is_owned) 1 else 0,
325
+ };
326
+
327
+ const data_ptr = @as([*]u8, @ptrCast(p)) + @sizeOf(SlotHeader);
328
+
329
+ if (is_owned) {
330
+ // Owned: copy bytes into our storage
331
+ @memcpy(data_ptr[0..header_ptr.len], bytes[0..header_ptr.len]);
332
+ } else {
333
+ // Unowned: store pointer to external memory
334
+ const ptr_storage = @as(*[*]const u8, @ptrCast(@alignCast(data_ptr)));
335
+ ptr_storage.* = bytes.ptr;
336
+ }
337
+
338
+ return slot_index;
339
+ }
340
+
341
+ pub fn getGeneration(self: *ClassPool, slot_index: u32) u32 {
342
+ if (slot_index >= self.num_slots) return 0;
343
+ const p = self.slotPtr(slot_index);
344
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
345
+ return header_ptr.generation;
346
+ }
347
+
348
+ pub fn incref(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
349
+ const p = self.slotPtr(slot_index);
350
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
351
+ if (header_ptr.generation != expected_generation) {
352
+ // Generation mismatch - this is a stale reference
353
+ return GraphemePoolError.WrongGeneration;
354
+ }
355
+ header_ptr.refcount +%= 1;
356
+ }
357
+
358
+ pub fn decref(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
359
+ const p = self.slotPtr(slot_index);
360
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
361
+
362
+ if (header_ptr.refcount == 0) return GraphemePoolError.InvalidId;
363
+ if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
364
+
365
+ header_ptr.refcount -%= 1;
366
+
367
+ if (header_ptr.refcount == 0) {
368
+ try self.free_list.append(self.allocator, slot_index);
369
+ }
370
+ }
371
+
372
+ /// Free a slot that has refcount=0 (freshly allocated, never incref'd).
373
+ /// This is used for cleanup when allocation succeeded but the caller
374
+ /// needs to abort before taking ownership via incref.
375
+ pub fn freeUnreferenced(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
376
+ if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
377
+ const p = self.slotPtr(slot_index);
378
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
379
+
380
+ if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
381
+ if (header_ptr.refcount != 0) return GraphemePoolError.InvalidId; // Not unreferenced
382
+
383
+ try self.free_list.append(self.allocator, slot_index);
384
+ }
385
+
386
+ pub fn get(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError![]const u8 {
387
+ if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
388
+
389
+ const p = self.slotPtr(slot_index);
390
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
391
+ // Validate generation to prevent accessing stale data
392
+ if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
393
+
394
+ const data_ptr = @as([*]u8, @ptrCast(p)) + @sizeOf(SlotHeader);
395
+
396
+ if (header_ptr.is_owned == 1) {
397
+ // Owned memory: return slice from our storage
398
+ return data_ptr[0..header_ptr.len];
399
+ } else {
400
+ // Unowned memory: dereference stored pointer
401
+ const ptr_storage = @as(*[*]const u8, @ptrCast(@alignCast(data_ptr)));
402
+ const external_ptr = ptr_storage.*;
403
+ return external_ptr[0..header_ptr.len];
404
+ }
405
+ }
406
+
407
+ pub fn getRefcount(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!u32 {
408
+ if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
409
+ const p = self.slotPtr(slot_index);
410
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
411
+ if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
412
+ return header_ptr.refcount;
413
+ }
414
+
415
+ pub fn isOwned(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!bool {
416
+ if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
417
+ const p = self.slotPtr(slot_index);
418
+ const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
419
+ if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
420
+ return header_ptr.is_owned == 1;
421
+ }
422
+ };
423
+ };
424
+
425
+ // Bit manipulation functions for encoded char values
426
+
427
+ pub fn isGraphemeChar(c: u32) bool {
428
+ return (c & 0xC000_0000) == CHAR_FLAG_GRAPHEME;
429
+ }
430
+
431
+ pub fn isContinuationChar(c: u32) bool {
432
+ return (c & 0xC000_0000) == CHAR_FLAG_CONTINUATION;
433
+ }
434
+
435
+ pub fn isClusterChar(c: u32) bool {
436
+ return (c & 0x8000_0000) == 0x8000_0000;
437
+ }
438
+
439
+ pub fn graphemeIdFromChar(c: u32) u32 {
440
+ return c & GRAPHEME_ID_MASK;
441
+ }
442
+
443
+ pub fn charRightExtent(c: u32) u32 {
444
+ return (c >> CHAR_EXT_RIGHT_SHIFT) & CHAR_EXT_MASK;
445
+ }
446
+
447
+ pub fn charLeftExtent(c: u32) u32 {
448
+ return (c >> CHAR_EXT_LEFT_SHIFT) & CHAR_EXT_MASK;
449
+ }
450
+
451
+ pub fn packGraphemeStart(gid: u32, total_width: u32) u32 {
452
+ const width_minus_one: u32 = if (total_width == 0) 0 else @intCast(@min(total_width - 1, 3));
453
+ const right: u32 = width_minus_one;
454
+ const left: u32 = 0;
455
+ return CHAR_FLAG_GRAPHEME |
456
+ ((right & CHAR_EXT_MASK) << CHAR_EXT_RIGHT_SHIFT) |
457
+ ((left & CHAR_EXT_MASK) << CHAR_EXT_LEFT_SHIFT) |
458
+ (gid & GRAPHEME_ID_MASK);
459
+ }
460
+
461
+ pub fn packContinuation(left: u32, right: u32, gid: u32) u32 {
462
+ return CHAR_FLAG_CONTINUATION |
463
+ ((@min(left, 3) & CHAR_EXT_MASK) << CHAR_EXT_LEFT_SHIFT) |
464
+ ((@min(right, 3) & CHAR_EXT_MASK) << CHAR_EXT_RIGHT_SHIFT) |
465
+ (gid & GRAPHEME_ID_MASK);
466
+ }
467
+
468
+ pub fn encodedCharWidth(c: u32) u32 {
469
+ if (isContinuationChar(c)) {
470
+ const left = charLeftExtent(c);
471
+ const right = charRightExtent(c);
472
+ return left + 1 + right;
473
+ } else if (isGraphemeChar(c)) {
474
+ return charRightExtent(c) + 1;
475
+ } else {
476
+ return 1;
477
+ }
478
+ }
479
+
480
+ var GLOBAL_POOL_STORAGE: ?GraphemePool = null;
481
+
482
+ pub fn initGlobalPool(allocator: std.mem.Allocator) *GraphemePool {
483
+ return initGlobalPoolWithOptions(allocator, .{});
484
+ }
485
+
486
+ pub fn initGlobalPoolWithOptions(allocator: std.mem.Allocator, options: GraphemePool.InitOptions) *GraphemePool {
487
+ if (GLOBAL_POOL_STORAGE == null) {
488
+ GLOBAL_POOL_STORAGE = GraphemePool.initWithOptions(allocator, options);
489
+ }
490
+ return &GLOBAL_POOL_STORAGE.?;
491
+ }
492
+
493
+ pub fn deinitGlobalPool() void {
494
+ if (GLOBAL_POOL_STORAGE) |*p| {
495
+ p.deinit();
496
+ GLOBAL_POOL_STORAGE = null;
497
+ }
498
+ }
499
+
500
+ pub const GraphemeTracker = struct {
501
+ pool: *GraphemePool,
502
+ used_ids: std.AutoHashMap(u32, u32), // id -> number of cells in this buffer
503
+
504
+ pub fn init(allocator: std.mem.Allocator, pool: *GraphemePool) GraphemeTracker {
505
+ return .{
506
+ .pool = pool,
507
+ .used_ids = std.AutoHashMap(u32, u32).init(allocator),
508
+ };
509
+ }
510
+
511
+ fn decRefAll(self: *GraphemeTracker) void {
512
+ var it = self.used_ids.keyIterator();
513
+ while (it.next()) |idp| {
514
+ // Pool refs are tracked per ID (first/last cell transition), so clear
515
+ // decrefs once per tracked ID, not once per per-buffer cell count.
516
+ self.pool.decref(idp.*) catch {};
517
+ }
518
+ }
519
+
520
+ pub fn deinit(self: *GraphemeTracker) void {
521
+ self.decRefAll();
522
+ self.used_ids.deinit();
523
+ }
524
+
525
+ pub fn clear(self: *GraphemeTracker) void {
526
+ self.decRefAll();
527
+ self.used_ids.clearRetainingCapacity();
528
+ }
529
+
530
+ pub fn add(self: *GraphemeTracker, id: u32) void {
531
+ const res = self.used_ids.getOrPut(id) catch |err| {
532
+ std.debug.panic("GraphemeTracker.add failed: {}\n", .{err});
533
+ };
534
+ if (!res.found_existing) {
535
+ res.value_ptr.* = 1;
536
+ self.pool.incref(id) catch |err| {
537
+ std.debug.panic("GraphemeTracker.add incref failed: {}\n", .{err});
538
+ };
539
+ } else {
540
+ res.value_ptr.* += 1;
541
+ }
542
+ }
543
+
544
+ pub fn remove(self: *GraphemeTracker, id: u32) void {
545
+ const count_ptr = self.used_ids.getPtr(id) orelse return;
546
+ if (count_ptr.* > 1) {
547
+ count_ptr.* -= 1;
548
+ return;
549
+ }
550
+
551
+ if (self.used_ids.remove(id)) {
552
+ self.pool.decref(id) catch {};
553
+ }
554
+ }
555
+
556
+ pub fn replace(self: *GraphemeTracker, old_id: ?u32, new_id: ?u32) void {
557
+ if (old_id != null and new_id != null and old_id.? == new_id.?) return;
558
+
559
+ if (new_id) |id| self.add(id);
560
+ if (old_id) |id| self.remove(id);
561
+ }
562
+
563
+ pub fn contains(self: *const GraphemeTracker, id: u32) bool {
564
+ return self.used_ids.contains(id);
565
+ }
566
+
567
+ pub fn hasAny(self: *const GraphemeTracker) bool {
568
+ return self.used_ids.count() > 0;
569
+ }
570
+
571
+ pub fn getGraphemeCount(self: *const GraphemeTracker) u32 {
572
+ return @intCast(self.used_ids.count());
573
+ }
574
+
575
+ pub fn getGraphemeCellCount(self: *const GraphemeTracker) u32 {
576
+ var total: u32 = 0;
577
+ var it = self.used_ids.valueIterator();
578
+ while (it.next()) |count_ptr| {
579
+ total += count_ptr.*;
580
+ }
581
+ return total;
582
+ }
583
+
584
+ pub fn getTotalGraphemeBytes(self: *const GraphemeTracker) u32 {
585
+ var total_bytes: u32 = 0;
586
+ var it = self.used_ids.iterator();
587
+ while (it.next()) |entry| {
588
+ const id = entry.key_ptr.*;
589
+ const count = entry.value_ptr.*;
590
+ if (self.pool.get(id)) |bytes| {
591
+ total_bytes += @as(u32, @intCast(bytes.len)) * count;
592
+ } else |_| {
593
+ // If we can't get the bytes, this shouldn't happen but handle gracefully
594
+ continue;
595
+ }
596
+ }
597
+ return total_bytes;
598
+ }
599
+ };