@fairyhunter13/opentui-core 0.1.112 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +63 -51
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-8fks7yv1.js +0 -411
  472. package/index-8fks7yv1.js.map +0 -10
  473. package/index-egy5e2rs.js +0 -12267
  474. package/index-egy5e2rs.js.map +0 -42
  475. package/index-tse8gzh0.js +0 -20614
  476. package/index-tse8gzh0.js.map +0 -67
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,3299 @@
1
+ const std = @import("std");
2
+ const editor_view = @import("../editor-view.zig");
3
+ const edit_buffer = @import("../edit-buffer.zig");
4
+ const text_buffer = @import("../text-buffer.zig");
5
+ const text_buffer_view = @import("../text-buffer-view.zig");
6
+ const opt_buffer_mod = @import("../buffer.zig");
7
+ const gp = @import("../grapheme.zig");
8
+ const link = @import("../link.zig");
9
+
10
+ const EditorView = editor_view.EditorView;
11
+ const EditBuffer = edit_buffer.EditBuffer;
12
+ const Cursor = edit_buffer.Cursor;
13
+ const Viewport = text_buffer_view.Viewport;
14
+
15
+ test "EditorView - init and deinit" {
16
+ const pool = gp.initGlobalPool(std.testing.allocator);
17
+ defer gp.deinitGlobalPool();
18
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
19
+ defer link.deinitGlobalLinkPool();
20
+
21
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
22
+ defer eb.deinit();
23
+
24
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
25
+ defer ev.deinit();
26
+
27
+ const vp = ev.getViewport();
28
+ try std.testing.expect(vp != null);
29
+ try std.testing.expectEqual(@as(u32, 80), vp.?.width);
30
+ try std.testing.expectEqual(@as(u32, 24), vp.?.height);
31
+ try std.testing.expectEqual(@as(u32, 0), vp.?.y);
32
+ }
33
+
34
+ test "EditorView - ensureCursorVisible scrolls down when cursor moves below viewport" {
35
+ const pool = gp.initGlobalPool(std.testing.allocator);
36
+ defer gp.deinitGlobalPool();
37
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
38
+ defer link.deinitGlobalLinkPool();
39
+
40
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
41
+ defer eb.deinit();
42
+
43
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
44
+ defer ev.deinit();
45
+
46
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
47
+
48
+ const cursor = ev.getPrimaryCursor();
49
+ try std.testing.expectEqual(@as(u32, 19), cursor.row);
50
+
51
+ _ = ev.getVirtualLines();
52
+
53
+ const vp = ev.getViewport().?;
54
+ try std.testing.expect(vp.y > 0);
55
+ try std.testing.expect(cursor.row >= vp.y);
56
+ try std.testing.expect(cursor.row < vp.y + vp.height);
57
+ }
58
+
59
+ test "EditorView - ensureCursorVisible scrolls up when cursor moves above viewport" {
60
+ const pool = gp.initGlobalPool(std.testing.allocator);
61
+ defer gp.deinitGlobalPool();
62
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
63
+ defer link.deinitGlobalLinkPool();
64
+
65
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
66
+ defer eb.deinit();
67
+
68
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
69
+ defer ev.deinit();
70
+
71
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
72
+
73
+ _ = ev.getVirtualLines();
74
+
75
+ var vp = ev.getViewport().?;
76
+ try std.testing.expect(vp.y > 0);
77
+
78
+ try eb.gotoLine(0);
79
+
80
+ _ = ev.getVirtualLines();
81
+
82
+ const cursor = ev.getPrimaryCursor();
83
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
84
+
85
+ vp = ev.getViewport().?;
86
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
87
+ }
88
+
89
+ test "EditorView - moveDown scrolls viewport automatically" {
90
+ const pool = gp.initGlobalPool(std.testing.allocator);
91
+ defer gp.deinitGlobalPool();
92
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
93
+ defer link.deinitGlobalLinkPool();
94
+
95
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
96
+ defer eb.deinit();
97
+
98
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
99
+ defer ev.deinit();
100
+
101
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
102
+
103
+ try eb.setCursor(0, 0);
104
+ _ = ev.getVirtualLines();
105
+ var vp = ev.getViewport().?;
106
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
107
+
108
+ var i: u32 = 0;
109
+ while (i < 15) : (i += 1) {
110
+ eb.moveDown();
111
+ }
112
+
113
+ _ = ev.getVirtualLines();
114
+
115
+ const cursor = ev.getPrimaryCursor();
116
+ try std.testing.expectEqual(@as(u32, 15), cursor.row);
117
+
118
+ vp = ev.getViewport().?;
119
+ try std.testing.expect(cursor.row >= vp.y);
120
+ try std.testing.expect(cursor.row < vp.y + vp.height);
121
+ }
122
+
123
+ test "EditorView - moveUp scrolls viewport automatically" {
124
+ const pool = gp.initGlobalPool(std.testing.allocator);
125
+ defer gp.deinitGlobalPool();
126
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
127
+ defer link.deinitGlobalLinkPool();
128
+
129
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
130
+ defer eb.deinit();
131
+
132
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
133
+ defer ev.deinit();
134
+
135
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
136
+
137
+ _ = ev.getVirtualLines();
138
+
139
+ var vp = ev.getViewport().?;
140
+ const initial_y = vp.y;
141
+ try std.testing.expect(initial_y > 0);
142
+
143
+ var i: u32 = 0;
144
+ while (i < 10) : (i += 1) {
145
+ eb.moveUp();
146
+ }
147
+
148
+ _ = ev.getVirtualLines();
149
+
150
+ const cursor = ev.getPrimaryCursor();
151
+ try std.testing.expectEqual(@as(u32, 9), cursor.row);
152
+
153
+ vp = ev.getViewport().?;
154
+ try std.testing.expect(vp.y < initial_y);
155
+ try std.testing.expect(cursor.row >= vp.y);
156
+ try std.testing.expect(cursor.row < vp.y + vp.height);
157
+ }
158
+
159
+ test "EditorView - scroll margin keeps cursor away from edges" {
160
+ const pool = gp.initGlobalPool(std.testing.allocator);
161
+ defer gp.deinitGlobalPool();
162
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
163
+ defer link.deinitGlobalLinkPool();
164
+
165
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
166
+ defer eb.deinit();
167
+
168
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
169
+ defer ev.deinit();
170
+
171
+ ev.setScrollMargin(0.2);
172
+
173
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
174
+
175
+ try eb.gotoLine(5);
176
+
177
+ const cursor = ev.getPrimaryCursor();
178
+ try std.testing.expectEqual(@as(u32, 5), cursor.row);
179
+
180
+ const vp = ev.getViewport().?;
181
+ const cursor_offset_in_viewport = cursor.row - vp.y;
182
+
183
+ try std.testing.expect(cursor_offset_in_viewport >= 2);
184
+ try std.testing.expect(cursor_offset_in_viewport < vp.height - 2);
185
+ }
186
+
187
+ test "EditorView - insertText with newlines maintains cursor visibility" {
188
+ const pool = gp.initGlobalPool(std.testing.allocator);
189
+ defer gp.deinitGlobalPool();
190
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
191
+ defer link.deinitGlobalLinkPool();
192
+
193
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
194
+ defer eb.deinit();
195
+
196
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
197
+ defer ev.deinit();
198
+
199
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
200
+
201
+ _ = ev.getVirtualLines();
202
+
203
+ const cursor = ev.getPrimaryCursor();
204
+ const vp = ev.getViewport().?;
205
+
206
+ try std.testing.expect(cursor.row >= vp.y);
207
+ try std.testing.expect(cursor.row < vp.y + vp.height);
208
+ }
209
+
210
+ test "EditorView - backspace at line start maintains visibility" {
211
+ const pool = gp.initGlobalPool(std.testing.allocator);
212
+ defer gp.deinitGlobalPool();
213
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
214
+ defer link.deinitGlobalLinkPool();
215
+
216
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
217
+ defer eb.deinit();
218
+
219
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
220
+ defer ev.deinit();
221
+
222
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
223
+
224
+ try eb.backspace();
225
+
226
+ _ = ev.getVirtualLines();
227
+
228
+ const cursor = ev.getPrimaryCursor();
229
+ const vp = ev.getViewport().?;
230
+
231
+ try std.testing.expect(cursor.row >= vp.y);
232
+ try std.testing.expect(cursor.row < vp.y + vp.height);
233
+ }
234
+
235
+ test "EditorView - deleteForward at line end maintains visibility" {
236
+ const pool = gp.initGlobalPool(std.testing.allocator);
237
+ defer gp.deinitGlobalPool();
238
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
239
+ defer link.deinitGlobalLinkPool();
240
+
241
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
242
+ defer eb.deinit();
243
+
244
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
245
+ defer ev.deinit();
246
+
247
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
248
+
249
+ try eb.setCursor(8, 6);
250
+
251
+ try eb.deleteForward();
252
+
253
+ _ = ev.getVirtualLines();
254
+
255
+ const cursor = ev.getPrimaryCursor();
256
+ const vp = ev.getViewport().?;
257
+
258
+ try std.testing.expect(cursor.row >= vp.y);
259
+ try std.testing.expect(cursor.row < vp.y + vp.height);
260
+ }
261
+
262
+ test "EditorView - deleteRange maintains cursor visibility" {
263
+ const pool = gp.initGlobalPool(std.testing.allocator);
264
+ defer gp.deinitGlobalPool();
265
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
266
+ defer link.deinitGlobalLinkPool();
267
+
268
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
269
+ defer eb.deinit();
270
+
271
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
272
+ defer ev.deinit();
273
+
274
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
275
+
276
+ try eb.deleteRange(.{ .row = 2, .col = 0 }, .{ .row = 7, .col = 6 });
277
+
278
+ _ = ev.getVirtualLines();
279
+
280
+ const cursor = ev.getPrimaryCursor();
281
+ const vp = ev.getViewport().?;
282
+
283
+ try std.testing.expect(cursor.row >= vp.y);
284
+ try std.testing.expect(cursor.row < vp.y + vp.height);
285
+ }
286
+
287
+ test "EditorView - deleteLine maintains cursor visibility" {
288
+ const pool = gp.initGlobalPool(std.testing.allocator);
289
+ defer gp.deinitGlobalPool();
290
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
291
+ defer link.deinitGlobalLinkPool();
292
+
293
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
294
+ defer eb.deinit();
295
+
296
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
297
+ defer ev.deinit();
298
+
299
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
300
+
301
+ try eb.gotoLine(7);
302
+
303
+ try eb.deleteLine();
304
+
305
+ _ = ev.getVirtualLines();
306
+
307
+ const cursor = ev.getPrimaryCursor();
308
+ const vp = ev.getViewport().?;
309
+
310
+ try std.testing.expect(cursor.row >= vp.y);
311
+ try std.testing.expect(cursor.row < vp.y + vp.height);
312
+ }
313
+
314
+ test "EditorView - setText resets viewport to top" {
315
+ const pool = gp.initGlobalPool(std.testing.allocator);
316
+ defer gp.deinitGlobalPool();
317
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
318
+ defer link.deinitGlobalLinkPool();
319
+
320
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
321
+ defer eb.deinit();
322
+
323
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 5);
324
+ defer ev.deinit();
325
+
326
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9");
327
+
328
+ _ = ev.getVirtualLines();
329
+
330
+ var vp = ev.getViewport().?;
331
+ try std.testing.expect(vp.y > 0);
332
+
333
+ try eb.setText("New Line 0\nNew Line 1\nNew Line 2");
334
+
335
+ const cursor = ev.getPrimaryCursor();
336
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
337
+
338
+ _ = ev.getVirtualLines();
339
+
340
+ vp = ev.getViewport().?;
341
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
342
+ }
343
+
344
+ test "EditorView - viewport respects total line count as max offset" {
345
+ const pool = gp.initGlobalPool(std.testing.allocator);
346
+ defer gp.deinitGlobalPool();
347
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
348
+ defer link.deinitGlobalLinkPool();
349
+
350
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
351
+ defer eb.deinit();
352
+
353
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
354
+ defer ev.deinit();
355
+
356
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4");
357
+
358
+ try eb.gotoLine(4);
359
+
360
+ const vp = ev.getViewport().?;
361
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
362
+ }
363
+
364
+ test "EditorView - horizontal movement doesn't affect vertical scroll" {
365
+ const pool = gp.initGlobalPool(std.testing.allocator);
366
+ defer gp.deinitGlobalPool();
367
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
368
+ defer link.deinitGlobalLinkPool();
369
+
370
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
371
+ defer eb.deinit();
372
+
373
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
374
+ defer ev.deinit();
375
+
376
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4");
377
+
378
+ try eb.setCursor(2, 0);
379
+
380
+ const vp_before = ev.getViewport().?;
381
+
382
+ eb.moveRight();
383
+ eb.moveRight();
384
+ eb.moveRight();
385
+
386
+ const vp_after = ev.getViewport().?;
387
+ try std.testing.expectEqual(vp_before.y, vp_after.y);
388
+ }
389
+
390
+ test "EditorView - cursor at boundaries doesn't cause invalid viewport" {
391
+ const pool = gp.initGlobalPool(std.testing.allocator);
392
+ defer gp.deinitGlobalPool();
393
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
394
+ defer link.deinitGlobalLinkPool();
395
+
396
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
397
+ defer eb.deinit();
398
+
399
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
400
+ defer ev.deinit();
401
+
402
+ try eb.setCursor(0, 0);
403
+
404
+ var vp = ev.getViewport().?;
405
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
406
+
407
+ try eb.insertText("First line");
408
+
409
+ try eb.setCursor(0, 0);
410
+
411
+ vp = ev.getViewport().?;
412
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
413
+
414
+ eb.moveLeft();
415
+ vp = ev.getViewport().?;
416
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
417
+
418
+ eb.moveUp();
419
+ vp = ev.getViewport().?;
420
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
421
+ }
422
+
423
+ test "EditorView - rapid cursor movements maintain visibility" {
424
+ const pool = gp.initGlobalPool(std.testing.allocator);
425
+ defer gp.deinitGlobalPool();
426
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
427
+ defer link.deinitGlobalLinkPool();
428
+
429
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
430
+ defer eb.deinit();
431
+
432
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
433
+ defer ev.deinit();
434
+
435
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19\nLine 20\nLine 21\nLine 22\nLine 23\nLine 24\nLine 25\nLine 26\nLine 27\nLine 28\nLine 29");
436
+
437
+ try eb.gotoLine(0);
438
+ try eb.gotoLine(29);
439
+ try eb.gotoLine(15);
440
+ try eb.gotoLine(5);
441
+ try eb.gotoLine(25);
442
+
443
+ _ = ev.getVirtualLines();
444
+
445
+ const cursor = ev.getPrimaryCursor();
446
+ const vp = ev.getViewport().?;
447
+
448
+ try std.testing.expect(cursor.row >= vp.y);
449
+ try std.testing.expect(cursor.row < vp.y + vp.height);
450
+ }
451
+
452
+ test "EditorView - VisualCursor without wrapping" {
453
+ const pool = gp.initGlobalPool(std.testing.allocator);
454
+ defer gp.deinitGlobalPool();
455
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
456
+ defer link.deinitGlobalLinkPool();
457
+
458
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
459
+ defer eb.deinit();
460
+
461
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
462
+ defer ev.deinit();
463
+
464
+ try eb.insertText("Hello World\nSecond Line\nThird Line");
465
+
466
+ try eb.setCursor(1, 3);
467
+
468
+ const vcursor = ev.getVisualCursor();
469
+ try std.testing.expectEqual(@as(u32, 1), vcursor.visual_row);
470
+ try std.testing.expectEqual(@as(u32, 3), vcursor.visual_col);
471
+ try std.testing.expectEqual(@as(u32, 1), vcursor.logical_row);
472
+ try std.testing.expectEqual(@as(u32, 3), vcursor.logical_col);
473
+ }
474
+
475
+ test "EditorView - VisualCursor with character wrapping" {
476
+ const pool = gp.initGlobalPool(std.testing.allocator);
477
+ defer gp.deinitGlobalPool();
478
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
479
+ defer link.deinitGlobalLinkPool();
480
+
481
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
482
+ defer eb.deinit();
483
+
484
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
485
+ defer ev.deinit();
486
+
487
+ ev.setWrapMode(.char);
488
+
489
+ try eb.setText("This is a very long line that will definitely wrap at 20 characters");
490
+
491
+ try eb.setCursor(0, 25);
492
+
493
+ const vcursor = ev.getVisualCursor();
494
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
495
+ try std.testing.expectEqual(@as(u32, 25), vcursor.logical_col);
496
+ try std.testing.expect(vcursor.visual_row > 0);
497
+ try std.testing.expect(vcursor.visual_col <= 20);
498
+ }
499
+
500
+ test "EditorView - VisualCursor with word wrapping" {
501
+ const pool = gp.initGlobalPool(std.testing.allocator);
502
+ defer gp.deinitGlobalPool();
503
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
504
+ defer link.deinitGlobalLinkPool();
505
+
506
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
507
+ defer eb.deinit();
508
+
509
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
510
+ defer ev.deinit();
511
+
512
+ ev.setWrapMode(.word);
513
+
514
+ try eb.setText("Hello world this is a test of word wrapping");
515
+
516
+ const line_count = eb.getTextBuffer().getLineCount();
517
+ try std.testing.expectEqual(@as(u32, 1), line_count);
518
+
519
+ _ = ev.getVisualCursor();
520
+ }
521
+
522
+ test "EditorView - moveUpVisual with wrapping" {
523
+ const pool = gp.initGlobalPool(std.testing.allocator);
524
+ defer gp.deinitGlobalPool();
525
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
526
+ defer link.deinitGlobalLinkPool();
527
+
528
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
529
+ defer eb.deinit();
530
+
531
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
532
+ defer ev.deinit();
533
+
534
+ ev.setWrapMode(.char);
535
+
536
+ try eb.setText("This is a very long line that will definitely wrap multiple times at twenty characters");
537
+
538
+ try eb.setCursor(0, 50);
539
+
540
+ const vcursor_before = ev.getVisualCursor();
541
+ const visual_row_before = vcursor_before.visual_row;
542
+
543
+ ev.moveUpVisual();
544
+
545
+ const vcursor_after = ev.getVisualCursor();
546
+
547
+ try std.testing.expectEqual(visual_row_before - 1, vcursor_after.visual_row);
548
+
549
+ try std.testing.expectEqual(@as(u32, 0), vcursor_after.logical_row);
550
+ }
551
+
552
+ test "EditorView - moveDownVisual with wrapping" {
553
+ const pool = gp.initGlobalPool(std.testing.allocator);
554
+ defer gp.deinitGlobalPool();
555
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
556
+ defer link.deinitGlobalLinkPool();
557
+
558
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
559
+ defer eb.deinit();
560
+
561
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
562
+ defer ev.deinit();
563
+
564
+ ev.setWrapMode(.char);
565
+
566
+ try eb.setText("This is a very long line that will definitely wrap multiple times at twenty characters");
567
+
568
+ try eb.setCursor(0, 0);
569
+
570
+ const vcursor_before = ev.getVisualCursor();
571
+ try std.testing.expectEqual(@as(u32, 0), vcursor_before.visual_row);
572
+
573
+ ev.moveDownVisual();
574
+
575
+ const vcursor_after = ev.getVisualCursor();
576
+
577
+ try std.testing.expectEqual(@as(u32, 1), vcursor_after.visual_row);
578
+
579
+ try std.testing.expectEqual(@as(u32, 0), vcursor_after.logical_row);
580
+ }
581
+
582
+ test "EditorView - visualToLogicalCursor conversion" {
583
+ const pool = gp.initGlobalPool(std.testing.allocator);
584
+ defer gp.deinitGlobalPool();
585
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
586
+ defer link.deinitGlobalLinkPool();
587
+
588
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
589
+ defer eb.deinit();
590
+
591
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
592
+ defer ev.deinit();
593
+
594
+ ev.setWrapMode(.char);
595
+
596
+ try eb.setText("12345678901234567890123456789012345");
597
+
598
+ if (ev.visualToLogicalCursor(1, 5)) |vcursor| {
599
+ try std.testing.expectEqual(@as(u32, 1), vcursor.visual_row);
600
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
601
+ try std.testing.expectEqual(@as(u32, 25), vcursor.logical_col);
602
+ }
603
+ }
604
+
605
+ test "EditorView - moveUpVisual at top boundary" {
606
+ const pool = gp.initGlobalPool(std.testing.allocator);
607
+ defer gp.deinitGlobalPool();
608
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
609
+ defer link.deinitGlobalLinkPool();
610
+
611
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
612
+ defer eb.deinit();
613
+
614
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
615
+ defer ev.deinit();
616
+
617
+ ev.setWrapMode(.char);
618
+ try eb.setText("Short line");
619
+
620
+ try eb.setCursor(0, 0);
621
+
622
+ const before = ev.getPrimaryCursor();
623
+ ev.moveUpVisual();
624
+ const after = ev.getPrimaryCursor();
625
+
626
+ try std.testing.expectEqual(before.row, after.row);
627
+ try std.testing.expectEqual(before.col, after.col);
628
+ }
629
+
630
+ test "EditorView - moveDownVisual at bottom boundary" {
631
+ const pool = gp.initGlobalPool(std.testing.allocator);
632
+ defer gp.deinitGlobalPool();
633
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
634
+ defer link.deinitGlobalLinkPool();
635
+
636
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
637
+ defer eb.deinit();
638
+
639
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
640
+ defer ev.deinit();
641
+
642
+ ev.setWrapMode(.char);
643
+ try eb.setText("Short line\nSecond line");
644
+
645
+ try eb.setCursor(1, 0);
646
+
647
+ const before = ev.getPrimaryCursor();
648
+ ev.moveDownVisual();
649
+ const after = ev.getPrimaryCursor();
650
+
651
+ try std.testing.expectEqual(before.row, after.row);
652
+ }
653
+
654
+ test "EditorView - VisualCursor preserves desired column across wrapped lines" {
655
+ const pool = gp.initGlobalPool(std.testing.allocator);
656
+ defer gp.deinitGlobalPool();
657
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
658
+ defer link.deinitGlobalLinkPool();
659
+
660
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
661
+ defer eb.deinit();
662
+
663
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
664
+ defer ev.deinit();
665
+
666
+ ev.setWrapMode(.char);
667
+
668
+ try eb.setText("12345678901234567890123456789012345678901234567890");
669
+
670
+ try eb.setCursor(0, 15);
671
+
672
+ ev.moveDownVisual();
673
+ ev.moveDownVisual();
674
+ ev.moveUpVisual();
675
+
676
+ const vcursor = ev.getVisualCursor();
677
+
678
+ try std.testing.expect(vcursor.visual_col <= 20);
679
+ }
680
+
681
+ test "EditorView - VisualCursor with multiple logical lines and wrapping" {
682
+ const pool = gp.initGlobalPool(std.testing.allocator);
683
+ defer gp.deinitGlobalPool();
684
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
685
+ defer link.deinitGlobalLinkPool();
686
+
687
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
688
+ defer eb.deinit();
689
+
690
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
691
+ defer ev.deinit();
692
+
693
+ ev.setWrapMode(.char);
694
+
695
+ try eb.setText("Short line 1\nThis is a very long line that will wrap multiple times\nShort line 3");
696
+
697
+ try eb.setCursor(1, 30);
698
+
699
+ const vcursor = ev.getVisualCursor();
700
+ try std.testing.expectEqual(@as(u32, 1), vcursor.logical_row);
701
+
702
+ try std.testing.expect(vcursor.visual_row > 1);
703
+ }
704
+
705
+ test "EditorView - logicalToVisualCursor handles cursor past line end" {
706
+ const pool = gp.initGlobalPool(std.testing.allocator);
707
+ defer gp.deinitGlobalPool();
708
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
709
+ defer link.deinitGlobalLinkPool();
710
+
711
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
712
+ defer eb.deinit();
713
+
714
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
715
+ defer ev.deinit();
716
+
717
+ try eb.setText("Short");
718
+
719
+ const vcursor = ev.logicalToVisualCursor(0, 100);
720
+
721
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
722
+ }
723
+
724
+ test "EditorView - getTextBufferView returns correct view" {
725
+ const pool = gp.initGlobalPool(std.testing.allocator);
726
+ defer gp.deinitGlobalPool();
727
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
728
+ defer link.deinitGlobalLinkPool();
729
+
730
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
731
+ defer eb.deinit();
732
+
733
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
734
+ defer ev.deinit();
735
+
736
+ const tbv = ev.getTextBufferView();
737
+ const vp = tbv.getViewport();
738
+ try std.testing.expect(vp != null);
739
+ }
740
+
741
+ test "EditorView - getEditBuffer returns correct buffer" {
742
+ const pool = gp.initGlobalPool(std.testing.allocator);
743
+ defer gp.deinitGlobalPool();
744
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
745
+ defer link.deinitGlobalLinkPool();
746
+
747
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
748
+ defer eb.deinit();
749
+
750
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
751
+ defer ev.deinit();
752
+
753
+ const returned_eb = ev.getEditBuffer();
754
+ try std.testing.expect(returned_eb == eb);
755
+ }
756
+
757
+ test "EditorView - setViewportSize maintains cursor visibility" {
758
+ const pool = gp.initGlobalPool(std.testing.allocator);
759
+ defer gp.deinitGlobalPool();
760
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
761
+ defer link.deinitGlobalLinkPool();
762
+
763
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
764
+ defer eb.deinit();
765
+
766
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
767
+ defer ev.deinit();
768
+
769
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14");
770
+
771
+ try eb.gotoLine(10);
772
+
773
+ ev.setViewportSize(80, 5);
774
+
775
+ const vp = ev.getViewport().?;
776
+ try std.testing.expectEqual(@as(u32, 80), vp.width);
777
+ try std.testing.expectEqual(@as(u32, 5), vp.height);
778
+ }
779
+
780
+ test "EditorView - moveDownVisual across empty line preserves desired column" {
781
+ const pool = gp.initGlobalPool(std.testing.allocator);
782
+ defer gp.deinitGlobalPool();
783
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
784
+ defer link.deinitGlobalLinkPool();
785
+
786
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
787
+ defer eb.deinit();
788
+
789
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
790
+ defer ev.deinit();
791
+
792
+ try eb.setText("Line with some text\n\nAnother line with text");
793
+
794
+ try eb.setCursor(0, 10);
795
+
796
+ const vcursor_before = ev.getVisualCursor();
797
+ try std.testing.expectEqual(@as(u32, 10), vcursor_before.visual_col);
798
+
799
+ ev.moveDownVisual();
800
+
801
+ const vcursor_empty = ev.getVisualCursor();
802
+ try std.testing.expectEqual(@as(u32, 1), vcursor_empty.logical_row);
803
+ try std.testing.expectEqual(@as(u32, 0), vcursor_empty.visual_col);
804
+
805
+ ev.moveDownVisual();
806
+
807
+ const vcursor_after = ev.getVisualCursor();
808
+ try std.testing.expectEqual(@as(u32, 2), vcursor_after.logical_row);
809
+ try std.testing.expectEqual(@as(u32, 10), vcursor_after.visual_col);
810
+ }
811
+
812
+ test "EditorView - moveUpVisual across empty line preserves desired column" {
813
+ const pool = gp.initGlobalPool(std.testing.allocator);
814
+ defer gp.deinitGlobalPool();
815
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
816
+ defer link.deinitGlobalLinkPool();
817
+
818
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
819
+ defer eb.deinit();
820
+
821
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
822
+ defer ev.deinit();
823
+
824
+ try eb.setText("Line with some text\n\nAnother line with text");
825
+
826
+ try eb.setCursor(2, 10);
827
+
828
+ const vcursor_before = ev.getVisualCursor();
829
+ try std.testing.expectEqual(@as(u32, 10), vcursor_before.visual_col);
830
+
831
+ ev.moveUpVisual();
832
+
833
+ const vcursor_empty = ev.getVisualCursor();
834
+ try std.testing.expectEqual(@as(u32, 1), vcursor_empty.logical_row);
835
+ try std.testing.expectEqual(@as(u32, 0), vcursor_empty.visual_col);
836
+
837
+ ev.moveUpVisual();
838
+
839
+ const vcursor_after = ev.getVisualCursor();
840
+ try std.testing.expectEqual(@as(u32, 0), vcursor_after.logical_row);
841
+ try std.testing.expectEqual(@as(u32, 10), vcursor_after.visual_col);
842
+ }
843
+
844
+ test "EditorView - horizontal movement resets desired visual column" {
845
+ const pool = gp.initGlobalPool(std.testing.allocator);
846
+ defer gp.deinitGlobalPool();
847
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
848
+ defer link.deinitGlobalLinkPool();
849
+
850
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
851
+ defer eb.deinit();
852
+
853
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
854
+ defer ev.deinit();
855
+
856
+ try eb.setText("Line with some text\n\nAnother line with text");
857
+
858
+ try eb.setCursor(0, 10);
859
+
860
+ const vcursor_initial = ev.getVisualCursor();
861
+ try std.testing.expectEqual(@as(u32, 10), vcursor_initial.visual_col);
862
+
863
+ ev.moveDownVisual();
864
+ ev.moveDownVisual();
865
+
866
+ const vcursor_after = ev.getVisualCursor();
867
+ try std.testing.expectEqual(@as(u32, 2), vcursor_after.logical_row);
868
+ try std.testing.expectEqual(@as(u32, 10), vcursor_after.visual_col);
869
+
870
+ eb.moveRight();
871
+
872
+ const vcursor_after_right = ev.getVisualCursor();
873
+ try std.testing.expectEqual(@as(u32, 11), vcursor_after_right.visual_col);
874
+
875
+ ev.moveUpVisual();
876
+ ev.moveUpVisual();
877
+
878
+ const vcursor_final = ev.getVisualCursor();
879
+ try std.testing.expectEqual(@as(u32, 0), vcursor_final.logical_row);
880
+ try std.testing.expectEqual(@as(u32, 11), vcursor_final.visual_col);
881
+ }
882
+
883
+ test "EditorView - inserting newlines maintains rope integrity" {
884
+ const pool = gp.initGlobalPool(std.testing.allocator);
885
+ defer gp.deinitGlobalPool();
886
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
887
+ defer link.deinitGlobalLinkPool();
888
+
889
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
890
+ defer eb.deinit();
891
+
892
+ try eb.insertText("Line 0\nLine 1\nLine 2");
893
+
894
+ const rope_init = eb.getTextBuffer().rope();
895
+ const line_count_init = eb.getTextBuffer().lineCount();
896
+ try std.testing.expectEqual(@as(u32, 3), line_count_init);
897
+
898
+ try eb.insertText("\n");
899
+
900
+ const line_count_1 = eb.getTextBuffer().lineCount();
901
+ try std.testing.expectEqual(@as(u32, 4), line_count_1);
902
+
903
+ if (rope_init.getMarker(.linestart, 2)) |m2| {
904
+ if (rope_init.getMarker(.linestart, 3)) |m3| {
905
+ try std.testing.expect(m2.global_weight != m3.global_weight);
906
+ }
907
+ }
908
+
909
+ try eb.insertText("\n");
910
+
911
+ const line_count_2 = eb.getTextBuffer().lineCount();
912
+ try std.testing.expectEqual(@as(u32, 5), line_count_2);
913
+
914
+ if (rope_init.getMarker(.linestart, 3)) |m3| {
915
+ if (rope_init.getMarker(.linestart, 4)) |m4| {
916
+ try std.testing.expect(m3.global_weight != m4.global_weight);
917
+ }
918
+ }
919
+ }
920
+
921
+ test "EditorView - visual cursor stays in sync after scrolling and moving up" {
922
+ const pool = gp.initGlobalPool(std.testing.allocator);
923
+ defer gp.deinitGlobalPool();
924
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
925
+ defer link.deinitGlobalLinkPool();
926
+
927
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
928
+ defer eb.deinit();
929
+
930
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
931
+ defer ev.deinit();
932
+
933
+ try eb.insertText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4");
934
+
935
+ var cursor = ev.getPrimaryCursor();
936
+ try std.testing.expectEqual(@as(u32, 4), cursor.row);
937
+ try std.testing.expectEqual(@as(u32, 6), cursor.col);
938
+
939
+ _ = ev.getVirtualLines();
940
+
941
+ var vp = ev.getViewport().?;
942
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
943
+
944
+ var i: u32 = 0;
945
+ while (i < 6) : (i += 1) {
946
+ try eb.insertText("\n");
947
+ _ = ev.getVirtualLines();
948
+ }
949
+
950
+ cursor = ev.getPrimaryCursor();
951
+ try std.testing.expectEqual(@as(u32, 10), cursor.row);
952
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
953
+
954
+ vp = ev.getViewport().?;
955
+ try std.testing.expect(vp.y > 0);
956
+
957
+ const vcursor_before = ev.getVisualCursor();
958
+ try std.testing.expectEqual(@as(u32, 10), vcursor_before.logical_row);
959
+
960
+ ev.moveUpVisual();
961
+ _ = ev.getVirtualLines();
962
+
963
+ const vcursor_after_up = ev.getVisualCursor();
964
+ const logical_cursor_after_up = ev.getPrimaryCursor();
965
+
966
+ try std.testing.expectEqual(@as(u32, 9), logical_cursor_after_up.row);
967
+ try std.testing.expectEqual(@as(u32, 9), vcursor_after_up.logical_row);
968
+
969
+ try std.testing.expect(vcursor_after_up.visual_row < vcursor_before.visual_row);
970
+
971
+ try eb.insertText("X");
972
+ _ = ev.getVirtualLines();
973
+
974
+ const cursor_after_insert = ev.getPrimaryCursor();
975
+ const vcursor_after_insert = ev.getVisualCursor();
976
+
977
+ try std.testing.expectEqual(@as(u32, 9), cursor_after_insert.row);
978
+ try std.testing.expectEqual(@as(u32, 1), cursor_after_insert.col);
979
+
980
+ try std.testing.expectEqual(@as(u32, 9), vcursor_after_insert.logical_row);
981
+ try std.testing.expectEqual(@as(u32, 1), vcursor_after_insert.logical_col);
982
+
983
+ var out_buffer: [200]u8 = undefined;
984
+ const written = eb.getText(&out_buffer);
985
+ const text = out_buffer[0..written];
986
+
987
+ var line_count: u32 = 0;
988
+ var line_start: usize = 0;
989
+ for (text, 0..) |c, idx| {
990
+ if (c == '\n') {
991
+ if (line_count == 9) {
992
+ const line_9 = text[line_start..idx];
993
+ try std.testing.expect(line_9.len >= 1);
994
+ try std.testing.expectEqual(@as(u8, 'X'), line_9[0]);
995
+ break;
996
+ }
997
+ line_count += 1;
998
+ line_start = idx + 1;
999
+ }
1000
+ }
1001
+ }
1002
+
1003
+ test "EditorView - cursor positioning after wide grapheme" {
1004
+ const pool = gp.initGlobalPool(std.testing.allocator);
1005
+ defer gp.deinitGlobalPool();
1006
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1007
+ defer link.deinitGlobalLinkPool();
1008
+
1009
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1010
+ defer eb.deinit();
1011
+
1012
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
1013
+ defer ev.deinit();
1014
+
1015
+ try eb.insertText("AB東CD");
1016
+
1017
+ const cursor = ev.getPrimaryCursor();
1018
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
1019
+ try std.testing.expectEqual(@as(u32, 6), cursor.col);
1020
+
1021
+ try eb.setCursor(0, 4);
1022
+ const cursor_after_move = ev.getPrimaryCursor();
1023
+ try std.testing.expectEqual(@as(u32, 4), cursor_after_move.col);
1024
+
1025
+ const vcursor = ev.getVisualCursor();
1026
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
1027
+ try std.testing.expectEqual(@as(u32, 4), vcursor.logical_col);
1028
+ try std.testing.expectEqual(@as(u32, 4), vcursor.visual_col);
1029
+ }
1030
+
1031
+ test "EditorView - backspace after wide grapheme updates cursor correctly" {
1032
+ const pool = gp.initGlobalPool(std.testing.allocator);
1033
+ defer gp.deinitGlobalPool();
1034
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1035
+ defer link.deinitGlobalLinkPool();
1036
+
1037
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1038
+ defer eb.deinit();
1039
+
1040
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
1041
+ defer ev.deinit();
1042
+
1043
+ try eb.insertText("AB東CD");
1044
+
1045
+ try eb.setCursor(0, 4);
1046
+
1047
+ try eb.backspace();
1048
+
1049
+ const cursor = ev.getPrimaryCursor();
1050
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
1051
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
1052
+
1053
+ const vcursor = ev.getVisualCursor();
1054
+ try std.testing.expectEqual(@as(u32, 2), vcursor.logical_col);
1055
+ try std.testing.expectEqual(@as(u32, 2), vcursor.visual_col);
1056
+
1057
+ var out_buffer: [100]u8 = undefined;
1058
+ const written = eb.getText(&out_buffer);
1059
+ try std.testing.expectEqualStrings("ABCD", out_buffer[0..written]);
1060
+ }
1061
+
1062
+ test "EditorView - viewport scrolling with wrapped lines: down + edit + up" {
1063
+ const pool = gp.initGlobalPool(std.testing.allocator);
1064
+ defer gp.deinitGlobalPool();
1065
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1066
+ defer link.deinitGlobalLinkPool();
1067
+
1068
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1069
+ defer eb.deinit();
1070
+
1071
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1072
+ defer ev.deinit();
1073
+
1074
+ const tbv = ev.getTextBufferView();
1075
+ tbv.setWrapMode(.char);
1076
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 10 }, true);
1077
+
1078
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1079
+
1080
+ try eb.setCursor(0, 0);
1081
+ _ = ev.getVirtualLines();
1082
+
1083
+ var vp = ev.getViewport().?;
1084
+ const initial_vp_y = vp.y;
1085
+ try std.testing.expectEqual(@as(u32, 0), initial_vp_y);
1086
+
1087
+ ev.moveDownVisual();
1088
+ ev.moveDownVisual();
1089
+ ev.moveDownVisual();
1090
+ _ = ev.getVirtualLines();
1091
+
1092
+ vp = ev.getViewport().?;
1093
+ _ = vp.y;
1094
+
1095
+ _ = ev.getVisualCursor();
1096
+
1097
+ try eb.insertText("X");
1098
+ _ = ev.getVirtualLines();
1099
+
1100
+ vp = ev.getViewport().?;
1101
+ _ = vp.y;
1102
+
1103
+ ev.moveUpVisual();
1104
+ ev.moveUpVisual();
1105
+ ev.moveUpVisual();
1106
+ _ = ev.getVirtualLines();
1107
+
1108
+ vp = ev.getViewport().?;
1109
+ const final_vp_y = vp.y;
1110
+
1111
+ const vcursor_final = ev.getVisualCursor();
1112
+
1113
+ try std.testing.expectEqual(@as(u32, 0), vcursor_final.visual_row);
1114
+ try std.testing.expectEqual(@as(u32, 0), final_vp_y);
1115
+ }
1116
+
1117
+ test "EditorView - viewport scrolling with wrapped lines: aggressive down + edit + up sequence" {
1118
+ const pool = gp.initGlobalPool(std.testing.allocator);
1119
+ defer gp.deinitGlobalPool();
1120
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1121
+ defer link.deinitGlobalLinkPool();
1122
+
1123
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1124
+ defer eb.deinit();
1125
+
1126
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1127
+ defer ev.deinit();
1128
+
1129
+ const tbv = ev.getTextBufferView();
1130
+ tbv.setWrapMode(.char);
1131
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 10 }, true);
1132
+
1133
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1134
+
1135
+ try eb.setCursor(0, 0);
1136
+ _ = ev.getVirtualLines();
1137
+
1138
+ const total_vlines = ev.getTotalVirtualLineCount();
1139
+ try std.testing.expect(total_vlines > 10);
1140
+
1141
+ var vp = ev.getViewport().?;
1142
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1143
+
1144
+ var i: u32 = 0;
1145
+ while (i < 12) : (i += 1) {
1146
+ ev.moveDownVisual();
1147
+ }
1148
+ _ = ev.getVirtualLines();
1149
+
1150
+ vp = ev.getViewport().?;
1151
+ try std.testing.expect(vp.y > 0);
1152
+
1153
+ try eb.insertText("TEST");
1154
+ _ = ev.getVirtualLines();
1155
+
1156
+ i = 0;
1157
+ while (i < 12) : (i += 1) {
1158
+ ev.moveUpVisual();
1159
+ }
1160
+ _ = ev.getVirtualLines();
1161
+
1162
+ vp = ev.getViewport().?;
1163
+ const vcursor = ev.getVisualCursor();
1164
+
1165
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_row);
1166
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1167
+ }
1168
+
1169
+ test "EditorView - viewport scrolling with wrapped lines: multiple edits and movements" {
1170
+ const pool = gp.initGlobalPool(std.testing.allocator);
1171
+ defer gp.deinitGlobalPool();
1172
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1173
+ defer link.deinitGlobalLinkPool();
1174
+
1175
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1176
+ defer eb.deinit();
1177
+
1178
+ var ev = try EditorView.init(std.testing.allocator, eb, 15, 8);
1179
+ defer ev.deinit();
1180
+
1181
+ const tbv = ev.getTextBufferView();
1182
+ tbv.setWrapMode(.char);
1183
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 15, .height = 8 }, true);
1184
+
1185
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVV");
1186
+
1187
+ try eb.setCursor(0, 0);
1188
+ _ = ev.getVirtualLines();
1189
+
1190
+ ev.moveDownVisual();
1191
+ ev.moveDownVisual();
1192
+ _ = ev.getVirtualLines();
1193
+
1194
+ try eb.insertText("A");
1195
+ _ = ev.getVirtualLines();
1196
+
1197
+ ev.moveDownVisual();
1198
+ _ = ev.getVirtualLines();
1199
+
1200
+ try eb.insertText("B");
1201
+ _ = ev.getVirtualLines();
1202
+
1203
+ ev.moveUpVisual();
1204
+ ev.moveUpVisual();
1205
+ ev.moveUpVisual();
1206
+ _ = ev.getVirtualLines();
1207
+
1208
+ const vp = ev.getViewport().?;
1209
+ const vcursor = ev.getVisualCursor();
1210
+
1211
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_row);
1212
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1213
+ }
1214
+
1215
+ test "EditorView - viewport scrolling with wrapped lines: verify viewport consistency" {
1216
+ const pool = gp.initGlobalPool(std.testing.allocator);
1217
+ defer gp.deinitGlobalPool();
1218
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1219
+ defer link.deinitGlobalLinkPool();
1220
+
1221
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1222
+ defer eb.deinit();
1223
+
1224
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1225
+ defer ev.deinit();
1226
+
1227
+ const tbv = ev.getTextBufferView();
1228
+ tbv.setWrapMode(.char);
1229
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 10 }, true);
1230
+
1231
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1232
+
1233
+ try eb.setCursor(0, 0);
1234
+ _ = ev.getVirtualLines();
1235
+
1236
+ const vline_count = ev.getTotalVirtualLineCount();
1237
+ try std.testing.expect(vline_count >= 10);
1238
+
1239
+ var movements_down: u32 = 0;
1240
+ var i: u32 = 0;
1241
+ while (i < 5) : (i += 1) {
1242
+ const vcursor_before = ev.getVisualCursor();
1243
+ ev.moveDownVisual();
1244
+ const vcursor_after = ev.getVisualCursor();
1245
+ if (true) {
1246
+ if (vcursor_after.visual_row > vcursor_before.visual_row) {
1247
+ movements_down += 1;
1248
+ }
1249
+ }
1250
+ }
1251
+ _ = ev.getVirtualLines();
1252
+
1253
+ _ = ev.getViewport().?;
1254
+ _ = ev.getVisualCursor();
1255
+
1256
+ try eb.insertText("EDITED");
1257
+ _ = ev.getVirtualLines();
1258
+
1259
+ i = 0;
1260
+ while (i < movements_down) : (i += 1) {
1261
+ ev.moveUpVisual();
1262
+ }
1263
+ _ = ev.getVirtualLines();
1264
+
1265
+ const vp_final = ev.getViewport().?;
1266
+ const vcursor_final = ev.getVisualCursor();
1267
+
1268
+ try std.testing.expectEqual(@as(u32, 0), vcursor_final.visual_row);
1269
+ try std.testing.expectEqual(@as(u32, 0), vp_final.y);
1270
+ }
1271
+
1272
+ test "EditorView - viewport scrolling with wrapped lines: backspace after scroll" {
1273
+ const pool = gp.initGlobalPool(std.testing.allocator);
1274
+ defer gp.deinitGlobalPool();
1275
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1276
+ defer link.deinitGlobalLinkPool();
1277
+
1278
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1279
+ defer eb.deinit();
1280
+
1281
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1282
+ defer ev.deinit();
1283
+
1284
+ const tbv = ev.getTextBufferView();
1285
+ tbv.setWrapMode(.char);
1286
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 10 }, true);
1287
+
1288
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1289
+
1290
+ try eb.setCursor(0, 0);
1291
+ _ = ev.getVirtualLines();
1292
+
1293
+ ev.moveDownVisual();
1294
+ ev.moveDownVisual();
1295
+ _ = ev.getVirtualLines();
1296
+
1297
+ try eb.backspace();
1298
+ _ = ev.getVirtualLines();
1299
+
1300
+ ev.moveUpVisual();
1301
+ ev.moveUpVisual();
1302
+ _ = ev.getVirtualLines();
1303
+
1304
+ const vp = ev.getViewport().?;
1305
+ const vcursor = ev.getVisualCursor();
1306
+
1307
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_row);
1308
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1309
+ }
1310
+
1311
+ test "EditorView - viewport scrolling with wrapped lines: viewport follows cursor precisely" {
1312
+ const pool = gp.initGlobalPool(std.testing.allocator);
1313
+ defer gp.deinitGlobalPool();
1314
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1315
+ defer link.deinitGlobalLinkPool();
1316
+
1317
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1318
+ defer eb.deinit();
1319
+
1320
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 5);
1321
+ defer ev.deinit();
1322
+
1323
+ const tbv = ev.getTextBufferView();
1324
+ tbv.setWrapMode(.char);
1325
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 5 }, true);
1326
+
1327
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1328
+
1329
+ try eb.setCursor(0, 0);
1330
+ _ = ev.getVirtualLines();
1331
+
1332
+ var i: u32 = 0;
1333
+ while (i < 10) : (i += 1) {
1334
+ ev.moveDownVisual();
1335
+ _ = ev.getVirtualLines();
1336
+
1337
+ const vp = ev.getViewport().?;
1338
+ const vcursor = ev.getVisualCursor();
1339
+
1340
+ try std.testing.expect(vcursor.visual_row >= 0);
1341
+ try std.testing.expect(vcursor.visual_row < vp.height);
1342
+ }
1343
+
1344
+ try eb.insertText("MIDDLE");
1345
+ _ = ev.getVirtualLines();
1346
+
1347
+ const vp_middle = ev.getViewport().?;
1348
+ const vcursor_middle = ev.getVisualCursor();
1349
+ try std.testing.expect(vcursor_middle.visual_row >= 0);
1350
+ try std.testing.expect(vcursor_middle.visual_row < vp_middle.height);
1351
+
1352
+ i = 0;
1353
+ while (i < 10) : (i += 1) {
1354
+ ev.moveUpVisual();
1355
+ _ = ev.getVirtualLines();
1356
+
1357
+ const vp = ev.getViewport().?;
1358
+ const vcursor = ev.getVisualCursor();
1359
+
1360
+ try std.testing.expect(vcursor.visual_row >= 0);
1361
+ try std.testing.expect(vcursor.visual_row < vp.height);
1362
+ }
1363
+
1364
+ const vp_final = ev.getViewport().?;
1365
+ const vcursor_final = ev.getVisualCursor();
1366
+
1367
+ try std.testing.expectEqual(@as(u32, 0), vcursor_final.visual_row);
1368
+ try std.testing.expectEqual(@as(u32, 0), vp_final.y);
1369
+ }
1370
+
1371
+ test "EditorView - wrapped lines: specific scenario with insert and deletions" {
1372
+ const pool = gp.initGlobalPool(std.testing.allocator);
1373
+ defer gp.deinitGlobalPool();
1374
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1375
+ defer link.deinitGlobalLinkPool();
1376
+
1377
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1378
+ defer eb.deinit();
1379
+
1380
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1381
+ defer ev.deinit();
1382
+
1383
+ const tbv = ev.getTextBufferView();
1384
+ tbv.setWrapMode(.char);
1385
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 20, .height = 10 }, true);
1386
+
1387
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVVWWWWWWWWWWXXXXXXXXXXYYYYYYYYYYZZZZZZZZZZ");
1388
+
1389
+ try eb.setCursor(0, 0);
1390
+ _ = ev.getVirtualLines();
1391
+
1392
+ var vp = ev.getViewport().?;
1393
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1394
+
1395
+ ev.moveDownVisual();
1396
+ ev.moveDownVisual();
1397
+ ev.moveDownVisual();
1398
+ ev.moveDownVisual();
1399
+ ev.moveDownVisual();
1400
+ _ = ev.getVirtualLines();
1401
+
1402
+ vp = ev.getViewport().?;
1403
+ const vcursor_mid = ev.getVisualCursor();
1404
+ try std.testing.expectEqual(@as(u32, 5), vcursor_mid.visual_row);
1405
+
1406
+ try eb.insertText("XXX");
1407
+ _ = ev.getVirtualLines();
1408
+
1409
+ vp = ev.getViewport().?;
1410
+ const vcursor_after_insert = ev.getVisualCursor();
1411
+ try std.testing.expect(vcursor_after_insert.visual_row >= 0);
1412
+ try std.testing.expect(vcursor_after_insert.visual_row < vp.height);
1413
+
1414
+ try eb.backspace();
1415
+ try eb.backspace();
1416
+ try eb.backspace();
1417
+ _ = ev.getVirtualLines();
1418
+
1419
+ ev.moveUpVisual();
1420
+ ev.moveUpVisual();
1421
+ ev.moveUpVisual();
1422
+ ev.moveUpVisual();
1423
+ ev.moveUpVisual();
1424
+ _ = ev.getVirtualLines();
1425
+
1426
+ vp = ev.getViewport().?;
1427
+ const vcursor_final2 = ev.getVisualCursor();
1428
+
1429
+ try std.testing.expectEqual(@as(u32, 0), vcursor_final2.visual_row);
1430
+ try std.testing.expectEqual(@as(u32, 0), vp.y);
1431
+ }
1432
+
1433
+ test "EditorView - wrapped lines: many small edits with viewport scrolling" {
1434
+ const pool = gp.initGlobalPool(std.testing.allocator);
1435
+ defer gp.deinitGlobalPool();
1436
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1437
+ defer link.deinitGlobalLinkPool();
1438
+
1439
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1440
+ defer eb.deinit();
1441
+
1442
+ var ev = try EditorView.init(std.testing.allocator, eb, 15, 8);
1443
+ defer ev.deinit();
1444
+
1445
+ const tbv = ev.getTextBufferView();
1446
+ tbv.setWrapMode(.char);
1447
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 15, .height = 8 }, true);
1448
+
1449
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPPQQQQQQQQQQRRRRRRRRRRSSSSSSSSSSTTTTTTTTTTUUUUUUUUUUVVVVVVVVVV");
1450
+
1451
+ try eb.setCursor(0, 0);
1452
+ _ = ev.getVirtualLines();
1453
+
1454
+ ev.moveDownVisual();
1455
+ ev.moveDownVisual();
1456
+ _ = ev.getVirtualLines();
1457
+
1458
+ try eb.insertText("1");
1459
+ _ = ev.getVirtualLines();
1460
+
1461
+ ev.moveDownVisual();
1462
+ _ = ev.getVirtualLines();
1463
+
1464
+ try eb.insertText("2");
1465
+ _ = ev.getVirtualLines();
1466
+
1467
+ ev.moveDownVisual();
1468
+ _ = ev.getVirtualLines();
1469
+
1470
+ try eb.insertText("3");
1471
+ _ = ev.getVirtualLines();
1472
+
1473
+ ev.moveUpVisual();
1474
+ _ = ev.getVirtualLines();
1475
+
1476
+ try eb.insertText("4");
1477
+ _ = ev.getVirtualLines();
1478
+
1479
+ ev.moveUpVisual();
1480
+ ev.moveUpVisual();
1481
+ ev.moveUpVisual();
1482
+ _ = ev.getVirtualLines();
1483
+
1484
+ const vp2 = ev.getViewport().?;
1485
+ const vcursor2 = ev.getVisualCursor();
1486
+
1487
+ try std.testing.expectEqual(@as(u32, 0), vcursor2.visual_row);
1488
+ try std.testing.expectEqual(@as(u32, 0), vp2.y);
1489
+ }
1490
+
1491
+ test "EditorView - horizontal scroll: cursor moves right beyond viewport" {
1492
+ const pool = gp.initGlobalPool(std.testing.allocator);
1493
+ defer gp.deinitGlobalPool();
1494
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1495
+ defer link.deinitGlobalLinkPool();
1496
+
1497
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1498
+ defer eb.deinit();
1499
+
1500
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1501
+ defer ev.deinit();
1502
+
1503
+ try eb.setText("This is a very long line that exceeds the viewport width of 20 characters");
1504
+
1505
+ try eb.setCursor(0, 0);
1506
+ _ = ev.getVirtualLines();
1507
+
1508
+ var vp = ev.getViewport().?;
1509
+ try std.testing.expectEqual(@as(u32, 0), vp.x);
1510
+
1511
+ try eb.setCursor(0, 50);
1512
+ _ = ev.getVirtualLines();
1513
+
1514
+ vp = ev.getViewport().?;
1515
+ try std.testing.expect(vp.x > 0);
1516
+
1517
+ const cursor = ev.getPrimaryCursor();
1518
+ try std.testing.expect(cursor.col >= vp.x);
1519
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1520
+ }
1521
+
1522
+ test "EditorView - horizontal scroll: cursor moves left to beginning" {
1523
+ const pool = gp.initGlobalPool(std.testing.allocator);
1524
+ defer gp.deinitGlobalPool();
1525
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1526
+ defer link.deinitGlobalLinkPool();
1527
+
1528
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1529
+ defer eb.deinit();
1530
+
1531
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1532
+ defer ev.deinit();
1533
+
1534
+ try eb.setText("This is a very long line that exceeds the viewport width of 20 characters");
1535
+
1536
+ try eb.setCursor(0, 50);
1537
+ _ = ev.getVirtualLines();
1538
+
1539
+ var vp = ev.getViewport().?;
1540
+ try std.testing.expect(vp.x > 0);
1541
+
1542
+ try eb.setCursor(0, 0);
1543
+ _ = ev.getVirtualLines();
1544
+
1545
+ vp = ev.getViewport().?;
1546
+ try std.testing.expectEqual(@as(u32, 0), vp.x);
1547
+ }
1548
+
1549
+ test "EditorView - horizontal scroll: moveRight scrolls viewport" {
1550
+ const pool = gp.initGlobalPool(std.testing.allocator);
1551
+ defer gp.deinitGlobalPool();
1552
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1553
+ defer link.deinitGlobalLinkPool();
1554
+
1555
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1556
+ defer eb.deinit();
1557
+
1558
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1559
+ defer ev.deinit();
1560
+
1561
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1562
+
1563
+ try eb.setCursor(0, 0);
1564
+ _ = ev.getVirtualLines();
1565
+
1566
+ var vp = ev.getViewport().?;
1567
+ try std.testing.expectEqual(@as(u32, 0), vp.x);
1568
+
1569
+ var i: u32 = 0;
1570
+ while (i < 50) : (i += 1) {
1571
+ eb.moveRight();
1572
+ }
1573
+
1574
+ _ = ev.getVirtualLines();
1575
+
1576
+ vp = ev.getViewport().?;
1577
+ try std.testing.expect(vp.x > 0);
1578
+
1579
+ const cursor = ev.getPrimaryCursor();
1580
+ try std.testing.expectEqual(@as(u32, 50), cursor.col);
1581
+ try std.testing.expect(cursor.col >= vp.x);
1582
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1583
+ }
1584
+
1585
+ test "EditorView - horizontal scroll: moveLeft scrolls viewport back" {
1586
+ const pool = gp.initGlobalPool(std.testing.allocator);
1587
+ defer gp.deinitGlobalPool();
1588
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1589
+ defer link.deinitGlobalLinkPool();
1590
+
1591
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1592
+ defer eb.deinit();
1593
+
1594
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1595
+ defer ev.deinit();
1596
+
1597
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1598
+
1599
+ try eb.setCursor(0, 50);
1600
+ _ = ev.getVirtualLines();
1601
+
1602
+ var vp = ev.getViewport().?;
1603
+ const initial_x = vp.x;
1604
+ try std.testing.expect(initial_x > 0);
1605
+
1606
+ var i: u32 = 0;
1607
+ while (i < 30) : (i += 1) {
1608
+ eb.moveLeft();
1609
+ }
1610
+
1611
+ _ = ev.getVirtualLines();
1612
+
1613
+ vp = ev.getViewport().?;
1614
+ try std.testing.expect(vp.x < initial_x);
1615
+
1616
+ const cursor = ev.getPrimaryCursor();
1617
+ try std.testing.expect(cursor.col >= vp.x);
1618
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1619
+ }
1620
+
1621
+ test "EditorView - horizontal scroll: editing in scrolled view" {
1622
+ const pool = gp.initGlobalPool(std.testing.allocator);
1623
+ defer gp.deinitGlobalPool();
1624
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1625
+ defer link.deinitGlobalLinkPool();
1626
+
1627
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1628
+ defer eb.deinit();
1629
+
1630
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1631
+ defer ev.deinit();
1632
+
1633
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1634
+
1635
+ try eb.setCursor(0, 50);
1636
+ _ = ev.getVirtualLines();
1637
+
1638
+ var vp = ev.getViewport().?;
1639
+ try std.testing.expect(vp.x > 0);
1640
+
1641
+ try eb.insertText("XYZ");
1642
+ _ = ev.getVirtualLines();
1643
+
1644
+ vp = ev.getViewport().?;
1645
+ const cursor = ev.getPrimaryCursor();
1646
+ try std.testing.expectEqual(@as(u32, 53), cursor.col);
1647
+ try std.testing.expect(cursor.col >= vp.x);
1648
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1649
+
1650
+ var out_buffer: [200]u8 = undefined;
1651
+ const written = eb.getText(&out_buffer);
1652
+ const text = out_buffer[0..written];
1653
+ try std.testing.expect(std.mem.indexOf(u8, text, "XYZ") != null);
1654
+ }
1655
+
1656
+ test "EditorView - horizontal scroll: backspace in scrolled view" {
1657
+ const pool = gp.initGlobalPool(std.testing.allocator);
1658
+ defer gp.deinitGlobalPool();
1659
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1660
+ defer link.deinitGlobalLinkPool();
1661
+
1662
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1663
+ defer eb.deinit();
1664
+
1665
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1666
+ defer ev.deinit();
1667
+
1668
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1669
+
1670
+ try eb.setCursor(0, 50);
1671
+ _ = ev.getVirtualLines();
1672
+
1673
+ var vp = ev.getViewport().?;
1674
+ try std.testing.expect(vp.x > 0);
1675
+
1676
+ try eb.backspace();
1677
+ try eb.backspace();
1678
+ try eb.backspace();
1679
+ _ = ev.getVirtualLines();
1680
+
1681
+ vp = ev.getViewport().?;
1682
+ const cursor = ev.getPrimaryCursor();
1683
+ try std.testing.expectEqual(@as(u32, 47), cursor.col);
1684
+ try std.testing.expect(cursor.col >= vp.x);
1685
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1686
+ }
1687
+
1688
+ test "EditorView - horizontal scroll: short lines reset scroll" {
1689
+ const pool = gp.initGlobalPool(std.testing.allocator);
1690
+ defer gp.deinitGlobalPool();
1691
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1692
+ defer link.deinitGlobalLinkPool();
1693
+
1694
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1695
+ defer eb.deinit();
1696
+
1697
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1698
+ defer ev.deinit();
1699
+
1700
+ try eb.setText("Short line\nAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJ\nAnother short");
1701
+
1702
+ try eb.setCursor(1, 50);
1703
+ _ = ev.getVirtualLines();
1704
+
1705
+ var vp = ev.getViewport().?;
1706
+ try std.testing.expect(vp.x > 0);
1707
+
1708
+ try eb.setCursor(0, 5);
1709
+ _ = ev.getVirtualLines();
1710
+
1711
+ vp = ev.getViewport().?;
1712
+ try std.testing.expect(vp.x <= 5);
1713
+
1714
+ try eb.setCursor(1, 50);
1715
+ _ = ev.getVirtualLines();
1716
+
1717
+ vp = ev.getViewport().?;
1718
+ try std.testing.expect(vp.x > 0);
1719
+ }
1720
+
1721
+ test "EditorView - horizontal scroll: scroll margin works" {
1722
+ const pool = gp.initGlobalPool(std.testing.allocator);
1723
+ defer gp.deinitGlobalPool();
1724
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1725
+ defer link.deinitGlobalLinkPool();
1726
+
1727
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1728
+ defer eb.deinit();
1729
+
1730
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1731
+ defer ev.deinit();
1732
+
1733
+ ev.setScrollMargin(0.2);
1734
+
1735
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1736
+
1737
+ try eb.setCursor(0, 0);
1738
+ _ = ev.getVirtualLines();
1739
+
1740
+ var i: u32 = 0;
1741
+ while (i < 25) : (i += 1) {
1742
+ eb.moveRight();
1743
+ }
1744
+
1745
+ _ = ev.getVirtualLines();
1746
+
1747
+ const vp = ev.getViewport().?;
1748
+ const cursor = ev.getPrimaryCursor();
1749
+
1750
+ const cursor_offset_in_viewport = cursor.col - vp.x;
1751
+ try std.testing.expect(cursor_offset_in_viewport >= 4);
1752
+ try std.testing.expect(cursor_offset_in_viewport < vp.width - 4);
1753
+ }
1754
+
1755
+ test "EditorView - horizontal scroll: no scrolling with wrapping enabled" {
1756
+ const pool = gp.initGlobalPool(std.testing.allocator);
1757
+ defer gp.deinitGlobalPool();
1758
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1759
+ defer link.deinitGlobalLinkPool();
1760
+
1761
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1762
+ defer eb.deinit();
1763
+
1764
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1765
+ defer ev.deinit();
1766
+
1767
+ ev.setWrapMode(.char);
1768
+
1769
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1770
+
1771
+ try eb.setCursor(0, 50);
1772
+ _ = ev.getVirtualLines();
1773
+
1774
+ const vp = ev.getViewport().?;
1775
+ try std.testing.expectEqual(@as(u32, 0), vp.x);
1776
+ }
1777
+
1778
+ test "EditorView - horizontal scroll: cursor position correct after scrolling" {
1779
+ const pool = gp.initGlobalPool(std.testing.allocator);
1780
+ defer gp.deinitGlobalPool();
1781
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1782
+ defer link.deinitGlobalLinkPool();
1783
+
1784
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1785
+ defer eb.deinit();
1786
+
1787
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1788
+ defer ev.deinit();
1789
+
1790
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1791
+
1792
+ try eb.setCursor(0, 0);
1793
+ _ = ev.getVirtualLines();
1794
+
1795
+ var i: u32 = 0;
1796
+ while (i < 50) : (i += 1) {
1797
+ eb.moveRight();
1798
+ _ = ev.getVirtualLines();
1799
+
1800
+ const cursor = ev.getPrimaryCursor();
1801
+ const vp = ev.getViewport().?;
1802
+ const vcursor = ev.getVisualCursor();
1803
+
1804
+ try std.testing.expectEqual(cursor.col, vcursor.logical_col);
1805
+ try std.testing.expect(cursor.col >= vp.x);
1806
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1807
+ }
1808
+ }
1809
+
1810
+ test "EditorView - horizontal scroll: rapid movements maintain visibility" {
1811
+ const pool = gp.initGlobalPool(std.testing.allocator);
1812
+ defer gp.deinitGlobalPool();
1813
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1814
+ defer link.deinitGlobalLinkPool();
1815
+
1816
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1817
+ defer eb.deinit();
1818
+
1819
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1820
+ defer ev.deinit();
1821
+
1822
+ try eb.setText("AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP");
1823
+
1824
+ try eb.setCursor(0, 0);
1825
+ try eb.setCursor(0, 80);
1826
+ try eb.setCursor(0, 40);
1827
+ try eb.setCursor(0, 10);
1828
+ try eb.setCursor(0, 60);
1829
+
1830
+ _ = ev.getVirtualLines();
1831
+
1832
+ const vp = ev.getViewport().?;
1833
+ const cursor = ev.getPrimaryCursor();
1834
+
1835
+ try std.testing.expectEqual(@as(u32, 60), cursor.col);
1836
+ try std.testing.expect(cursor.col >= vp.x);
1837
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1838
+ }
1839
+
1840
+ test "EditorView - horizontal scroll: goto end of long line" {
1841
+ const pool = gp.initGlobalPool(std.testing.allocator);
1842
+ defer gp.deinitGlobalPool();
1843
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1844
+ defer link.deinitGlobalLinkPool();
1845
+
1846
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1847
+ defer eb.deinit();
1848
+
1849
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
1850
+ defer ev.deinit();
1851
+
1852
+ const long_line = "AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJKKKKKKKKKKLLLLLLLLLLMMMMMMMMMMNNNNNNNNNNOOOOOOOOOOPPPPPPPPPP";
1853
+ try eb.setText(long_line);
1854
+
1855
+ try eb.setCursor(0, 0);
1856
+ _ = ev.getVirtualLines();
1857
+
1858
+ try eb.setCursor(0, @intCast(long_line.len));
1859
+ _ = ev.getVirtualLines();
1860
+
1861
+ const vp = ev.getViewport().?;
1862
+ const cursor = ev.getPrimaryCursor();
1863
+
1864
+ try std.testing.expect(vp.x > 0);
1865
+ try std.testing.expect(cursor.col >= vp.x);
1866
+ try std.testing.expect(cursor.col < vp.x + vp.width);
1867
+ }
1868
+
1869
+ test "EditorView - cursor at second cell of width=2 grapheme moveLeft should jump to before grapheme" {
1870
+ const pool = gp.initGlobalPool(std.testing.allocator);
1871
+ defer gp.deinitGlobalPool();
1872
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1873
+ defer link.deinitGlobalLinkPool();
1874
+
1875
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1876
+ defer eb.deinit();
1877
+
1878
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
1879
+ defer ev.deinit();
1880
+
1881
+ try eb.setText("(emoji 🌟 and CJK 世界)");
1882
+
1883
+ try eb.setCursor(0, 7);
1884
+ var cursor = eb.getPrimaryCursor();
1885
+ try std.testing.expectEqual(@as(u32, 7), cursor.col);
1886
+
1887
+ // Move right - should jump over emoji to col 9
1888
+ eb.moveRight();
1889
+ cursor = eb.getPrimaryCursor();
1890
+ try std.testing.expectEqual(@as(u32, 9), cursor.col);
1891
+
1892
+ // Manually set cursor to col 8 (second cell of emoji at 7-8)
1893
+ // TODO: setCursor should probably also snap to beginning of grapheme?
1894
+ // When the width/cell based cursor is visual only and EditBuffer/Rope cursor is byte based
1895
+ try eb.setCursor(0, 8);
1896
+ cursor = eb.getPrimaryCursor();
1897
+ try std.testing.expectEqual(@as(u32, 8), cursor.col);
1898
+
1899
+ // Should jump to col 9 (after the emoji), not col 10
1900
+ eb.moveRight();
1901
+ cursor = eb.getPrimaryCursor();
1902
+ try std.testing.expectEqual(@as(u32, 9), cursor.col);
1903
+
1904
+ try eb.setCursor(0, 8);
1905
+ cursor = eb.getPrimaryCursor();
1906
+ try std.testing.expectEqual(@as(u32, 8), cursor.col);
1907
+
1908
+ eb.moveLeft();
1909
+ cursor = eb.getPrimaryCursor();
1910
+ try std.testing.expectEqual(@as(u32, 6), cursor.col);
1911
+ }
1912
+
1913
+ test "EditorView - cursor should be able to land after closing paren on line with wide graphemes" {
1914
+ const pool = gp.initGlobalPool(std.testing.allocator);
1915
+ defer gp.deinitGlobalPool();
1916
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1917
+ defer link.deinitGlobalLinkPool();
1918
+
1919
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1920
+ defer eb.deinit();
1921
+
1922
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
1923
+ defer ev.deinit();
1924
+
1925
+ try eb.setText("(emoji 🌟 and CJK 世界)\nNext line");
1926
+
1927
+ try eb.setCursor(0, 0);
1928
+ var cursor = eb.getPrimaryCursor();
1929
+
1930
+ var i: u32 = 0;
1931
+ while (i < 30) : (i += 1) {
1932
+ const prev_col = cursor.col;
1933
+ const prev_row = cursor.row;
1934
+ eb.moveRight();
1935
+ cursor = eb.getPrimaryCursor();
1936
+
1937
+ // Should not jump to next line until we've reached the end of the current line
1938
+ if (prev_row == 0 and cursor.row == 1) {
1939
+ // We jumped to the next line - check that we were at the end
1940
+ const iter_mod = @import("../text-buffer-iterators.zig");
1941
+ const line_width = iter_mod.lineWidthAt(eb.getTextBuffer().rope(), 0);
1942
+ try std.testing.expectEqual(line_width, prev_col);
1943
+ break;
1944
+ }
1945
+
1946
+ if (i > 25) {
1947
+ break;
1948
+ }
1949
+ }
1950
+
1951
+ try std.testing.expectEqual(@as(u32, 1), cursor.row);
1952
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
1953
+ }
1954
+
1955
+ test "EditorView - visual cursor should stay on same line when moving to line end with wide graphemes" {
1956
+ const pool = gp.initGlobalPool(std.testing.allocator);
1957
+ defer gp.deinitGlobalPool();
1958
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1959
+ defer link.deinitGlobalLinkPool();
1960
+
1961
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1962
+ defer eb.deinit();
1963
+
1964
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
1965
+ defer ev.deinit();
1966
+
1967
+ try eb.setText("(emoji 🌟 and CJK 世界)\nNext line");
1968
+
1969
+ try eb.setCursor(0, 0);
1970
+
1971
+ var i: u32 = 0;
1972
+ while (i < 30) : (i += 1) {
1973
+ eb.moveRight();
1974
+ const cursor = eb.getPrimaryCursor();
1975
+ const vcursor = ev.getVisualCursor();
1976
+
1977
+ // Visual cursor should stay on row 0 until we move past the line end
1978
+ if (cursor.row == 0) {
1979
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_row);
1980
+ try std.testing.expectEqual(cursor.col, vcursor.visual_col);
1981
+ }
1982
+
1983
+ if (cursor.row == 1) {
1984
+ try std.testing.expectEqual(@as(u32, 1), vcursor.visual_row);
1985
+ break;
1986
+ }
1987
+
1988
+ if (i > 25) break;
1989
+ }
1990
+ }
1991
+
1992
+ test "EditorView - placeholder with styled text renders with correct highlights" {
1993
+ const pool = gp.initGlobalPool(std.testing.allocator);
1994
+ defer gp.deinitGlobalPool();
1995
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
1996
+ defer link.deinitGlobalLinkPool();
1997
+
1998
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
1999
+ defer eb.deinit();
2000
+
2001
+ const ss = @import("../syntax-style.zig");
2002
+ const style = try ss.SyntaxStyle.init(std.testing.allocator);
2003
+ defer style.deinit();
2004
+ eb.getTextBuffer().setSyntaxStyle(style);
2005
+
2006
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2007
+ defer ev.deinit();
2008
+
2009
+ const text_part1 = "Enter ";
2010
+ const text_part2 = "something";
2011
+ const text_part3 = " here";
2012
+
2013
+ const fg_gray = [4]f32{ 0.5, 0.5, 0.5, 1.0 };
2014
+ const fg_blue = [4]f32{ 0.3, 0.5, 0.9, 1.0 };
2015
+
2016
+ const chunks = [_]text_buffer.StyledChunk{
2017
+ .{
2018
+ .text_ptr = text_part1.ptr,
2019
+ .text_len = text_part1.len,
2020
+ .fg_ptr = @ptrCast(&fg_gray),
2021
+ .bg_ptr = null,
2022
+ .attributes = 0,
2023
+ },
2024
+ .{
2025
+ .text_ptr = text_part2.ptr,
2026
+ .text_len = text_part2.len,
2027
+ .fg_ptr = @ptrCast(&fg_blue),
2028
+ .bg_ptr = null,
2029
+ .attributes = 0,
2030
+ },
2031
+ .{
2032
+ .text_ptr = text_part3.ptr,
2033
+ .text_len = text_part3.len,
2034
+ .fg_ptr = @ptrCast(&fg_gray),
2035
+ .bg_ptr = null,
2036
+ .attributes = 0,
2037
+ },
2038
+ };
2039
+
2040
+ try ev.setPlaceholderStyledText(&chunks);
2041
+
2042
+ var out_buffer: [100]u8 = undefined;
2043
+ const written = eb.getText(&out_buffer);
2044
+ try std.testing.expectEqual(@as(usize, 0), written);
2045
+
2046
+ ev.updateBeforeRender();
2047
+
2048
+ const tbv_ptr = ev.getTextBufferView();
2049
+
2050
+ var opt_buffer = try opt_buffer_mod.OptimizedBuffer.init(
2051
+ std.testing.allocator,
2052
+ 80,
2053
+ 24,
2054
+ .{ .pool = pool, .width_method = .wcwidth },
2055
+ );
2056
+ defer opt_buffer.deinit();
2057
+
2058
+ try opt_buffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, 32);
2059
+ try opt_buffer.drawTextBuffer(tbv_ptr, 0, 0);
2060
+
2061
+ const epsilon: f32 = 0.01;
2062
+
2063
+ const cell_0 = opt_buffer.get(0, 0) orelse unreachable;
2064
+ try std.testing.expectEqual(@as(u32, 'E'), cell_0.char);
2065
+
2066
+ const cell_6 = opt_buffer.get(6, 0) orelse unreachable;
2067
+ try std.testing.expectEqual(@as(u32, 's'), cell_6.char);
2068
+
2069
+ const cell_15 = opt_buffer.get(15, 0) orelse unreachable;
2070
+ try std.testing.expectEqual(@as(u32, ' '), cell_15.char);
2071
+
2072
+ const fg_0 = opt_buffer.buffer.fg[0];
2073
+ try std.testing.expect(@abs(fg_0[0] - fg_gray[0]) < epsilon);
2074
+ try std.testing.expect(@abs(fg_0[1] - fg_gray[1]) < epsilon);
2075
+ try std.testing.expect(@abs(fg_0[2] - fg_gray[2]) < epsilon);
2076
+
2077
+ const fg_6 = opt_buffer.buffer.fg[6];
2078
+ try std.testing.expect(@abs(fg_6[0] - fg_blue[0]) < epsilon);
2079
+ try std.testing.expect(@abs(fg_6[1] - fg_blue[1]) < epsilon);
2080
+ try std.testing.expect(@abs(fg_6[2] - fg_blue[2]) < epsilon);
2081
+
2082
+ const fg_15 = opt_buffer.buffer.fg[15];
2083
+ try std.testing.expect(@abs(fg_15[0] - fg_gray[0]) < epsilon);
2084
+ try std.testing.expect(@abs(fg_15[1] - fg_gray[1]) < epsilon);
2085
+ try std.testing.expect(@abs(fg_15[2] - fg_gray[2]) < epsilon);
2086
+ }
2087
+
2088
+ test "EditorView - getNextWordBoundary returns VisualCursor" {
2089
+ const pool = gp.initGlobalPool(std.testing.allocator);
2090
+ defer gp.deinitGlobalPool();
2091
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2092
+ defer link.deinitGlobalLinkPool();
2093
+
2094
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2095
+ defer eb.deinit();
2096
+
2097
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2098
+ defer ev.deinit();
2099
+
2100
+ try eb.insertText("Hello World Test");
2101
+ try eb.setCursor(0, 0);
2102
+
2103
+ const next_vcursor = ev.getNextWordBoundary();
2104
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.logical_row);
2105
+ try std.testing.expectEqual(@as(u32, 6), next_vcursor.logical_col);
2106
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.visual_row);
2107
+ try std.testing.expectEqual(@as(u32, 6), next_vcursor.visual_col);
2108
+ }
2109
+
2110
+ test "EditorView - getPrevWordBoundary returns VisualCursor" {
2111
+ const pool = gp.initGlobalPool(std.testing.allocator);
2112
+ defer gp.deinitGlobalPool();
2113
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2114
+ defer link.deinitGlobalLinkPool();
2115
+
2116
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2117
+ defer eb.deinit();
2118
+
2119
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2120
+ defer ev.deinit();
2121
+
2122
+ try eb.insertText("Hello World Test");
2123
+ try eb.setCursor(0, 12);
2124
+
2125
+ const prev_vcursor = ev.getPrevWordBoundary();
2126
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.logical_row);
2127
+ try std.testing.expectEqual(@as(u32, 6), prev_vcursor.logical_col);
2128
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.visual_row);
2129
+ try std.testing.expectEqual(@as(u32, 6), prev_vcursor.visual_col);
2130
+ }
2131
+
2132
+ test "EditorView - word boundary with wrapping" {
2133
+ const pool = gp.initGlobalPool(std.testing.allocator);
2134
+ defer gp.deinitGlobalPool();
2135
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2136
+ defer link.deinitGlobalLinkPool();
2137
+
2138
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2139
+ defer eb.deinit();
2140
+
2141
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
2142
+ defer ev.deinit();
2143
+
2144
+ ev.setWrapMode(.char);
2145
+
2146
+ try eb.setText("This is a very long line that will wrap and has multiple words");
2147
+ try eb.setCursor(0, 0);
2148
+
2149
+ const next_vcursor = ev.getNextWordBoundary();
2150
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.logical_row);
2151
+ try std.testing.expectEqual(@as(u32, 5), next_vcursor.logical_col);
2152
+
2153
+ try std.testing.expect(next_vcursor.visual_col <= 20);
2154
+ }
2155
+
2156
+ test "EditorView - word boundary across lines" {
2157
+ const pool = gp.initGlobalPool(std.testing.allocator);
2158
+ defer gp.deinitGlobalPool();
2159
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2160
+ defer link.deinitGlobalLinkPool();
2161
+
2162
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2163
+ defer eb.deinit();
2164
+
2165
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2166
+ defer ev.deinit();
2167
+
2168
+ try eb.insertText("Hello\nWorld");
2169
+ try eb.setCursor(0, 5);
2170
+
2171
+ const next_vcursor = ev.getNextWordBoundary();
2172
+ try std.testing.expectEqual(@as(u32, 1), next_vcursor.logical_row);
2173
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.logical_col);
2174
+ try std.testing.expectEqual(@as(u32, 1), next_vcursor.visual_row);
2175
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.visual_col);
2176
+ }
2177
+
2178
+ test "EditorView - word boundary prev across lines" {
2179
+ const pool = gp.initGlobalPool(std.testing.allocator);
2180
+ defer gp.deinitGlobalPool();
2181
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2182
+ defer link.deinitGlobalLinkPool();
2183
+
2184
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2185
+ defer eb.deinit();
2186
+
2187
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2188
+ defer ev.deinit();
2189
+
2190
+ try eb.insertText("Hello\nWorld");
2191
+ try eb.setCursor(1, 0);
2192
+
2193
+ const prev_vcursor = ev.getPrevWordBoundary();
2194
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.logical_row);
2195
+ try std.testing.expectEqual(@as(u32, 5), prev_vcursor.logical_col);
2196
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.visual_row);
2197
+ try std.testing.expectEqual(@as(u32, 5), prev_vcursor.visual_col);
2198
+ }
2199
+
2200
+ test "EditorView - word boundary with punctuation" {
2201
+ const pool = gp.initGlobalPool(std.testing.allocator);
2202
+ defer gp.deinitGlobalPool();
2203
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2204
+ defer link.deinitGlobalLinkPool();
2205
+
2206
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2207
+ defer eb.deinit();
2208
+
2209
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2210
+ defer ev.deinit();
2211
+
2212
+ try eb.insertText("self-contained multi-word");
2213
+ try eb.setCursor(0, 0);
2214
+
2215
+ const next_vcursor = ev.getNextWordBoundary();
2216
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.logical_row);
2217
+ try std.testing.expectEqual(@as(u32, 5), next_vcursor.logical_col);
2218
+ }
2219
+
2220
+ test "EditorView - word boundary at end of buffer" {
2221
+ const pool = gp.initGlobalPool(std.testing.allocator);
2222
+ defer gp.deinitGlobalPool();
2223
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2224
+ defer link.deinitGlobalLinkPool();
2225
+
2226
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2227
+ defer eb.deinit();
2228
+
2229
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2230
+ defer ev.deinit();
2231
+
2232
+ try eb.insertText("Hello World");
2233
+ try eb.setCursor(0, 11);
2234
+
2235
+ const next_vcursor = ev.getNextWordBoundary();
2236
+ try std.testing.expectEqual(@as(u32, 0), next_vcursor.logical_row);
2237
+ try std.testing.expectEqual(@as(u32, 11), next_vcursor.logical_col);
2238
+ }
2239
+
2240
+ test "EditorView - word boundary at start of buffer" {
2241
+ const pool = gp.initGlobalPool(std.testing.allocator);
2242
+ defer gp.deinitGlobalPool();
2243
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2244
+ defer link.deinitGlobalLinkPool();
2245
+
2246
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2247
+ defer eb.deinit();
2248
+
2249
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2250
+ defer ev.deinit();
2251
+
2252
+ try eb.insertText("Hello World");
2253
+ try eb.setCursor(0, 0);
2254
+
2255
+ const prev_vcursor = ev.getPrevWordBoundary();
2256
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.logical_row);
2257
+ try std.testing.expectEqual(@as(u32, 0), prev_vcursor.logical_col);
2258
+ }
2259
+
2260
+ test "EditorView - horizontal scroll: combined vertical and horizontal scrolling" {
2261
+ const pool = gp.initGlobalPool(std.testing.allocator);
2262
+ defer gp.deinitGlobalPool();
2263
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2264
+ defer link.deinitGlobalLinkPool();
2265
+
2266
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2267
+ defer eb.deinit();
2268
+
2269
+ var ev = try EditorView.init(std.testing.allocator, eb, 20, 10);
2270
+ defer ev.deinit();
2271
+
2272
+ const line0 = "AAAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJ";
2273
+ const repeated_line = "\nAAAAAAAABBBBBBBBBBCCCCCCCCCCDDDDDDDDDDEEEEEEEEEEFFFFFFFFFFGGGGGGGGGGHHHHHHHHHHIIIIIIIIIIJJJJJJJJJJ";
2274
+
2275
+ var buffer: [3000]u8 = undefined;
2276
+ var fbs = std.io.fixedBufferStream(&buffer);
2277
+ const writer = fbs.writer();
2278
+ writer.writeAll(line0) catch unreachable;
2279
+ var i: u32 = 1;
2280
+ while (i < 20) : (i += 1) {
2281
+ writer.writeAll(repeated_line) catch unreachable;
2282
+ }
2283
+
2284
+ const text = fbs.getWritten();
2285
+ try eb.setText(text);
2286
+
2287
+ try eb.setCursor(15, 60);
2288
+ _ = ev.getVirtualLines();
2289
+
2290
+ const vp = ev.getViewport().?;
2291
+ const cursor = ev.getPrimaryCursor();
2292
+
2293
+ try std.testing.expect(vp.y > 0);
2294
+ try std.testing.expect(vp.x > 0);
2295
+
2296
+ try std.testing.expect(cursor.row >= vp.y);
2297
+ try std.testing.expect(cursor.row < vp.y + vp.height);
2298
+ try std.testing.expect(cursor.col >= vp.x);
2299
+ try std.testing.expect(cursor.col < vp.x + vp.width);
2300
+ }
2301
+
2302
+ test "EditorView - deleteSelectedText single line" {
2303
+ const pool = gp.initGlobalPool(std.testing.allocator);
2304
+ defer gp.deinitGlobalPool();
2305
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2306
+ defer link.deinitGlobalLinkPool();
2307
+
2308
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2309
+ defer eb_inst.deinit();
2310
+
2311
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 80, 24);
2312
+ defer ev.deinit();
2313
+
2314
+ try eb_inst.setText("Hello World");
2315
+
2316
+ ev.text_buffer_view.setSelection(0, 5, null, null);
2317
+
2318
+ const sel_before = ev.text_buffer_view.getSelection();
2319
+ try std.testing.expect(sel_before != null);
2320
+ try std.testing.expectEqual(@as(u32, 0), sel_before.?.start);
2321
+ try std.testing.expectEqual(@as(u32, 5), sel_before.?.end);
2322
+
2323
+ try ev.deleteSelectedText();
2324
+
2325
+ var out_buffer: [100]u8 = undefined;
2326
+ const written = ev.getText(&out_buffer);
2327
+ try std.testing.expectEqualStrings(" World", out_buffer[0..written]);
2328
+
2329
+ const cursor = ev.getPrimaryCursor();
2330
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2331
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
2332
+
2333
+ const sel_after = ev.text_buffer_view.getSelection();
2334
+ try std.testing.expect(sel_after == null);
2335
+ }
2336
+
2337
+ test "EditorView - deleteSelectedText multi-line" {
2338
+ const pool = gp.initGlobalPool(std.testing.allocator);
2339
+ defer gp.deinitGlobalPool();
2340
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2341
+ defer link.deinitGlobalLinkPool();
2342
+
2343
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2344
+ defer eb_inst.deinit();
2345
+
2346
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 80, 24);
2347
+ defer ev.deinit();
2348
+
2349
+ try eb_inst.setText("Line 1\nLine 2\nLine 3");
2350
+
2351
+ ev.text_buffer_view.setSelection(2, 15, null, null);
2352
+
2353
+ try ev.deleteSelectedText();
2354
+
2355
+ var out_buffer: [100]u8 = undefined;
2356
+ const written = ev.getText(&out_buffer);
2357
+ try std.testing.expectEqualStrings("Liine 3", out_buffer[0..written]);
2358
+
2359
+ const cursor = ev.getPrimaryCursor();
2360
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2361
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
2362
+ }
2363
+
2364
+ test "EditorView - deleteSelectedText with wrapping" {
2365
+ const pool = gp.initGlobalPool(std.testing.allocator);
2366
+ defer gp.deinitGlobalPool();
2367
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2368
+ defer link.deinitGlobalLinkPool();
2369
+
2370
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2371
+ defer eb_inst.deinit();
2372
+
2373
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 20, 10);
2374
+ defer ev.deinit();
2375
+
2376
+ ev.setWrapMode(.char);
2377
+
2378
+ try eb_inst.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ");
2379
+
2380
+ const vline_count = ev.getTotalVirtualLineCount();
2381
+ try std.testing.expect(vline_count >= 2);
2382
+
2383
+ ev.text_buffer_view.setSelection(5, 15, null, null);
2384
+
2385
+ try ev.deleteSelectedText();
2386
+
2387
+ var out_buffer: [100]u8 = undefined;
2388
+ const written = ev.getText(&out_buffer);
2389
+ try std.testing.expectEqualStrings("ABCDEPQRSTUVWXYZ", out_buffer[0..written]);
2390
+
2391
+ const cursor = ev.getPrimaryCursor();
2392
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2393
+ try std.testing.expectEqual(@as(u32, 5), cursor.col);
2394
+ }
2395
+
2396
+ test "EditorView - deleteSelectedText with viewport scrolled" {
2397
+ const pool = gp.initGlobalPool(std.testing.allocator);
2398
+ defer gp.deinitGlobalPool();
2399
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2400
+ defer link.deinitGlobalLinkPool();
2401
+
2402
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2403
+ defer eb_inst.deinit();
2404
+
2405
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 40, 5);
2406
+ defer ev.deinit();
2407
+
2408
+ try eb_inst.setText("Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9\nLine 10\nLine 11\nLine 12\nLine 13\nLine 14\nLine 15\nLine 16\nLine 17\nLine 18\nLine 19");
2409
+
2410
+ try eb_inst.gotoLine(10);
2411
+ _ = ev.getVirtualLines();
2412
+
2413
+ var vp = ev.getViewport().?;
2414
+ try std.testing.expect(vp.y > 0);
2415
+
2416
+ ev.text_buffer_view.setSelection(50, 70, null, null);
2417
+
2418
+ try ev.deleteSelectedText();
2419
+
2420
+ _ = ev.getVirtualLines();
2421
+ vp = ev.getViewport().?;
2422
+ const cursor = ev.getPrimaryCursor();
2423
+
2424
+ try std.testing.expect(cursor.row >= vp.y);
2425
+ try std.testing.expect(cursor.row < vp.y + vp.height);
2426
+ }
2427
+
2428
+ test "EditorView - deleteSelectedText with no selection" {
2429
+ const pool = gp.initGlobalPool(std.testing.allocator);
2430
+ defer gp.deinitGlobalPool();
2431
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2432
+ defer link.deinitGlobalLinkPool();
2433
+
2434
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2435
+ defer eb_inst.deinit();
2436
+
2437
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 80, 24);
2438
+ defer ev.deinit();
2439
+
2440
+ try eb_inst.setText("Hello World");
2441
+
2442
+ try ev.deleteSelectedText();
2443
+
2444
+ var out_buffer: [100]u8 = undefined;
2445
+ const written = ev.getText(&out_buffer);
2446
+ try std.testing.expectEqualStrings("Hello World", out_buffer[0..written]);
2447
+ }
2448
+
2449
+ test "EditorView - deleteSelectedText entire line" {
2450
+ const pool = gp.initGlobalPool(std.testing.allocator);
2451
+ defer gp.deinitGlobalPool();
2452
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2453
+ defer link.deinitGlobalLinkPool();
2454
+
2455
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2456
+ defer eb_inst.deinit();
2457
+
2458
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 80, 24);
2459
+ defer ev.deinit();
2460
+
2461
+ try eb_inst.setText("First\nSecond\nThird\n");
2462
+
2463
+ ev.text_buffer_view.setSelection(5, 13, null, null);
2464
+
2465
+ try ev.deleteSelectedText();
2466
+
2467
+ var out_buffer: [100]u8 = undefined;
2468
+ const written = ev.getText(&out_buffer);
2469
+ try std.testing.expectEqualStrings("FirstThird\n", out_buffer[0..written]);
2470
+
2471
+ const cursor = ev.getPrimaryCursor();
2472
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2473
+ try std.testing.expectEqual(@as(u32, 5), cursor.col);
2474
+ }
2475
+
2476
+ test "EditorView - deleteSelectedText respects selection with empty lines" {
2477
+ const pool = gp.initGlobalPool(std.testing.allocator);
2478
+ defer gp.deinitGlobalPool();
2479
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2480
+ defer link.deinitGlobalLinkPool();
2481
+
2482
+ var eb_inst = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2483
+ defer eb_inst.deinit();
2484
+
2485
+ var ev = try EditorView.init(std.testing.allocator, eb_inst, 40, 10);
2486
+ defer ev.deinit();
2487
+
2488
+ ev.setWrapMode(.word);
2489
+
2490
+ try eb_inst.setText("AAAA\n\nBBBB\n\nCCCC");
2491
+
2492
+ try eb_inst.setCursor(2, 0);
2493
+
2494
+ _ = ev.text_buffer_view.setLocalSelection(0, 2, 4, 2, null, null);
2495
+
2496
+ const sel = ev.text_buffer_view.getSelection();
2497
+ try std.testing.expect(sel != null);
2498
+
2499
+ try std.testing.expectEqual(@as(u32, 6), sel.?.start);
2500
+ try std.testing.expectEqual(@as(u32, 10), sel.?.end);
2501
+
2502
+ var selected_buffer: [100]u8 = undefined;
2503
+ const selected_len = ev.text_buffer_view.getSelectedTextIntoBuffer(&selected_buffer);
2504
+ const selected_text = selected_buffer[0..selected_len];
2505
+ try std.testing.expectEqualStrings("BBBB", selected_text);
2506
+
2507
+ try ev.deleteSelectedText();
2508
+
2509
+ var out_buffer: [100]u8 = undefined;
2510
+ const written = ev.getText(&out_buffer);
2511
+ try std.testing.expectEqualStrings("AAAA\n\n\n\nCCCC", out_buffer[0..written]);
2512
+
2513
+ const cursor = ev.getPrimaryCursor();
2514
+ try std.testing.expectEqual(@as(u32, 2), cursor.row);
2515
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
2516
+ }
2517
+
2518
+ test "EditorView - word wrapping with space insertion maintains cursor sync" {
2519
+ const pool = gp.initGlobalPool(std.testing.allocator);
2520
+ defer gp.deinitGlobalPool();
2521
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2522
+ defer link.deinitGlobalLinkPool();
2523
+
2524
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2525
+ defer eb.deinit();
2526
+
2527
+ var ev = try EditorView.init(std.testing.allocator, eb, 15, 10);
2528
+ defer ev.deinit();
2529
+
2530
+ ev.setWrapMode(.word);
2531
+ ev.setViewport(Viewport{ .x = 0, .y = 0, .width = 15, .height = 10 }, true);
2532
+
2533
+ try eb.setText("AAAAAAAAAAAAAAAAAAA");
2534
+ try eb.setCursor(0, 7);
2535
+ try eb.insertText(" ");
2536
+
2537
+ const logical_cursor_after_space = eb.getPrimaryCursor();
2538
+ const vcursor_after_space = ev.getVisualCursor();
2539
+
2540
+ try std.testing.expectEqual(@as(u32, 0), logical_cursor_after_space.row);
2541
+ try std.testing.expectEqual(@as(u32, 8), logical_cursor_after_space.col);
2542
+
2543
+ try std.testing.expectEqual(@as(u32, 0), vcursor_after_space.logical_row);
2544
+ try std.testing.expectEqual(@as(u32, 1), vcursor_after_space.visual_row);
2545
+
2546
+ try eb.backspace();
2547
+
2548
+ const logical_cursor_after_backspace = eb.getPrimaryCursor();
2549
+ try std.testing.expectEqual(@as(u32, 0), logical_cursor_after_backspace.row);
2550
+ try std.testing.expectEqual(@as(u32, 7), logical_cursor_after_backspace.col);
2551
+ }
2552
+
2553
+ test "EditorView - getVisualCursor always returns on empty buffer" {
2554
+ const pool = gp.initGlobalPool(std.testing.allocator);
2555
+ defer gp.deinitGlobalPool();
2556
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2557
+ defer link.deinitGlobalLinkPool();
2558
+
2559
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2560
+ defer eb.deinit();
2561
+
2562
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2563
+ defer ev.deinit();
2564
+
2565
+ const vcursor = ev.getVisualCursor();
2566
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_row);
2567
+ try std.testing.expectEqual(@as(u32, 0), vcursor.visual_col);
2568
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
2569
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_col);
2570
+ }
2571
+
2572
+ test "EditorView - logicalToVisualCursor clamps row beyond last line" {
2573
+ const pool = gp.initGlobalPool(std.testing.allocator);
2574
+ defer gp.deinitGlobalPool();
2575
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2576
+ defer link.deinitGlobalLinkPool();
2577
+
2578
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2579
+ defer eb.deinit();
2580
+
2581
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2582
+ defer ev.deinit();
2583
+
2584
+ try eb.setText("Line 1\nLine 2\nLine 3");
2585
+
2586
+ const vcursor = ev.logicalToVisualCursor(100, 0);
2587
+ try std.testing.expectEqual(@as(u32, 2), vcursor.logical_row);
2588
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_col);
2589
+ }
2590
+
2591
+ test "EditorView - logicalToVisualCursor clamps col beyond line width" {
2592
+ const pool = gp.initGlobalPool(std.testing.allocator);
2593
+ defer gp.deinitGlobalPool();
2594
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2595
+ defer link.deinitGlobalLinkPool();
2596
+
2597
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2598
+ defer eb.deinit();
2599
+
2600
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2601
+ defer ev.deinit();
2602
+
2603
+ try eb.setText("Hello");
2604
+
2605
+ const vcursor = ev.logicalToVisualCursor(0, 100);
2606
+ try std.testing.expectEqual(@as(u32, 0), vcursor.logical_row);
2607
+ try std.testing.expectEqual(@as(u32, 5), vcursor.logical_col);
2608
+ try std.testing.expectEqual(@as(u32, 5), vcursor.visual_col);
2609
+ }
2610
+
2611
+ test "EditorView - placeholder shows when empty" {
2612
+ const pool = gp.initGlobalPool(std.testing.allocator);
2613
+ defer gp.deinitGlobalPool();
2614
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2615
+ defer link.deinitGlobalLinkPool();
2616
+
2617
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2618
+ defer eb.deinit();
2619
+
2620
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2621
+ defer ev.deinit();
2622
+
2623
+ const text = "Enter text here...";
2624
+ const gray_color = text_buffer.RGBA{ 0.4, 0.4, 0.4, 1.0 };
2625
+ const chunks = [_]text_buffer.StyledChunk{.{
2626
+ .text_ptr = text.ptr,
2627
+ .text_len = text.len,
2628
+ .fg_ptr = @ptrCast(&gray_color),
2629
+ .bg_ptr = null,
2630
+ .attributes = 0,
2631
+ }};
2632
+ try ev.setPlaceholderStyledText(&chunks);
2633
+
2634
+ var out_buffer: [100]u8 = undefined;
2635
+ const text_len = eb.getText(&out_buffer);
2636
+ try std.testing.expectEqual(@as(usize, 0), text_len);
2637
+
2638
+ try std.testing.expect(ev.placeholder_buffer != null);
2639
+ const placeholder = ev.placeholder_buffer.?;
2640
+ try std.testing.expectEqual(@as(u32, 18), placeholder.getLength());
2641
+ }
2642
+
2643
+ test "EditorView - placeholder cleared when set to empty" {
2644
+ const pool = gp.initGlobalPool(std.testing.allocator);
2645
+ defer gp.deinitGlobalPool();
2646
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2647
+ defer link.deinitGlobalLinkPool();
2648
+
2649
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2650
+ defer eb.deinit();
2651
+
2652
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2653
+ defer ev.deinit();
2654
+
2655
+ const text = "Placeholder";
2656
+ const gray_color = text_buffer.RGBA{ 0.4, 0.4, 0.4, 1.0 };
2657
+ const chunks = [_]text_buffer.StyledChunk{.{
2658
+ .text_ptr = text.ptr,
2659
+ .text_len = text.len,
2660
+ .fg_ptr = @ptrCast(&gray_color),
2661
+ .bg_ptr = null,
2662
+ .attributes = 0,
2663
+ }};
2664
+ try ev.setPlaceholderStyledText(&chunks);
2665
+
2666
+ try std.testing.expect(ev.placeholder_buffer != null);
2667
+
2668
+ const empty_chunks = [_]text_buffer.StyledChunk{};
2669
+ try ev.setPlaceholderStyledText(&empty_chunks);
2670
+
2671
+ try std.testing.expect(ev.placeholder_buffer == null);
2672
+ }
2673
+
2674
+ test "EditorView - placeholder with styled text" {
2675
+ const pool = gp.initGlobalPool(std.testing.allocator);
2676
+ defer gp.deinitGlobalPool();
2677
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2678
+ defer link.deinitGlobalLinkPool();
2679
+
2680
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2681
+ defer eb.deinit();
2682
+
2683
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2684
+ defer ev.deinit();
2685
+
2686
+ const text1 = "Hello ";
2687
+ const text2 = "World";
2688
+ const red_color = text_buffer.RGBA{ 1.0, 0.0, 0.0, 1.0 };
2689
+ const blue_color = text_buffer.RGBA{ 0.0, 0.0, 1.0, 1.0 };
2690
+
2691
+ const chunks = [_]text_buffer.StyledChunk{
2692
+ .{
2693
+ .text_ptr = text1.ptr,
2694
+ .text_len = text1.len,
2695
+ .fg_ptr = @ptrCast(&red_color),
2696
+ .bg_ptr = null,
2697
+ .attributes = 0,
2698
+ },
2699
+ .{
2700
+ .text_ptr = text2.ptr,
2701
+ .text_len = text2.len,
2702
+ .fg_ptr = @ptrCast(&blue_color),
2703
+ .bg_ptr = null,
2704
+ .attributes = 0,
2705
+ },
2706
+ };
2707
+
2708
+ try ev.setPlaceholderStyledText(&chunks);
2709
+
2710
+ try std.testing.expect(ev.placeholder_buffer != null);
2711
+ const placeholder = ev.placeholder_buffer.?;
2712
+ try std.testing.expectEqual(@as(u32, 11), placeholder.getLength());
2713
+ }
2714
+
2715
+ test "EditorView - placeholder renders to buffer when empty" {
2716
+ const pool = gp.initGlobalPool(std.testing.allocator);
2717
+ defer gp.deinitGlobalPool();
2718
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2719
+ defer link.deinitGlobalLinkPool();
2720
+
2721
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2722
+ defer eb.deinit();
2723
+
2724
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2725
+ defer ev.deinit();
2726
+
2727
+ const placeholder_text = "Type something...";
2728
+ const gray_color = text_buffer.RGBA{ 0.5, 0.5, 0.5, 1.0 };
2729
+ const placeholder_chunks = [_]text_buffer.StyledChunk{.{
2730
+ .text_ptr = placeholder_text.ptr,
2731
+ .text_len = placeholder_text.len,
2732
+ .fg_ptr = @ptrCast(&gray_color),
2733
+ .bg_ptr = null,
2734
+ .attributes = 0,
2735
+ }};
2736
+ try ev.setPlaceholderStyledText(&placeholder_chunks);
2737
+
2738
+ try std.testing.expect(ev.placeholder_buffer != null);
2739
+ try std.testing.expect(ev.placeholder_active);
2740
+
2741
+ var opt_buffer = try opt_buffer_mod.OptimizedBuffer.init(
2742
+ std.testing.allocator,
2743
+ 80,
2744
+ 10,
2745
+ .{ .pool = pool, .width_method = .wcwidth },
2746
+ );
2747
+ defer opt_buffer.deinit();
2748
+
2749
+ try opt_buffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, 32);
2750
+ try opt_buffer.drawEditorView(ev, 0, 0);
2751
+
2752
+ var out_buffer: [1000]u8 = undefined;
2753
+ const written = try opt_buffer.writeResolvedChars(&out_buffer, false);
2754
+ const result = out_buffer[0..written];
2755
+
2756
+ try std.testing.expect(std.mem.startsWith(u8, result, "Type something..."));
2757
+
2758
+ try eb.insertText("Hello");
2759
+
2760
+ try opt_buffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, 32);
2761
+ try opt_buffer.drawEditorView(ev, 0, 0);
2762
+ try std.testing.expect(!ev.placeholder_active);
2763
+
2764
+ const written2 = try opt_buffer.writeResolvedChars(&out_buffer, false);
2765
+ const result2 = out_buffer[0..written2];
2766
+
2767
+ try std.testing.expect(std.mem.startsWith(u8, result2, "Hello"));
2768
+ try std.testing.expect(!std.mem.startsWith(u8, result2, "Type something..."));
2769
+ }
2770
+
2771
+ test "EditorView - placeholder shrink clears tail and preserves background" {
2772
+ const pool = gp.initGlobalPool(std.testing.allocator);
2773
+ defer gp.deinitGlobalPool();
2774
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2775
+ defer link.deinitGlobalLinkPool();
2776
+
2777
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2778
+ defer eb.deinit();
2779
+
2780
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 10);
2781
+ defer ev.deinit();
2782
+
2783
+ const long_text = "Ask anything... \"Fix a TODO in the codebase\"";
2784
+ const short_text = "Run a command... \"pwd\"";
2785
+ const fg = text_buffer.RGBA{ 0.6, 0.6, 0.6, 1.0 };
2786
+ const panel_bg = text_buffer.RGBA{ 0.14, 0.14, 0.16, 1.0 };
2787
+
2788
+ const long_chunks = [_]text_buffer.StyledChunk{.{
2789
+ .text_ptr = long_text.ptr,
2790
+ .text_len = long_text.len,
2791
+ .fg_ptr = @ptrCast(&fg),
2792
+ .bg_ptr = null,
2793
+ .attributes = 0,
2794
+ }};
2795
+ const short_chunks = [_]text_buffer.StyledChunk{.{
2796
+ .text_ptr = short_text.ptr,
2797
+ .text_len = short_text.len,
2798
+ .fg_ptr = @ptrCast(&fg),
2799
+ .bg_ptr = null,
2800
+ .attributes = 0,
2801
+ }};
2802
+
2803
+ var opt_buffer = try opt_buffer_mod.OptimizedBuffer.init(
2804
+ std.testing.allocator,
2805
+ 120,
2806
+ 10,
2807
+ .{ .pool = pool, .width_method = .wcwidth },
2808
+ );
2809
+ defer opt_buffer.deinit();
2810
+
2811
+ try opt_buffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, 32);
2812
+
2813
+ var x: u32 = 0;
2814
+ while (x < 80) : (x += 1) {
2815
+ opt_buffer.set(x, 0, .{ .char = 32, .fg = fg, .bg = panel_bg, .attributes = 0 });
2816
+ }
2817
+
2818
+ try ev.setPlaceholderStyledText(&long_chunks);
2819
+ try opt_buffer.drawEditorView(ev, 0, 0);
2820
+
2821
+ x = 0;
2822
+ while (x < 80) : (x += 1) {
2823
+ opt_buffer.set(x, 0, .{ .char = 32, .fg = fg, .bg = panel_bg, .attributes = 0 });
2824
+ }
2825
+
2826
+ try ev.setPlaceholderStyledText(&short_chunks);
2827
+ try opt_buffer.drawEditorView(ev, 0, 0);
2828
+
2829
+ var out_buffer: [1600]u8 = undefined;
2830
+ const written = try opt_buffer.writeResolvedChars(&out_buffer, false);
2831
+ const line = out_buffer[0..written];
2832
+
2833
+ try std.testing.expect(std.mem.indexOf(u8, line, short_text) != null);
2834
+ try std.testing.expect(std.mem.indexOf(u8, line, "roken tests") == null);
2835
+ try std.testing.expect(std.mem.indexOf(u8, line, "TODO in the codebase") == null);
2836
+
2837
+ const tail = opt_buffer.get(35, 0) orelse return error.TestUnexpectedResult;
2838
+ try std.testing.expectEqual(@as(u32, 32), tail.char);
2839
+ try std.testing.expectEqual(@as(f32, panel_bg[0]), tail.bg[0]);
2840
+ try std.testing.expectEqual(@as(f32, panel_bg[1]), tail.bg[1]);
2841
+ try std.testing.expectEqual(@as(f32, panel_bg[2]), tail.bg[2]);
2842
+ try std.testing.expectEqual(@as(f32, panel_bg[3]), tail.bg[3]);
2843
+ }
2844
+
2845
+ test "EditorView - tab indicator set and get" {
2846
+ const pool = gp.initGlobalPool(std.testing.allocator);
2847
+ defer gp.deinitGlobalPool();
2848
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2849
+ defer link.deinitGlobalLinkPool();
2850
+
2851
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2852
+ defer eb.deinit();
2853
+
2854
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2855
+ defer ev.deinit();
2856
+
2857
+ try std.testing.expect(ev.getTabIndicator() == null);
2858
+ try std.testing.expect(ev.getTabIndicatorColor() == null);
2859
+
2860
+ ev.setTabIndicator('·');
2861
+ ev.setTabIndicatorColor(.{ 0.5, 0.5, 0.5, 1.0 });
2862
+
2863
+ try std.testing.expectEqual(@as(u32, '·'), ev.getTabIndicator().?);
2864
+ try std.testing.expectEqual(@as(f32, 0.5), ev.getTabIndicatorColor().?[0]);
2865
+ }
2866
+
2867
+ test "EditorView - tab indicator renders in buffer" {
2868
+ const pool = gp.initGlobalPool(std.testing.allocator);
2869
+ defer gp.deinitGlobalPool();
2870
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2871
+ defer link.deinitGlobalLinkPool();
2872
+
2873
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2874
+ defer eb.deinit();
2875
+
2876
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
2877
+ defer ev.deinit();
2878
+
2879
+ eb.tb.setTabWidth(4);
2880
+ try eb.insertText("A\tB");
2881
+
2882
+ ev.setTabIndicator('→');
2883
+ ev.setTabIndicatorColor(.{ 0.3, 0.3, 0.3, 1.0 });
2884
+
2885
+ var opt_buffer = try opt_buffer_mod.OptimizedBuffer.init(
2886
+ std.testing.allocator,
2887
+ 20,
2888
+ 10,
2889
+ .{ .pool = pool, .width_method = .wcwidth },
2890
+ );
2891
+ defer opt_buffer.deinit();
2892
+
2893
+ try opt_buffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, 32);
2894
+ try opt_buffer.drawEditorView(ev, 0, 0);
2895
+
2896
+ const cell_0 = opt_buffer.get(0, 0);
2897
+ try std.testing.expect(cell_0 != null);
2898
+ try std.testing.expectEqual(@as(u32, 'A'), cell_0.?.char);
2899
+
2900
+ const cell_1 = opt_buffer.get(1, 0);
2901
+ try std.testing.expect(cell_1 != null);
2902
+ try std.testing.expectEqual(@as(u32, '→'), cell_1.?.char);
2903
+ try std.testing.expectEqual(@as(f32, 0.3), cell_1.?.fg[0]);
2904
+
2905
+ const cell_2 = opt_buffer.get(2, 0);
2906
+ try std.testing.expect(cell_2 != null);
2907
+ try std.testing.expectEqual(@as(u32, 32), cell_2.?.char);
2908
+
2909
+ const cell_3 = opt_buffer.get(3, 0);
2910
+ try std.testing.expect(cell_3 != null);
2911
+ try std.testing.expectEqual(@as(u32, 32), cell_3.?.char);
2912
+
2913
+ const cell_4 = opt_buffer.get(4, 0);
2914
+ try std.testing.expect(cell_4 != null);
2915
+ try std.testing.expectEqual(@as(u32, 32), cell_4.?.char);
2916
+
2917
+ const cell_5 = opt_buffer.get(5, 0);
2918
+ try std.testing.expect(cell_5 != null);
2919
+ try std.testing.expectEqual(@as(u32, 'B'), cell_5.?.char);
2920
+ }
2921
+
2922
+ test "EditorView - word wrapping during editing: typing with incremental wrapping" {
2923
+ const pool = gp.initGlobalPool(std.testing.allocator);
2924
+ defer gp.deinitGlobalPool();
2925
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
2926
+ defer link.deinitGlobalLinkPool();
2927
+
2928
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
2929
+ defer eb.deinit();
2930
+
2931
+ var ev = try EditorView.init(std.testing.allocator, eb, 17, 10);
2932
+ defer ev.deinit();
2933
+
2934
+ ev.setWrapMode(.word);
2935
+
2936
+ // Type "Hello world ddddddddd" character by character
2937
+ // Width=17
2938
+ // "Hello world " = 12 chars
2939
+ // "Hello world ddddd" = 17 chars (fits exactly on one line)
2940
+ // "Hello world dddddd" = 18 chars (should wrap after "world ", moving ALL d's to next line)
2941
+ //
2942
+ // The key issue: word wrapping should keep the break point AFTER "world " consistently
2943
+ // When "Hello world dddddd" (18 chars) wraps, it should become:
2944
+ // Line 1: "Hello world " (12 chars)
2945
+ // Line 2: "dddddd" (6 chars)
2946
+ // NOT:
2947
+ // Line 1: "Hello world ddddd" (17 chars)
2948
+ // Line 2: "d" (1 char)
2949
+ const text_to_type = "Hello world ddddddddd";
2950
+
2951
+ for (text_to_type, 0..) |char, i| {
2952
+ var char_buf: [1]u8 = .{char};
2953
+ try eb.insertText(&char_buf);
2954
+ _ = ev.getVirtualLines();
2955
+
2956
+ const vline_count = ev.getTotalVirtualLineCount();
2957
+ const cursor = ev.getPrimaryCursor();
2958
+
2959
+ // "Hello world " = 12 chars (i=11 completes this)
2960
+ // "Hello world d" through "Hello world ddddd" = 13-17 chars (i=12 to i=16)
2961
+ // "Hello world dddddd" = 18 chars (i=17) - should wrap AFTER "world "
2962
+ const current_len = i + 1;
2963
+ if (current_len <= 17) {
2964
+ // Should fit on 1 line
2965
+ try std.testing.expectEqual(@as(u32, 1), vline_count);
2966
+ } else {
2967
+ // Should wrap AFTER "world ", moving ALL d's to line 2
2968
+ try std.testing.expectEqual(@as(u32, 2), vline_count);
2969
+
2970
+ // Cursor should still be on row 0 (single logical line that wrapped)
2971
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2972
+ }
2973
+ }
2974
+
2975
+ // Now we have "Hello world ddddddddd" (21 chars) with word wrapping at width=17
2976
+ // Should be: "Hello world " (12 chars) on vline 1, "ddddddddd" (9 chars) on vline 2
2977
+ var vline_count = ev.getTotalVirtualLineCount();
2978
+ try std.testing.expectEqual(@as(u32, 2), vline_count);
2979
+
2980
+ // Backspace to remove d's until only 2 remain: "Hello world dd"
2981
+ // We need to delete 7 d's (from 9 d's to 2 d's)
2982
+ var i: usize = 0;
2983
+ while (i < 7) : (i += 1) {
2984
+ try eb.backspace();
2985
+ _ = ev.getVirtualLines();
2986
+ }
2987
+
2988
+ // After removing 7 d's, we should have "Hello world dd" (14 chars)
2989
+ // This should fit on one line at width=17
2990
+ vline_count = ev.getTotalVirtualLineCount();
2991
+ try std.testing.expectEqual(@as(u32, 1), vline_count);
2992
+
2993
+ var cursor = ev.getPrimaryCursor();
2994
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
2995
+ try std.testing.expectEqual(@as(u32, 14), cursor.col);
2996
+
2997
+ // Now type more d's again - should wrap correctly after "world "
2998
+ // Starting with "Hello world dd" (14 chars)
2999
+ const more_ds = "ddddddd";
3000
+ for (more_ds, 0..) |char, j| {
3001
+ var char_buf: [1]u8 = .{char};
3002
+ try eb.insertText(&char_buf);
3003
+ _ = ev.getVirtualLines();
3004
+
3005
+ vline_count = ev.getTotalVirtualLineCount();
3006
+
3007
+ // After each d:
3008
+ // j=0: "Hello world ddd" (15) - fits on 1 line
3009
+ // j=1: "Hello world dddd" (16) - fits on 1 line
3010
+ // j=2: "Hello world ddddd" (17) - fits exactly on 1 line
3011
+ // j=3: "Hello world dddddd" (18) - should wrap AFTER "world ", moving ALL d's to line 2
3012
+ // j=4: "Hello world ddddddd" (19) - still wrapped same way
3013
+ // j=5: "Hello world dddddddd" (20) - still wrapped same way
3014
+ // j=6: "Hello world ddddddddd" (21) - still wrapped same way
3015
+ const current_len = 14 + j + 1;
3016
+ if (current_len <= 17) {
3017
+ // Should fit on 1 line
3018
+ try std.testing.expectEqual(@as(u32, 1), vline_count);
3019
+ } else {
3020
+ // Should wrap AFTER "world ", moving ALL d's to line 2
3021
+ // This is the key: the wrap point should stay at "world " boundary
3022
+ try std.testing.expectEqual(@as(u32, 2), vline_count);
3023
+
3024
+ // CRITICAL: Check that first virtual line is "Hello world " (12 chars)
3025
+ // and second virtual line has all the d's
3026
+ const vlines = ev.getVirtualLines();
3027
+ try std.testing.expect(vlines.len == 2);
3028
+
3029
+ // First vline should be "Hello world " with width 12
3030
+ try std.testing.expectEqual(@as(u32, 12), vlines[0].width_cols);
3031
+
3032
+ // Second vline should have all the d's (the original "dd" plus newly typed d's)
3033
+ const expected_d_count: u32 = @as(u32, 2) + @as(u32, @intCast(j + 1)); // dd + newly typed d's
3034
+ try std.testing.expectEqual(expected_d_count, vlines[1].width_cols);
3035
+ }
3036
+ }
3037
+
3038
+ // After adding 7 more d's, we have "Hello world ddddddddd" (21 chars) again
3039
+ // Should wrap after "world " into 2 lines
3040
+ vline_count = ev.getTotalVirtualLineCount();
3041
+ try std.testing.expectEqual(@as(u32, 2), vline_count);
3042
+
3043
+ cursor = ev.getPrimaryCursor();
3044
+ try std.testing.expectEqual(@as(u32, 0), cursor.row);
3045
+
3046
+ // Verify the text is correct
3047
+ var out_buffer: [100]u8 = undefined;
3048
+ const written = eb.getText(&out_buffer);
3049
+ try std.testing.expectEqualStrings("Hello world ddddddddd", out_buffer[0..written]);
3050
+ }
3051
+
3052
+ test "EditorView - cursor movement with emoji skin tone modifier wcwidth" {
3053
+ const pool = gp.initGlobalPool(std.testing.allocator);
3054
+ defer gp.deinitGlobalPool();
3055
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3056
+ defer link.deinitGlobalLinkPool();
3057
+
3058
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
3059
+ defer eb.deinit();
3060
+
3061
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
3062
+ defer ev.deinit();
3063
+
3064
+ // "👋🏿" is a waving hand emoji with dark skin tone modifier
3065
+ // In wcwidth mode (tmux-style), each codepoint has width 2, total = 4 columns
3066
+ // IMPORTANT: In wcwidth mode, each codepoint is treated as a separate char for cursor movement
3067
+ try eb.setText("👋🏿");
3068
+
3069
+ // Start at position 0 (before the first codepoint)
3070
+ try eb.setCursor(0, 0);
3071
+ var cursor = eb.getPrimaryCursor();
3072
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3073
+
3074
+ // Move right once - should move past the FIRST codepoint (2 columns)
3075
+ // In wcwidth mode, each codepoint is a separate char, so this moves to col 2
3076
+ eb.moveRight();
3077
+ cursor = eb.getPrimaryCursor();
3078
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3079
+
3080
+ // Move right again - should move past the SECOND codepoint (2 more columns)
3081
+ eb.moveRight();
3082
+ cursor = eb.getPrimaryCursor();
3083
+ try std.testing.expectEqual(@as(u32, 4), cursor.col);
3084
+
3085
+ // Move left once - should move back to col 2 (before second codepoint)
3086
+ eb.moveLeft();
3087
+ cursor = eb.getPrimaryCursor();
3088
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3089
+
3090
+ // Move left again - should move back to the beginning
3091
+ eb.moveLeft();
3092
+ cursor = eb.getPrimaryCursor();
3093
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3094
+ }
3095
+
3096
+ test "EditorView - cursor movement with emoji skin tone modifier unicode" {
3097
+ const pool = gp.initGlobalPool(std.testing.allocator);
3098
+ defer gp.deinitGlobalPool();
3099
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3100
+ defer link.deinitGlobalLinkPool();
3101
+
3102
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .unicode);
3103
+ defer eb.deinit();
3104
+
3105
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
3106
+ defer ev.deinit();
3107
+
3108
+ // "👋🏿" is a waving hand emoji with dark skin tone modifier
3109
+ // In unicode mode (modern terminals), skin tone is 0-width, total = 2 columns
3110
+ try eb.setText("👋🏿");
3111
+
3112
+ // Start at position 0 (before the grapheme cluster)
3113
+ try eb.setCursor(0, 0);
3114
+ var cursor = eb.getPrimaryCursor();
3115
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3116
+
3117
+ // Move right once - should move past the entire grapheme cluster (2 columns in unicode)
3118
+ eb.moveRight();
3119
+ cursor = eb.getPrimaryCursor();
3120
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3121
+
3122
+ // Move left once - should move back to the beginning
3123
+ eb.moveLeft();
3124
+ cursor = eb.getPrimaryCursor();
3125
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3126
+ }
3127
+
3128
+ test "EditorView - backspace emoji with skin tone modifier wcwidth" {
3129
+ const pool = gp.initGlobalPool(std.testing.allocator);
3130
+ defer gp.deinitGlobalPool();
3131
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3132
+ defer link.deinitGlobalLinkPool();
3133
+
3134
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
3135
+ defer eb.deinit();
3136
+
3137
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
3138
+ defer ev.deinit();
3139
+
3140
+ // "👋🏿" is a waving hand emoji with dark skin tone modifier
3141
+ // In wcwidth mode, this renders as 4 columns (2+2)
3142
+ // In wcwidth mode, each codepoint is treated as a separate char
3143
+ try eb.setText("👋🏿");
3144
+
3145
+ // Move cursor to col 2 (after first codepoint)
3146
+ eb.moveRight();
3147
+ var cursor = eb.getPrimaryCursor();
3148
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3149
+
3150
+ // Move cursor to col 4 (after second codepoint)
3151
+ eb.moveRight();
3152
+ cursor = eb.getPrimaryCursor();
3153
+ try std.testing.expectEqual(@as(u32, 4), cursor.col);
3154
+
3155
+ // Get text before backspace to verify it contains the emoji
3156
+ var buffer_before: [100]u8 = undefined;
3157
+ const len_before = eb.getText(&buffer_before);
3158
+ try std.testing.expectEqualStrings("👋🏿", buffer_before[0..len_before]);
3159
+
3160
+ // First backspace should delete just the skin tone modifier (second codepoint)
3161
+ try eb.backspace();
3162
+
3163
+ // Cursor should now be at position 2 (after the first codepoint)
3164
+ cursor = eb.getPrimaryCursor();
3165
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3166
+
3167
+ // Text buffer should contain just the hand emoji without skin tone
3168
+ var buffer_middle: [100]u8 = undefined;
3169
+ const len_middle = eb.getText(&buffer_middle);
3170
+ try std.testing.expectEqualStrings("👋", buffer_middle[0..len_middle]);
3171
+
3172
+ // Second backspace should delete the hand emoji (first codepoint)
3173
+ try eb.backspace();
3174
+
3175
+ // Cursor should now be at position 0
3176
+ cursor = eb.getPrimaryCursor();
3177
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3178
+
3179
+ // Text buffer should be empty
3180
+ var buffer_after: [100]u8 = undefined;
3181
+ const len_after = eb.getText(&buffer_after);
3182
+ try std.testing.expectEqual(@as(usize, 0), len_after);
3183
+ try std.testing.expectEqualStrings("", buffer_after[0..len_after]);
3184
+ }
3185
+
3186
+ test "EditorView - backspace emoji with skin tone modifier unicode" {
3187
+ const pool = gp.initGlobalPool(std.testing.allocator);
3188
+ defer gp.deinitGlobalPool();
3189
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3190
+ defer link.deinitGlobalLinkPool();
3191
+
3192
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .unicode);
3193
+ defer eb.deinit();
3194
+
3195
+ var ev = try EditorView.init(std.testing.allocator, eb, 80, 24);
3196
+ defer ev.deinit();
3197
+
3198
+ // "👋🏿" is a waving hand emoji with dark skin tone modifier
3199
+ // In unicode mode, this renders as 2 columns (modifier is 0-width)
3200
+ try eb.setText("👋🏿");
3201
+
3202
+ // Move cursor to AFTER the grapheme cluster (2 columns total in unicode mode)
3203
+ eb.moveRight();
3204
+ var cursor = eb.getPrimaryCursor();
3205
+ try std.testing.expectEqual(@as(u32, 2), cursor.col);
3206
+
3207
+ // Get text before backspace to verify it contains the emoji
3208
+ var buffer_before: [100]u8 = undefined;
3209
+ const len_before = eb.getText(&buffer_before);
3210
+ try std.testing.expectEqualStrings("👋🏿", buffer_before[0..len_before]);
3211
+
3212
+ // Backspace should delete the entire grapheme cluster (both codepoints)
3213
+ try eb.backspace();
3214
+
3215
+ // Cursor should now be at position 0
3216
+ cursor = eb.getPrimaryCursor();
3217
+ try std.testing.expectEqual(@as(u32, 0), cursor.col);
3218
+
3219
+ // Text buffer should be empty
3220
+ var buffer_after: [100]u8 = undefined;
3221
+ const len_after = eb.getText(&buffer_after);
3222
+ try std.testing.expectEqual(@as(usize, 0), len_after);
3223
+ try std.testing.expectEqualStrings("", buffer_after[0..len_after]);
3224
+ }
3225
+
3226
+ test "EditorView - mouse selection doesn't scroll when focus is within viewport" {
3227
+ const pool = gp.initGlobalPool(std.testing.allocator);
3228
+ defer gp.deinitGlobalPool();
3229
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3230
+ defer link.deinitGlobalLinkPool();
3231
+
3232
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
3233
+ defer eb.deinit();
3234
+
3235
+ var ev = try EditorView.init(std.testing.allocator, eb, 40, 10);
3236
+ defer ev.deinit();
3237
+
3238
+ // Create 50 lines of text
3239
+ var i: u32 = 0;
3240
+ while (i < 50) : (i += 1) {
3241
+ if (i > 0) try eb.insertText("\n");
3242
+ try eb.insertText("Line ");
3243
+ var num_buf: [10]u8 = undefined;
3244
+ const num_str = try std.fmt.bufPrint(&num_buf, "{d}", .{i});
3245
+ try eb.insertText(num_str);
3246
+ }
3247
+
3248
+ // Reset cursor to top
3249
+ try eb.setCursor(0, 0);
3250
+ _ = ev.getVirtualLines();
3251
+
3252
+ const vp_initial = ev.getViewport().?;
3253
+ try std.testing.expectEqual(@as(u32, 0), vp_initial.y);
3254
+
3255
+ // Simulate selection within the viewport (lines 0-5, all visible)
3256
+ _ = ev.setLocalSelection(0, 0, 5, 5, null, null, true);
3257
+ _ = ev.getVirtualLines();
3258
+
3259
+ const vp_after = ev.getViewport().?;
3260
+
3261
+ // Viewport should not have changed
3262
+ try std.testing.expectEqual(vp_initial.y, vp_after.y);
3263
+ try std.testing.expectEqual(vp_initial.x, vp_after.x);
3264
+ }
3265
+
3266
+ test "EditorView - mouse selection focus outside buffer bounds clamps correctly" {
3267
+ const pool = gp.initGlobalPool(std.testing.allocator);
3268
+ defer gp.deinitGlobalPool();
3269
+ const link_pool = link.initGlobalLinkPool(std.testing.allocator);
3270
+ defer link.deinitGlobalLinkPool();
3271
+
3272
+ var eb = try EditBuffer.init(std.testing.allocator, pool, link_pool, .wcwidth);
3273
+ defer eb.deinit();
3274
+
3275
+ var ev = try EditorView.init(std.testing.allocator, eb, 40, 10);
3276
+ defer ev.deinit();
3277
+
3278
+ // Create just 10 lines
3279
+ var i: u32 = 0;
3280
+ while (i < 10) : (i += 1) {
3281
+ if (i > 0) try eb.insertText("\n");
3282
+ try eb.insertText("Line ");
3283
+ var num_buf: [10]u8 = undefined;
3284
+ const num_str = try std.fmt.bufPrint(&num_buf, "{d}", .{i});
3285
+ try eb.insertText(num_str);
3286
+ }
3287
+
3288
+ try eb.setCursor(0, 0);
3289
+ _ = ev.getVirtualLines();
3290
+
3291
+ // Try to select way beyond buffer (to line 100)
3292
+ _ = ev.setLocalSelection(0, 0, 5, 100, null, null, true);
3293
+ _ = ev.getVirtualLines();
3294
+
3295
+ const cursor = ev.getPrimaryCursor();
3296
+
3297
+ // Cursor should be clamped to last line (line 9)
3298
+ try std.testing.expectEqual(@as(u32, 9), cursor.row);
3299
+ }