@fairyhunter13/opentui-core 0.1.113 β†’ 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 +62 -53
  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-9vwc3fg6.js +0 -12260
  472. package/index-9vwc3fg6.js.map +0 -42
  473. package/index-dcj62y8t.js +0 -20614
  474. package/index-dcj62y8t.js.map +0 -67
  475. package/index-f7n39gpy.js +0 -411
  476. package/index-f7n39gpy.js.map +0 -10
  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,267 @@
1
+ const std = @import("std");
2
+ const testing = std.testing;
3
+ const utf8 = @import("../utf8.zig");
4
+
5
+ test "wcwidth: cursor movement through emoji with skin tone" {
6
+ const text = "πŸ‘‹πŸΏ"; // Wave + dark skin tone = 4 columns
7
+
8
+ const width_wave = utf8.getWidthAt(text, 0, 4, .wcwidth);
9
+ try testing.expectEqual(@as(u32, 2), width_wave);
10
+
11
+ const width_skin = utf8.getWidthAt(text, 4, 4, .wcwidth);
12
+ try testing.expectEqual(@as(u32, 2), width_skin);
13
+
14
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
15
+ try testing.expectEqual(@as(u32, 4), total);
16
+ }
17
+
18
+ test "wcwidth: cursor movement through ZWJ sequence" {
19
+ const text = "πŸ‘©β€πŸš€"; // Woman + ZWJ + Rocket = 4 columns (2+0+2)
20
+
21
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
22
+ try testing.expectEqual(@as(u32, 4), total);
23
+
24
+ const width_woman = utf8.getWidthAt(text, 0, 4, .wcwidth);
25
+ try testing.expectEqual(@as(u32, 2), width_woman);
26
+
27
+ const width_zwj = utf8.getWidthAt(text, 4, 4, .wcwidth);
28
+ try testing.expectEqual(@as(u32, 0), width_zwj);
29
+
30
+ const width_rocket = utf8.getWidthAt(text, 7, 4, .wcwidth);
31
+ try testing.expectEqual(@as(u32, 2), width_rocket);
32
+ }
33
+
34
+ test "wcwidth: cursor movement through family emoji" {
35
+ const text = "πŸ‘¨β€πŸ‘©β€πŸ‘§"; // Man + ZWJ + Woman + ZWJ + Girl = 6 columns (2+0+2+0+2)
36
+
37
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
38
+ try testing.expectEqual(@as(u32, 6), total);
39
+
40
+ const width_man = utf8.getWidthAt(text, 0, 4, .wcwidth);
41
+ try testing.expectEqual(@as(u32, 2), width_man);
42
+
43
+ const width_zwj1 = utf8.getWidthAt(text, 4, 4, .wcwidth);
44
+ try testing.expectEqual(@as(u32, 0), width_zwj1);
45
+
46
+ const width_woman = utf8.getWidthAt(text, 7, 4, .wcwidth);
47
+ try testing.expectEqual(@as(u32, 2), width_woman);
48
+
49
+ const width_zwj2 = utf8.getWidthAt(text, 11, 4, .wcwidth);
50
+ try testing.expectEqual(@as(u32, 0), width_zwj2);
51
+
52
+ const width_girl = utf8.getWidthAt(text, 14, 4, .wcwidth);
53
+ try testing.expectEqual(@as(u32, 2), width_girl);
54
+ }
55
+
56
+ test "wcwidth: getPrevGraphemeStart through emoji with skin tone" {
57
+ const text = "AπŸ‘‹πŸΏB"; // A(1) + πŸ‘‹(2) + 🏿(2) + B(1) = 6 columns
58
+
59
+ const r_end = utf8.getPrevGraphemeStart(text, text.len, 4, .wcwidth);
60
+ try testing.expect(r_end != null);
61
+ try testing.expectEqual(@as(u32, 1), r_end.?.width);
62
+
63
+ const r_b = utf8.getPrevGraphemeStart(text, r_end.?.start_offset, 4, .wcwidth);
64
+ try testing.expect(r_b != null);
65
+ try testing.expectEqual(@as(u32, 2), r_b.?.width);
66
+
67
+ const r_skin = utf8.getPrevGraphemeStart(text, r_b.?.start_offset, 4, .wcwidth);
68
+ try testing.expect(r_skin != null);
69
+ try testing.expectEqual(@as(u32, 2), r_skin.?.width);
70
+
71
+ const r_wave = utf8.getPrevGraphemeStart(text, r_skin.?.start_offset, 4, .wcwidth);
72
+ try testing.expect(r_wave != null);
73
+ try testing.expectEqual(@as(u32, 1), r_wave.?.width);
74
+ }
75
+
76
+ test "wcwidth: getPrevGraphemeStart through ZWJ sequence" {
77
+ const text = "XπŸ‘©β€πŸš€Y"; // X(1) + πŸ‘©(2) + ZWJ(0) + πŸš€(2) + Y(1) = 6 columns
78
+
79
+ const r_end = utf8.getPrevGraphemeStart(text, text.len, 4, .wcwidth);
80
+ try testing.expect(r_end != null);
81
+ try testing.expectEqual(@as(u32, 1), r_end.?.width);
82
+
83
+ const r_y = utf8.getPrevGraphemeStart(text, r_end.?.start_offset, 4, .wcwidth);
84
+ try testing.expect(r_y != null);
85
+ try testing.expectEqual(@as(u32, 2), r_y.?.width);
86
+
87
+ const r_rocket = utf8.getPrevGraphemeStart(text, r_y.?.start_offset, 4, .wcwidth);
88
+ try testing.expect(r_rocket != null);
89
+ try testing.expectEqual(@as(u32, 2), r_rocket.?.width);
90
+
91
+ const r_woman = utf8.getPrevGraphemeStart(text, r_rocket.?.start_offset, 4, .wcwidth);
92
+ try testing.expect(r_woman != null);
93
+ try testing.expectEqual(@as(u32, 1), r_woman.?.width);
94
+ }
95
+
96
+ test "wcwidth: findPosByWidth through emoji sequence" {
97
+ const text = "ABπŸ‘‹πŸΏCD"; // A(1) B(1) πŸ‘‹(2) 🏿(2) C(1) D(1) = 8 columns
98
+
99
+ const pos_start = utf8.findPosByWidth(text, 3, 4, false, false, .wcwidth);
100
+ try testing.expectEqual(@as(u32, 2), pos_start.byte_offset);
101
+
102
+ const pos_end = utf8.findPosByWidth(text, 3, 4, false, true, .wcwidth);
103
+ try testing.expectEqual(@as(u32, 6), pos_end.byte_offset);
104
+ }
105
+
106
+ test "wcwidth: findWrapPosByWidth through emoji" {
107
+ const text = "HiπŸ‘‹πŸΏBye"; // H(1) i(1) πŸ‘‹(2) 🏿(2) B(1) y(1) e(1) = 10 columns
108
+
109
+ const wrap_4 = utf8.findWrapPosByWidth(text, 4, 4, false, .wcwidth);
110
+ try testing.expectEqual(@as(u32, 6), wrap_4.byte_offset);
111
+ try testing.expectEqual(@as(u32, 4), wrap_4.columns_used);
112
+
113
+ const wrap_5 = utf8.findWrapPosByWidth(text, 5, 4, false, .wcwidth);
114
+ try testing.expectEqual(@as(u32, 6), wrap_5.byte_offset);
115
+ try testing.expectEqual(@as(u32, 4), wrap_5.columns_used);
116
+
117
+ const wrap_6 = utf8.findWrapPosByWidth(text, 6, 4, false, .wcwidth);
118
+ try testing.expectEqual(@as(u32, 10), wrap_6.byte_offset);
119
+ try testing.expectEqual(@as(u32, 6), wrap_6.columns_used);
120
+ }
121
+
122
+ test "wcwidth: combining marks have zero width" {
123
+ const text = "e\u{0301}"; // e + combining acute
124
+
125
+ const width_e = utf8.getWidthAt(text, 0, 4, .wcwidth);
126
+ try testing.expectEqual(@as(u32, 1), width_e);
127
+
128
+ const width_combining = utf8.getWidthAt(text, 1, 4, .wcwidth);
129
+ try testing.expectEqual(@as(u32, 0), width_combining);
130
+
131
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
132
+ try testing.expectEqual(@as(u32, 1), total);
133
+ }
134
+
135
+ test "wcwidth: CJK characters have width 2" {
136
+ const text = "δ½ ε₯½δΈ–η•Œ"; // 4 CJK characters
137
+
138
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
139
+ try testing.expectEqual(@as(u32, 8), total);
140
+
141
+ const width_char1 = utf8.getWidthAt(text, 0, 4, .wcwidth);
142
+ try testing.expectEqual(@as(u32, 2), width_char1);
143
+
144
+ const width_char2 = utf8.getWidthAt(text, 3, 4, .wcwidth);
145
+ try testing.expectEqual(@as(u32, 2), width_char2);
146
+ }
147
+
148
+ test "wcwidth: variation selectors have zero width" {
149
+ const text = "☺\u{FE0F}"; // Smiling face + VS16
150
+
151
+ const width_face = utf8.getWidthAt(text, 0, 4, .wcwidth);
152
+ try testing.expectEqual(@as(u32, 1), width_face);
153
+
154
+ const width_vs = utf8.getWidthAt(text, 3, 4, .wcwidth);
155
+ try testing.expectEqual(@as(u32, 0), width_vs);
156
+
157
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
158
+ try testing.expectEqual(@as(u32, 1), total);
159
+ }
160
+
161
+ test "wcwidth: flag emoji counts both regional indicators" {
162
+ const text = "πŸ‡ΊπŸ‡Έ"; // US flag (two regional indicators)
163
+
164
+ const width_ri1 = utf8.getWidthAt(text, 0, 4, .wcwidth);
165
+ try testing.expectEqual(@as(u32, 1), width_ri1);
166
+
167
+ const width_ri2 = utf8.getWidthAt(text, 4, 4, .wcwidth);
168
+ try testing.expectEqual(@as(u32, 1), width_ri2);
169
+
170
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
171
+ try testing.expectEqual(@as(u32, 2), total);
172
+ }
173
+
174
+ test "wcwidth: mixed content with cursor movement" {
175
+ const text = "AπŸ‘‹πŸΏBδΈ–C"; // A(1) πŸ‘‹(2) 🏿(2) B(1) δΈ–(2) C(1) = 9 columns
176
+
177
+ const r_end = utf8.getPrevGraphemeStart(text, text.len, 4, .wcwidth);
178
+ try testing.expect(r_end != null);
179
+ try testing.expectEqual(@as(u32, 1), r_end.?.width);
180
+
181
+ const r_cjk = utf8.getPrevGraphemeStart(text, r_end.?.start_offset, 4, .wcwidth);
182
+ try testing.expect(r_cjk != null);
183
+ try testing.expectEqual(@as(u32, 2), r_cjk.?.width);
184
+
185
+ const r_b = utf8.getPrevGraphemeStart(text, r_cjk.?.start_offset, 4, .wcwidth);
186
+ try testing.expect(r_b != null);
187
+ try testing.expectEqual(@as(u32, 1), r_b.?.width);
188
+
189
+ const r_skin = utf8.getPrevGraphemeStart(text, r_b.?.start_offset, 4, .wcwidth);
190
+ try testing.expect(r_skin != null);
191
+ try testing.expectEqual(@as(u32, 2), r_skin.?.width);
192
+
193
+ const r_wave = utf8.getPrevGraphemeStart(text, r_skin.?.start_offset, 4, .wcwidth);
194
+ try testing.expect(r_wave != null);
195
+ try testing.expectEqual(@as(u32, 2), r_wave.?.width);
196
+
197
+ const r_a = utf8.getPrevGraphemeStart(text, r_wave.?.start_offset, 4, .wcwidth);
198
+ try testing.expect(r_a != null);
199
+ try testing.expectEqual(@as(u32, 1), r_a.?.width);
200
+ }
201
+
202
+ test "wcwidth: findGraphemeInfo with emoji" {
203
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
204
+ defer result.deinit(testing.allocator);
205
+
206
+ const text = "πŸ‘‹πŸΏ"; // Wave + skin tone modifier
207
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
208
+
209
+ try testing.expectEqual(@as(usize, 1), result.items.len);
210
+
211
+ try testing.expectEqual(@as(u32, 0), result.items[0].byte_offset);
212
+ try testing.expectEqual(@as(u8, 8), result.items[0].byte_len);
213
+ try testing.expectEqual(@as(u8, 4), result.items[0].width);
214
+ }
215
+
216
+ test "wcwidth: findGraphemeInfo with ZWJ sequence" {
217
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
218
+ defer result.deinit(testing.allocator);
219
+
220
+ const text = "πŸ‘©β€πŸš€"; // Woman + ZWJ + Rocket
221
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
222
+
223
+ try testing.expectEqual(@as(usize, 1), result.items.len);
224
+
225
+ try testing.expectEqual(@as(u32, 0), result.items[0].byte_offset);
226
+ try testing.expectEqual(@as(u8, 11), result.items[0].byte_len);
227
+ try testing.expectEqual(@as(u8, 4), result.items[0].width);
228
+ }
229
+
230
+ test "wcwidth: findGraphemeInfo with combining marks" {
231
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
232
+ defer result.deinit(testing.allocator);
233
+
234
+ const text = "e\u{0301}"; // e + combining acute
235
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
236
+
237
+ try testing.expectEqual(@as(usize, 1), result.items.len);
238
+ try testing.expectEqual(@as(u32, 0), result.items[0].byte_offset);
239
+ try testing.expectEqual(@as(u8, 3), result.items[0].byte_len);
240
+ try testing.expectEqual(@as(u8, 1), result.items[0].width);
241
+ }
242
+
243
+ test "wcwidth: tab width handling" {
244
+ const text = "A\tB"; // A + tab + B
245
+
246
+ const total = utf8.calculateTextWidth(text, 4, false, .wcwidth);
247
+ try testing.expectEqual(@as(u32, 6), total);
248
+
249
+ const tab_width = utf8.getWidthAt(text, 1, 4, .wcwidth);
250
+ try testing.expectEqual(@as(u32, 4), tab_width);
251
+ }
252
+
253
+ test "wcwidth: boundary at wide character" {
254
+ const text = "δΈ–X"; // δΈ–(2) X(1) = 3 columns
255
+
256
+ const pos_start = utf8.findPosByWidth(text, 2, 4, false, false, .wcwidth);
257
+ try testing.expectEqual(@as(u32, 3), pos_start.byte_offset);
258
+ try testing.expectEqual(@as(u32, 2), pos_start.columns_used);
259
+
260
+ const pos_end = utf8.findPosByWidth(text, 2, 4, false, true, .wcwidth);
261
+ try testing.expectEqual(@as(u32, 3), pos_end.byte_offset);
262
+ try testing.expectEqual(@as(u32, 2), pos_end.columns_used);
263
+
264
+ const pos_3 = utf8.findPosByWidth(text, 3, 4, false, true, .wcwidth);
265
+ try testing.expectEqual(@as(u32, 4), pos_3.byte_offset);
266
+ try testing.expectEqual(@as(u32, 3), pos_3.columns_used);
267
+ }
@@ -0,0 +1,357 @@
1
+ const std = @import("std");
2
+ const testing = std.testing;
3
+ const utf8 = @import("../utf8.zig");
4
+
5
+ test "findGraphemeInfo wcwidth: empty string" {
6
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
7
+ defer result.deinit(testing.allocator);
8
+
9
+ try utf8.findGraphemeInfo("", 4, false, .wcwidth, testing.allocator, &result);
10
+ try testing.expectEqual(@as(usize, 0), result.items.len);
11
+ }
12
+
13
+ test "findGraphemeInfo wcwidth: ASCII-only returns empty" {
14
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
15
+ defer result.deinit(testing.allocator);
16
+
17
+ try utf8.findGraphemeInfo("hello world", 4, true, .wcwidth, testing.allocator, &result);
18
+ try testing.expectEqual(@as(usize, 0), result.items.len);
19
+ }
20
+
21
+ test "findGraphemeInfo wcwidth: ASCII with tab" {
22
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
23
+ defer result.deinit(testing.allocator);
24
+
25
+ try utf8.findGraphemeInfo("hello\tworld", 4, false, .wcwidth, testing.allocator, &result);
26
+
27
+ // Should have one entry for the tab
28
+ try testing.expectEqual(@as(usize, 1), result.items.len);
29
+ try testing.expectEqual(@as(u32, 5), result.items[0].byte_offset);
30
+ try testing.expectEqual(@as(u8, 1), result.items[0].byte_len);
31
+ try testing.expectEqual(@as(u8, 4), result.items[0].width);
32
+ try testing.expectEqual(@as(u32, 5), result.items[0].col_offset);
33
+ }
34
+
35
+ test "findGraphemeInfo wcwidth: CJK characters" {
36
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
37
+ defer result.deinit(testing.allocator);
38
+
39
+ const text = "helloδΈ–η•Œ";
40
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
41
+
42
+ // Should have two entries for the CJK characters (each codepoint separately)
43
+ try testing.expectEqual(@as(usize, 2), result.items.len);
44
+
45
+ // First CJK char 'δΈ–' at byte 5
46
+ try testing.expectEqual(@as(u32, 5), result.items[0].byte_offset);
47
+ try testing.expectEqual(@as(u8, 3), result.items[0].byte_len);
48
+ try testing.expectEqual(@as(u8, 2), result.items[0].width);
49
+ try testing.expectEqual(@as(u32, 5), result.items[0].col_offset);
50
+
51
+ // Second CJK char 'η•Œ' at byte 8
52
+ try testing.expectEqual(@as(u32, 8), result.items[1].byte_offset);
53
+ try testing.expectEqual(@as(u8, 3), result.items[1].byte_len);
54
+ try testing.expectEqual(@as(u8, 2), result.items[1].width);
55
+ try testing.expectEqual(@as(u32, 7), result.items[1].col_offset);
56
+ }
57
+
58
+ test "findGraphemeInfo wcwidth: emoji with skin tone - single grapheme cluster" {
59
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
60
+ defer result.deinit(testing.allocator);
61
+
62
+ const text = "πŸ‘‹πŸΏ"; // Wave + skin tone modifier
63
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
64
+
65
+ try testing.expectEqual(@as(usize, 1), result.items.len);
66
+
67
+ try testing.expectEqual(@as(u32, 0), result.items[0].byte_offset);
68
+ try testing.expectEqual(@as(u8, 8), result.items[0].byte_len);
69
+ try testing.expectEqual(@as(u8, 4), result.items[0].width);
70
+ }
71
+
72
+ test "findGraphemeInfo wcwidth: emoji with ZWJ - single grapheme cluster" {
73
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
74
+ defer result.deinit(testing.allocator);
75
+
76
+ const text = "πŸ‘©β€πŸš€"; // Woman + ZWJ + Rocket (11 bytes total)
77
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
78
+
79
+ try testing.expectEqual(@as(usize, 1), result.items.len);
80
+
81
+ try testing.expectEqual(@as(u8, 11), result.items[0].byte_len);
82
+ try testing.expectEqual(@as(u8, 4), result.items[0].width);
83
+ }
84
+
85
+ test "findGraphemeInfo wcwidth: combining mark - part of base grapheme" {
86
+ var result: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
87
+ defer result.deinit(testing.allocator);
88
+
89
+ const text = "e\u{0301}test"; // e + combining acute accent + test
90
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result);
91
+
92
+ try testing.expectEqual(@as(usize, 1), result.items.len);
93
+ try testing.expectEqual(@as(u32, 0), result.items[0].byte_offset);
94
+ try testing.expectEqual(@as(u8, 3), result.items[0].byte_len);
95
+ try testing.expectEqual(@as(u8, 1), result.items[0].width);
96
+ }
97
+
98
+ test "findGraphemeInfo wcwidth vs unicode: emoji with skin tone" {
99
+ var result_wcwidth: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
100
+ defer result_wcwidth.deinit(testing.allocator);
101
+ var result_unicode: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
102
+ defer result_unicode.deinit(testing.allocator);
103
+
104
+ const text = "HiπŸ‘‹πŸΏBye";
105
+
106
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result_wcwidth);
107
+ try utf8.findGraphemeInfo(text, 4, false, .unicode, testing.allocator, &result_unicode);
108
+
109
+ try testing.expectEqual(@as(usize, 1), result_wcwidth.items.len);
110
+ try testing.expectEqual(@as(usize, 1), result_unicode.items.len);
111
+
112
+ try testing.expectEqual(@as(u32, 2), result_wcwidth.items[0].byte_offset);
113
+ try testing.expectEqual(@as(u8, 8), result_wcwidth.items[0].byte_len);
114
+
115
+ try testing.expectEqual(@as(u32, 2), result_unicode.items[0].byte_offset);
116
+ try testing.expectEqual(@as(u8, 8), result_unicode.items[0].byte_len);
117
+
118
+ try testing.expectEqual(@as(u8, 4), result_wcwidth.items[0].width);
119
+ try testing.expectEqual(@as(u8, 2), result_unicode.items[0].width);
120
+ }
121
+
122
+ test "findGraphemeInfo wcwidth vs unicode: flag emoji" {
123
+ var result_wcwidth: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
124
+ defer result_wcwidth.deinit(testing.allocator);
125
+ var result_unicode: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
126
+ defer result_unicode.deinit(testing.allocator);
127
+
128
+ const text = "πŸ‡ΊπŸ‡Έ"; // US flag (two regional indicators)
129
+
130
+ try utf8.findGraphemeInfo(text, 4, false, .wcwidth, testing.allocator, &result_wcwidth);
131
+ try utf8.findGraphemeInfo(text, 4, false, .unicode, testing.allocator, &result_unicode);
132
+
133
+ try testing.expectEqual(@as(usize, 1), result_wcwidth.items.len);
134
+ try testing.expectEqual(@as(usize, 1), result_unicode.items.len);
135
+
136
+ try testing.expectEqual(@as(u8, 2), result_wcwidth.items[0].width);
137
+ try testing.expectEqual(@as(u8, 2), result_unicode.items[0].width);
138
+ }
139
+
140
+ // ============================================================================
141
+ // WIDTH CALCULATION TESTS - WCWIDTH MODE
142
+ // ============================================================================
143
+
144
+ test "getWidthAt wcwidth: combining mark has zero width" {
145
+ const text = "e\u{0301}"; // e + combining acute accent
146
+
147
+ // In wcwidth mode, combining mark is a separate codepoint
148
+ const width_e = utf8.getWidthAt(text, 0, 8, .wcwidth);
149
+ try testing.expectEqual(@as(u32, 1), width_e); // Just 'e'
150
+
151
+ const width_combining = utf8.getWidthAt(text, 1, 8, .wcwidth);
152
+ try testing.expectEqual(@as(u32, 0), width_combining); // Combining mark has width 0
153
+ }
154
+
155
+ test "calculateTextWidth wcwidth: emoji with skin tone counts both codepoints" {
156
+ const text = "πŸ‘‹πŸΏ"; // Wave + dark skin tone
157
+
158
+ const width_wcwidth = utf8.calculateTextWidth(text, 4, false, .wcwidth);
159
+ const width_unicode = utf8.calculateTextWidth(text, 4, false, .unicode);
160
+
161
+ // wcwidth: counts both codepoints (2 + 2 = 4)
162
+ try testing.expectEqual(@as(u32, 4), width_wcwidth);
163
+
164
+ // unicode: single grapheme cluster (width 2)
165
+ try testing.expectEqual(@as(u32, 2), width_unicode);
166
+ }
167
+
168
+ test "calculateTextWidth wcwidth: flag emoji counts both RIs" {
169
+ const text = "πŸ‡ΊπŸ‡Έ"; // US flag
170
+
171
+ const width_wcwidth = utf8.calculateTextWidth(text, 4, false, .wcwidth);
172
+ const width_unicode = utf8.calculateTextWidth(text, 4, false, .unicode);
173
+
174
+ // wcwidth: counts both regional indicators (1 + 1 = 2)
175
+ try testing.expectEqual(@as(u32, 2), width_wcwidth);
176
+
177
+ // unicode: single flag grapheme (width 2)
178
+ try testing.expectEqual(@as(u32, 2), width_unicode);
179
+ }
180
+
181
+ // ============================================================================
182
+ // FIND WRAP POS BY WIDTH TESTS - WCWIDTH MODE
183
+ // ============================================================================
184
+
185
+ test "findWrapPosByWidth wcwidth: emoji with skin tone stops earlier" {
186
+ const text = "HiπŸ‘‹πŸΏBye"; // H(1) i(1) wave(2) skin(2) B(1) y(1) e(1) = 10 cols wcwidth
187
+
188
+ const result_wcwidth = utf8.findWrapPosByWidth(text, 4, 4, false, .wcwidth);
189
+ const result_unicode = utf8.findWrapPosByWidth(text, 4, 4, false, .unicode);
190
+
191
+ // wcwidth: stops after "HiπŸ‘‹" = 4 columns (1+1+2)
192
+ try testing.expectEqual(@as(u32, 6), result_wcwidth.byte_offset);
193
+ try testing.expectEqual(@as(u32, 4), result_wcwidth.columns_used);
194
+
195
+ // unicode: stops after "HiπŸ‘‹πŸΏ" = 4 columns (1+1+2 for whole grapheme)
196
+ try testing.expectEqual(@as(u32, 10), result_unicode.byte_offset);
197
+ try testing.expectEqual(@as(u32, 4), result_unicode.columns_used);
198
+ }
199
+
200
+ test "findPosByWidth wcwidth: emoji boundary behavior" {
201
+ const text = "ABπŸ‘‹πŸΏCD"; // A(1) B(1) wave(2) skin(2) C(1) D(1)
202
+
203
+ // With include_start_before=false (selection start)
204
+ const start3 = utf8.findPosByWidth(text, 3, 4, false, false, .wcwidth);
205
+ // wcwidth: stops after "AB" at 2 columns (wave would exceed)
206
+ try testing.expectEqual(@as(u32, 2), start3.byte_offset);
207
+
208
+ // With include_start_before=true (selection end)
209
+ const end3 = utf8.findPosByWidth(text, 3, 4, false, true, .wcwidth);
210
+ // wcwidth: includes wave since it starts at column 2 which is < 3
211
+ try testing.expectEqual(@as(u32, 6), end3.byte_offset);
212
+ try testing.expectEqual(@as(u32, 4), end3.columns_used);
213
+ }
214
+
215
+ test "getPrevGraphemeStart wcwidth: each codepoint separate" {
216
+ const text = "HiπŸ‘‹πŸΏ";
217
+
218
+ // From end of text (after skin tone)
219
+ const r_end = utf8.getPrevGraphemeStart(text, text.len, 4, .wcwidth);
220
+ try testing.expect(r_end != null);
221
+ try testing.expectEqual(@as(usize, 6), r_end.?.start_offset); // Skin tone starts at byte 6
222
+ try testing.expectEqual(@as(u32, 2), r_end.?.width);
223
+
224
+ // From start of skin tone (byte 6)
225
+ const r_wave = utf8.getPrevGraphemeStart(text, 6, 4, .wcwidth);
226
+ try testing.expect(r_wave != null);
227
+ try testing.expectEqual(@as(usize, 2), r_wave.?.start_offset); // Wave starts at byte 2
228
+ try testing.expectEqual(@as(u32, 2), r_wave.?.width);
229
+ }
230
+
231
+ // ============================================================================
232
+ // ADDITIONAL COMPREHENSIVE WCWIDTH TESTS
233
+ // ============================================================================
234
+
235
+ test "wcwidth: zero-width characters are handled correctly" {
236
+ // ZWJ (Zero Width Joiner) should have width 0
237
+ const text_zwj = "\u{200D}";
238
+ const width_zwj = utf8.calculateTextWidth(text_zwj, 4, false, .wcwidth);
239
+ try testing.expectEqual(@as(u32, 0), width_zwj);
240
+
241
+ // Combining marks should have width 0
242
+ const text_combining = "e\u{0301}"; // e + combining acute
243
+ const width = utf8.calculateTextWidth(text_combining, 4, false, .wcwidth);
244
+ try testing.expectEqual(@as(u32, 1), width); // Only 'e' contributes
245
+ }
246
+
247
+ test "wcwidth: variation selectors" {
248
+ // VS15 (text presentation) and VS16 (emoji presentation)
249
+ const text_vs16 = "☺\u{FE0F}"; // Smiling face + VS16
250
+ const width_vs16 = utf8.calculateTextWidth(text_vs16, 4, false, .wcwidth);
251
+ try testing.expectEqual(@as(u32, 1), width_vs16); // Smiling face (1) + VS16 (0) = 1
252
+ }
253
+
254
+ test "wcwidth: regional indicators counted separately" {
255
+ // Each regional indicator should contribute width 1
256
+ const text = "πŸ‡ΊπŸ‡Έ"; // US flag = two regional indicators
257
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
258
+ try testing.expectEqual(@as(u32, 2), width); // Each RI has width 1
259
+ }
260
+
261
+ test "wcwidth: emoji ZWJ sequences split" {
262
+ // Woman astronaut = woman + ZWJ + rocket
263
+ const text = "πŸ‘©β€πŸš€";
264
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
265
+ // Woman (2) + ZWJ (0) + Rocket (2) = 4
266
+ try testing.expectEqual(@as(u32, 4), width);
267
+ }
268
+
269
+ test "wcwidth: family emoji split into components" {
270
+ // Family emoji with ZWJ
271
+ const text = "πŸ‘¨β€πŸ‘©β€πŸ‘§"; // Man + ZWJ + Woman + ZWJ + Girl
272
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
273
+ // Man (2) + ZWJ (0) + Woman (2) + ZWJ (0) + Girl (2) = 6
274
+ try testing.expectEqual(@as(u32, 6), width);
275
+ }
276
+
277
+ test "wcwidth: skin tone modifiers counted separately" {
278
+ // Emoji with skin tone modifier
279
+ const text = "πŸ‘‹πŸ»"; // Wave + light skin tone
280
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
281
+ // Wave (2) + Skin tone modifier (2) = 4
282
+ try testing.expectEqual(@as(u32, 4), width);
283
+ }
284
+
285
+ test "wcwidth: CJK characters have width 2" {
286
+ const text = "δ½ ε₯½δΈ–η•Œ"; // 4 CJK characters
287
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
288
+ try testing.expectEqual(@as(u32, 8), width); // 4 * 2 = 8
289
+ }
290
+
291
+ test "wcwidth: mixed ASCII and emoji" {
292
+ const text = "HelloπŸ‘‹World";
293
+ // H(1) e(1) l(1) l(1) o(1) πŸ‘‹(2) W(1) o(1) r(1) l(1) d(1) = 12
294
+ const width = utf8.calculateTextWidth(text, 4, false, .wcwidth);
295
+ try testing.expectEqual(@as(u32, 12), width);
296
+ }
297
+
298
+ test "wcwidth: findWrapPosByWidth with ZWJ sequences" {
299
+ const text = "ABπŸ‘©β€πŸš€CD"; // A(1) B(1) woman(2) ZWJ(0) rocket(2) C(1) D(1) = 8
300
+
301
+ // Should wrap after woman emoji (before ZWJ)
302
+ const result = utf8.findWrapPosByWidth(text, 4, 4, false, .wcwidth);
303
+ try testing.expectEqual(@as(u32, 6), result.byte_offset); // After woman emoji
304
+ try testing.expectEqual(@as(u32, 4), result.columns_used);
305
+ }
306
+
307
+ test "wcwidth: findPosByWidth with skin tone modifier" {
308
+ const text = "ABπŸ‘‹πŸ»CD"; // A(1) B(1) wave(2) skin(2) C(1) D(1) = 8
309
+
310
+ // With include_start_before=false, include codepoints that end at or before max_columns
311
+ // Wave ends at column 4, which is at max_columns=4, so it's included
312
+ const start4 = utf8.findPosByWidth(text, 4, 4, false, false, .wcwidth);
313
+ try testing.expectEqual(@as(u32, 6), start4.byte_offset); // After wave
314
+ try testing.expectEqual(@as(u32, 4), start4.columns_used);
315
+
316
+ // With include_start_before=true, include codepoints that start before max_columns
317
+ // Wave starts at column 2 which is < 4, so it's included
318
+ const end4 = utf8.findPosByWidth(text, 4, 4, false, true, .wcwidth);
319
+ try testing.expectEqual(@as(u32, 6), end4.byte_offset); // After wave
320
+ try testing.expectEqual(@as(u32, 4), end4.columns_used);
321
+ }
322
+
323
+ test "wcwidth: getWidthAt with combining marks" {
324
+ const text = "e\u{0301}test"; // e + combining acute
325
+
326
+ // Width at 'e' should be 1
327
+ const width_e = utf8.getWidthAt(text, 0, 4, .wcwidth);
328
+ try testing.expectEqual(@as(u32, 1), width_e);
329
+
330
+ // Width at combining mark should be 0 (but next non-zero is 't')
331
+ const width_combining = utf8.getWidthAt(text, 1, 4, .wcwidth);
332
+ try testing.expectEqual(@as(u32, 0), width_combining);
333
+ }
334
+
335
+ test "wcwidth: getPrevGraphemeStart with ZWJ sequence" {
336
+ const text = "ABπŸ‘©β€πŸš€"; // A B woman ZWJ rocket
337
+
338
+ // From end (after rocket)
339
+ const r1 = utf8.getPrevGraphemeStart(text, text.len, 4, .wcwidth);
340
+ try testing.expect(r1 != null);
341
+ // Should point to rocket emoji (after ZWJ)
342
+ try testing.expectEqual(@as(u32, 2), r1.?.width);
343
+
344
+ // From rocket start, should go to ZWJ
345
+ const r2 = utf8.getPrevGraphemeStart(text, r1.?.start_offset, 4, .wcwidth);
346
+ try testing.expect(r2 != null);
347
+
348
+ // Eventually should reach woman emoji
349
+ var pos = text.len;
350
+ var count: usize = 0;
351
+ while (utf8.getPrevGraphemeStart(text, pos, 4, .wcwidth)) |prev| {
352
+ pos = prev.start_offset;
353
+ count += 1;
354
+ if (count > 10) break; // Safety limit
355
+ }
356
+ try testing.expect(count >= 3); // At least rocket, ZWJ, woman
357
+ }