@fairyhunter13/opentui-core 0.1.88 → 0.1.90

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (570) hide show
  1. package/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 +141 -0
  7. package/docs/env-vars.md +140 -0
  8. package/docs/getting-started.md +353 -0
  9. package/docs/renderables-vs-constructs.md +159 -0
  10. package/docs/tree-sitter.md +311 -0
  11. package/package.json +61 -52
  12. package/scripts/build.ts +400 -0
  13. package/scripts/publish.ts +60 -0
  14. package/src/3d/SpriteResourceManager.ts +286 -0
  15. package/src/3d/SpriteUtils.ts +71 -0
  16. package/src/3d/TextureUtils.ts +196 -0
  17. package/src/3d/ThreeRenderable.ts +197 -0
  18. package/src/3d/WGPURenderer.ts +294 -0
  19. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  20. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  21. package/src/3d/animation/SpriteAnimator.ts +633 -0
  22. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  23. package/src/3d/canvas.ts +464 -0
  24. package/src/3d/index.ts +12 -0
  25. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  26. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  27. package/src/3d/physics/physics-interface.ts +31 -0
  28. package/src/3d/shaders/supersampling.wgsl +201 -0
  29. package/src/3d.ts +3 -0
  30. package/src/NativeSpanFeed.ts +300 -0
  31. package/src/Renderable.ts +1698 -0
  32. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  33. package/src/animation/Timeline.test.ts +2709 -0
  34. package/src/animation/Timeline.ts +598 -0
  35. package/src/ansi.ts +18 -0
  36. package/src/benchmark/latest-all-bench-run.json +707 -0
  37. package/src/benchmark/latest-async-bench-run.json +336 -0
  38. package/src/benchmark/latest-default-bench-run.json +657 -0
  39. package/src/benchmark/latest-large-bench-run.json +707 -0
  40. package/src/benchmark/latest-quick-bench-run.json +207 -0
  41. package/src/benchmark/markdown-benchmark.ts +1804 -0
  42. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  43. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  44. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  45. package/src/benchmark/native-span-feed-compare.ts +280 -0
  46. package/src/benchmark/renderer-benchmark.ts +754 -0
  47. package/src/benchmark/text-table-benchmark.ts +947 -0
  48. package/src/buffer.test.ts +291 -0
  49. package/src/buffer.ts +519 -0
  50. package/src/console.test.ts +612 -0
  51. package/src/console.ts +1255 -0
  52. package/src/edit-buffer.test.ts +1769 -0
  53. package/src/edit-buffer.ts +411 -0
  54. package/src/editor-view.test.ts +1032 -0
  55. package/src/editor-view.ts +284 -0
  56. package/src/examples/ascii-font-selection-demo.ts +245 -0
  57. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  58. package/src/examples/assets/concrete.png +0 -0
  59. package/src/examples/assets/crate.png +0 -0
  60. package/src/examples/assets/crate_emissive.png +0 -0
  61. package/src/examples/assets/forrest_background.png +0 -0
  62. package/src/examples/assets/hast-example.json +1018 -0
  63. package/src/examples/assets/heart.png +0 -0
  64. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  65. package/src/examples/assets/main_char_idle.png +0 -0
  66. package/src/examples/assets/main_char_jump_end.png +0 -0
  67. package/src/examples/assets/main_char_jump_landing.png +0 -0
  68. package/src/examples/assets/main_char_jump_start.png +0 -0
  69. package/src/examples/assets/main_char_run_loop.png +0 -0
  70. package/src/examples/assets/roughness_map.jpg +0 -0
  71. package/src/examples/build.ts +115 -0
  72. package/src/examples/code-demo.ts +584 -0
  73. package/src/examples/console-demo.ts +358 -0
  74. package/src/examples/core-plugin-slots-demo.ts +759 -0
  75. package/src/examples/diff-demo.ts +699 -0
  76. package/src/examples/draggable-three-demo.ts +259 -0
  77. package/src/examples/editor-demo.ts +322 -0
  78. package/src/examples/extmarks-demo.ts +204 -0
  79. package/src/examples/focus-restore-demo.ts +310 -0
  80. package/src/examples/fonts.ts +245 -0
  81. package/src/examples/fractal-shader-demo.ts +268 -0
  82. package/src/examples/framebuffer-demo.ts +674 -0
  83. package/src/examples/full-unicode-demo.ts +181 -0
  84. package/src/examples/golden-star-demo.ts +933 -0
  85. package/src/examples/grayscale-buffer-demo.ts +249 -0
  86. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  87. package/src/examples/index.ts +925 -0
  88. package/src/examples/input-demo.ts +377 -0
  89. package/src/examples/input-select-layout-demo.ts +425 -0
  90. package/src/examples/install.sh +143 -0
  91. package/src/examples/keypress-debug-demo.ts +452 -0
  92. package/src/examples/lib/HexList.ts +122 -0
  93. package/src/examples/lib/PaletteGrid.ts +125 -0
  94. package/src/examples/lib/standalone-keys.ts +25 -0
  95. package/src/examples/lib/tab-controller.ts +243 -0
  96. package/src/examples/lights-phong-demo.ts +290 -0
  97. package/src/examples/link-demo.ts +220 -0
  98. package/src/examples/live-state-demo.ts +480 -0
  99. package/src/examples/markdown-demo.ts +620 -0
  100. package/src/examples/mouse-interaction-demo.ts +428 -0
  101. package/src/examples/nested-zindex-demo.ts +357 -0
  102. package/src/examples/opacity-example.ts +235 -0
  103. package/src/examples/opentui-demo.ts +1057 -0
  104. package/src/examples/physx-planck-2d-demo.ts +507 -0
  105. package/src/examples/physx-rapier-2d-demo.ts +526 -0
  106. package/src/examples/relative-positioning-demo.ts +323 -0
  107. package/src/examples/scroll-example.ts +214 -0
  108. package/src/examples/scrollbox-mouse-test.ts +112 -0
  109. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  110. package/src/examples/select-demo.ts +237 -0
  111. package/src/examples/shader-cube-demo.ts +772 -0
  112. package/src/examples/simple-layout-example.ts +591 -0
  113. package/src/examples/slider-demo.ts +617 -0
  114. package/src/examples/split-mode-demo.ts +445 -0
  115. package/src/examples/sprite-animation-demo.ts +443 -0
  116. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  117. package/src/examples/static-sprite-demo.ts +193 -0
  118. package/src/examples/sticky-scroll-example.ts +308 -0
  119. package/src/examples/styled-text-demo.ts +282 -0
  120. package/src/examples/tab-select-demo.ts +219 -0
  121. package/src/examples/terminal-title.ts +29 -0
  122. package/src/examples/terminal.ts +305 -0
  123. package/src/examples/text-node-demo.ts +416 -0
  124. package/src/examples/text-selection-demo.ts +377 -0
  125. package/src/examples/text-table-demo.ts +503 -0
  126. package/src/examples/text-truncation-demo.ts +481 -0
  127. package/src/examples/text-wrap.ts +757 -0
  128. package/src/examples/texture-loading-demo.ts +259 -0
  129. package/src/examples/timeline-example.ts +670 -0
  130. package/src/examples/transparency-demo.ts +241 -0
  131. package/src/examples/vnode-composition-demo.ts +404 -0
  132. package/src/index.ts +22 -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 +168 -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 +31 -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 +280 -0
  168. package/src/lib/keymapping.ts +87 -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 +1676 -0
  187. package/src/lib/stdin-parser.ts +1248 -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 +331 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +270 -0
  197. package/src/lib/tree-sitter/client.test.ts +1061 -0
  198. package/src/lib/tree-sitter/client.ts +615 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +80 -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 +1001 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +75 -0
  204. package/src/lib/tree-sitter/resolve-ft.ts +62 -0
  205. package/src/lib/tree-sitter/types.ts +81 -0
  206. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  207. package/src/lib/tree-sitter-styled-text.ts +306 -0
  208. package/src/lib/validate-dir-name.ts +55 -0
  209. package/src/lib/yoga.options.test.ts +628 -0
  210. package/src/lib/yoga.options.ts +346 -0
  211. package/src/plugins/core-slot.ts +579 -0
  212. package/src/plugins/registry.ts +377 -0
  213. package/src/plugins/types.ts +46 -0
  214. package/src/post/filters.ts +888 -0
  215. package/src/renderables/ASCIIFont.ts +219 -0
  216. package/src/renderables/Box.test.ts +160 -0
  217. package/src/renderables/Box.ts +295 -0
  218. package/src/renderables/Code.test.ts +2062 -0
  219. package/src/renderables/Code.ts +357 -0
  220. package/src/renderables/Diff.regression.test.ts +226 -0
  221. package/src/renderables/Diff.test.ts +3027 -0
  222. package/src/renderables/Diff.ts +1209 -0
  223. package/src/renderables/EditBufferRenderable.ts +764 -0
  224. package/src/renderables/FrameBuffer.ts +47 -0
  225. package/src/renderables/Input.test.ts +1228 -0
  226. package/src/renderables/Input.ts +245 -0
  227. package/src/renderables/LineNumberRenderable.ts +675 -0
  228. package/src/renderables/Markdown.ts +1106 -0
  229. package/src/renderables/ScrollBar.ts +422 -0
  230. package/src/renderables/ScrollBox.ts +883 -0
  231. package/src/renderables/Select.test.ts +1010 -0
  232. package/src/renderables/Select.ts +523 -0
  233. package/src/renderables/Slider.test.ts +456 -0
  234. package/src/renderables/Slider.ts +347 -0
  235. package/src/renderables/TabSelect.test.ts +197 -0
  236. package/src/renderables/TabSelect.ts +455 -0
  237. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  238. package/src/renderables/Text.test.ts +2660 -0
  239. package/src/renderables/Text.ts +147 -0
  240. package/src/renderables/TextBufferRenderable.ts +518 -0
  241. package/src/renderables/TextNode.test.ts +1058 -0
  242. package/src/renderables/TextNode.ts +325 -0
  243. package/src/renderables/TextTable.test.ts +1421 -0
  244. package/src/renderables/TextTable.ts +1344 -0
  245. package/src/renderables/Textarea.ts +732 -0
  246. package/src/renderables/TimeToFirstDraw.ts +89 -0
  247. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  248. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  249. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  250. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  251. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  252. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  253. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1787 -0
  254. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  255. package/src/renderables/__tests__/Markdown.test.ts +2287 -0
  256. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  257. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  258. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  259. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  260. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  261. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  262. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  263. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  264. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  265. package/src/renderables/__tests__/Textarea.rendering.test.ts +1864 -0
  266. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  267. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  268. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  269. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  270. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  271. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  272. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  273. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  274. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  275. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  276. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  277. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  278. package/src/renderables/composition/README.md +8 -0
  279. package/src/renderables/composition/VRenderable.ts +32 -0
  280. package/src/renderables/composition/constructs.ts +127 -0
  281. package/src/renderables/composition/vnode.ts +289 -0
  282. package/src/renderables/index.ts +22 -0
  283. package/src/renderables/markdown-parser.ts +66 -0
  284. package/src/renderer.ts +2363 -0
  285. package/src/runtime-plugin-support.ts +39 -0
  286. package/src/runtime-plugin.ts +144 -0
  287. package/src/syntax-style.test.ts +841 -0
  288. package/src/syntax-style.ts +264 -0
  289. package/src/testing/README.md +210 -0
  290. package/src/testing/capture-spans.test.ts +194 -0
  291. package/src/testing/integration.test.ts +276 -0
  292. package/src/testing/manual-clock.ts +106 -0
  293. package/src/testing/mock-keys.test.ts +1356 -0
  294. package/src/testing/mock-keys.ts +449 -0
  295. package/src/testing/mock-mouse.test.ts +218 -0
  296. package/src/testing/mock-mouse.ts +247 -0
  297. package/src/testing/mock-tree-sitter-client.ts +73 -0
  298. package/src/testing/spy.ts +13 -0
  299. package/src/testing/test-recorder.test.ts +415 -0
  300. package/src/testing/test-recorder.ts +145 -0
  301. package/src/testing/test-renderer.ts +116 -0
  302. package/src/testing.ts +7 -0
  303. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  304. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  305. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  306. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  307. package/src/tests/allocator-stats.test.ts +38 -0
  308. package/src/tests/destroy-during-render.test.ts +200 -0
  309. package/src/tests/hover-cursor.test.ts +98 -0
  310. package/src/tests/native-span-feed-async.test.ts +173 -0
  311. package/src/tests/native-span-feed-close.test.ts +120 -0
  312. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  313. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  314. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  315. package/src/tests/opacity.test.ts +123 -0
  316. package/src/tests/renderable.snapshot.test.ts +524 -0
  317. package/src/tests/renderable.test.ts +1281 -0
  318. package/src/tests/renderer.console-startup.test.ts +65 -0
  319. package/src/tests/renderer.control.test.ts +364 -0
  320. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  321. package/src/tests/renderer.cursor.test.ts +26 -0
  322. package/src/tests/renderer.destroy-during-render.test.ts +110 -0
  323. package/src/tests/renderer.focus-restore.test.ts +228 -0
  324. package/src/tests/renderer.focus.test.ts +251 -0
  325. package/src/tests/renderer.idle.test.ts +219 -0
  326. package/src/tests/renderer.input.test.ts +2145 -0
  327. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  328. package/src/tests/renderer.mouse.test.ts +1269 -0
  329. package/src/tests/renderer.palette.test.ts +629 -0
  330. package/src/tests/renderer.selection.test.ts +49 -0
  331. package/src/tests/renderer.slot-registry.test.ts +649 -0
  332. package/src/tests/renderer.useMouse.test.ts +50 -0
  333. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  334. package/src/tests/runtime-plugin-support.test.ts +28 -0
  335. package/src/tests/runtime-plugin.fixture.ts +40 -0
  336. package/src/tests/runtime-plugin.test.ts +190 -0
  337. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  338. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  339. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  340. package/src/tests/scrollbox.test.ts +1530 -0
  341. package/src/tests/wrap-resize-perf.test.ts +229 -0
  342. package/src/tests/yoga-setters.test.ts +921 -0
  343. package/src/text-buffer-view.test.ts +705 -0
  344. package/src/text-buffer-view.ts +189 -0
  345. package/src/text-buffer.test.ts +347 -0
  346. package/src/text-buffer.ts +250 -0
  347. package/src/types.ts +152 -0
  348. package/src/utils.ts +88 -0
  349. package/src/zig/ansi.zig +268 -0
  350. package/src/zig/bench/README.md +50 -0
  351. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  352. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  353. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  354. package/src/zig/bench/rope-markers_bench.zig +713 -0
  355. package/src/zig/bench/rope_bench.zig +514 -0
  356. package/src/zig/bench/styled-text_bench.zig +470 -0
  357. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  358. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  359. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  360. package/src/zig/bench/utf8_bench.zig +799 -0
  361. package/src/zig/bench-utils.zig +431 -0
  362. package/src/zig/bench.zig +217 -0
  363. package/src/zig/buffer.zig +2223 -0
  364. package/src/zig/build.zig +289 -0
  365. package/src/zig/build.zig.zon +16 -0
  366. package/src/zig/edit-buffer.zig +825 -0
  367. package/src/zig/editor-view.zig +802 -0
  368. package/src/zig/event-bus.zig +13 -0
  369. package/src/zig/event-emitter.zig +65 -0
  370. package/src/zig/file-logger.zig +92 -0
  371. package/src/zig/grapheme.zig +599 -0
  372. package/src/zig/lib.zig +1834 -0
  373. package/src/zig/link.zig +333 -0
  374. package/src/zig/logger.zig +43 -0
  375. package/src/zig/mem-registry.zig +125 -0
  376. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  377. package/src/zig/native-span-feed.zig +708 -0
  378. package/src/zig/renderer.zig +1386 -0
  379. package/src/zig/rope.zig +1220 -0
  380. package/src/zig/syntax-style.zig +161 -0
  381. package/src/zig/terminal.zig +975 -0
  382. package/src/zig/test.zig +70 -0
  383. package/src/zig/tests/README.md +18 -0
  384. package/src/zig/tests/buffer_test.zig +2526 -0
  385. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  386. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  387. package/src/zig/tests/editor-view_test.zig +3299 -0
  388. package/src/zig/tests/event-emitter_test.zig +249 -0
  389. package/src/zig/tests/grapheme_test.zig +1304 -0
  390. package/src/zig/tests/link_test.zig +190 -0
  391. package/src/zig/tests/mem-registry_test.zig +473 -0
  392. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  393. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  394. package/src/zig/tests/renderer_test.zig +1010 -0
  395. package/src/zig/tests/rope-nested_test.zig +712 -0
  396. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  397. package/src/zig/tests/rope_test.zig +2362 -0
  398. package/src/zig/tests/segment-merge.test.zig +148 -0
  399. package/src/zig/tests/syntax-style_test.zig +557 -0
  400. package/src/zig/tests/terminal_test.zig +719 -0
  401. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  402. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  403. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  404. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  405. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  406. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  407. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  408. package/src/zig/tests/text-buffer_test.zig +2191 -0
  409. package/src/zig/tests/unicode-width-map.zon +3909 -0
  410. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  411. package/src/zig/tests/utf8_test.zig +4057 -0
  412. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  413. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  414. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  415. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  416. package/src/zig/text-buffer-iterators.zig +499 -0
  417. package/src/zig/text-buffer-segment.zig +404 -0
  418. package/src/zig/text-buffer-view.zig +1371 -0
  419. package/src/zig/text-buffer.zig +1180 -0
  420. package/src/zig/utf8.zig +1948 -0
  421. package/src/zig/utils.zig +9 -0
  422. package/src/zig-structs.ts +261 -0
  423. package/src/zig.ts +3843 -0
  424. package/tsconfig.build.json +22 -0
  425. package/tsconfig.json +28 -0
  426. package/3d/SpriteResourceManager.d.ts +0 -74
  427. package/3d/SpriteUtils.d.ts +0 -13
  428. package/3d/TextureUtils.d.ts +0 -24
  429. package/3d/ThreeRenderable.d.ts +0 -40
  430. package/3d/WGPURenderer.d.ts +0 -61
  431. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  432. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  433. package/3d/animation/SpriteAnimator.d.ts +0 -124
  434. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  435. package/3d/canvas.d.ts +0 -44
  436. package/3d/index.d.ts +0 -12
  437. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  438. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  439. package/3d/physics/physics-interface.d.ts +0 -27
  440. package/3d.d.ts +0 -2
  441. package/3d.js +0 -34042
  442. package/3d.js.map +0 -155
  443. package/LICENSE +0 -21
  444. package/NativeSpanFeed.d.ts +0 -41
  445. package/Renderable.d.ts +0 -334
  446. package/animation/Timeline.d.ts +0 -126
  447. package/ansi.d.ts +0 -13
  448. package/buffer.d.ts +0 -107
  449. package/console.d.ts +0 -143
  450. package/edit-buffer.d.ts +0 -98
  451. package/editor-view.d.ts +0 -73
  452. package/index-e4hzc2j2.js +0 -113
  453. package/index-e4hzc2j2.js.map +0 -10
  454. package/index-nkrr8a4c.js +0 -18415
  455. package/index-nkrr8a4c.js.map +0 -64
  456. package/index-nyw5p3ep.js +0 -12619
  457. package/index-nyw5p3ep.js.map +0 -43
  458. package/index.d.ts +0 -21
  459. package/index.js +0 -430
  460. package/index.js.map +0 -9
  461. package/lib/KeyHandler.d.ts +0 -61
  462. package/lib/RGBA.d.ts +0 -25
  463. package/lib/ascii.font.d.ts +0 -508
  464. package/lib/border.d.ts +0 -49
  465. package/lib/bunfs.d.ts +0 -7
  466. package/lib/clipboard.d.ts +0 -17
  467. package/lib/clock.d.ts +0 -15
  468. package/lib/data-paths.d.ts +0 -26
  469. package/lib/debounce.d.ts +0 -42
  470. package/lib/detect-links.d.ts +0 -6
  471. package/lib/env.d.ts +0 -42
  472. package/lib/extmarks-history.d.ts +0 -17
  473. package/lib/extmarks.d.ts +0 -89
  474. package/lib/hast-styled-text.d.ts +0 -17
  475. package/lib/index.d.ts +0 -21
  476. package/lib/keymapping.d.ts +0 -25
  477. package/lib/objects-in-viewport.d.ts +0 -24
  478. package/lib/output.capture.d.ts +0 -24
  479. package/lib/parse.keypress-kitty.d.ts +0 -2
  480. package/lib/parse.keypress.d.ts +0 -26
  481. package/lib/parse.mouse.d.ts +0 -30
  482. package/lib/paste.d.ts +0 -7
  483. package/lib/queue.d.ts +0 -15
  484. package/lib/renderable.validations.d.ts +0 -12
  485. package/lib/scroll-acceleration.d.ts +0 -43
  486. package/lib/selection.d.ts +0 -63
  487. package/lib/singleton.d.ts +0 -7
  488. package/lib/stdin-parser.d.ts +0 -76
  489. package/lib/styled-text.d.ts +0 -63
  490. package/lib/terminal-capability-detection.d.ts +0 -30
  491. package/lib/terminal-palette.d.ts +0 -50
  492. package/lib/tree-sitter/assets/update.d.ts +0 -11
  493. package/lib/tree-sitter/client.d.ts +0 -47
  494. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  495. package/lib/tree-sitter/download-utils.d.ts +0 -21
  496. package/lib/tree-sitter/index.d.ts +0 -8
  497. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  498. package/lib/tree-sitter/parsers-config.d.ts +0 -38
  499. package/lib/tree-sitter/resolve-ft.d.ts +0 -2
  500. package/lib/tree-sitter/types.d.ts +0 -81
  501. package/lib/tree-sitter-styled-text.d.ts +0 -14
  502. package/lib/validate-dir-name.d.ts +0 -1
  503. package/lib/yoga.options.d.ts +0 -32
  504. package/parser.worker.js +0 -869
  505. package/parser.worker.js.map +0 -12
  506. package/plugins/core-slot.d.ts +0 -72
  507. package/plugins/registry.d.ts +0 -38
  508. package/plugins/types.d.ts +0 -34
  509. package/post/filters.d.ts +0 -105
  510. package/renderables/ASCIIFont.d.ts +0 -52
  511. package/renderables/Box.d.ts +0 -72
  512. package/renderables/Code.d.ts +0 -78
  513. package/renderables/Diff.d.ts +0 -142
  514. package/renderables/EditBufferRenderable.d.ts +0 -162
  515. package/renderables/FrameBuffer.d.ts +0 -16
  516. package/renderables/Input.d.ts +0 -67
  517. package/renderables/LineNumberRenderable.d.ts +0 -74
  518. package/renderables/Markdown.d.ts +0 -173
  519. package/renderables/ScrollBar.d.ts +0 -77
  520. package/renderables/ScrollBox.d.ts +0 -124
  521. package/renderables/Select.d.ts +0 -115
  522. package/renderables/Slider.d.ts +0 -44
  523. package/renderables/TabSelect.d.ts +0 -96
  524. package/renderables/Text.d.ts +0 -36
  525. package/renderables/TextBufferRenderable.d.ts +0 -105
  526. package/renderables/TextNode.d.ts +0 -91
  527. package/renderables/TextTable.d.ts +0 -140
  528. package/renderables/Textarea.d.ts +0 -114
  529. package/renderables/TimeToFirstDraw.d.ts +0 -24
  530. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  531. package/renderables/composition/VRenderable.d.ts +0 -16
  532. package/renderables/composition/constructs.d.ts +0 -35
  533. package/renderables/composition/vnode.d.ts +0 -46
  534. package/renderables/index.d.ts +0 -22
  535. package/renderables/markdown-parser.d.ts +0 -10
  536. package/renderer.d.ts +0 -388
  537. package/runtime-plugin-support.d.ts +0 -3
  538. package/runtime-plugin-support.js +0 -29
  539. package/runtime-plugin-support.js.map +0 -10
  540. package/runtime-plugin.d.ts +0 -11
  541. package/runtime-plugin.js +0 -16
  542. package/runtime-plugin.js.map +0 -9
  543. package/syntax-style.d.ts +0 -54
  544. package/testing/manual-clock.d.ts +0 -16
  545. package/testing/mock-keys.d.ts +0 -81
  546. package/testing/mock-mouse.d.ts +0 -38
  547. package/testing/mock-tree-sitter-client.d.ts +0 -23
  548. package/testing/spy.d.ts +0 -7
  549. package/testing/test-recorder.d.ts +0 -61
  550. package/testing/test-renderer.d.ts +0 -23
  551. package/testing.d.ts +0 -6
  552. package/testing.js +0 -675
  553. package/testing.js.map +0 -15
  554. package/text-buffer-view.d.ts +0 -42
  555. package/text-buffer.d.ts +0 -67
  556. package/types.d.ts +0 -131
  557. package/utils.d.ts +0 -14
  558. package/zig-structs.d.ts +0 -155
  559. package/zig.d.ts +0 -351
  560. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  561. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  562. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  563. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  564. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  565. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  566. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  567. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  568. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  569. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  570. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,2362 @@
