@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,1032 @@
1
+ import { describe, expect, it, beforeEach, afterEach } from "bun:test"
2
+ import { EditBuffer } from "./edit-buffer.js"
3
+ import { EditorView } from "./editor-view.js"
4
+ import { RGBA } from "./lib/RGBA.js"
5
+
6
+ describe("EditorView", () => {
7
+ let buffer: EditBuffer
8
+ let view: EditorView
9
+
10
+ beforeEach(() => {
11
+ buffer = EditBuffer.create("wcwidth")
12
+ view = EditorView.create(buffer, 40, 10)
13
+ })
14
+
15
+ afterEach(() => {
16
+ view.destroy()
17
+ buffer.destroy()
18
+ })
19
+
20
+ describe("initialization", () => {
21
+ it("should create view with specified viewport dimensions", () => {
22
+ const viewport = view.getViewport()
23
+ expect(viewport.width).toBe(40)
24
+ expect(viewport.height).toBe(10)
25
+ expect(viewport.offsetY).toBe(0)
26
+ expect(viewport.offsetX).toBe(0)
27
+ })
28
+
29
+ it("should start with wrap mode set to none", () => {
30
+ expect(view.getVirtualLineCount()).toBeGreaterThanOrEqual(0)
31
+ })
32
+ })
33
+
34
+ describe("viewport management", () => {
35
+ it("should update viewport size", () => {
36
+ view.setViewportSize(80, 20)
37
+ const viewport = view.getViewport()
38
+ expect(viewport.width).toBe(80)
39
+ expect(viewport.height).toBe(20)
40
+ })
41
+
42
+ it("should set scroll margin", () => {
43
+ view.setScrollMargin(0.2)
44
+ expect(true).toBe(true)
45
+ })
46
+
47
+ it("should return correct virtual line count for simple text", () => {
48
+ buffer.setText("Line 1\nLine 2\nLine 3")
49
+ expect(view.getVirtualLineCount()).toBe(3)
50
+ })
51
+ })
52
+
53
+ describe("text wrapping", () => {
54
+ it("should enable and disable wrapping via wrap mode", () => {
55
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789ABCDEFGHIJKLMNOPQRST")
56
+
57
+ expect(view.getVirtualLineCount()).toBe(1)
58
+
59
+ view.setWrapMode("char")
60
+ expect(view.getVirtualLineCount()).toBeGreaterThan(1)
61
+
62
+ view.setWrapMode("none")
63
+ expect(view.getVirtualLineCount()).toBe(1)
64
+ })
65
+
66
+ it("should wrap at viewport width", () => {
67
+ buffer.setText("ABCDEFGHIJKLMNOPQRST")
68
+
69
+ view.setWrapMode("char")
70
+ view.setViewportSize(10, 10)
71
+
72
+ expect(view.getVirtualLineCount()).toBe(2)
73
+
74
+ view.setViewportSize(5, 10)
75
+ expect(view.getVirtualLineCount()).toBe(4)
76
+
77
+ view.setViewportSize(20, 10)
78
+ expect(view.getVirtualLineCount()).toBe(1)
79
+ })
80
+
81
+ it("should change wrap mode", () => {
82
+ buffer.setText("Hello wonderful world")
83
+
84
+ view.setViewportSize(10, 10)
85
+
86
+ view.setWrapMode("char")
87
+ const charCount = view.getVirtualLineCount()
88
+ expect(charCount).toBeGreaterThanOrEqual(2)
89
+
90
+ view.setWrapMode("word")
91
+ const wordCount = view.getVirtualLineCount()
92
+ expect(wordCount).toBeGreaterThanOrEqual(2)
93
+
94
+ view.setWrapMode("none")
95
+ const noneCount = view.getVirtualLineCount()
96
+ expect(noneCount).toBe(1)
97
+ })
98
+
99
+ it("should preserve newlines when wrapping", () => {
100
+ buffer.setText("Short\nAnother short line\nLast")
101
+
102
+ view.setWrapMode("char")
103
+ view.setViewportSize(50, 10)
104
+
105
+ expect(view.getVirtualLineCount()).toBe(3)
106
+ })
107
+
108
+ it("should wrap long lines with wrapping enabled", () => {
109
+ const longLine = "This is a very long line that will definitely wrap when the viewport is narrow"
110
+ buffer.setText(longLine)
111
+
112
+ view.setWrapMode("char")
113
+ view.setViewportSize(20, 10)
114
+
115
+ const vlineCount = view.getVirtualLineCount()
116
+ expect(vlineCount).toBeGreaterThan(1)
117
+ })
118
+ })
119
+
120
+ describe("integration with EditBuffer", () => {
121
+ it("should reflect edits made to EditBuffer", () => {
122
+ buffer.setText("Line 1\nLine 2\nLine 3")
123
+ expect(view.getVirtualLineCount()).toBe(3)
124
+
125
+ buffer.gotoLine(9999)
126
+ buffer.newLine()
127
+ buffer.insertText("Line 4")
128
+
129
+ expect(view.getVirtualLineCount()).toBe(4)
130
+ })
131
+
132
+ it("should update after text deletion", () => {
133
+ buffer.setText("Line 1\nLine 2\nLine 3")
134
+ expect(view.getVirtualLineCount()).toBe(3)
135
+
136
+ buffer.gotoLine(1)
137
+ buffer.deleteLine()
138
+
139
+ expect(view.getVirtualLineCount()).toBe(2)
140
+ })
141
+ })
142
+
143
+ describe("viewport with wrapping and editing", () => {
144
+ it("should maintain wrapping after edits", () => {
145
+ buffer.setText("Short line")
146
+
147
+ view.setWrapMode("char")
148
+ view.setViewportSize(20, 10)
149
+
150
+ expect(view.getVirtualLineCount()).toBe(1)
151
+
152
+ buffer.gotoLine(9999)
153
+ buffer.insertText(" that becomes very long and should wrap now")
154
+
155
+ expect(view.getVirtualLineCount()).toBeGreaterThan(1)
156
+ })
157
+
158
+ it("should handle viewport resize with wrapped content", () => {
159
+ const longText = "This is a very long line that will wrap when the viewport is narrow"
160
+ buffer.setText(longText)
161
+
162
+ view.setWrapMode("char")
163
+ view.setViewportSize(20, 10)
164
+
165
+ const count20 = view.getVirtualLineCount()
166
+ expect(count20).toBeGreaterThan(1)
167
+
168
+ view.setViewportSize(40, 10)
169
+ const count40 = view.getVirtualLineCount()
170
+ expect(count40).toBeLessThan(count20)
171
+ })
172
+ })
173
+
174
+ describe("selection", () => {
175
+ it("should set and reset selection", () => {
176
+ buffer.setText("Hello World")
177
+
178
+ view.setSelection(0, 5)
179
+ expect(view.hasSelection()).toBe(true)
180
+
181
+ view.resetSelection()
182
+ expect(view.hasSelection()).toBe(false)
183
+ })
184
+
185
+ it("should set selection with colors", () => {
186
+ buffer.setText("Hello World")
187
+
188
+ const bgColor = RGBA.fromValues(0, 0, 1, 0.3)
189
+ const fgColor = RGBA.fromValues(1, 1, 1, 1)
190
+
191
+ view.setSelection(0, 5, bgColor, fgColor)
192
+ expect(view.hasSelection()).toBe(true)
193
+
194
+ const selection = view.getSelection()
195
+ expect(selection).toEqual({ start: 0, end: 5 })
196
+ })
197
+
198
+ it("should update selection end position", () => {
199
+ buffer.setText("Hello World")
200
+
201
+ view.setSelection(0, 5)
202
+ expect(view.getSelectedText()).toBe("Hello")
203
+
204
+ view.updateSelection(11)
205
+ expect(view.getSelectedText()).toBe("Hello World")
206
+
207
+ const selection = view.getSelection()
208
+ expect(selection).toEqual({ start: 0, end: 11 })
209
+ })
210
+
211
+ it("should shrink selection with updateSelection", () => {
212
+ buffer.setText("Hello World")
213
+
214
+ view.setSelection(0, 11)
215
+ expect(view.getSelectedText()).toBe("Hello World")
216
+
217
+ view.updateSelection(5)
218
+ expect(view.getSelectedText()).toBe("Hello")
219
+ })
220
+
221
+ it("should update local selection focus position", () => {
222
+ buffer.setText("Hello World")
223
+
224
+ const changed1 = view.setLocalSelection(0, 0, 5, 0)
225
+ expect(changed1).toBe(true)
226
+ expect(view.getSelectedText()).toBe("Hello")
227
+
228
+ const changed2 = view.updateLocalSelection(0, 0, 11, 0)
229
+ expect(changed2).toBe(true)
230
+ expect(view.getSelectedText()).toBe("Hello World")
231
+ })
232
+
233
+ it("should update local selection across lines", () => {
234
+ buffer.setText("Line 1\nLine 2\nLine 3")
235
+
236
+ view.setLocalSelection(2, 0, 2, 0)
237
+
238
+ const changed = view.updateLocalSelection(2, 0, 4, 1)
239
+ expect(changed).toBe(true)
240
+
241
+ const selectedText = view.getSelectedText()
242
+ expect(selectedText).toContain("ne 1")
243
+ expect(selectedText).toContain("Line")
244
+ })
245
+
246
+ it("should fallback to setLocalSelection when updateLocalSelection called with no existing anchor", () => {
247
+ buffer.setText("Hello World")
248
+
249
+ const changed = view.updateLocalSelection(0, 0, 5, 0)
250
+ expect(changed).toBe(true)
251
+ expect(view.hasSelection()).toBe(true)
252
+ expect(view.getSelectedText()).toBe("Hello")
253
+ })
254
+
255
+ it("should preserve anchor when updating local selection", () => {
256
+ buffer.setText("Hello World")
257
+
258
+ view.setLocalSelection(0, 0, 5, 0)
259
+ expect(view.getSelectedText()).toBe("Hello")
260
+
261
+ view.updateLocalSelection(0, 0, 11, 0)
262
+ expect(view.getSelectedText()).toBe("Hello World")
263
+
264
+ view.updateLocalSelection(0, 0, 3, 0)
265
+ expect(view.getSelectedText()).toBe("Hel")
266
+ })
267
+
268
+ it("should handle backward selection with updateLocalSelection", () => {
269
+ buffer.setText("Hello World")
270
+
271
+ view.setLocalSelection(11, 0, 11, 0)
272
+
273
+ const changed = view.updateLocalSelection(11, 0, 6, 0)
274
+ expect(changed).toBe(true)
275
+ expect(view.getSelectedText()).toBe("World")
276
+ })
277
+
278
+ it("should handle wrapped lines with updateLocalSelection", () => {
279
+ buffer.setText("ABCDEFGHIJKLMNOPQRST")
280
+
281
+ view.setWrapMode("char")
282
+ view.setViewportSize(10, 10)
283
+
284
+ view.setLocalSelection(0, 0, 0, 0)
285
+
286
+ const changed = view.updateLocalSelection(0, 0, 5, 1)
287
+ expect(changed).toBe(true)
288
+ expect(view.getSelectedText()).toBe("ABCDEFGHIJKLMNO")
289
+ })
290
+ })
291
+
292
+ describe("word boundary navigation", () => {
293
+ it("should get next word boundary with visual cursor", () => {
294
+ buffer.setText("hello world foo")
295
+ buffer.setCursorToLineCol(0, 0)
296
+
297
+ const nextBoundary = view.getNextWordBoundary()
298
+ expect(nextBoundary).toBeDefined()
299
+ expect(nextBoundary.visualCol).toBeGreaterThan(0)
300
+ })
301
+
302
+ it("should get previous word boundary with visual cursor", () => {
303
+ buffer.setText("hello world foo")
304
+ buffer.setCursorToLineCol(0, 15)
305
+
306
+ const prevBoundary = view.getPrevWordBoundary()
307
+ expect(prevBoundary).toBeDefined()
308
+ expect(prevBoundary.visualCol).toBeLessThan(15)
309
+ })
310
+
311
+ it("should handle word boundary at start", () => {
312
+ buffer.setText("hello world")
313
+ buffer.setCursorToLineCol(0, 0)
314
+
315
+ const prevBoundary = view.getPrevWordBoundary()
316
+ expect(prevBoundary.logicalRow).toBe(0)
317
+ expect(prevBoundary.visualCol).toBe(0)
318
+ })
319
+
320
+ it("should handle word boundary at end", () => {
321
+ buffer.setText("hello world")
322
+ buffer.setCursorToLineCol(0, 11)
323
+
324
+ const nextBoundary = view.getNextWordBoundary()
325
+ expect(nextBoundary.visualCol).toBe(11)
326
+ })
327
+
328
+ it("should navigate across lines with visual coordinates", () => {
329
+ buffer.setText("hello\nworld")
330
+ buffer.setCursorToLineCol(0, 5)
331
+
332
+ const nextBoundary = view.getNextWordBoundary()
333
+ expect(nextBoundary.logicalRow).toBeGreaterThanOrEqual(0)
334
+ })
335
+
336
+ it("should handle wrapping when getting word boundaries", () => {
337
+ buffer.setText("hello world test foo bar")
338
+ view.setWrapMode("word")
339
+ view.setViewportSize(10, 10)
340
+
341
+ buffer.setCursorToLineCol(0, 0)
342
+ const nextBoundary = view.getNextWordBoundary()
343
+
344
+ expect(nextBoundary).toBeDefined()
345
+ expect(nextBoundary.visualRow).toBeGreaterThanOrEqual(0)
346
+ expect(nextBoundary.logicalRow).toBeGreaterThanOrEqual(0)
347
+ })
348
+ })
349
+
350
+ describe("large content", () => {
351
+ it("should handle many lines", () => {
352
+ const lines = Array.from({ length: 100 }, (_, i) => `Line ${i}`).join("\n")
353
+ buffer.setText(lines)
354
+
355
+ expect(view.getTotalVirtualLineCount()).toBe(100)
356
+ })
357
+
358
+ it("should handle very long single line with wrapping", () => {
359
+ const longLine = "A".repeat(1000)
360
+ buffer.setText(longLine)
361
+
362
+ view.setWrapMode("char")
363
+ view.setViewportSize(80, 24)
364
+
365
+ const vlineCount = view.getVirtualLineCount()
366
+ expect(vlineCount).toBeGreaterThan(10)
367
+ })
368
+ })
369
+
370
+ describe("viewport slicing", () => {
371
+ it("should show subset of content in viewport", () => {
372
+ const lines = Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n")
373
+ buffer.setText(lines)
374
+
375
+ const smallView = EditorView.create(buffer, 40, 5)
376
+
377
+ expect(smallView.getTotalVirtualLineCount()).toBe(20)
378
+
379
+ smallView.destroy()
380
+ })
381
+ })
382
+
383
+ describe("error handling", () => {
384
+ it("should throw error when using destroyed view", () => {
385
+ view.destroy()
386
+
387
+ expect(() => view.getVirtualLineCount()).toThrow("EditorView is destroyed")
388
+ expect(() => view.setViewportSize(80, 24)).toThrow("EditorView is destroyed")
389
+ expect(() => view.setWrapMode("char")).toThrow("EditorView is destroyed")
390
+ })
391
+ })
392
+
393
+ describe("Unicode edge cases", () => {
394
+ it("should handle emoji with wrapping", () => {
395
+ buffer.setText("🌟".repeat(20))
396
+
397
+ view.setWrapMode("char")
398
+ view.setViewportSize(10, 10)
399
+
400
+ expect(view.getVirtualLineCount()).toBeGreaterThan(1)
401
+ })
402
+
403
+ it("should handle CJK characters with wrapping", () => {
404
+ buffer.setText("测试文字处理功能")
405
+
406
+ view.setWrapMode("char")
407
+ view.setViewportSize(10, 10)
408
+
409
+ const vlineCount = view.getVirtualLineCount()
410
+ expect(vlineCount).toBeGreaterThanOrEqual(1)
411
+ })
412
+
413
+ it("should handle mixed ASCII and wide characters", () => {
414
+ buffer.setText("AB测试CD文字EF")
415
+
416
+ view.setWrapMode("char")
417
+ view.setViewportSize(8, 10)
418
+
419
+ expect(view.getVirtualLineCount()).toBeGreaterThanOrEqual(1)
420
+ })
421
+
422
+ it("should navigate visual cursor correctly through emoji and CJK", () => {
423
+ buffer.setText("(emoji 🌟 and CJK 世界)")
424
+
425
+ let cursor = view.getVisualCursor()
426
+ expect(cursor.visualRow).toBe(0)
427
+ expect(cursor.visualCol).toBe(0)
428
+ expect(cursor.offset).toBe(0)
429
+
430
+ for (let i = 0; i < 6; i++) {
431
+ buffer.moveCursorRight()
432
+ }
433
+ cursor = view.getVisualCursor()
434
+ expect(cursor.offset).toBe(6)
435
+
436
+ buffer.moveCursorRight()
437
+ cursor = view.getVisualCursor()
438
+ expect(cursor.offset).toBe(7)
439
+
440
+ buffer.moveCursorRight()
441
+ cursor = view.getVisualCursor()
442
+ expect(cursor.offset).toBe(9)
443
+
444
+ buffer.moveCursorLeft()
445
+ cursor = view.getVisualCursor()
446
+ expect(cursor.offset).toBe(7)
447
+
448
+ buffer.moveCursorLeft()
449
+ cursor = view.getVisualCursor()
450
+ expect(cursor.offset).toBe(6)
451
+ })
452
+
453
+ it("should handle vertical navigation through emoji cells correctly", () => {
454
+ buffer.setText("1234567890123456789\n(emoji 🌟 and CJK 世界)\n1234567890123456789")
455
+
456
+ buffer.setCursorToLineCol(0, 7)
457
+ let cursor = view.getVisualCursor()
458
+ expect(cursor.visualRow).toBe(0)
459
+ expect(cursor.visualCol).toBe(7)
460
+
461
+ view.moveDownVisual()
462
+ cursor = view.getVisualCursor()
463
+ expect(cursor.visualRow).toBe(1)
464
+ expect(cursor.visualCol).toBe(7)
465
+
466
+ buffer.moveCursorRight()
467
+ cursor = view.getVisualCursor()
468
+ expect(cursor.visualCol).toBe(9)
469
+
470
+ view.moveUpVisual()
471
+ cursor = view.getVisualCursor()
472
+ expect(cursor.visualRow).toBe(0)
473
+ expect(cursor.visualCol).toBe(9)
474
+
475
+ buffer.moveCursorLeft()
476
+ cursor = view.getVisualCursor()
477
+ expect(cursor.visualCol).toBe(8)
478
+
479
+ view.moveDownVisual()
480
+ cursor = view.getVisualCursor()
481
+ expect(cursor.visualRow).toBe(1)
482
+ expect(cursor.visualCol).toBe(8)
483
+
484
+ buffer.moveCursorLeft()
485
+ cursor = view.getVisualCursor()
486
+ expect(cursor.visualCol).toBe(6)
487
+ })
488
+ })
489
+
490
+ describe("cursor movement around multi-cell graphemes", () => {
491
+ // These tests verify that the cursor correctly handles multi-cell graphemes like emojis (🌟)
492
+ // and CJK characters (世界). Multi-cell graphemes occupy 2 visual columns but are treated
493
+ // as a single logical unit for cursor movement and deletion.
494
+ //
495
+ // Key behaviors:
496
+ // - moveCursorRight/Left skips over entire graphemes (no intermediate positions)
497
+ // - deleteCharBackward deletes the entire grapheme, not individual cells
498
+ // - Visual column positions reflect the actual display width (2 cells per wide grapheme)
499
+ // - Logical column positions mark grapheme boundaries (skipping intermediate cell positions)
500
+
501
+ it("should understand logical vs visual cursor positions", () => {
502
+ buffer.setText("a🌟b")
503
+
504
+ buffer.setCursorToLineCol(0, 0)
505
+ expect(view.getVisualCursor().visualCol).toBe(0)
506
+
507
+ buffer.setCursorToLineCol(0, 1)
508
+ expect(view.getVisualCursor().visualCol).toBe(1)
509
+
510
+ buffer.setCursorToLineCol(0, 3)
511
+ expect(view.getVisualCursor().visualCol).toBe(3)
512
+
513
+ buffer.setCursorToLineCol(0, 4)
514
+ expect(view.getVisualCursor().visualCol).toBe(4)
515
+
516
+ buffer.setCursorToLineCol(0, 0)
517
+ buffer.moveCursorRight()
518
+ expect(buffer.getCursorPosition().col).toBe(1)
519
+
520
+ buffer.moveCursorRight()
521
+ expect(buffer.getCursorPosition().col).toBe(3)
522
+ expect(view.getVisualCursor().visualCol).toBe(3)
523
+
524
+ buffer.moveCursorRight()
525
+ expect(buffer.getCursorPosition().col).toBe(4)
526
+ })
527
+
528
+ it("should move cursor correctly around emoji (🌟) with visual positions", () => {
529
+ buffer.setText("a🌟b")
530
+
531
+ buffer.setCursorToLineCol(0, 1)
532
+ let visualCursor = view.getVisualCursor()
533
+ expect(visualCursor.visualCol).toBe(1)
534
+
535
+ buffer.moveCursorRight()
536
+ visualCursor = view.getVisualCursor()
537
+ expect(visualCursor.visualCol).toBe(3)
538
+
539
+ buffer.moveCursorRight()
540
+ visualCursor = view.getVisualCursor()
541
+ expect(visualCursor.visualCol).toBe(4)
542
+
543
+ buffer.moveCursorLeft()
544
+ visualCursor = view.getVisualCursor()
545
+ expect(visualCursor.visualCol).toBe(3)
546
+
547
+ buffer.moveCursorLeft()
548
+ visualCursor = view.getVisualCursor()
549
+ expect(visualCursor.visualCol).toBe(1)
550
+ })
551
+
552
+ it("should move cursor correctly around CJK characters (世界) with visual positions", () => {
553
+ buffer.setText("a世界b")
554
+
555
+ buffer.setCursorToLineCol(0, 0)
556
+ expect(view.getVisualCursor().visualCol).toBe(0)
557
+
558
+ buffer.moveCursorRight()
559
+ expect(view.getVisualCursor().visualCol).toBe(1)
560
+
561
+ buffer.moveCursorRight()
562
+ expect(view.getVisualCursor().visualCol).toBe(3)
563
+
564
+ buffer.moveCursorRight()
565
+ expect(view.getVisualCursor().visualCol).toBe(5)
566
+
567
+ buffer.moveCursorRight()
568
+ expect(view.getVisualCursor().visualCol).toBe(6)
569
+
570
+ buffer.moveCursorLeft()
571
+ expect(view.getVisualCursor().visualCol).toBe(5)
572
+
573
+ buffer.moveCursorLeft()
574
+ expect(view.getVisualCursor().visualCol).toBe(3)
575
+
576
+ buffer.moveCursorLeft()
577
+ expect(view.getVisualCursor().visualCol).toBe(1)
578
+ })
579
+
580
+ it("should handle backspace correctly after emoji", () => {
581
+ buffer.setText("a🌟b")
582
+
583
+ buffer.setCursorToLineCol(0, 3)
584
+ expect(view.getVisualCursor().visualCol).toBe(3)
585
+
586
+ buffer.deleteCharBackward()
587
+ expect(buffer.getText()).toBe("ab")
588
+ expect(view.getVisualCursor().visualCol).toBe(1)
589
+ })
590
+
591
+ it("should handle backspace correctly after CJK character", () => {
592
+ buffer.setText("世界")
593
+
594
+ buffer.setCursorToLineCol(0, 4)
595
+ expect(view.getVisualCursor().visualCol).toBe(4)
596
+
597
+ buffer.deleteCharBackward()
598
+ expect(buffer.getText()).toBe("世")
599
+ expect(view.getVisualCursor().visualCol).toBe(2)
600
+
601
+ buffer.deleteCharBackward()
602
+ expect(buffer.getText()).toBe("")
603
+ expect(view.getVisualCursor().visualCol).toBe(0)
604
+ })
605
+
606
+ it("should treat multi-cell graphemes as single units for cursor movement", () => {
607
+ buffer.setText("🌟世界🎉")
608
+
609
+ buffer.setCursorToLineCol(0, 0)
610
+ expect(view.getVisualCursor().visualCol).toBe(0)
611
+
612
+ buffer.moveCursorRight()
613
+ expect(view.getVisualCursor().visualCol).toBe(2)
614
+
615
+ buffer.moveCursorRight()
616
+ expect(view.getVisualCursor().visualCol).toBe(4)
617
+
618
+ buffer.moveCursorRight()
619
+ expect(view.getVisualCursor().visualCol).toBe(6)
620
+
621
+ buffer.moveCursorRight()
622
+ expect(view.getVisualCursor().visualCol).toBe(8)
623
+
624
+ buffer.moveCursorLeft()
625
+ expect(view.getVisualCursor().visualCol).toBe(6)
626
+
627
+ buffer.moveCursorLeft()
628
+ expect(view.getVisualCursor().visualCol).toBe(4)
629
+
630
+ buffer.moveCursorLeft()
631
+ expect(view.getVisualCursor().visualCol).toBe(2)
632
+
633
+ buffer.moveCursorLeft()
634
+ expect(view.getVisualCursor().visualCol).toBe(0)
635
+ })
636
+
637
+ it("should handle backspace through mixed multi-cell graphemes", () => {
638
+ buffer.setText("a🌟b世c")
639
+
640
+ buffer.setCursorToLineCol(0, 7)
641
+ expect(view.getVisualCursor().visualCol).toBe(7)
642
+
643
+ buffer.deleteCharBackward()
644
+ expect(buffer.getText()).toBe("a🌟b世")
645
+ expect(view.getVisualCursor().visualCol).toBe(6)
646
+
647
+ buffer.deleteCharBackward()
648
+ expect(buffer.getText()).toBe("a🌟b")
649
+ expect(view.getVisualCursor().visualCol).toBe(4)
650
+
651
+ buffer.deleteCharBackward()
652
+ expect(buffer.getText()).toBe("a🌟")
653
+ expect(view.getVisualCursor().visualCol).toBe(3)
654
+
655
+ buffer.deleteCharBackward()
656
+ expect(buffer.getText()).toBe("a")
657
+ expect(view.getVisualCursor().visualCol).toBe(1)
658
+
659
+ buffer.deleteCharBackward()
660
+ expect(buffer.getText()).toBe("")
661
+ expect(view.getVisualCursor().visualCol).toBe(0)
662
+ })
663
+
664
+ it("should handle delete key correctly before multi-cell graphemes", () => {
665
+ buffer.setText("a🌟b")
666
+
667
+ buffer.setCursorToLineCol(0, 1)
668
+ expect(view.getVisualCursor().visualCol).toBe(1)
669
+
670
+ buffer.deleteChar()
671
+ expect(buffer.getText()).toBe("ab")
672
+ expect(view.getVisualCursor().visualCol).toBe(1)
673
+
674
+ buffer.setCursorToLineCol(0, 0)
675
+
676
+ buffer.deleteChar()
677
+ expect(buffer.getText()).toBe("b")
678
+ expect(view.getVisualCursor().visualCol).toBe(0)
679
+ })
680
+
681
+ it("should handle line start and end with multi-cell graphemes", () => {
682
+ buffer.setText("🌟世界🎉")
683
+
684
+ buffer.setCursorToLineCol(0, 0)
685
+ expect(view.getVisualCursor().visualCol).toBe(0)
686
+
687
+ const eol = view.getEOL()
688
+ buffer.setCursorToLineCol(eol.logicalRow, eol.logicalCol)
689
+ expect(view.getVisualCursor().visualCol).toBe(8)
690
+ })
691
+ })
692
+
693
+ describe("visual line navigation (SOL/EOL)", () => {
694
+ describe("without wrapping", () => {
695
+ it("should get visual SOL on single line", () => {
696
+ buffer.setText("Hello World")
697
+ buffer.setCursorToLineCol(0, 6) // Middle of line
698
+
699
+ const sol = view.getVisualSOL()
700
+ expect(sol.logicalRow).toBe(0)
701
+ expect(sol.logicalCol).toBe(0)
702
+ expect(sol.visualRow).toBe(0)
703
+ expect(sol.visualCol).toBe(0)
704
+ expect(sol.offset).toBe(0)
705
+ })
706
+
707
+ it("should get visual EOL on single line", () => {
708
+ buffer.setText("Hello World")
709
+ buffer.setCursorToLineCol(0, 6) // Middle of line
710
+
711
+ const eol = view.getVisualEOL()
712
+ expect(eol.logicalRow).toBe(0)
713
+ expect(eol.logicalCol).toBe(11)
714
+ expect(eol.visualRow).toBe(0)
715
+ expect(eol.visualCol).toBe(11)
716
+ })
717
+
718
+ it("should get visual SOL/EOL on multi-line text", () => {
719
+ buffer.setText("Line 1\nLine 2\nLine 3")
720
+
721
+ // Test on second line
722
+ buffer.setCursorToLineCol(1, 3)
723
+
724
+ const sol = view.getVisualSOL()
725
+ expect(sol.logicalRow).toBe(1)
726
+ expect(sol.logicalCol).toBe(0)
727
+ expect(sol.visualRow).toBe(1)
728
+ expect(sol.visualCol).toBe(0)
729
+
730
+ const eol = view.getVisualEOL()
731
+ expect(eol.logicalRow).toBe(1)
732
+ expect(eol.logicalCol).toBe(6)
733
+ expect(eol.visualRow).toBe(1)
734
+ expect(eol.visualCol).toBe(6)
735
+ })
736
+
737
+ it("should handle visual SOL/EOL at line boundaries", () => {
738
+ buffer.setText("ABC\nDEF")
739
+
740
+ // At start of line 0
741
+ buffer.setCursorToLineCol(0, 0)
742
+ let sol = view.getVisualSOL()
743
+ expect(sol.logicalCol).toBe(0)
744
+
745
+ // At end of line 0
746
+ buffer.setCursorToLineCol(0, 3)
747
+ let eol = view.getVisualEOL()
748
+ expect(eol.logicalCol).toBe(3)
749
+
750
+ // At start of line 1
751
+ buffer.setCursorToLineCol(1, 0)
752
+ sol = view.getVisualSOL()
753
+ expect(sol.logicalRow).toBe(1)
754
+ expect(sol.logicalCol).toBe(0)
755
+ })
756
+ })
757
+
758
+ describe("with wrapping", () => {
759
+ it("should get SOL of first wrapped line", () => {
760
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
761
+ view.setWrapMode("char")
762
+ view.setViewportSize(10, 10)
763
+
764
+ // Cursor at position 0 (first visual line)
765
+ buffer.setCursorToLineCol(0, 0)
766
+
767
+ const sol = view.getVisualSOL()
768
+ expect(sol.logicalRow).toBe(0)
769
+ expect(sol.logicalCol).toBe(0)
770
+ expect(sol.visualRow).toBe(0)
771
+ expect(sol.visualCol).toBe(0)
772
+ })
773
+
774
+ it("should get EOL of first wrapped line", () => {
775
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
776
+ view.setWrapMode("char")
777
+ view.setViewportSize(10, 10)
778
+
779
+ buffer.setCursorToLineCol(0, 5)
780
+
781
+ const eol = view.getVisualEOL()
782
+ expect(eol.logicalRow).toBe(0)
783
+ expect(eol.logicalCol).toBe(9)
784
+ expect(eol.visualRow).toBe(0)
785
+ expect(eol.visualCol).toBe(9)
786
+ })
787
+
788
+ it("should get SOL of second wrapped line", () => {
789
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
790
+ view.setWrapMode("char")
791
+ view.setViewportSize(10, 10)
792
+
793
+ buffer.setCursorToLineCol(0, 15)
794
+
795
+ const sol = view.getVisualSOL()
796
+ expect(sol.logicalRow).toBe(0)
797
+ expect(sol.logicalCol).toBe(10)
798
+ expect(sol.visualRow).toBe(1)
799
+ expect(sol.visualCol).toBe(0)
800
+ })
801
+
802
+ it("should get EOL of second wrapped line", () => {
803
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
804
+ view.setWrapMode("char")
805
+ view.setViewportSize(10, 10)
806
+
807
+ buffer.setCursorToLineCol(0, 15)
808
+
809
+ const eol = view.getVisualEOL()
810
+ expect(eol.logicalRow).toBe(0)
811
+ expect(eol.logicalCol).toBe(19)
812
+ expect(eol.visualRow).toBe(1)
813
+ expect(eol.visualCol).toBe(9)
814
+ })
815
+
816
+ it("should get EOL of last wrapped line (end of logical line)", () => {
817
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
818
+ view.setWrapMode("char")
819
+ view.setViewportSize(10, 10)
820
+
821
+ buffer.setCursorToLineCol(0, 25)
822
+
823
+ const eol = view.getVisualEOL()
824
+ expect(eol.logicalRow).toBe(0)
825
+ expect(eol.logicalCol).toBe(26)
826
+ expect(eol.visualRow).toBe(2)
827
+ expect(eol.visualCol).toBe(6)
828
+ })
829
+
830
+ it("should handle word wrapping correctly", () => {
831
+ buffer.setText("Hello wonderful world of text")
832
+ view.setWrapMode("word")
833
+ view.setViewportSize(15, 10)
834
+
835
+ buffer.setCursorToLineCol(0, 20)
836
+
837
+ const vcursor = view.getVisualCursor()
838
+ expect(vcursor.visualRow).toBeGreaterThan(0)
839
+
840
+ const sol = view.getVisualSOL()
841
+ expect(sol.visualRow).toBe(vcursor.visualRow)
842
+ expect(sol.visualCol).toBe(0)
843
+ expect(sol.logicalRow).toBe(0)
844
+ expect(sol.logicalCol).toBeGreaterThan(0)
845
+
846
+ const eol = view.getVisualEOL()
847
+ expect(eol.visualRow).toBe(vcursor.visualRow)
848
+ expect(eol.logicalRow).toBe(0)
849
+ expect(eol.logicalCol).toBeGreaterThan(sol.logicalCol)
850
+ })
851
+
852
+ it("should move cursor to END of current visual line, NOT start of next line", () => {
853
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
854
+ view.setWrapMode("char")
855
+ view.setViewportSize(10, 10)
856
+
857
+ buffer.setCursorToLineCol(0, 5)
858
+ let vcursor = view.getVisualCursor()
859
+ expect(vcursor.visualRow).toBe(0)
860
+ expect(vcursor.logicalCol).toBe(5)
861
+
862
+ const eol = view.getVisualEOL()
863
+ buffer.setCursor(eol.logicalRow, eol.logicalCol)
864
+
865
+ const finalCursor = buffer.getCursorPosition()
866
+ const finalVCursor = view.getVisualCursor()
867
+
868
+ expect(finalVCursor.visualRow).toBe(0)
869
+ expect(finalCursor.col).toBe(9)
870
+ })
871
+
872
+ it("should navigate through multiple wrapped lines", () => {
873
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789")
874
+ view.setWrapMode("char")
875
+ view.setViewportSize(10, 10)
876
+
877
+ const positions = [0, 10, 20, 30]
878
+
879
+ for (const pos of positions) {
880
+ buffer.setCursorToLineCol(0, pos)
881
+
882
+ const vcursor = view.getVisualCursor()
883
+ const sol = view.getVisualSOL()
884
+ const eol = view.getVisualEOL()
885
+
886
+ expect(sol.visualCol).toBe(0)
887
+ expect(sol.visualRow).toBe(vcursor.visualRow)
888
+
889
+ expect(eol.logicalCol).toBeGreaterThan(sol.logicalCol)
890
+ expect(eol.visualRow).toBe(vcursor.visualRow)
891
+ }
892
+ })
893
+ })
894
+
895
+ describe("with multi-byte characters", () => {
896
+ it("should handle emoji in visual SOL/EOL", () => {
897
+ buffer.setText("Hello 🌟 World")
898
+ buffer.setCursorToLineCol(0, 8) // After emoji
899
+
900
+ const sol = view.getVisualSOL()
901
+ expect(sol.logicalCol).toBe(0)
902
+ expect(sol.visualCol).toBe(0)
903
+
904
+ const eol = view.getVisualEOL()
905
+ expect(eol.logicalCol).toBe(14)
906
+ expect(eol.visualCol).toBe(14) // Visual width of the line
907
+ })
908
+
909
+ it("should handle CJK characters in visual SOL/EOL", () => {
910
+ buffer.setText("测试文字")
911
+ buffer.setCursorToLineCol(0, 2) // Middle
912
+
913
+ const sol = view.getVisualSOL()
914
+ expect(sol.logicalCol).toBe(0)
915
+ expect(sol.visualCol).toBe(0)
916
+
917
+ const eol = view.getVisualEOL()
918
+ expect(eol.logicalRow).toBe(0)
919
+ expect(eol.logicalCol).toBe(8) // CJK text line width
920
+ expect(eol.visualCol).toBe(8) // Visual width
921
+ })
922
+
923
+ it("should handle wrapped emoji correctly", () => {
924
+ buffer.setText("🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟") // 10 emoji
925
+ view.setWrapMode("char")
926
+ view.setViewportSize(10, 10)
927
+
928
+ // First wrapped line
929
+ buffer.setCursorToLineCol(0, 2)
930
+ let sol = view.getVisualSOL()
931
+ let eol = view.getVisualEOL()
932
+ let vcursor = view.getVisualCursor()
933
+
934
+ expect(vcursor.visualRow).toBe(0)
935
+ expect(sol.logicalCol).toBe(0)
936
+ expect(sol.visualCol).toBe(0)
937
+ expect(eol.logicalCol).toBeGreaterThan(0)
938
+ expect(eol.visualCol).toBeGreaterThan(0)
939
+
940
+ // Second wrapped line - need to be far enough to be on next visual line
941
+ buffer.setCursorToLineCol(0, 12) // Past first 5 emoji (10 logical cols)
942
+ vcursor = view.getVisualCursor()
943
+ sol = view.getVisualSOL()
944
+ eol = view.getVisualEOL()
945
+
946
+ expect(vcursor.visualRow).toBe(1) // Should be on second visual line
947
+ expect(sol.visualCol).toBe(0)
948
+ expect(sol.logicalCol).toBeGreaterThan(0)
949
+ expect(eol.logicalCol).toBe(20) // End of logical line
950
+ })
951
+
952
+ it("should handle mixed ASCII and CJK with wrapping", () => {
953
+ buffer.setText("AB测试CD文字EF") // Mixed width chars
954
+ view.setWrapMode("char")
955
+ view.setViewportSize(8, 10)
956
+
957
+ buffer.setCursorToLineCol(0, 5)
958
+
959
+ const vcursor = view.getVisualCursor()
960
+ const sol = view.getVisualSOL()
961
+ const eol = view.getVisualEOL()
962
+
963
+ expect(sol.visualRow).toBe(vcursor.visualRow)
964
+ expect(sol.visualCol).toBe(0)
965
+ expect(eol.visualRow).toBe(vcursor.visualRow)
966
+ expect(eol.visualCol).toBeGreaterThan(0)
967
+ })
968
+ })
969
+
970
+ describe("edge cases", () => {
971
+ it("should handle empty line", () => {
972
+ buffer.setText("\n")
973
+ buffer.setCursorToLineCol(0, 0)
974
+
975
+ const sol = view.getVisualSOL()
976
+ const eol = view.getVisualEOL()
977
+
978
+ expect(sol.logicalRow).toBe(0)
979
+ expect(sol.logicalCol).toBe(0)
980
+ expect(eol.logicalRow).toBe(0)
981
+ expect(eol.logicalCol).toBe(0)
982
+ })
983
+
984
+ it("should handle cursor at exact wrap boundary", () => {
985
+ buffer.setText("0123456789ABCDEFGHIJ")
986
+ view.setWrapMode("char")
987
+ view.setViewportSize(10, 10)
988
+
989
+ // Cursor at position 10 (start of second visual line)
990
+ buffer.setCursorToLineCol(0, 10)
991
+
992
+ const vcursor = view.getVisualCursor()
993
+ expect(vcursor.visualRow).toBe(1)
994
+
995
+ const sol = view.getVisualSOL()
996
+ expect(sol.logicalCol).toBe(10)
997
+ expect(sol.visualRow).toBe(1)
998
+ expect(sol.visualCol).toBe(0)
999
+
1000
+ const eol = view.getVisualEOL()
1001
+ expect(eol.logicalCol).toBe(20)
1002
+ expect(eol.visualRow).toBe(1)
1003
+ })
1004
+
1005
+ it("should handle single character line", () => {
1006
+ buffer.setText("X")
1007
+ buffer.setCursorToLineCol(0, 0)
1008
+
1009
+ const sol = view.getVisualSOL()
1010
+ const eol = view.getVisualEOL()
1011
+
1012
+ expect(sol.logicalCol).toBe(0)
1013
+ expect(eol.logicalCol).toBe(1)
1014
+ })
1015
+
1016
+ it("should compare logical EOL vs visual EOL on wrapped line", () => {
1017
+ buffer.setText("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
1018
+ view.setWrapMode("char")
1019
+ view.setViewportSize(10, 10)
1020
+
1021
+ buffer.setCursorToLineCol(0, 5)
1022
+
1023
+ const logicalEOL = view.getEOL()
1024
+ const visualEOL = view.getVisualEOL()
1025
+
1026
+ expect(logicalEOL.logicalCol).toBe(26)
1027
+ expect(visualEOL.logicalCol).toBe(9)
1028
+ expect(visualEOL.visualRow).toBe(0)
1029
+ })
1030
+ })
1031
+ })
1032
+ })