1
+ const std = @import("std");
2
+ const rope_mod = @import("../rope.zig");
3
+
4
+ const SimpleItem = struct {
5
+ value: u32,
6
+
7
+ pub fn empty() SimpleItem {
8
+ return .{ .value = 0 };
9
+ }
10
+
11
+ pub fn is_empty(self: *const SimpleItem) bool {
12
+ return self.value == 0;
13
+ }
14
+ };
15
+
16
+ const ItemWithMetrics = struct {
17
+ value: u32,
18
+ size: u32,
19
+
20
+ pub const Metrics = struct {
21
+ total_size: u32 = 0,
22
+
23
+ pub fn add(self: *Metrics, other: Metrics) void {
24
+ self.total_size += other.total_size;
25
+ }
26
+ };
27
+
28
+ pub fn measure(self: *const ItemWithMetrics) Metrics {
29
+ return .{ .total_size = self.size };
30
+ }
31
+
32
+ pub fn empty() ItemWithMetrics {
33
+ return .{ .value = 0, .size = 0 };
34
+ }
35
+
36
+ pub fn is_empty(self: *const ItemWithMetrics) bool {
37
+ return self.value == 0 and self.size == 0;
38
+ }
39
+ };
40
+
41
+ //===== Basic Rope Tests =====
42
+
43
+ test "Rope - can initialize with arena allocator" {
44
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
45
+ defer arena.deinit();
46
+
47
+ const RopeType = rope_mod.Rope(SimpleItem);
48
+ var rope = try RopeType.init(arena.allocator());
49
+ try std.testing.expectEqual(@as(u32, 0), rope.count()); // Sentinel filtered
50
+ }
51
+
52
+ test "Rope - from_item creates single item rope" {
53
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
54
+ defer arena.deinit();
55
+
56
+ const RopeType = rope_mod.Rope(SimpleItem);
57
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 42 });
58
+
59
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
60
+ const item = rope.get(0);
61
+ try std.testing.expect(item != null);
62
+ try std.testing.expectEqual(@as(u32, 42), item.?.value);
63
+ }
64
+
65
+ test "Rope - from_slice creates rope from multiple items" {
66
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
67
+ defer arena.deinit();
68
+
69
+ const RopeType = rope_mod.Rope(SimpleItem);
70
+ const items = [_]SimpleItem{
71
+ .{ .value = 1 },
72
+ .{ .value = 2 },
73
+ .{ .value = 3 },
74
+ };
75
+
76
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
77
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
78
+
79
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
80
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
81
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
82
+ }
83
+
84
+ test "Rope - get with out of bounds returns null" {
85
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
86
+ defer arena.deinit();
87
+
88
+ const RopeType = rope_mod.Rope(SimpleItem);
89
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
90
+
91
+ try std.testing.expect(rope.get(100) == null);
92
+ }
93
+
94
+ //===== Insert Tests =====
95
+
96
+ test "Rope - insert at beginning" {
97
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
98
+ defer arena.deinit();
99
+
100
+ const RopeType = rope_mod.Rope(SimpleItem);
101
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
102
+
103
+ try rope.insert(0, .{ .value = 0 });
104
+
105
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
106
+ try std.testing.expectEqual(@as(u32, 0), rope.get(0).?.value);
107
+ try std.testing.expectEqual(@as(u32, 1), rope.get(1).?.value);
108
+ }
109
+
110
+ test "Rope - insert at end" {
111
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
112
+ defer arena.deinit();
113
+
114
+ const RopeType = rope_mod.Rope(SimpleItem);
115
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
116
+
117
+ try rope.insert(1, .{ .value = 2 });
118
+
119
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
120
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
121
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
122
+ }
123
+
124
+ test "Rope - multiple inserts" {
125
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
126
+ defer arena.deinit();
127
+
128
+ const RopeType = rope_mod.Rope(SimpleItem);
129
+ var rope = try RopeType.init(arena.allocator());
130
+
131
+ try rope.insert(0, .{ .value = 1 });
132
+ try rope.insert(1, .{ .value = 2 });
133
+ try rope.insert(2, .{ .value = 3 });
134
+
135
+ try std.testing.expectEqual(@as(u32, 3), rope.count()); // Sentinel filtered
136
+ }
137
+
138
+ //===== Delete Tests =====
139
+
140
+ test "Rope - delete at beginning" {
141
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
142
+ defer arena.deinit();
143
+
144
+ const RopeType = rope_mod.Rope(SimpleItem);
145
+ const items = [_]SimpleItem{
146
+ .{ .value = 1 },
147
+ .{ .value = 2 },
148
+ .{ .value = 3 },
149
+ };
150
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
151
+
152
+ try rope.delete(0);
153
+
154
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
155
+ try std.testing.expectEqual(@as(u32, 2), rope.get(0).?.value);
156
+ }
157
+
158
+ test "Rope - delete in middle" {
159
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
160
+ defer arena.deinit();
161
+
162
+ const RopeType = rope_mod.Rope(SimpleItem);
163
+ const items = [_]SimpleItem{
164
+ .{ .value = 1 },
165
+ .{ .value = 2 },
166
+ .{ .value = 3 },
167
+ };
168
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
169
+
170
+ try rope.delete(1);
171
+
172
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
173
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
174
+ try std.testing.expectEqual(@as(u32, 3), rope.get(1).?.value);
175
+ }
176
+
177
+ //===== Walk Tests =====
178
+
179
+ test "Rope - walk all items" {
180
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
181
+ defer arena.deinit();
182
+
183
+ const RopeType = rope_mod.Rope(SimpleItem);
184
+ const items = [_]SimpleItem{
185
+ .{ .value = 10 },
186
+ .{ .value = 20 },
187
+ .{ .value = 30 },
188
+ };
189
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
190
+
191
+ const Context = struct {
192
+ sum: u32 = 0,
193
+
194
+ fn walker(ctx: *anyopaque, data: *const SimpleItem, index: u32) RopeType.Node.WalkerResult {
195
+ _ = index;
196
+ const self = @as(*@This(), @ptrCast(@alignCast(ctx)));
197
+ self.sum += data.value;
198
+ return .{};
199
+ }
200
+ };
201
+
202
+ var ctx = Context{};
203
+ try rope.walk(&ctx, Context.walker);
204
+
205
+ try std.testing.expectEqual(@as(u32, 60), ctx.sum);
206
+ }
207
+
208
+ test "Rope - walk with early exit" {
209
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
210
+ defer arena.deinit();
211
+
212
+ const RopeType = rope_mod.Rope(SimpleItem);
213
+ const items = [_]SimpleItem{
214
+ .{ .value = 1 },
215
+ .{ .value = 2 },
216
+ .{ .value = 3 },
217
+ };
218
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
219
+
220
+ const Context = struct {
221
+ count: u32 = 0,
222
+
223
+ fn walker(ctx: *anyopaque, data: *const SimpleItem, index: u32) RopeType.Node.WalkerResult {
224
+ _ = index;
225
+ const self = @as(*@This(), @ptrCast(@alignCast(ctx)));
226
+ self.count += 1;
227
+ if (data.value == 2) {
228
+ return .{ .keep_walking = false };
229
+ }
230
+ return .{};
231
+ }
232
+ };
233
+
234
+ var ctx = Context{};
235
+ try rope.walk(&ctx, Context.walker);
236
+
237
+ try std.testing.expectEqual(@as(u32, 2), ctx.count);
238
+ }
239
+
240
+ //===== Metrics Tests =====
241
+
242
+ test "Rope - custom metrics are tracked" {
243
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
244
+ defer arena.deinit();
245
+
246
+ const RopeType = rope_mod.Rope(ItemWithMetrics);
247
+ const items = [_]ItemWithMetrics{
248
+ .{ .value = 1, .size = 10 },
249
+ .{ .value = 2, .size = 20 },
250
+ .{ .value = 3, .size = 30 },
251
+ };
252
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
253
+
254
+ const metrics = rope.root.metrics();
255
+ try std.testing.expectEqual(@as(u32, 3), metrics.count);
256
+ try std.testing.expectEqual(@as(u32, 60), metrics.custom.total_size);
257
+ }
258
+
259
+ test "Rope - rebalance maintains data" {
260
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
261
+ defer arena.deinit();
262
+
263
+ const RopeType = rope_mod.Rope(SimpleItem);
264
+ var items: [20]SimpleItem = undefined;
265
+ for (&items, 0..) |*item, i| {
266
+ item.* = .{ .value = @intCast(i) };
267
+ }
268
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
269
+
270
+ try rope.rebalance(arena.allocator());
271
+
272
+ // Data should be preserved
273
+ try std.testing.expectEqual(@as(u32, 20), rope.count());
274
+ try std.testing.expectEqual(@as(u32, 0), rope.get(0).?.value);
275
+ try std.testing.expectEqual(@as(u32, 10), rope.get(10).?.value);
276
+ try std.testing.expectEqual(@as(u32, 19), rope.get(19).?.value);
277
+ }
278
+
279
+ //===== Stress Tests =====
280
+
281
+ test "Rope - large number of items" {
282
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
283
+ defer arena.deinit();
284
+
285
+ const RopeType = rope_mod.Rope(SimpleItem);
286
+ var items: [100]SimpleItem = undefined;
287
+ for (&items, 0..) |*item, i| {
288
+ item.* = .{ .value = @intCast(i) };
289
+ }
290
+
291
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
292
+
293
+ try std.testing.expectEqual(@as(u32, 100), rope.count());
294
+ try std.testing.expectEqual(@as(u32, 0), rope.get(0).?.value);
295
+ try std.testing.expectEqual(@as(u32, 50), rope.get(50).?.value);
296
+ try std.testing.expectEqual(@as(u32, 99), rope.get(99).?.value);
297
+ }
298
+
299
+ test "Rope - many insert operations" {
300
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
301
+ defer arena.deinit();
302
+
303
+ const RopeType = rope_mod.Rope(SimpleItem);
304
+ var rope = try RopeType.init(arena.allocator());
305
+
306
+ for (0..50) |i| {
307
+ try rope.insert(@intCast(i), .{ .value = @intCast(i) });
308
+ }
309
+
310
+ try std.testing.expectEqual(@as(u32, 50), rope.count()); // Sentinel filtered
311
+ }
312
+
313
+ //===== Nested Rope Tests (Lines→Chunks Pattern) =====
314
+
315
+ // Chunk type similar to what would be used in text buffer
316
+ const Chunk = struct {
317
+ data: []const u8,
318
+ width: u32,
319
+
320
+ pub const Metrics = struct {
321
+ total_width: u32 = 0,
322
+ total_bytes: u32 = 0,
323
+
324
+ pub fn add(self: *Metrics, other: Metrics) void {
325
+ self.total_width += other.total_width;
326
+ self.total_bytes += other.total_bytes;
327
+ }
328
+ };
329
+
330
+ pub fn measure(self: *const Chunk) Metrics {
331
+ return .{
332
+ .total_width = self.width,
333
+ .total_bytes = @intCast(self.data.len),
334
+ };
335
+ }
336
+
337
+ pub fn empty() Chunk {
338
+ return .{ .data = "", .width = 0 };
339
+ }
340
+
341
+ pub fn is_empty(self: *const Chunk) bool {
342
+ return self.data.len == 0;
343
+ }
344
+ };
345
+
346
+ // Static empty chunk rope node for Line.empty()
347
+ const empty_chunk_leaf_node = rope_mod.Rope(Chunk).Node{
348
+ .leaf = .{
349
+ .data = Chunk.empty(),
350
+ },
351
+ };
352
+
353
+ // Line type containing a rope of chunks
354
+ const Line = struct {
355
+ chunks: rope_mod.Rope(Chunk),
356
+ line_id: u32,
357
+
358
+ pub const Metrics = struct {
359
+ total_width: u32 = 0,
360
+ total_lines: u32 = 1,
361
+
362
+ pub fn add(self: *Metrics, other: Metrics) void {
363
+ self.total_width += other.total_width;
364
+ self.total_lines += other.total_lines;
365
+ }
366
+ };
367
+
368
+ pub fn measure(self: *const Line) Metrics {
369
+ const chunk_metrics = self.chunks.root.metrics();
370
+ return .{
371
+ .total_width = chunk_metrics.custom.total_width,
372
+ .total_lines = 1,
373
+ };
374
+ }
375
+
376
+ pub fn empty() Line {
377
+ // Use static empty chunk rope - safe because it's immutable
378
+ const ChunkRope = rope_mod.Rope(Chunk);
379
+ return .{
380
+ .chunks = .{
381
+ .root = &empty_chunk_leaf_node,
382
+ .allocator = undefined, // Never used for empty
383
+ .empty_leaf = &empty_chunk_leaf_node,
384
+ .marker_cache = ChunkRope.MarkerCache.init(undefined),
385
+ },
386
+ .line_id = 0,
387
+ };
388
+ }
389
+
390
+ pub fn is_empty(self: *const Line) bool {
391
+ return self.line_id == 0 and self.chunks.count() == 1;
392
+ }
393
+ };
394
+
395
+ test "Rope - replace item at index" {
396
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
397
+ defer arena.deinit();
398
+
399
+ const RopeType = rope_mod.Rope(SimpleItem);
400
+ const items = [_]SimpleItem{
401
+ .{ .value = 1 },
402
+ .{ .value = 2 },
403
+ .{ .value = 3 },
404
+ };
405
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
406
+
407
+ try rope.replace(1, .{ .value = 20 });
408
+
409
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
410
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
411
+ try std.testing.expectEqual(@as(u32, 20), rope.get(1).?.value);
412
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
413
+ }
414
+
415
+ test "Rope - append item" {
416
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
417
+ defer arena.deinit();
418
+
419
+ const RopeType = rope_mod.Rope(SimpleItem);
420
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
421
+
422
+ try rope.append(.{ .value = 2 });
423
+ try rope.append(.{ .value = 3 });
424
+
425
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
426
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
427
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
428
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
429
+ }
430
+
431
+ test "Rope - prepend item" {
432
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
433
+ defer arena.deinit();
434
+
435
+ const RopeType = rope_mod.Rope(SimpleItem);
436
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 3 });
437
+
438
+ try rope.prepend(.{ .value = 2 });
439
+ try rope.prepend(.{ .value = 1 });
440
+
441
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
442
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
443
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
444
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
445
+ }
446
+
447
+ test "Rope - concatenate two ropes" {
448
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
449
+ defer arena.deinit();
450
+
451
+ const RopeType = rope_mod.Rope(SimpleItem);
452
+
453
+ const items1 = [_]SimpleItem{
454
+ .{ .value = 1 },
455
+ .{ .value = 2 },
456
+ };
457
+ var rope1 = try RopeType.from_slice(arena.allocator(), &items1);
458
+
459
+ const items2 = [_]SimpleItem{
460
+ .{ .value = 3 },
461
+ .{ .value = 4 },
462
+ };
463
+ const rope2 = try RopeType.from_slice(arena.allocator(), &items2);
464
+
465
+ try rope1.concat(&rope2);
466
+
467
+ try std.testing.expectEqual(@as(u32, 4), rope1.count());
468
+ try std.testing.expectEqual(@as(u32, 1), rope1.get(0).?.value);
469
+ try std.testing.expectEqual(@as(u32, 2), rope1.get(1).?.value);
470
+ try std.testing.expectEqual(@as(u32, 3), rope1.get(2).?.value);
471
+ try std.testing.expectEqual(@as(u32, 4), rope1.get(3).?.value);
472
+ }
473
+
474
+ //===== Undo/Redo Tests =====
475
+
476
+ test "Rope - basic undo operation" {
477
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
478
+ defer arena.deinit();
479
+
480
+ const RopeType = rope_mod.Rope(SimpleItem);
481
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
482
+
483
+ try rope.store_undo("initial");
484
+
485
+ try rope.insert(1, .{ .value = 2 });
486
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
487
+
488
+ const meta = try rope.undo("before undo");
489
+ try std.testing.expectEqualStrings("initial", meta);
490
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
491
+ }
492
+
493
+ test "Rope - basic redo operation" {
494
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
495
+ defer arena.deinit();
496
+
497
+ const RopeType = rope_mod.Rope(SimpleItem);
498
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
499
+
500
+ try rope.store_undo("initial");
501
+ try rope.insert(1, .{ .value = 2 });
502
+
503
+ _ = try rope.undo("before undo");
504
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
505
+
506
+ const meta = try rope.redo();
507
+ try std.testing.expectEqualStrings("before undo", meta);
508
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
509
+ }
510
+
511
+ test "Rope - multiple undo/redo operations" {
512
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
513
+ defer arena.deinit();
514
+
515
+ const RopeType = rope_mod.Rope(SimpleItem);
516
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
517
+
518
+ try rope.store_undo("state1");
519
+ try rope.append(.{ .value = 2 });
520
+
521
+ try rope.store_undo("state2");
522
+ try rope.append(.{ .value = 3 });
523
+
524
+ try rope.store_undo("state3");
525
+ try rope.append(.{ .value = 4 });
526
+
527
+ try std.testing.expectEqual(@as(u32, 4), rope.count());
528
+
529
+ _ = try rope.undo("current");
530
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
531
+
532
+ _ = try rope.undo("current");
533
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
534
+
535
+ _ = try rope.redo();
536
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
537
+ }
538
+
539
+ test "Rope - undo/redo with delete operations" {
540
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
541
+ defer arena.deinit();
542
+
543
+ const RopeType = rope_mod.Rope(SimpleItem);
544
+ const items = [_]SimpleItem{
545
+ .{ .value = 1 },
546
+ .{ .value = 2 },
547
+ .{ .value = 3 },
548
+ };
549
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
550
+
551
+ try rope.store_undo("before delete");
552
+ try rope.delete(1);
553
+
554
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
555
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
556
+ try std.testing.expectEqual(@as(u32, 3), rope.get(1).?.value);
557
+
558
+ _ = try rope.undo("after delete");
559
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
560
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
561
+ }
562
+
563
+ test "Rope - undo/redo with replace operations" {
564
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
565
+ defer arena.deinit();
566
+
567
+ const RopeType = rope_mod.Rope(SimpleItem);
568
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 10 });
569
+
570
+ try rope.store_undo("original");
571
+ try rope.replace(0, .{ .value = 20 });
572
+ try std.testing.expectEqual(@as(u32, 20), rope.get(0).?.value);
573
+
574
+ _ = try rope.undo("after replace");
575
+ try std.testing.expectEqual(@as(u32, 10), rope.get(0).?.value);
576
+
577
+ _ = try rope.redo();
578
+ try std.testing.expectEqual(@as(u32, 20), rope.get(0).?.value);
579
+ }
580
+
581
+ test "Rope - can_undo and can_redo" {
582
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
583
+ defer arena.deinit();
584
+
585
+ const RopeType = rope_mod.Rope(SimpleItem);
586
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
587
+
588
+ try std.testing.expect(!rope.can_undo());
589
+ try std.testing.expect(!rope.can_redo());
590
+
591
+ try rope.store_undo("state1");
592
+ try std.testing.expect(rope.can_undo());
593
+ try std.testing.expect(!rope.can_redo());
594
+
595
+ _ = try rope.undo("current");
596
+ try std.testing.expect(!rope.can_undo()); // No more undo (only one state)
597
+ try std.testing.expect(rope.can_redo());
598
+ }
599
+
600
+ test "Rope - clear history" {
601
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
602
+ defer arena.deinit();
603
+
604
+ const RopeType = rope_mod.Rope(SimpleItem);
605
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
606
+
607
+ try rope.store_undo("state1");
608
+ try rope.append(.{ .value = 2 });
609
+ try rope.store_undo("state2");
610
+
611
+ try std.testing.expect(rope.can_undo());
612
+
613
+ rope.clear_history();
614
+ try std.testing.expect(!rope.can_undo());
615
+ try std.testing.expect(!rope.can_redo());
616
+ }
617
+
618
+ test "Rope - undo fails when no history" {
619
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
620
+ defer arena.deinit();
621
+
622
+ const RopeType = rope_mod.Rope(SimpleItem);
623
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
624
+
625
+ // No history stored, undo should fail
626
+ const result = rope.undo("current");
627
+ try std.testing.expectError(error.Stop, result);
628
+ }
629
+
630
+ test "Rope - redo fails when no redo history" {
631
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
632
+ defer arena.deinit();
633
+
634
+ const RopeType = rope_mod.Rope(SimpleItem);
635
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
636
+
637
+ // No redo history, redo should fail
638
+ const result = rope.redo();
639
+ try std.testing.expectError(error.Stop, result);
640
+ }
641
+
642
+ test "Rope - complex undo/redo workflow" {
643
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
644
+ defer arena.deinit();
645
+
646
+ const RopeType = rope_mod.Rope(SimpleItem);
647
+ var rope = try RopeType.init(arena.allocator());
648
+
649
+ // Build up a sequence of operations
650
+ try rope.store_undo("empty");
651
+ try rope.insert(0, .{ .value = 1 });
652
+
653
+ try rope.store_undo("one item");
654
+ try rope.append(.{ .value = 2 });
655
+
656
+ try rope.store_undo("two items");
657
+ try rope.append(.{ .value = 3 });
658
+
659
+ try rope.store_undo("three items");
660
+ try rope.delete(1); // Remove middle
661
+
662
+ // State: [1, 3]
663
+ try std.testing.expectEqual(@as(u32, 2), rope.count()); // Sentinel filtered
664
+
665
+ // Undo delete
666
+ _ = try rope.undo("current");
667
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
668
+
669
+ // Undo append
670
+ _ = try rope.undo("current");
671
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
672
+
673
+ // Redo append
674
+ _ = try rope.redo();
675
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
676
+ }
677
+
678
+ test "Rope - undo/redo with metadata tracking" {
679
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
680
+ defer arena.deinit();
681
+
682
+ const RopeType = rope_mod.Rope(SimpleItem);
683
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
684
+
685
+ try rope.store_undo("insert operation");
686
+ try rope.append(.{ .value = 2 });
687
+
688
+ try rope.store_undo("delete operation");
689
+ try rope.delete(0);
690
+
691
+ // Undo and check metadata
692
+ const meta1 = try rope.undo("current state");
693
+ try std.testing.expectEqualStrings("delete operation", meta1);
694
+
695
+ const meta2 = try rope.undo("current state");
696
+ try std.testing.expectEqualStrings("insert operation", meta2);
697
+ }
698
+
699
+ test "Rope - undo invalidates redo after new operation" {
700
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
701
+ defer arena.deinit();
702
+
703
+ const RopeType = rope_mod.Rope(SimpleItem);
704
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
705
+
706
+ try rope.store_undo("state1");
707
+ try rope.append(.{ .value = 2 });
708
+
709
+ try rope.store_undo("state2");
710
+ try rope.append(.{ .value = 3 });
711
+
712
+ // Undo once
713
+ _ = try rope.undo("current");
714
+ try std.testing.expect(rope.can_redo());
715
+
716
+ // Make a new change - this stores the old redo as a branch and clears redo
717
+ try rope.store_undo("new branch");
718
+ try rope.append(.{ .value = 99 });
719
+
720
+ // Redo should NOT work anymore (it was saved as a branch)
721
+ try std.testing.expect(!rope.can_redo());
722
+
723
+ // But we can still undo
724
+ try std.testing.expect(rope.can_undo());
725
+ }
726
+
727
+ test "Rope - undo/redo with nested ropes" {
728
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
729
+ defer arena.deinit();
730
+ const allocator = arena.allocator();
731
+
732
+ const chunks1 = [_]Chunk{.{ .data = "Line 1", .width = 6 }};
733
+ const line1 = Line{
734
+ .chunks = try rope_mod.Rope(Chunk).from_slice(allocator, &chunks1),
735
+ .line_id = 1,
736
+ };
737
+
738
+ const RopeType = rope_mod.Rope(Line);
739
+ var rope = try RopeType.from_item(allocator, line1);
740
+
741
+ try rope.store_undo("before append");
742
+
743
+ const chunks2 = [_]Chunk{.{ .data = "Line 2", .width = 6 }};
744
+ const line2 = Line{
745
+ .chunks = try rope_mod.Rope(Chunk).from_slice(allocator, &chunks2),
746
+ .line_id = 2,
747
+ };
748
+ try rope.append(line2);
749
+
750
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
751
+
752
+ // Undo
753
+ _ = try rope.undo("after append");
754
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
755
+
756
+ // Redo
757
+ _ = try rope.redo();
758
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
759
+ }
760
+
761
+ test "Rope - stress test undo/redo with many operations" {
762
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
763
+ defer arena.deinit();
764
+
765
+ const RopeType = rope_mod.Rope(SimpleItem);
766
+ var rope = try RopeType.init(arena.allocator());
767
+
768
+ // Perform 20 operations
769
+ for (0..20) |i| {
770
+ try rope.store_undo("operation");
771
+ try rope.append(.{ .value = @intCast(i) });
772
+ }
773
+
774
+ try std.testing.expectEqual(@as(u32, 20), rope.count()); // Sentinel filtered
775
+
776
+ // Undo 10 operations
777
+ for (0..10) |_| {
778
+ _ = try rope.undo("current");
779
+ }
780
+ try std.testing.expectEqual(@as(u32, 10), rope.count());
781
+
782
+ // Redo 5 operations
783
+ for (0..5) |_| {
784
+ _ = try rope.redo();
785
+ }
786
+ try std.testing.expectEqual(@as(u32, 15), rope.count());
787
+ }
788
+
789
+ //===== Bulk/Range Operations Tests =====
790
+
791
+ test "Rope - split at beginning" {
792
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
793
+ defer arena.deinit();
794
+
795
+ const RopeType = rope_mod.Rope(SimpleItem);
796
+ const items = [_]SimpleItem{
797
+ .{ .value = 1 },
798
+ .{ .value = 2 },
799
+ .{ .value = 3 },
800
+ };
801
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
802
+
803
+ const right = try rope.split(0);
804
+
805
+ try std.testing.expectEqual(@as(u32, 0), rope.count()); // Sentinel filtered
806
+ try std.testing.expectEqual(@as(u32, 3), right.count());
807
+ try std.testing.expectEqual(@as(u32, 1), right.get(0).?.value);
808
+ }
809
+
810
+ test "Rope - split at middle" {
811
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
812
+ defer arena.deinit();
813
+
814
+ const RopeType = rope_mod.Rope(SimpleItem);
815
+ const items = [_]SimpleItem{
816
+ .{ .value = 1 },
817
+ .{ .value = 2 },
818
+ .{ .value = 3 },
819
+ .{ .value = 4 },
820
+ .{ .value = 5 },
821
+ };
822
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
823
+
824
+ const right = try rope.split(2);
825
+
826
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
827
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
828
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
829
+
830
+ try std.testing.expectEqual(@as(u32, 3), right.count());
831
+ try std.testing.expectEqual(@as(u32, 3), right.get(0).?.value);
832
+ try std.testing.expectEqual(@as(u32, 4), right.get(1).?.value);
833
+ try std.testing.expectEqual(@as(u32, 5), right.get(2).?.value);
834
+ }
835
+
836
+ test "Rope - split at end" {
837
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
838
+ defer arena.deinit();
839
+
840
+ const RopeType = rope_mod.Rope(SimpleItem);
841
+ const items = [_]SimpleItem{
842
+ .{ .value = 1 },
843
+ .{ .value = 2 },
844
+ .{ .value = 3 },
845
+ };
846
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
847
+
848
+ const right = try rope.split(3);
849
+
850
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
851
+ try std.testing.expectEqual(@as(u32, 0), right.count()); // Sentinel filtered
852
+ }
853
+
854
+ test "Rope - slice full range" {
855
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
856
+ defer arena.deinit();
857
+
858
+ const RopeType = rope_mod.Rope(SimpleItem);
859
+ const items = [_]SimpleItem{
860
+ .{ .value = 1 },
861
+ .{ .value = 2 },
862
+ .{ .value = 3 },
863
+ };
864
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
865
+
866
+ const sliced = try rope.slice(0, 3, arena.allocator());
867
+ defer arena.allocator().free(sliced);
868
+
869
+ try std.testing.expectEqual(@as(usize, 3), sliced.len);
870
+ try std.testing.expectEqual(@as(u32, 1), sliced[0].value);
871
+ try std.testing.expectEqual(@as(u32, 2), sliced[1].value);
872
+ try std.testing.expectEqual(@as(u32, 3), sliced[2].value);
873
+ }
874
+
875
+ test "Rope - slice partial range" {
876
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
877
+ defer arena.deinit();
878
+
879
+ const RopeType = rope_mod.Rope(SimpleItem);
880
+ const items = [_]SimpleItem{
881
+ .{ .value = 10 },
882
+ .{ .value = 20 },
883
+ .{ .value = 30 },
884
+ .{ .value = 40 },
885
+ .{ .value = 50 },
886
+ };
887
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
888
+
889
+ const sliced = try rope.slice(1, 4, arena.allocator());
890
+ defer arena.allocator().free(sliced);
891
+
892
+ try std.testing.expectEqual(@as(usize, 3), sliced.len);
893
+ try std.testing.expectEqual(@as(u32, 20), sliced[0].value);
894
+ try std.testing.expectEqual(@as(u32, 30), sliced[1].value);
895
+ try std.testing.expectEqual(@as(u32, 40), sliced[2].value);
896
+ }
897
+
898
+ test "Rope - slice empty range" {
899
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
900
+ defer arena.deinit();
901
+
902
+ const RopeType = rope_mod.Rope(SimpleItem);
903
+ const items = [_]SimpleItem{
904
+ .{ .value = 1 },
905
+ .{ .value = 2 },
906
+ };
907
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
908
+
909
+ const sliced = try rope.slice(1, 1, arena.allocator());
910
+ defer arena.allocator().free(sliced);
911
+
912
+ try std.testing.expectEqual(@as(usize, 0), sliced.len);
913
+ }
914
+
915
+ test "Rope - delete_range at beginning" {
916
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
917
+ defer arena.deinit();
918
+
919
+ const RopeType = rope_mod.Rope(SimpleItem);
920
+ const items = [_]SimpleItem{
921
+ .{ .value = 1 },
922
+ .{ .value = 2 },
923
+ .{ .value = 3 },
924
+ .{ .value = 4 },
925
+ .{ .value = 5 },
926
+ };
927
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
928
+
929
+ try rope.delete_range(0, 2);
930
+
931
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
932
+ try std.testing.expectEqual(@as(u32, 3), rope.get(0).?.value);
933
+ try std.testing.expectEqual(@as(u32, 4), rope.get(1).?.value);
934
+ try std.testing.expectEqual(@as(u32, 5), rope.get(2).?.value);
935
+ }
936
+
937
+ test "Rope - delete_range in middle" {
938
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
939
+ defer arena.deinit();
940
+
941
+ const RopeType = rope_mod.Rope(SimpleItem);
942
+ const items = [_]SimpleItem{
943
+ .{ .value = 1 },
944
+ .{ .value = 2 },
945
+ .{ .value = 3 },
946
+ .{ .value = 4 },
947
+ .{ .value = 5 },
948
+ };
949
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
950
+
951
+ try rope.delete_range(1, 4);
952
+
953
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
954
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
955
+ try std.testing.expectEqual(@as(u32, 5), rope.get(1).?.value);
956
+ }
957
+
958
+ test "Rope - delete_range at end" {
959
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
960
+ defer arena.deinit();
961
+
962
+ const RopeType = rope_mod.Rope(SimpleItem);
963
+ const items = [_]SimpleItem{
964
+ .{ .value = 1 },
965
+ .{ .value = 2 },
966
+ .{ .value = 3 },
967
+ };
968
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
969
+
970
+ try rope.delete_range(1, 3);
971
+
972
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
973
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
974
+ }
975
+
976
+ test "Rope - delete_range empty (same indices)" {
977
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
978
+ defer arena.deinit();
979
+
980
+ const RopeType = rope_mod.Rope(SimpleItem);
981
+ const items = [_]SimpleItem{
982
+ .{ .value = 1 },
983
+ .{ .value = 2 },
984
+ };
985
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
986
+
987
+ try rope.delete_range(1, 1);
988
+
989
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
990
+ }
991
+
992
+ test "Rope - insert_slice at beginning" {
993
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
994
+ defer arena.deinit();
995
+
996
+ const RopeType = rope_mod.Rope(SimpleItem);
997
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 3 });
998
+
999
+ const to_insert = [_]SimpleItem{
1000
+ .{ .value = 1 },
1001
+ .{ .value = 2 },
1002
+ };
1003
+ try rope.insert_slice(0, &to_insert);
1004
+
1005
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
1006
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
1007
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
1008
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
1009
+ }
1010
+
1011
+ test "Rope - insert_slice in middle" {
1012
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1013
+ defer arena.deinit();
1014
+
1015
+ const RopeType = rope_mod.Rope(SimpleItem);
1016
+ const items = [_]SimpleItem{
1017
+ .{ .value = 1 },
1018
+ .{ .value = 4 },
1019
+ };
1020
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1021
+
1022
+ const to_insert = [_]SimpleItem{
1023
+ .{ .value = 2 },
1024
+ .{ .value = 3 },
1025
+ };
1026
+ try rope.insert_slice(1, &to_insert);
1027
+
1028
+ try std.testing.expectEqual(@as(u32, 4), rope.count());
1029
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
1030
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
1031
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
1032
+ try std.testing.expectEqual(@as(u32, 4), rope.get(3).?.value);
1033
+ }
1034
+
1035
+ test "Rope - insert_slice at end" {
1036
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1037
+ defer arena.deinit();
1038
+
1039
+ const RopeType = rope_mod.Rope(SimpleItem);
1040
+ const items = [_]SimpleItem{
1041
+ .{ .value = 1 },
1042
+ .{ .value = 2 },
1043
+ };
1044
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1045
+
1046
+ const to_insert = [_]SimpleItem{
1047
+ .{ .value = 3 },
1048
+ .{ .value = 4 },
1049
+ };
1050
+ try rope.insert_slice(2, &to_insert);
1051
+
1052
+ try std.testing.expectEqual(@as(u32, 4), rope.count());
1053
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
1054
+ try std.testing.expectEqual(@as(u32, 4), rope.get(3).?.value);
1055
+ }
1056
+
1057
+ test "Rope - insert_slice empty array" {
1058
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1059
+ defer arena.deinit();
1060
+
1061
+ const RopeType = rope_mod.Rope(SimpleItem);
1062
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1063
+
1064
+ const to_insert: []const SimpleItem = &[_]SimpleItem{};
1065
+ try rope.insert_slice(0, to_insert);
1066
+
1067
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
1068
+ }
1069
+
1070
+ test "Rope - to_array with simple items" {
1071
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1072
+ defer arena.deinit();
1073
+
1074
+ const RopeType = rope_mod.Rope(SimpleItem);
1075
+ const items = [_]SimpleItem{
1076
+ .{ .value = 10 },
1077
+ .{ .value = 20 },
1078
+ .{ .value = 30 },
1079
+ };
1080
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1081
+
1082
+ const array = try rope.to_array(arena.allocator());
1083
+ defer arena.allocator().free(array);
1084
+
1085
+ try std.testing.expectEqual(@as(usize, 3), array.len);
1086
+ try std.testing.expectEqual(@as(u32, 10), array[0].value);
1087
+ try std.testing.expectEqual(@as(u32, 20), array[1].value);
1088
+ try std.testing.expectEqual(@as(u32, 30), array[2].value);
1089
+ }
1090
+
1091
+ test "Rope - to_array empty rope" {
1092
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1093
+ defer arena.deinit();
1094
+
1095
+ const RopeType = rope_mod.Rope(SimpleItem);
1096
+ var rope = try RopeType.init(arena.allocator());
1097
+
1098
+ const array = try rope.to_array(arena.allocator());
1099
+ defer arena.allocator().free(array);
1100
+
1101
+ try std.testing.expectEqual(@as(usize, 0), array.len); // Sentinel filtered
1102
+ }
1103
+
1104
+ test "Rope - combined bulk operations" {
1105
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1106
+ defer arena.deinit();
1107
+
1108
+ const RopeType = rope_mod.Rope(SimpleItem);
1109
+ const items = [_]SimpleItem{
1110
+ .{ .value = 1 },
1111
+ .{ .value = 2 },
1112
+ .{ .value = 3 },
1113
+ .{ .value = 4 },
1114
+ .{ .value = 5 },
1115
+ };
1116
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1117
+
1118
+ try rope.delete_range(2, 4);
1119
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
1120
+
1121
+ const to_insert = [_]SimpleItem{
1122
+ .{ .value = 30 },
1123
+ .{ .value = 40 },
1124
+ };
1125
+ try rope.insert_slice(1, &to_insert);
1126
+ try std.testing.expectEqual(@as(u32, 5), rope.count());
1127
+
1128
+ const array = try rope.to_array(arena.allocator());
1129
+ defer arena.allocator().free(array);
1130
+
1131
+ try std.testing.expectEqual(@as(u32, 1), array[0].value);
1132
+ try std.testing.expectEqual(@as(u32, 30), array[1].value);
1133
+ try std.testing.expectEqual(@as(u32, 40), array[2].value);
1134
+ try std.testing.expectEqual(@as(u32, 2), array[3].value);
1135
+ try std.testing.expectEqual(@as(u32, 5), array[4].value);
1136
+ }
1137
+
1138
+ test "Rope - undo/redo with bulk operations" {
1139
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1140
+ defer arena.deinit();
1141
+
1142
+ const RopeType = rope_mod.Rope(SimpleItem);
1143
+ const items = [_]SimpleItem{
1144
+ .{ .value = 1 },
1145
+ .{ .value = 2 },
1146
+ .{ .value = 3 },
1147
+ };
1148
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1149
+
1150
+ // Store state
1151
+ try rope.store_undo("before bulk");
1152
+
1153
+ // Bulk insert
1154
+ const to_insert = [_]SimpleItem{
1155
+ .{ .value = 10 },
1156
+ .{ .value = 20 },
1157
+ };
1158
+ try rope.insert_slice(1, &to_insert);
1159
+ try std.testing.expectEqual(@as(u32, 5), rope.count());
1160
+
1161
+ // Undo
1162
+ _ = try rope.undo("after bulk");
1163
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
1164
+
1165
+ // Redo
1166
+ _ = try rope.redo();
1167
+ try std.testing.expectEqual(@as(u32, 5), rope.count());
1168
+ }
1169
+
1170
+ //===== Edge Case Tests =====
1171
+
1172
+ test "Rope - slice with start > end returns empty" {
1173
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1174
+ defer arena.deinit();
1175
+
1176
+ const RopeType = rope_mod.Rope(SimpleItem);
1177
+ const items = [_]SimpleItem{
1178
+ .{ .value = 1 },
1179
+ .{ .value = 2 },
1180
+ .{ .value = 3 },
1181
+ };
1182
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1183
+
1184
+ const sliced = try rope.slice(2, 1, arena.allocator());
1185
+ defer arena.allocator().free(sliced);
1186
+
1187
+ try std.testing.expectEqual(@as(usize, 0), sliced.len);
1188
+ }
1189
+
1190
+ test "Rope - slice beyond bounds" {
1191
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1192
+ defer arena.deinit();
1193
+
1194
+ const RopeType = rope_mod.Rope(SimpleItem);
1195
+ const items = [_]SimpleItem{
1196
+ .{ .value = 1 },
1197
+ .{ .value = 2 },
1198
+ };
1199
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1200
+
1201
+ // Should only get items that exist
1202
+ const sliced = try rope.slice(0, 100, arena.allocator());
1203
+ defer arena.allocator().free(sliced);
1204
+
1205
+ try std.testing.expectEqual(@as(usize, 2), sliced.len);
1206
+ }
1207
+
1208
+ test "Rope - delete_range with start > end does nothing" {
1209
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1210
+ defer arena.deinit();
1211
+
1212
+ const RopeType = rope_mod.Rope(SimpleItem);
1213
+ const items = [_]SimpleItem{
1214
+ .{ .value = 1 },
1215
+ .{ .value = 2 },
1216
+ };
1217
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1218
+
1219
+ try rope.delete_range(2, 1);
1220
+
1221
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
1222
+ }
1223
+
1224
+ test "Rope - insert_slice beyond count appends" {
1225
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1226
+ defer arena.deinit();
1227
+
1228
+ const RopeType = rope_mod.Rope(SimpleItem);
1229
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1230
+
1231
+ const to_insert = [_]SimpleItem{
1232
+ .{ .value = 2 },
1233
+ .{ .value = 3 },
1234
+ };
1235
+ try rope.insert_slice(100, &to_insert);
1236
+
1237
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
1238
+ try std.testing.expectEqual(@as(u32, 2), rope.get(1).?.value);
1239
+ try std.testing.expectEqual(@as(u32, 3), rope.get(2).?.value);
1240
+ }
1241
+
1242
+ test "Rope - replace at out of bounds does nothing" {
1243
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1244
+ defer arena.deinit();
1245
+
1246
+ const RopeType = rope_mod.Rope(SimpleItem);
1247
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1248
+
1249
+ try rope.replace(100, .{ .value = 999 });
1250
+
1251
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
1252
+ try std.testing.expectEqual(@as(u32, 1), rope.get(0).?.value);
1253
+ }
1254
+
1255
+ test "Rope - delete at out of bounds" {
1256
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1257
+ defer arena.deinit();
1258
+
1259
+ const RopeType = rope_mod.Rope(SimpleItem);
1260
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1261
+
1262
+ // This should handle gracefully (delete beyond bounds)
1263
+ try rope.delete(100);
1264
+
1265
+ // Count unchanged
1266
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
1267
+ }
1268
+
1269
+ test "Rope - split at zero creates empty left" {
1270
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1271
+ defer arena.deinit();
1272
+
1273
+ const RopeType = rope_mod.Rope(SimpleItem);
1274
+ const items = [_]SimpleItem{
1275
+ .{ .value = 1 },
1276
+ .{ .value = 2 },
1277
+ };
1278
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1279
+
1280
+ const right = try rope.split(0);
1281
+
1282
+ try std.testing.expectEqual(@as(u32, 0), rope.count()); // Sentinel filtered
1283
+ try std.testing.expectEqual(@as(u32, 2), right.count());
1284
+ }
1285
+
1286
+ test "Rope - split beyond count" {
1287
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1288
+ defer arena.deinit();
1289
+
1290
+ const RopeType = rope_mod.Rope(SimpleItem);
1291
+ const items = [_]SimpleItem{
1292
+ .{ .value = 1 },
1293
+ .{ .value = 2 },
1294
+ };
1295
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1296
+
1297
+ const right = try rope.split(100);
1298
+
1299
+ try std.testing.expectEqual(@as(u32, 2), rope.count());
1300
+ try std.testing.expectEqual(@as(u32, 0), right.count()); // Sentinel filtered
1301
+ }
1302
+
1303
+ test "Rope - multiple undo without operations" {
1304
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1305
+ defer arena.deinit();
1306
+
1307
+ const RopeType = rope_mod.Rope(SimpleItem);
1308
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1309
+
1310
+ try rope.store_undo("state1");
1311
+ try rope.store_undo("state2");
1312
+
1313
+ // Two undos back to back
1314
+ _ = try rope.undo("current");
1315
+ _ = try rope.undo("current");
1316
+
1317
+ // Should fail on third
1318
+ const result = rope.undo("current");
1319
+ try std.testing.expectError(error.Stop, result);
1320
+ }
1321
+
1322
+ test "Rope - stress test with 1000 items" {
1323
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1324
+ defer arena.deinit();
1325
+
1326
+ const RopeType = rope_mod.Rope(SimpleItem);
1327
+ var items: [1000]SimpleItem = undefined;
1328
+ for (&items, 0..) |*item, i| {
1329
+ item.* = .{ .value = @intCast(i) };
1330
+ }
1331
+
1332
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1333
+
1334
+ try std.testing.expectEqual(@as(u32, 1000), rope.count());
1335
+ try std.testing.expectEqual(@as(u32, 0), rope.get(0).?.value);
1336
+ try std.testing.expectEqual(@as(u32, 500), rope.get(500).?.value);
1337
+ try std.testing.expectEqual(@as(u32, 999), rope.get(999).?.value);
1338
+
1339
+ const depth = rope.root.depth();
1340
+ try std.testing.expect(depth < 20); // log2(1000) ≈ 10, allow some slack
1341
+ }
1342
+
1343
+ test "Rope - delete_range entire rope" {
1344
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1345
+ defer arena.deinit();
1346
+
1347
+ const RopeType = rope_mod.Rope(SimpleItem);
1348
+ const items = [_]SimpleItem{
1349
+ .{ .value = 1 },
1350
+ .{ .value = 2 },
1351
+ .{ .value = 3 },
1352
+ };
1353
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1354
+
1355
+ try rope.delete_range(0, 3);
1356
+
1357
+ // Should be empty (sentinel filtered)
1358
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
1359
+ }
1360
+
1361
+ test "Rope - to_array single item" {
1362
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1363
+ defer arena.deinit();
1364
+
1365
+ const RopeType = rope_mod.Rope(SimpleItem);
1366
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 42 });
1367
+
1368
+ const array = try rope.to_array(arena.allocator());
1369
+ defer arena.allocator().free(array);
1370
+
1371
+ try std.testing.expectEqual(@as(usize, 1), array.len);
1372
+ try std.testing.expectEqual(@as(u32, 42), array[0].value);
1373
+ }
1374
+
1375
+ test "Rope - concat with empty rope" {
1376
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1377
+ defer arena.deinit();
1378
+
1379
+ const RopeType = rope_mod.Rope(SimpleItem);
1380
+ var rope1 = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1381
+ const rope2 = try RopeType.init(arena.allocator());
1382
+
1383
+ try rope1.concat(&rope2);
1384
+
1385
+ try std.testing.expectEqual(@as(u32, 1), rope1.count()); // original only (empty filtered)
1386
+ }
1387
+
1388
+ test "Rope - redo after modifying tree fails" {
1389
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1390
+ defer arena.deinit();
1391
+
1392
+ const RopeType = rope_mod.Rope(SimpleItem);
1393
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1394
+
1395
+ try rope.store_undo("state1");
1396
+ try rope.append(.{ .value = 2 });
1397
+
1398
+ _ = try rope.undo("current");
1399
+
1400
+ // Manually modify the rope (breaking the redo contract)
1401
+ try rope.append(.{ .value = 3 });
1402
+
1403
+ // Redo should fail because tree was modified
1404
+ const result = rope.redo();
1405
+ try std.testing.expectError(error.Stop, result);
1406
+ }
1407
+
1408
+ test "Rope - rebalance extremely unbalanced tree" {
1409
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1410
+ defer arena.deinit();
1411
+
1412
+ const RopeType = rope_mod.Rope(SimpleItem);
1413
+ var rope = try RopeType.init(arena.allocator());
1414
+
1415
+ for (0..50) |i| {
1416
+ try rope.append(.{ .value = @intCast(i) });
1417
+ }
1418
+
1419
+ const depth_before = rope.root.depth();
1420
+
1421
+ // Rebalance
1422
+ try rope.rebalance(arena.allocator());
1423
+
1424
+ const depth_after = rope.root.depth();
1425
+
1426
+ // Should be more balanced now
1427
+ try std.testing.expect(depth_after <= depth_before);
1428
+ try std.testing.expect(depth_after < 15); // log2(50) ≈ 6
1429
+
1430
+ // Data should be preserved
1431
+ try std.testing.expectEqual(@as(u32, 50), rope.count());
1432
+ try std.testing.expectEqual(@as(u32, 0), rope.get(0).?.value); // Fixed index
1433
+ }
1434
+
1435
+ //===== Weight-aware Tests =====
1436
+
1437
+ // Type with custom weight for testing weight-based operations
1438
+ const WeightedItem = struct {
1439
+ value: u32,
1440
+ weight: u32,
1441
+
1442
+ pub const Metrics = struct {
1443
+ total_weight: u32 = 0,
1444
+
1445
+ pub fn add(self: *Metrics, other: Metrics) void {
1446
+ self.total_weight += other.total_weight;
1447
+ }
1448
+
1449
+ pub fn weight(self: *const Metrics) u32 {
1450
+ return self.total_weight;
1451
+ }
1452
+ };
1453
+
1454
+ pub fn measure(self: *const WeightedItem) Metrics {
1455
+ return .{ .total_weight = self.weight };
1456
+ }
1457
+
1458
+ pub fn empty() WeightedItem {
1459
+ return .{ .value = 0, .weight = 0 };
1460
+ }
1461
+
1462
+ pub fn is_empty(self: *const WeightedItem) bool {
1463
+ return self.value == 0 and self.weight == 0;
1464
+ }
1465
+ };
1466
+
1467
+ // Leaf split function for testing (callback format)
1468
+ const WeightedRope = rope_mod.Rope(WeightedItem);
1469
+
1470
+ fn splitWeightedItemCallback(
1471
+ ctx: ?*anyopaque,
1472
+ allocator: std.mem.Allocator,
1473
+ leaf: *const WeightedItem,
1474
+ weight_in_leaf: u32,
1475
+ ) error{ OutOfBounds, OutOfMemory }!WeightedRope.Node.LeafSplitResult {
1476
+ _ = ctx;
1477
+ _ = allocator;
1478
+ if (weight_in_leaf == 0) {
1479
+ return .{
1480
+ .left = WeightedItem.empty(),
1481
+ .right = leaf.*,
1482
+ };
1483
+ } else if (weight_in_leaf >= leaf.weight) {
1484
+ return .{
1485
+ .left = leaf.*,
1486
+ .right = WeightedItem.empty(),
1487
+ };
1488
+ }
1489
+
1490
+ // Split proportionally
1491
+ return .{
1492
+ .left = .{ .value = leaf.value, .weight = weight_in_leaf },
1493
+ .right = .{ .value = leaf.value + 1000, .weight = leaf.weight - weight_in_leaf },
1494
+ };
1495
+ }
1496
+
1497
+ // Helper to create the callback struct
1498
+ fn makeWeightedSplitter() WeightedRope.Node.LeafSplitFn {
1499
+ return .{
1500
+ .ctx = null,
1501
+ .splitFn = splitWeightedItemCallback,
1502
+ };
1503
+ }
1504
+
1505
+ test "Rope - totalWeight returns correct weight" {
1506
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1507
+ defer arena.deinit();
1508
+ const items = [_]WeightedItem{
1509
+ .{ .value = 1, .weight = 10 },
1510
+ .{ .value = 2, .weight = 20 },
1511
+ .{ .value = 3, .weight = 30 },
1512
+ };
1513
+
1514
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
1515
+ try std.testing.expectEqual(@as(u32, 60), rope.totalWeight());
1516
+ }
1517
+
1518
+ test "Rope - split_at_weight at boundary" {
1519
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1520
+ defer arena.deinit();
1521
+ const items = [_]WeightedItem{
1522
+ .{ .value = 1, .weight = 10 },
1523
+ .{ .value = 2, .weight = 20 },
1524
+ .{ .value = 3, .weight = 30 },
1525
+ };
1526
+
1527
+ const rope = try WeightedRope.from_slice(arena.allocator(), &items);
1528
+
1529
+ // Split at weight 30 (boundary between second and third item)
1530
+ const splitter = makeWeightedSplitter();
1531
+ const result = try WeightedRope.Node.split_at_weight(rope.root, 30, arena.allocator(), rope.empty_leaf, &splitter);
1532
+
1533
+ // Left should have weight 30 (first two items)
1534
+ try std.testing.expectEqual(@as(u32, 30), result.left.metrics().weight());
1535
+
1536
+ // Right should have weight 30 (third item)
1537
+ try std.testing.expectEqual(@as(u32, 30), result.right.metrics().weight());
1538
+ }
1539
+
1540
+ test "Rope - split_at_weight inside leaf" {
1541
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1542
+ defer arena.deinit();
1543
+ const rope = try WeightedRope.from_item(arena.allocator(), .{ .value = 1, .weight = 100 });
1544
+
1545
+ // Split at weight 40 (inside the single leaf)
1546
+ const splitter = makeWeightedSplitter();
1547
+ const result = try WeightedRope.Node.split_at_weight(rope.root, 40, arena.allocator(), rope.empty_leaf, &splitter);
1548
+
1549
+ // Left should have weight 40
1550
+ try std.testing.expectEqual(@as(u32, 40), result.left.metrics().weight());
1551
+
1552
+ // Right should have weight 60
1553
+ try std.testing.expectEqual(@as(u32, 60), result.right.metrics().weight());
1554
+ }
1555
+
1556
+ test "Rope - splitByWeight" {
1557
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1558
+ defer arena.deinit();
1559
+ const items = [_]WeightedItem{
1560
+ .{ .value = 1, .weight = 10 },
1561
+ .{ .value = 2, .weight = 20 },
1562
+ .{ .value = 3, .weight = 30 },
1563
+ };
1564
+
1565
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
1566
+ try std.testing.expectEqual(@as(u32, 60), rope.totalWeight());
1567
+
1568
+ // Split at weight 30
1569
+ const splitter = makeWeightedSplitter();
1570
+ const right_half = try rope.splitByWeight(30, &splitter);
1571
+
1572
+ // Left half should have weight 30
1573
+ try std.testing.expectEqual(@as(u32, 30), rope.totalWeight());
1574
+
1575
+ // Right half should have weight 30
1576
+ try std.testing.expectEqual(@as(u32, 30), right_half.totalWeight());
1577
+ }
1578
+
1579
+ test "Rope - deleteRangeByWeight" {
1580
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1581
+ defer arena.deinit();
1582
+ const items = [_]WeightedItem{
1583
+ .{ .value = 1, .weight = 10 },
1584
+ .{ .value = 2, .weight = 20 },
1585
+ .{ .value = 3, .weight = 30 },
1586
+ .{ .value = 4, .weight = 40 },
1587
+ };
1588
+
1589
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
1590
+ try std.testing.expectEqual(@as(u32, 100), rope.totalWeight());
1591
+
1592
+ // Delete weight range [10, 30) - removes the second item (weight 20)
1593
+ const splitter = makeWeightedSplitter();
1594
+ try rope.deleteRangeByWeight(10, 30, &splitter);
1595
+
1596
+ // Should have removed weight 20
1597
+ try std.testing.expectEqual(@as(u32, 80), rope.totalWeight());
1598
+ }
1599
+
1600
+ test "Rope - insertSliceByWeight" {
1601
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1602
+ defer arena.deinit();
1603
+ const items = [_]WeightedItem{
1604
+ .{ .value = 1, .weight = 10 },
1605
+ .{ .value = 3, .weight = 30 },
1606
+ };
1607
+
1608
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
1609
+ try std.testing.expectEqual(@as(u32, 40), rope.totalWeight());
1610
+
1611
+ // Insert at weight 10 (after first item)
1612
+ const insert_items = [_]WeightedItem{
1613
+ .{ .value = 2, .weight = 20 },
1614
+ };
1615
+ const splitter = makeWeightedSplitter();
1616
+ try rope.insertSliceByWeight(10, &insert_items, &splitter);
1617
+
1618
+ // Should have added weight 20
1619
+ try std.testing.expectEqual(@as(u32, 60), rope.totalWeight());
1620
+ }
1621
+
1622
+ test "Rope - findByWeight" {
1623
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1624
+ defer arena.deinit();
1625
+ const items = [_]WeightedItem{
1626
+ .{ .value = 1, .weight = 10 },
1627
+ .{ .value = 2, .weight = 20 },
1628
+ .{ .value = 3, .weight = 30 },
1629
+ };
1630
+
1631
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
1632
+
1633
+ // Find leaf containing weight 0 (first item)
1634
+ const result0 = rope.findByWeight(0);
1635
+ try std.testing.expect(result0 != null);
1636
+ try std.testing.expectEqual(@as(u32, 1), result0.?.leaf.value);
1637
+ try std.testing.expectEqual(@as(u32, 0), result0.?.start_weight);
1638
+
1639
+ // Find leaf containing weight 15 (second item)
1640
+ const result15 = rope.findByWeight(15);
1641
+ try std.testing.expect(result15 != null);
1642
+ try std.testing.expectEqual(@as(u32, 2), result15.?.leaf.value);
1643
+ try std.testing.expectEqual(@as(u32, 10), result15.?.start_weight);
1644
+
1645
+ // Find leaf containing weight 35 (third item)
1646
+ const result35 = rope.findByWeight(35);
1647
+ try std.testing.expect(result35 != null);
1648
+ try std.testing.expectEqual(@as(u32, 3), result35.?.leaf.value);
1649
+ try std.testing.expectEqual(@as(u32, 30), result35.?.start_weight);
1650
+
1651
+ // Out of bounds
1652
+ const result100 = rope.findByWeight(100);
1653
+ try std.testing.expect(result100 == null);
1654
+ }
1655
+
1656
+ //===== Integrated Marker Tracking Tests (Union Types) =====
1657
+
1658
+ // Simple union type for testing automatic marker tracking
1659
+ const TokenType = union(enum) {
1660
+ word: u32,
1661
+ space: u32,
1662
+ newline: void, // Marker type
1663
+
1664
+ // Define which tags are markers (only track these!)
1665
+ pub const MarkerTypes = &[_]std.meta.Tag(TokenType){.newline};
1666
+
1667
+ pub const Metrics = struct {
1668
+ width: u32 = 0,
1669
+
1670
+ pub fn add(self: *Metrics, other: Metrics) void {
1671
+ self.width += other.width;
1672
+ }
1673
+
1674
+ pub fn weight(self: *const Metrics) u32 {
1675
+ return self.width;
1676
+ }
1677
+ };
1678
+
1679
+ pub fn measure(self: *const TokenType) Metrics {
1680
+ return switch (self.*) {
1681
+ .word => |w| .{ .width = w },
1682
+ .space => |s| .{ .width = s },
1683
+ .newline => .{ .width = 0 },
1684
+ };
1685
+ }
1686
+
1687
+ pub fn empty() TokenType {
1688
+ return .{ .space = 0 };
1689
+ }
1690
+
1691
+ pub fn is_empty(self: *const TokenType) bool {
1692
+ return switch (self.*) {
1693
+ .space => |s| s == 0,
1694
+ else => false,
1695
+ };
1696
+ }
1697
+ };
1698
+
1699
+ test "Rope - automatic marker tracking with union type" {
1700
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1701
+ defer arena.deinit();
1702
+
1703
+ const RopeType = rope_mod.Rope(TokenType);
1704
+
1705
+ // Create rope with marker tracking enabled
1706
+ const tokens = [_]TokenType{
1707
+ .{ .word = 5 }, // "Hello"
1708
+ .{ .space = 1 }, // " "
1709
+ .{ .word = 5 }, // "World"
1710
+ .{ .newline = {} }, // Line break marker
1711
+ .{ .word = 6 }, // "Second"
1712
+ .{ .space = 1 }, // " "
1713
+ .{ .word = 4 }, // "Line"
1714
+ .{ .newline = {} }, // Line break marker
1715
+ .{ .word = 5 }, // "Third"
1716
+ };
1717
+
1718
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
1719
+
1720
+ // O(1) lookup: find newline markers (only .newline is tracked, not .word or .space)
1721
+ try std.testing.expectEqual(@as(u32, 2), rope.markerCount(.newline));
1722
+
1723
+ // Get first newline (end of line 0)
1724
+ const nl0 = rope.getMarker(.newline, 0);
1725
+ try std.testing.expect(nl0 != null);
1726
+ try std.testing.expectEqual(@as(u32, 3), nl0.?.leaf_index); // After word, space, word
1727
+ try std.testing.expectEqual(@as(u32, 11), nl0.?.global_weight); // 5 + 1 + 5
1728
+
1729
+ // Get second newline (end of line 1)
1730
+ const nl1 = rope.getMarker(.newline, 1);
1731
+ try std.testing.expect(nl1 != null);
1732
+ try std.testing.expectEqual(@as(u32, 7), nl1.?.leaf_index);
1733
+ try std.testing.expectEqual(@as(u32, 22), nl1.?.global_weight); // 11 + 6 + 1 + 4
1734
+
1735
+ // Word and space are NOT markers - should return 0
1736
+ try std.testing.expectEqual(@as(u32, 0), rope.markerCount(.word));
1737
+ try std.testing.expectEqual(@as(u32, 0), rope.markerCount(.space));
1738
+ }
1739
+
1740
+ test "Rope - marker tracking with empty rope" {
1741
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1742
+ defer arena.deinit();
1743
+
1744
+ const RopeType = rope_mod.Rope(TokenType);
1745
+ var rope = try RopeType.init(arena.allocator());
1746
+
1747
+ try std.testing.expectEqual(@as(u32, 0), rope.markerCount(.newline));
1748
+ try std.testing.expectEqual(@as(u32, 0), rope.markerCount(.word)); // Not a marker type
1749
+ try std.testing.expect(rope.getMarker(.newline, 0) == null);
1750
+ }
1751
+
1752
+ test "Rope - marker tracking requires rebuild" {
1753
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1754
+ defer arena.deinit();
1755
+
1756
+ const RopeType = rope_mod.Rope(TokenType);
1757
+ const tokens = [_]TokenType{
1758
+ .{ .word = 5 },
1759
+ .{ .newline = {} },
1760
+ };
1761
+
1762
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
1763
+
1764
+ // Markers are automatically tracked in the tree
1765
+ try std.testing.expectEqual(@as(u32, 1), rope.markerCount(.newline));
1766
+ try std.testing.expect(rope.getMarker(.newline, 0) != null);
1767
+ }
1768
+
1769
+ test "Rope - marker tracking with many markers" {
1770
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1771
+ defer arena.deinit();
1772
+
1773
+ const RopeType = rope_mod.Rope(TokenType);
1774
+
1775
+ // Create 100 lines
1776
+ var tokens_array: [199]TokenType = undefined; // 100 words + 99 newlines
1777
+ for (0..100) |i| {
1778
+ if (i > 0) {
1779
+ tokens_array[i * 2 - 1] = .{ .newline = {} };
1780
+ }
1781
+ tokens_array[i * 2] = .{ .word = 5 };
1782
+ }
1783
+
1784
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens_array);
1785
+
1786
+ // Should have 99 newlines (only newlines are tracked as markers)
1787
+ try std.testing.expectEqual(@as(u32, 99), rope.markerCount(.newline));
1788
+
1789
+ // Test O(1) random access to specific lines
1790
+ const nl50 = rope.getMarker(.newline, 50).?;
1791
+ try std.testing.expectEqual(@as(u32, 101), nl50.leaf_index); // word, nl, word, nl, ... (50th newline is at index 101)
1792
+ try std.testing.expectEqual(@as(u32, 255), nl50.global_weight); // 51 words * 5 width
1793
+
1794
+ const nl98 = rope.getMarker(.newline, 98).?;
1795
+ try std.testing.expectEqual(@as(u32, 197), nl98.leaf_index);
1796
+ }
1797
+ //===== Debug toText Tests =====
1798
+
1799
+ test "Rope - toText shows basic structure" {
1800
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1801
+ defer arena.deinit();
1802
+
1803
+ const RopeType = rope_mod.Rope(SimpleItem);
1804
+ const items = [_]SimpleItem{
1805
+ .{ .value = 1 },
1806
+ .{ .value = 2 },
1807
+ .{ .value = 3 },
1808
+ };
1809
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1810
+
1811
+ const debug_text = try rope.toText(arena.allocator());
1812
+
1813
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[root") != null);
1814
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "branch") != null or std.mem.indexOf(u8, debug_text, "leaf") != null);
1815
+ }
1816
+
1817
+ test "Rope - toText shows empty rope" {
1818
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1819
+ defer arena.deinit();
1820
+
1821
+ const RopeType = rope_mod.Rope(SimpleItem);
1822
+ var rope = try RopeType.init(arena.allocator());
1823
+
1824
+ const debug_text = try rope.toText(arena.allocator());
1825
+
1826
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[root") != null);
1827
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[empty]") != null);
1828
+ }
1829
+
1830
+ test "Rope - toText with union type shows tags" {
1831
+ const TestSegment = union(enum) {
1832
+ text: struct { width: u32 },
1833
+ brk: void,
1834
+ linestart: void,
1835
+
1836
+ pub const MarkerTypes = &[_]std.meta.Tag(@This()){ .brk, .linestart };
1837
+
1838
+ pub const Metrics = struct {
1839
+ width: u32 = 0,
1840
+
1841
+ pub fn add(self: *Metrics, other: Metrics) void {
1842
+ self.width += other.width;
1843
+ }
1844
+
1845
+ pub fn weight(self: *const Metrics) u32 {
1846
+ return self.width;
1847
+ }
1848
+ };
1849
+
1850
+ pub fn measure(self: *const @This()) Metrics {
1851
+ return switch (self.*) {
1852
+ .text => |t| Metrics{ .width = t.width },
1853
+ .brk => Metrics{ .width = 1 },
1854
+ .linestart => Metrics{ .width = 0 },
1855
+ };
1856
+ }
1857
+
1858
+ pub fn empty() @This() {
1859
+ return .{ .text = .{ .width = 0 } };
1860
+ }
1861
+
1862
+ pub fn is_empty(self: *const @This()) bool {
1863
+ return switch (self.*) {
1864
+ .text => |t| t.width == 0,
1865
+ else => false,
1866
+ };
1867
+ }
1868
+ };
1869
+
1870
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1871
+ defer arena.deinit();
1872
+
1873
+ const TestRope = rope_mod.Rope(TestSegment);
1874
+ var rope = try TestRope.from_slice(arena.allocator(), &[_]TestSegment{
1875
+ .linestart,
1876
+ .{ .text = .{ .width = 5 } },
1877
+ .brk,
1878
+ .linestart,
1879
+ .{ .text = .{ .width = 10 } },
1880
+ });
1881
+
1882
+ const debug_text = try rope.toText(arena.allocator());
1883
+
1884
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "text") != null);
1885
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "brk") != null);
1886
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "linestart") != null);
1887
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "w5") != null);
1888
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "w10") != null);
1889
+ }
1890
+
1891
+ test "Rope - toText with nested structure" {
1892
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1893
+ defer arena.deinit();
1894
+
1895
+ const RopeType = rope_mod.Rope(SimpleItem);
1896
+
1897
+ // Create a larger rope that will have branches
1898
+ var items: [10]SimpleItem = undefined;
1899
+ for (&items, 0..) |*item, i| {
1900
+ item.* = .{ .value = @intCast(i) };
1901
+ }
1902
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1903
+
1904
+ const debug_text = try rope.toText(arena.allocator());
1905
+
1906
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[root") != null);
1907
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[branch") != null);
1908
+ }
1909
+
1910
+ test "Rope - toText after insertions shows updated structure" {
1911
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1912
+ defer arena.deinit();
1913
+
1914
+ const RopeType = rope_mod.Rope(SimpleItem);
1915
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
1916
+
1917
+ const before = try rope.toText(arena.allocator());
1918
+ try std.testing.expect(std.mem.indexOf(u8, before, "[root") != null);
1919
+
1920
+ try rope.append(.{ .value = 2 });
1921
+ try rope.append(.{ .value = 3 });
1922
+
1923
+ const after = try rope.toText(arena.allocator());
1924
+ try std.testing.expect(std.mem.indexOf(u8, after, "[root") != null);
1925
+ try std.testing.expect(after.len >= before.len);
1926
+ }
1927
+
1928
+ test "Rope - toText with custom metrics shows width info" {
1929
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1930
+ defer arena.deinit();
1931
+
1932
+ const RopeType = rope_mod.Rope(ItemWithMetrics);
1933
+ const items = [_]ItemWithMetrics{
1934
+ .{ .value = 1, .size = 100 },
1935
+ .{ .value = 2, .size = 200 },
1936
+ };
1937
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
1938
+
1939
+ const debug_text = try rope.toText(arena.allocator());
1940
+
1941
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[root") != null);
1942
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "w") != null);
1943
+ }
1944
+
1945
+ test "Rope - toText shows single leaf" {
1946
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1947
+ defer arena.deinit();
1948
+
1949
+ const RopeType = rope_mod.Rope(SimpleItem);
1950
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 42 });
1951
+
1952
+ const debug_text = try rope.toText(arena.allocator());
1953
+
1954
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[root") != null);
1955
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "[leaf") != null);
1956
+ try std.testing.expect(std.mem.indexOf(u8, debug_text, "]") != null);
1957
+ }
1958
+
1959
+ test "Rope - marker cache MUST update after delete operations" {
1960
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1961
+ defer arena.deinit();
1962
+
1963
+ const RopeType = rope_mod.Rope(TokenType);
1964
+
1965
+ // Create: word(5) newline word(5) newline word(5)
1966
+ // 3 lines total, 2 newlines
1967
+ const tokens = [_]TokenType{
1968
+ .{ .word = 5 },
1969
+ .{ .newline = {} },
1970
+ .{ .word = 5 },
1971
+ .{ .newline = {} },
1972
+ .{ .word = 5 },
1973
+ };
1974
+
1975
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
1976
+
1977
+ try std.testing.expectEqual(@as(u32, 2), rope.markerCount(.newline));
1978
+
1979
+ try rope.delete(4);
1980
+
1981
+ try std.testing.expectEqual(@as(u32, 2), rope.markerCount(.newline));
1982
+
1983
+ // The critical test: marker positions MUST be correct after delete!
1984
+ const nl1_after = rope.getMarker(.newline, 1);
1985
+ try std.testing.expect(nl1_after != null);
1986
+
1987
+ // After deleting the last word at index 4, the second newline should be at index 3
1988
+ // (was at index 3 before, stays at 3 after deleting index 4)
1989
+ try std.testing.expectEqual(@as(u32, 3), nl1_after.?.leaf_index);
1990
+ }
1991
+
1992
+ test "Rope - marker cache MUST update after undo" {
1993
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
1994
+ defer arena.deinit();
1995
+
1996
+ const RopeType = rope_mod.Rope(TokenType);
1997
+
1998
+ // Create: word(10) newline word(5)
1999
+ const tokens = [_]TokenType{
2000
+ .{ .word = 10 },
2001
+ .{ .newline = {} },
2002
+ .{ .word = 5 },
2003
+ };
2004
+
2005
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
2006
+
2007
+ // Initial state: 1 newline at weight 10
2008
+ const nl_before = rope.getMarker(.newline, 0);
2009
+ try std.testing.expect(nl_before != null);
2010
+ try std.testing.expectEqual(@as(u32, 10), nl_before.?.global_weight);
2011
+
2012
+ // Store undo point
2013
+ try rope.store_undo("before delete");
2014
+
2015
+ // Delete part of first word: delete range [0, 1) removes first word
2016
+ try rope.delete_range(0, 1);
2017
+
2018
+ // After delete: newline should be at weight 0 (no word before it)
2019
+ const nl_after_delete = rope.getMarker(.newline, 0);
2020
+ try std.testing.expect(nl_after_delete != null);
2021
+ try std.testing.expectEqual(@as(u32, 0), nl_after_delete.?.global_weight);
2022
+
2023
+ // Undo the delete
2024
+ _ = try rope.undo("after delete");
2025
+
2026
+ // CRITICAL: After undo, marker cache MUST be recalculated!
2027
+ // Newline should be back at weight 10
2028
+ const nl_after_undo = rope.getMarker(.newline, 0);
2029
+ try std.testing.expect(nl_after_undo != null);
2030
+ try std.testing.expectEqual(@as(u32, 10), nl_after_undo.?.global_weight);
2031
+ }
2032
+
2033
+ test "Rope - marker cache MUST update after redo" {
2034
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2035
+ defer arena.deinit();
2036
+
2037
+ const RopeType = rope_mod.Rope(TokenType);
2038
+
2039
+ const tokens = [_]TokenType{
2040
+ .{ .word = 10 },
2041
+ .{ .newline = {} },
2042
+ .{ .word = 5 },
2043
+ };
2044
+
2045
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
2046
+
2047
+ try rope.store_undo("initial");
2048
+ try rope.delete_range(0, 1);
2049
+
2050
+ const nl_after_delete = rope.getMarker(.newline, 0);
2051
+ try std.testing.expectEqual(@as(u32, 0), nl_after_delete.?.global_weight);
2052
+
2053
+ // Undo
2054
+ _ = try rope.undo("after delete");
2055
+ const nl_after_undo = rope.getMarker(.newline, 0);
2056
+ try std.testing.expectEqual(@as(u32, 10), nl_after_undo.?.global_weight);
2057
+
2058
+ // Redo
2059
+ _ = try rope.redo();
2060
+
2061
+ // CRITICAL: After redo, marker cache MUST be recalculated!
2062
+ const nl_after_redo = rope.getMarker(.newline, 0);
2063
+ try std.testing.expect(nl_after_redo != null);
2064
+ try std.testing.expectEqual(@as(u32, 0), nl_after_redo.?.global_weight);
2065
+ }
2066
+
2067
+ test "Rope - marker cache survives multiple undo/redo cycles" {
2068
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2069
+ defer arena.deinit();
2070
+
2071
+ const RopeType = rope_mod.Rope(TokenType);
2072
+
2073
+ var rope = try RopeType.from_slice(arena.allocator(), &[_]TokenType{
2074
+ .{ .word = 5 },
2075
+ .{ .newline = {} },
2076
+ .{ .word = 5 },
2077
+ });
2078
+
2079
+ try rope.store_undo("state1");
2080
+ try rope.append(.{ .newline = {} });
2081
+ try rope.append(.{ .word = 5 });
2082
+
2083
+ // Should have 2 newlines now
2084
+ try std.testing.expectEqual(@as(u32, 2), rope.markerCount(.newline));
2085
+ const nl1_orig = rope.getMarker(.newline, 1);
2086
+ try std.testing.expectEqual(@as(u32, 10), nl1_orig.?.global_weight);
2087
+
2088
+ try rope.store_undo("state2");
2089
+ try rope.delete(0); // Delete first word
2090
+
2091
+ // Markers should update: first newline now at weight 0
2092
+ const nl0_after_delete = rope.getMarker(.newline, 0);
2093
+ try std.testing.expectEqual(@as(u32, 0), nl0_after_delete.?.global_weight);
2094
+
2095
+ // Undo twice
2096
+ _ = try rope.undo("current");
2097
+ _ = try rope.undo("current");
2098
+
2099
+ // Back to original: 1 newline at weight 5
2100
+ try std.testing.expectEqual(@as(u32, 1), rope.markerCount(.newline));
2101
+ const nl_final = rope.getMarker(.newline, 0);
2102
+ try std.testing.expectEqual(@as(u32, 5), nl_final.?.global_weight);
2103
+
2104
+ // Redo twice
2105
+ _ = try rope.redo();
2106
+ _ = try rope.redo();
2107
+
2108
+ // Should match the post-delete state
2109
+ const nl0_redo = rope.getMarker(.newline, 0);
2110
+ try std.testing.expectEqual(@as(u32, 0), nl0_redo.?.global_weight);
2111
+ }
2112
+
2113
+ //===== Configurable Undo Depth Tests =====
2114
+
2115
+ test "Rope - weight-based balancing with custom weight function" {
2116
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2117
+ defer arena.deinit();
2118
+
2119
+ // Create items with different sizes
2120
+ const items = [_]WeightedItem{
2121
+ .{ .value = 1, .weight = 100 }, // Large item
2122
+ .{ .value = 2, .weight = 10 }, // Small item
2123
+ .{ .value = 3, .weight = 200 }, // Very large item
2124
+ .{ .value = 4, .weight = 50 }, // Medium item
2125
+ };
2126
+
2127
+ var rope = try WeightedRope.from_slice(arena.allocator(), &items);
2128
+
2129
+ // Check that metrics are tracked
2130
+ const metrics = rope.root.metrics();
2131
+ try std.testing.expectEqual(@as(u32, 4), metrics.count);
2132
+ try std.testing.expectEqual(@as(u32, 360), metrics.custom.total_weight);
2133
+ try std.testing.expectEqual(@as(u32, 360), metrics.weight()); // Should use weight()
2134
+ }
2135
+
2136
+ test "Rope - unlimited undo depth by default" {
2137
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2138
+ defer arena.deinit();
2139
+
2140
+ const RopeType = rope_mod.Rope(SimpleItem);
2141
+ var rope = try RopeType.init(arena.allocator());
2142
+
2143
+ // Store many undo states
2144
+ for (0..100) |i| {
2145
+ try rope.store_undo("state");
2146
+ try rope.append(.{ .value = @intCast(i) });
2147
+ }
2148
+
2149
+ // Should have all 100 states
2150
+ try std.testing.expectEqual(@as(usize, 100), rope.undo_depth);
2151
+ try std.testing.expect(rope.can_undo());
2152
+ }
2153
+
2154
+ test "Rope - max_undo_depth limits history" {
2155
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2156
+ defer arena.deinit();
2157
+
2158
+ const RopeType = rope_mod.Rope(SimpleItem);
2159
+ var rope = try RopeType.initWithConfig(arena.allocator(), .{ .max_undo_depth = 10 });
2160
+
2161
+ // Store 20 undo states
2162
+ for (0..20) |i| {
2163
+ try rope.store_undo("state");
2164
+ try rope.append(.{ .value = @intCast(i) });
2165
+ }
2166
+
2167
+ // Should only keep 10 states
2168
+ try std.testing.expectEqual(@as(usize, 10), rope.undo_depth);
2169
+ try std.testing.expect(rope.can_undo());
2170
+
2171
+ // Can undo at most 10 times (may be less due to how history works)
2172
+ var undo_count: usize = 0;
2173
+ while (rope.can_undo()) : (undo_count += 1) {
2174
+ _ = rope.undo("current") catch break;
2175
+ }
2176
+ // Should have undone at least some operations, but not more than 10
2177
+ try std.testing.expect(undo_count > 0);
2178
+ try std.testing.expect(undo_count <= 10);
2179
+ }
2180
+
2181
+ test "Rope - trimUndoHistory works correctly" {
2182
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2183
+ defer arena.deinit();
2184
+
2185
+ const RopeType = rope_mod.Rope(SimpleItem);
2186
+ var rope = try RopeType.initWithConfig(arena.allocator(), .{ .max_undo_depth = 5 });
2187
+
2188
+ for (0..10) |i| {
2189
+ try rope.store_undo("state");
2190
+ try rope.append(.{ .value = @intCast(i) });
2191
+
2192
+ try std.testing.expect(rope.undo_depth <= 5);
2193
+ }
2194
+
2195
+ try std.testing.expectEqual(@as(usize, 5), rope.undo_depth);
2196
+ }
2197
+
2198
+ test "Rope - weight-based join_balanced respects weight ratio" {
2199
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2200
+ defer arena.deinit();
2201
+
2202
+ const left_items = [_]WeightedItem{
2203
+ .{ .value = 1, .weight = 1000 },
2204
+ .{ .value = 2, .weight = 1000 },
2205
+ .{ .value = 3, .weight = 1000 },
2206
+ };
2207
+ var rope_left = try WeightedRope.from_slice(arena.allocator(), &left_items);
2208
+
2209
+ const right_items = [_]WeightedItem{
2210
+ .{ .value = 4, .weight = 100 },
2211
+ };
2212
+ const rope_right = try WeightedRope.from_slice(arena.allocator(), &right_items);
2213
+
2214
+ try rope_left.concat(&rope_right);
2215
+
2216
+ try std.testing.expectEqual(@as(u32, 4), rope_left.count());
2217
+
2218
+ try std.testing.expect(rope_left.root.is_balanced());
2219
+ }
2220
+
2221
+ test "Rope - integration weight-based balancing with history limits" {
2222
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2223
+ defer arena.deinit();
2224
+
2225
+ var rope = try WeightedRope.initWithConfig(arena.allocator(), .{ .max_undo_depth = 5 });
2226
+
2227
+ var expected_count: u32 = 0;
2228
+ for (0..10) |i| {
2229
+ try rope.store_undo("insert");
2230
+ try rope.append(.{
2231
+ .value = @intCast(i),
2232
+ .weight = @intCast((i + 1) * 10),
2233
+ });
2234
+ expected_count += 1;
2235
+ }
2236
+
2237
+ try std.testing.expectEqual(expected_count, rope.count());
2238
+
2239
+ try rope.insert(5, .{ .value = 999, .weight = 50 });
2240
+ expected_count += 1;
2241
+
2242
+ try std.testing.expectEqual(expected_count, rope.count());
2243
+
2244
+ try std.testing.expect(rope.undo_depth <= 5);
2245
+
2246
+ try std.testing.expect(rope.root.is_balanced());
2247
+ }
2248
+
2249
+ test "Rope - clear removes all items" {
2250
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2251
+ defer arena.deinit();
2252
+
2253
+ const RopeType = rope_mod.Rope(SimpleItem);
2254
+ const items = [_]SimpleItem{
2255
+ .{ .value = 1 },
2256
+ .{ .value = 2 },
2257
+ .{ .value = 3 },
2258
+ };
2259
+ var rope = try RopeType.from_slice(arena.allocator(), &items);
2260
+
2261
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
2262
+
2263
+ rope.clear();
2264
+
2265
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2266
+ }
2267
+
2268
+ test "Rope - clear on empty rope" {
2269
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2270
+ defer arena.deinit();
2271
+
2272
+ const RopeType = rope_mod.Rope(SimpleItem);
2273
+ var rope = try RopeType.init(arena.allocator());
2274
+
2275
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2276
+
2277
+ rope.clear();
2278
+
2279
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2280
+ }
2281
+
2282
+ test "Rope - clear then insert works" {
2283
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2284
+ defer arena.deinit();
2285
+
2286
+ const RopeType = rope_mod.Rope(SimpleItem);
2287
+ var rope = try RopeType.from_item(arena.allocator(), .{ .value = 1 });
2288
+
2289
+ rope.clear();
2290
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2291
+
2292
+ try rope.append(.{ .value = 42 });
2293
+ try std.testing.expectEqual(@as(u32, 1), rope.count());
2294
+ try std.testing.expectEqual(@as(u32, 42), rope.get(0).?.value);
2295
+ }
2296
+
2297
+ test "Rope - clear with markers resets marker cache" {
2298
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2299
+ defer arena.deinit();
2300
+
2301
+ const RopeType = rope_mod.Rope(TokenType);
2302
+ const tokens = [_]TokenType{
2303
+ .{ .word = 5 },
2304
+ .{ .newline = {} },
2305
+ .{ .word = 5 },
2306
+ .{ .newline = {} },
2307
+ };
2308
+
2309
+ var rope = try RopeType.from_slice(arena.allocator(), &tokens);
2310
+ try std.testing.expectEqual(@as(u32, 2), rope.markerCount(.newline));
2311
+
2312
+ rope.clear();
2313
+
2314
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2315
+ try std.testing.expectEqual(@as(u32, 0), rope.markerCount(.newline));
2316
+ }
2317
+
2318
+ test "Rope - integration all features working together" {
2319
+ var arena = std.heap.ArenaAllocator.init(std.testing.allocator);
2320
+ defer arena.deinit();
2321
+
2322
+ const RopeType = rope_mod.Rope(SimpleItem);
2323
+ var rope = try RopeType.initWithConfig(arena.allocator(), .{ .max_undo_depth = 3 });
2324
+
2325
+ try std.testing.expectEqual(@as(u32, 0), rope.count());
2326
+
2327
+ try rope.store_undo("empty");
2328
+ try rope.append(.{ .value = 1 });
2329
+
2330
+ try rope.store_undo("one");
2331
+ try rope.append(.{ .value = 2 });
2332
+
2333
+ try rope.store_undo("two");
2334
+ try rope.append(.{ .value = 3 });
2335
+
2336
+ try rope.store_undo("three");
2337
+ try rope.append(.{ .value = 4 });
2338
+
2339
+ try std.testing.expectEqual(@as(u32, 4), rope.count());
2340
+
2341
+ try std.testing.expectEqual(@as(usize, 3), rope.undo_depth);
2342
+
2343
+ const val = rope.get(2);
2344
+ try std.testing.expectEqual(@as(u32, 3), val.?.value);
2345
+
2346
+ _ = try rope.undo("current");
2347
+ try std.testing.expectEqual(@as(u32, 3), rope.count());
2348
+
2349
+ const Context = struct {
2350
+ count: u32 = 0,
2351
+ fn walker(ctx: *anyopaque, data: *const SimpleItem, index: u32) RopeType.Node.WalkerResult {
2352
+ _ = data;
2353
+ _ = index;
2354
+ const self = @as(*@This(), @ptrCast(@alignCast(ctx)));
2355
+ self.count += 1;
2356
+ return .{};
2357
+ }
2358
+ };
2359
+ var ctx = Context{};
2360
+ try rope.walk(&ctx, Context.walker);
2361
+ try std.testing.expectEqual(@as(u32, 3), ctx.count);
2362
+ }