@fairyhunter13/opentui-core 0.1.113 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +62 -53
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-9vwc3fg6.js +0 -12260
  472. package/index-9vwc3fg6.js.map +0 -42
  473. package/index-dcj62y8t.js +0 -20614
  474. package/index-dcj62y8t.js.map +0 -67
  475. package/index-f7n39gpy.js +0 -411
  476. package/index-f7n39gpy.js.map +0 -10
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,1854 @@
1
+ const std = @import("std");
2
+ const build_options = @import("build_options");
3
+ const Allocator = std.mem.Allocator;
4
+
5
+ const ansi = @import("ansi.zig");
6
+ const buffer = @import("buffer.zig");
7
+ const renderer = @import("renderer.zig");
8
+ const gp = @import("grapheme.zig");
9
+ const link = @import("link.zig");
10
+ const text_buffer = @import("text-buffer.zig");
11
+ const text_buffer_view = @import("text-buffer-view.zig");
12
+ const edit_buffer_mod = @import("edit-buffer.zig");
13
+ const editor_view = @import("editor-view.zig");
14
+ const syntax_style = @import("syntax-style.zig");
15
+ const terminal = @import("terminal.zig");
16
+ const utf8 = @import("utf8.zig");
17
+ const logger = @import("logger.zig");
18
+ const event_bus = @import("event-bus.zig");
19
+ const utils = @import("utils.zig");
20
+ const native_span_feed = @import("native-span-feed.zig");
21
+ const buffer_effects = @import("buffer-methods.zig");
22
+
23
+ pub const OptimizedBuffer = buffer.OptimizedBuffer;
24
+ pub const CliRenderer = renderer.CliRenderer;
25
+ pub const Terminal = terminal.Terminal;
26
+ pub const RGBA = buffer.RGBA;
27
+
28
+ comptime {
29
+ _ = native_span_feed;
30
+ }
31
+
32
+ export fn setLogCallback(callback: ?*const fn (level: u8, msgPtr: [*]const u8, msgLen: usize) callconv(.c) void) void {
33
+ logger.setLogCallback(callback);
34
+ }
35
+
36
+ export fn setEventCallback(callback: ?*const fn (namePtr: [*]const u8, nameLen: usize, dataPtr: [*]const u8, dataLen: usize) callconv(.c) void) void {
37
+ event_bus.setEventCallback(callback);
38
+ }
39
+
40
+ var gpa = std.heap.GeneralPurposeAllocator(.{
41
+ .enable_memory_limit = build_options.gpa_safe_stats,
42
+ .safety = build_options.gpa_safe_stats,
43
+ }){};
44
+ const globalAllocator = gpa.allocator();
45
+ var arena = std.heap.ArenaAllocator.init(globalAllocator);
46
+ const globalArena = arena.allocator();
47
+
48
+ pub const ExternalBuildOptions = extern struct {
49
+ gpa_safe_stats: bool,
50
+ gpa_memory_limit_tracking: bool,
51
+ };
52
+
53
+ pub const ExternalAllocatorStats = extern struct {
54
+ total_requested_bytes: u64,
55
+ active_allocations: u64,
56
+ small_allocations: u64,
57
+ large_allocations: u64,
58
+ requested_bytes_valid: bool,
59
+ };
60
+
61
+ fn toNonNegativeU64(value: anytype) u64 {
62
+ const ValueType = @TypeOf(value);
63
+
64
+ return switch (@typeInfo(ValueType)) {
65
+ .int => |int_info| if (int_info.signedness == .signed) blk: {
66
+ const signed_value: i64 = @intCast(value);
67
+ if (signed_value <= 0) break :blk 0;
68
+ break :blk @intCast(signed_value);
69
+ } else @intCast(value),
70
+ .comptime_int => blk: {
71
+ if (value <= 0) break :blk 0;
72
+ break :blk @intCast(value);
73
+ },
74
+ else => 0,
75
+ };
76
+ }
77
+
78
+ const RequestedBytesInfo = struct {
79
+ bytes: u64,
80
+ valid: bool,
81
+ };
82
+
83
+ fn sanitizeRequestedBytes(value: u64) RequestedBytesInfo {
84
+ const signed_value: i64 = @bitCast(value);
85
+ if (signed_value < 0) {
86
+ return .{ .bytes = 0, .valid = false };
87
+ }
88
+
89
+ return .{ .bytes = @intCast(signed_value), .valid = true };
90
+ }
91
+
92
+ fn queryStatsField(comptime field_names: []const []const u8) ?u64 {
93
+ if (!@hasDecl(@TypeOf(gpa), "queryStats")) {
94
+ return null;
95
+ }
96
+
97
+ const stats = gpa.queryStats();
98
+ const StatsType = @TypeOf(stats);
99
+
100
+ inline for (field_names) |field_name| {
101
+ if (@hasField(StatsType, field_name)) {
102
+ return toNonNegativeU64(@field(stats, field_name));
103
+ }
104
+ }
105
+
106
+ return null;
107
+ }
108
+
109
+ fn getTotalRequestedBytesInfo() RequestedBytesInfo {
110
+ if (!build_options.gpa_safe_stats) {
111
+ return .{ .bytes = 0, .valid = false };
112
+ }
113
+
114
+ if (queryStatsField(&.{"total_requested_bytes"})) |value| {
115
+ return sanitizeRequestedBytes(value);
116
+ }
117
+
118
+ if (@hasField(@TypeOf(gpa), "total_requested_bytes")) {
119
+ if (@TypeOf(gpa.total_requested_bytes) == void) {
120
+ return .{ .bytes = 0, .valid = false };
121
+ }
122
+
123
+ return sanitizeRequestedBytes(toNonNegativeU64(gpa.total_requested_bytes));
124
+ }
125
+
126
+ return .{ .bytes = 0, .valid = false };
127
+ }
128
+
129
+ fn getSmallAllocationCount() u64 {
130
+ if (queryStatsField(&.{ "small_allocations", "small_allocation_count" })) |value| {
131
+ return value;
132
+ }
133
+
134
+ var total: u64 = 0;
135
+ for (gpa.buckets) |bucket_head| {
136
+ var current = bucket_head;
137
+ while (current) |bucket| {
138
+ const allocated: u64 = @intCast(bucket.allocated_count);
139
+ const freed: u64 = @intCast(bucket.freed_count);
140
+ if (allocated >= freed) {
141
+ total += allocated - freed;
142
+ }
143
+ current = bucket.next;
144
+ }
145
+ }
146
+
147
+ return total;
148
+ }
149
+
150
+ fn getLargeAllocationCount() u64 {
151
+ if (queryStatsField(&.{ "large_allocations", "large_allocation_count" })) |value| {
152
+ return value;
153
+ }
154
+
155
+ return @intCast(gpa.large_allocations.count());
156
+ }
157
+
158
+ export fn createNativeSpanFeed(options_ptr: ?*const native_span_feed.Options) ?*native_span_feed.Stream {
159
+ return native_span_feed.createNativeSpanFeedWithAllocator(globalAllocator, options_ptr);
160
+ }
161
+
162
+ export fn getArenaAllocatedBytes() usize {
163
+ return arena.queryCapacity();
164
+ }
165
+
166
+ export fn getBuildOptions(out_ptr: *ExternalBuildOptions) void {
167
+ out_ptr.* = .{
168
+ .gpa_safe_stats = build_options.gpa_safe_stats,
169
+ .gpa_memory_limit_tracking = build_options.gpa_safe_stats,
170
+ };
171
+ }
172
+
173
+ export fn getAllocatorStats(out_ptr: *ExternalAllocatorStats) void {
174
+ const small_allocations = getSmallAllocationCount();
175
+ const large_allocations = getLargeAllocationCount();
176
+ const active_allocations = small_allocations + large_allocations;
177
+ const requested_bytes = getTotalRequestedBytesInfo();
178
+
179
+ out_ptr.* = .{
180
+ .total_requested_bytes = requested_bytes.bytes,
181
+ .active_allocations = active_allocations,
182
+ .small_allocations = small_allocations,
183
+ .large_allocations = large_allocations,
184
+ .requested_bytes_valid = requested_bytes.valid,
185
+ };
186
+ }
187
+
188
+ export fn createRenderer(width: u32, height: u32, testing: bool, remote: bool) ?*renderer.CliRenderer {
189
+ if (width == 0 or height == 0) {
190
+ logger.warn("Invalid renderer dimensions: {}x{}", .{ width, height });
191
+ return null;
192
+ }
193
+
194
+ const pool = gp.initGlobalPool(globalArena);
195
+ _ = link.initGlobalLinkPool(globalArena);
196
+ return renderer.CliRenderer.createWithOptions(globalAllocator, width, height, pool, testing, remote) catch |err| {
197
+ logger.err("Failed to create renderer: {}", .{err});
198
+ return null;
199
+ };
200
+ }
201
+
202
+ export fn setTerminalEnvVar(rendererPtr: *renderer.CliRenderer, keyPtr: [*]const u8, keyLen: usize, valuePtr: [*]const u8, valueLen: usize) bool {
203
+ const key = keyPtr[0..keyLen];
204
+ const value = valuePtr[0..valueLen];
205
+ return rendererPtr.setTerminalEnvVar(key, value);
206
+ }
207
+
208
+ export fn setUseThread(rendererPtr: *renderer.CliRenderer, useThread: bool) void {
209
+ rendererPtr.setUseThread(useThread);
210
+ }
211
+
212
+ export fn destroyRenderer(rendererPtr: *renderer.CliRenderer) void {
213
+ rendererPtr.destroy();
214
+ }
215
+
216
+ export fn setBackgroundColor(rendererPtr: *renderer.CliRenderer, color: [*]const f32) void {
217
+ rendererPtr.setBackgroundColor(utils.f32PtrToRGBA(color));
218
+ }
219
+
220
+ export fn setRenderOffset(rendererPtr: *renderer.CliRenderer, offset: u32) void {
221
+ rendererPtr.setRenderOffset(offset);
222
+ }
223
+
224
+ export fn updateStats(rendererPtr: *renderer.CliRenderer, time: f64, fps: u32, frameCallbackTime: f64) void {
225
+ rendererPtr.updateStats(time, fps, frameCallbackTime);
226
+ }
227
+
228
+ export fn updateMemoryStats(rendererPtr: *renderer.CliRenderer, heapUsed: u32, heapTotal: u32, arrayBuffers: u32) void {
229
+ rendererPtr.updateMemoryStats(heapUsed, heapTotal, arrayBuffers);
230
+ }
231
+
232
+ export fn getNextBuffer(rendererPtr: *renderer.CliRenderer) *buffer.OptimizedBuffer {
233
+ return rendererPtr.getNextBuffer();
234
+ }
235
+
236
+ export fn getCurrentBuffer(rendererPtr: *renderer.CliRenderer) *buffer.OptimizedBuffer {
237
+ return rendererPtr.getCurrentBuffer();
238
+ }
239
+
240
+ const OutputSlice = extern struct {
241
+ ptr: [*]const u8,
242
+ len: usize,
243
+ };
244
+
245
+ export fn getLastOutputForTest(rendererPtr: *renderer.CliRenderer, outSlice: *OutputSlice) void {
246
+ const output = rendererPtr.getLastOutputForTest();
247
+ outSlice.ptr = output.ptr;
248
+ outSlice.len = output.len;
249
+ }
250
+
251
+ export fn setHyperlinksCapability(rendererPtr: *renderer.CliRenderer, enabled: bool) void {
252
+ rendererPtr.terminal.caps.hyperlinks = enabled;
253
+ }
254
+
255
+ export fn clearGlobalLinkPool() void {
256
+ link.deinitGlobalLinkPool();
257
+ }
258
+
259
+ export fn getBufferWidth(bufferPtr: *buffer.OptimizedBuffer) u32 {
260
+ return bufferPtr.width;
261
+ }
262
+
263
+ export fn getBufferHeight(bufferPtr: *buffer.OptimizedBuffer) u32 {
264
+ return bufferPtr.height;
265
+ }
266
+
267
+ export fn render(rendererPtr: *renderer.CliRenderer, force: bool) void {
268
+ rendererPtr.render(force);
269
+ }
270
+
271
+ export fn createOptimizedBuffer(width: u32, height: u32, respectAlpha: bool, widthMethod: u8, idPtr: [*]const u8, idLen: usize) ?*buffer.OptimizedBuffer {
272
+ if (width == 0 or height == 0) {
273
+ logger.warn("Invalid buffer dimensions: {}x{}", .{ width, height });
274
+ return null;
275
+ }
276
+
277
+ const pool = gp.initGlobalPool(globalArena);
278
+ const link_pool = link.initGlobalLinkPool(globalArena);
279
+ const wMethod: utf8.WidthMethod = if (widthMethod == 0) .wcwidth else .unicode;
280
+ const id = idPtr[0..idLen];
281
+
282
+ return buffer.OptimizedBuffer.init(globalAllocator, width, height, .{
283
+ .respectAlpha = respectAlpha,
284
+ .pool = pool,
285
+ .width_method = wMethod,
286
+ .id = id,
287
+ .link_pool = link_pool,
288
+ }) catch |err| {
289
+ logger.err("Failed to create optimized buffer: {}", .{err});
290
+ return null;
291
+ };
292
+ }
293
+
294
+ export fn destroyOptimizedBuffer(bufferPtr: *buffer.OptimizedBuffer) void {
295
+ bufferPtr.deinit();
296
+ }
297
+
298
+ export fn destroyFrameBuffer(frameBufferPtr: *buffer.OptimizedBuffer) void {
299
+ destroyOptimizedBuffer(frameBufferPtr);
300
+ }
301
+
302
+ export fn drawFrameBuffer(targetPtr: *buffer.OptimizedBuffer, destX: i32, destY: i32, frameBuffer: *buffer.OptimizedBuffer, sourceX: u32, sourceY: u32, sourceWidth: u32, sourceHeight: u32) void {
303
+ const srcX = if (sourceX == 0) null else sourceX;
304
+ const srcY = if (sourceY == 0) null else sourceY;
305
+ const srcWidth = if (sourceWidth == 0) null else sourceWidth;
306
+ const srcHeight = if (sourceHeight == 0) null else sourceHeight;
307
+
308
+ targetPtr.drawFrameBuffer(destX, destY, frameBuffer, srcX, srcY, srcWidth, srcHeight);
309
+ }
310
+
311
+ export fn setCursorPosition(rendererPtr: *renderer.CliRenderer, x: i32, y: i32, visible: bool) void {
312
+ rendererPtr.terminal.setCursorPosition(@intCast(@max(1, x)), @intCast(@max(1, y)), visible);
313
+ }
314
+
315
+ pub const ExternalCapabilities = extern struct {
316
+ kitty_keyboard: bool,
317
+ kitty_graphics: bool,
318
+ rgb: bool,
319
+ unicode: u8, // 0 = wcwidth, 1 = unicode
320
+ sgr_pixels: bool,
321
+ color_scheme_updates: bool,
322
+ explicit_width: bool,
323
+ scaled_text: bool,
324
+ sixel: bool,
325
+ focus_tracking: bool,
326
+ sync: bool,
327
+ bracketed_paste: bool,
328
+ hyperlinks: bool,
329
+ osc52: bool,
330
+ explicit_cursor_positioning: bool,
331
+ term_name_ptr: [*]const u8,
332
+ term_name_len: usize,
333
+ term_version_ptr: [*]const u8,
334
+ term_version_len: usize,
335
+ term_from_xtversion: bool,
336
+ };
337
+
338
+ export fn getTerminalCapabilities(rendererPtr: *renderer.CliRenderer, capsPtr: *ExternalCapabilities) void {
339
+ const caps = rendererPtr.getTerminalCapabilities();
340
+ const term = &rendererPtr.terminal;
341
+
342
+ capsPtr.* = .{
343
+ .kitty_keyboard = caps.kitty_keyboard,
344
+ .kitty_graphics = caps.kitty_graphics,
345
+ .rgb = caps.rgb,
346
+ .unicode = if (caps.unicode == .wcwidth) 0 else 1,
347
+ .sgr_pixels = caps.sgr_pixels,
348
+ .color_scheme_updates = caps.color_scheme_updates,
349
+ .explicit_width = caps.explicit_width,
350
+ .scaled_text = caps.scaled_text,
351
+ .sixel = caps.sixel,
352
+ .focus_tracking = caps.focus_tracking,
353
+ .sync = caps.sync,
354
+ .bracketed_paste = caps.bracketed_paste,
355
+ .hyperlinks = caps.hyperlinks,
356
+ .osc52 = caps.osc52,
357
+ .explicit_cursor_positioning = caps.explicit_cursor_positioning,
358
+ .term_name_ptr = &term.term_info.name,
359
+ .term_name_len = term.term_info.name_len,
360
+ .term_version_ptr = &term.term_info.version,
361
+ .term_version_len = term.term_info.version_len,
362
+ .term_from_xtversion = term.term_info.from_xtversion,
363
+ };
364
+ }
365
+
366
+ export fn processCapabilityResponse(rendererPtr: *renderer.CliRenderer, responsePtr: [*]const u8, responseLen: usize) void {
367
+ const response = responsePtr[0..responseLen];
368
+ rendererPtr.processCapabilityResponse(response);
369
+ }
370
+
371
+ export fn setCursorColor(rendererPtr: *renderer.CliRenderer, color: [*]const f32) void {
372
+ rendererPtr.terminal.setCursorColor(utils.f32PtrToRGBA(color));
373
+ }
374
+
375
+ pub const CursorStyleOptions = extern struct {
376
+ style: u8,
377
+ blinking: u8,
378
+ color: ?[*]const f32,
379
+ cursor: u8,
380
+ };
381
+
382
+ export fn setCursorStyleOptions(rendererPtr: *renderer.CliRenderer, options: *const CursorStyleOptions) void {
383
+ const current = rendererPtr.terminal.getCursorStyle();
384
+
385
+ const style = if (options.style <= 3) @as(terminal.CursorStyle, @enumFromInt(options.style)) else current.style;
386
+ const blinking = if (options.blinking <= 1) options.blinking == 1 else current.blinking;
387
+
388
+ if (options.style <= 3 or options.blinking <= 1) {
389
+ rendererPtr.terminal.setCursorStyle(style, blinking);
390
+ }
391
+ if (options.color) |rgba| {
392
+ rendererPtr.terminal.setCursorColor(utils.f32PtrToRGBA(rgba));
393
+ }
394
+ if (options.cursor <= 5) {
395
+ rendererPtr.terminal.setMousePointerStyle(@enumFromInt(options.cursor));
396
+ }
397
+ }
398
+
399
+ pub const ExternalCursorState = extern struct {
400
+ x: u32,
401
+ y: u32,
402
+ visible: bool,
403
+ style: u8,
404
+ blinking: bool,
405
+ r: f32,
406
+ g: f32,
407
+ b: f32,
408
+ a: f32,
409
+ };
410
+
411
+ export fn getCursorState(rendererPtr: *renderer.CliRenderer, outPtr: *ExternalCursorState) void {
412
+ const pos = rendererPtr.terminal.getCursorPosition();
413
+ const style = rendererPtr.terminal.getCursorStyle();
414
+ const color = rendererPtr.terminal.getCursorColor();
415
+
416
+ const styleTag: u8 = switch (style.style) {
417
+ .block => 0,
418
+ .line => 1,
419
+ .underline => 2,
420
+ .default => 3,
421
+ };
422
+
423
+ outPtr.* = .{
424
+ .x = pos.x,
425
+ .y = pos.y,
426
+ .visible = pos.visible,
427
+ .style = styleTag,
428
+ .blinking = style.blinking,
429
+ .r = color[0],
430
+ .g = color[1],
431
+ .b = color[2],
432
+ .a = color[3],
433
+ };
434
+ }
435
+
436
+ export fn setDebugOverlay(rendererPtr: *renderer.CliRenderer, enabled: bool, corner: u8) void {
437
+ const cornerEnum: renderer.DebugOverlayCorner = switch (corner) {
438
+ 0 => .topLeft,
439
+ 1 => .topRight,
440
+ 2 => .bottomLeft,
441
+ else => .bottomRight,
442
+ };
443
+
444
+ rendererPtr.setDebugOverlay(enabled, cornerEnum);
445
+ }
446
+
447
+ export fn clearTerminal(rendererPtr: *renderer.CliRenderer) void {
448
+ rendererPtr.clearTerminal();
449
+ }
450
+
451
+ export fn setTerminalTitle(rendererPtr: *renderer.CliRenderer, titlePtr: [*]const u8, titleLen: usize) void {
452
+ const title = titlePtr[0..titleLen];
453
+ rendererPtr.setTerminalTitle(title);
454
+ }
455
+
456
+ export fn copyToClipboardOSC52(rendererPtr: *renderer.CliRenderer, target: u8, payloadPtr: [*]const u8, payloadLen: usize) bool {
457
+ const targetEnum = std.meta.intToEnum(terminal.ClipboardTarget, target) catch .clipboard;
458
+ const payload = payloadPtr[0..payloadLen];
459
+ return rendererPtr.copyToClipboardOSC52(targetEnum, payload);
460
+ }
461
+
462
+ export fn clearClipboardOSC52(rendererPtr: *renderer.CliRenderer, target: u8) bool {
463
+ const targetEnum = std.meta.intToEnum(terminal.ClipboardTarget, target) catch .clipboard;
464
+ return rendererPtr.clearClipboardOSC52(targetEnum);
465
+ }
466
+
467
+ // Buffer functions
468
+ export fn bufferClear(bufferPtr: *buffer.OptimizedBuffer, bg: [*]const f32) void {
469
+ bufferPtr.clear(utils.f32PtrToRGBA(bg), null) catch {};
470
+ }
471
+
472
+ export fn bufferGetCharPtr(bufferPtr: *buffer.OptimizedBuffer) [*]u32 {
473
+ return bufferPtr.getCharPtr();
474
+ }
475
+
476
+ export fn bufferGetFgPtr(bufferPtr: *buffer.OptimizedBuffer) [*]RGBA {
477
+ return bufferPtr.getFgPtr();
478
+ }
479
+
480
+ export fn bufferGetBgPtr(bufferPtr: *buffer.OptimizedBuffer) [*]RGBA {
481
+ return bufferPtr.getBgPtr();
482
+ }
483
+
484
+ export fn bufferGetAttributesPtr(bufferPtr: *buffer.OptimizedBuffer) [*]u32 {
485
+ return bufferPtr.getAttributesPtr();
486
+ }
487
+
488
+ export fn bufferGetRespectAlpha(bufferPtr: *buffer.OptimizedBuffer) bool {
489
+ return bufferPtr.getRespectAlpha();
490
+ }
491
+
492
+ export fn bufferSetRespectAlpha(bufferPtr: *buffer.OptimizedBuffer, respectAlpha: bool) void {
493
+ bufferPtr.setRespectAlpha(respectAlpha);
494
+ }
495
+
496
+ export fn bufferGetId(bufferPtr: *buffer.OptimizedBuffer, outPtr: [*]u8, maxLen: usize) usize {
497
+ const id = bufferPtr.getId();
498
+ const copyLen = @min(id.len, maxLen);
499
+ @memcpy(outPtr[0..copyLen], id[0..copyLen]);
500
+ return copyLen;
501
+ }
502
+
503
+ export fn bufferGetRealCharSize(bufferPtr: *buffer.OptimizedBuffer) u32 {
504
+ return bufferPtr.getRealCharSize();
505
+ }
506
+
507
+ export fn bufferWriteResolvedChars(bufferPtr: *buffer.OptimizedBuffer, outputPtr: [*]u8, outputLen: usize, addLineBreaks: bool) u32 {
508
+ const output_slice = outputPtr[0..outputLen];
509
+ return bufferPtr.writeResolvedChars(output_slice, addLineBreaks) catch 0;
510
+ }
511
+
512
+ export fn bufferDrawText(bufferPtr: *buffer.OptimizedBuffer, text: [*]const u8, textLen: usize, x: u32, y: u32, fg: [*]const f32, bg: ?[*]const f32, attributes: u32) void {
513
+ const rgbaFg = utils.f32PtrToRGBA(fg);
514
+ const rgbaBg = if (bg) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
515
+ bufferPtr.drawText(text[0..textLen], x, y, rgbaFg, rgbaBg, attributes) catch {};
516
+ }
517
+
518
+ export fn bufferSetCellWithAlphaBlending(bufferPtr: *buffer.OptimizedBuffer, x: u32, y: u32, char: u32, fg: [*]const f32, bg: [*]const f32, attributes: u32) void {
519
+ const rgbaFg = utils.f32PtrToRGBA(fg);
520
+ const rgbaBg = utils.f32PtrToRGBA(bg);
521
+ bufferPtr.setCellWithAlphaBlending(x, y, char, rgbaFg, rgbaBg, attributes) catch {};
522
+ }
523
+
524
+ export fn bufferSetCell(bufferPtr: *buffer.OptimizedBuffer, x: u32, y: u32, char: u32, fg: [*]const f32, bg: [*]const f32, attributes: u32) void {
525
+ const rgbaFg = utils.f32PtrToRGBA(fg);
526
+ const rgbaBg = utils.f32PtrToRGBA(bg);
527
+ const cell = buffer.Cell{
528
+ .char = char,
529
+ .fg = rgbaFg,
530
+ .bg = rgbaBg,
531
+ .attributes = attributes,
532
+ };
533
+ bufferPtr.set(x, y, cell);
534
+ }
535
+
536
+ export fn bufferFillRect(bufferPtr: *buffer.OptimizedBuffer, x: u32, y: u32, width: u32, height: u32, bg: [*]const f32) void {
537
+ const rgbaBg = utils.f32PtrToRGBA(bg);
538
+ bufferPtr.fillRect(x, y, width, height, rgbaBg) catch {};
539
+ }
540
+
541
+ export fn bufferColorMatrix(bufferPtr: *buffer.OptimizedBuffer, matrixPtr: [*]const f32, cellMaskPtr: [*]const f32, cellMaskCount: usize, strength: f32, target: u8) void {
542
+ if (cellMaskCount == 0) return;
543
+ const matrix = matrixPtr[0..16];
544
+ const len = cellMaskCount * 3;
545
+ const cellMask = cellMaskPtr[0..len];
546
+ const targetEnum: buffer_effects.ColorTarget = @enumFromInt(target);
547
+ buffer_effects.colorMatrix(bufferPtr, matrix, cellMask, strength, targetEnum);
548
+ }
549
+
550
+ export fn bufferColorMatrixUniform(bufferPtr: *buffer.OptimizedBuffer, matrixPtr: [*]const f32, strength: f32, target: u8) void {
551
+ const matrix = matrixPtr[0..16];
552
+ const targetEnum: buffer_effects.ColorTarget = @enumFromInt(target);
553
+ buffer_effects.colorMatrixUniform(bufferPtr, matrix, strength, targetEnum);
554
+ }
555
+
556
+ export fn bufferDrawPackedBuffer(bufferPtr: *buffer.OptimizedBuffer, data: [*]const u8, dataLen: usize, posX: u32, posY: u32, terminalWidthCells: u32, terminalHeightCells: u32) void {
557
+ bufferPtr.drawPackedBuffer(data, dataLen, posX, posY, terminalWidthCells, terminalHeightCells);
558
+ }
559
+
560
+ export fn bufferDrawGrayscaleBuffer(bufferPtr: *buffer.OptimizedBuffer, posX: i32, posY: i32, intensities: [*]const f32, srcWidth: u32, srcHeight: u32, fg: ?[*]const f32, bg: ?[*]const f32) void {
561
+ const rgbaFg = if (fg) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
562
+ const rgbaBg = if (bg) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
563
+ bufferPtr.drawGrayscaleBuffer(posX, posY, intensities, srcWidth, srcHeight, rgbaFg, rgbaBg);
564
+ }
565
+
566
+ export fn bufferDrawGrayscaleBufferSupersampled(bufferPtr: *buffer.OptimizedBuffer, posX: i32, posY: i32, intensities: [*]const f32, srcWidth: u32, srcHeight: u32, fg: ?[*]const f32, bg: ?[*]const f32) void {
567
+ const rgbaFg = if (fg) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
568
+ const rgbaBg = if (bg) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
569
+ bufferPtr.drawGrayscaleBufferSupersampled(posX, posY, intensities, srcWidth, srcHeight, rgbaFg, rgbaBg);
570
+ }
571
+
572
+ export fn bufferPushScissorRect(bufferPtr: *buffer.OptimizedBuffer, x: i32, y: i32, width: u32, height: u32) void {
573
+ bufferPtr.pushScissorRect(x, y, width, height) catch {};
574
+ }
575
+
576
+ export fn bufferPopScissorRect(bufferPtr: *buffer.OptimizedBuffer) void {
577
+ bufferPtr.popScissorRect();
578
+ }
579
+
580
+ export fn bufferClearScissorRects(bufferPtr: *buffer.OptimizedBuffer) void {
581
+ bufferPtr.clearScissorRects();
582
+ }
583
+
584
+ // Opacity stack functions
585
+ export fn bufferPushOpacity(bufferPtr: *buffer.OptimizedBuffer, opacity: f32) void {
586
+ bufferPtr.pushOpacity(opacity) catch {};
587
+ }
588
+
589
+ export fn bufferPopOpacity(bufferPtr: *buffer.OptimizedBuffer) void {
590
+ bufferPtr.popOpacity();
591
+ }
592
+
593
+ export fn bufferGetCurrentOpacity(bufferPtr: *buffer.OptimizedBuffer) f32 {
594
+ return bufferPtr.getCurrentOpacity();
595
+ }
596
+
597
+ export fn bufferClearOpacity(bufferPtr: *buffer.OptimizedBuffer) void {
598
+ bufferPtr.clearOpacity();
599
+ }
600
+
601
+ export fn bufferDrawSuperSampleBuffer(bufferPtr: *buffer.OptimizedBuffer, x: u32, y: u32, pixelData: [*]const u8, len: usize, format: u8, alignedBytesPerRow: u32) void {
602
+ bufferPtr.drawSuperSampleBuffer(x, y, pixelData, len, format, alignedBytesPerRow) catch {};
603
+ }
604
+
605
+ export fn linkAlloc(urlPtr: [*]const u8, urlLen: usize) u32 {
606
+ const url = urlPtr[0..urlLen];
607
+ const link_pool = link.initGlobalLinkPool(globalArena);
608
+ return link_pool.alloc(url) catch 0;
609
+ }
610
+
611
+ export fn linkGetUrl(id: u32, outPtr: [*]u8, maxLen: usize) usize {
612
+ const link_pool = link.initGlobalLinkPool(globalArena);
613
+ const url_bytes = link_pool.get(id) catch return 0;
614
+ const copyLen = @min(url_bytes.len, maxLen);
615
+ @memcpy(outPtr[0..copyLen], url_bytes[0..copyLen]);
616
+ return copyLen;
617
+ }
618
+
619
+ export fn attributesWithLink(baseAttributes: u32, linkId: u32) u32 {
620
+ return ansi.TextAttributes.setLinkId(baseAttributes, linkId);
621
+ }
622
+
623
+ export fn attributesGetLinkId(attributes: u32) u32 {
624
+ return ansi.TextAttributes.getLinkId(attributes);
625
+ }
626
+
627
+ pub const ExternalGridDrawOptions = extern struct {
628
+ draw_inner: bool,
629
+ draw_outer: bool,
630
+ };
631
+
632
+ export fn bufferDrawGrid(
633
+ bufferPtr: *buffer.OptimizedBuffer,
634
+ borderChars: [*]const u32,
635
+ borderFg: [*]const f32,
636
+ borderBg: [*]const f32,
637
+ columnOffsets: [*]const i32,
638
+ columnCount: u32,
639
+ rowOffsets: [*]const i32,
640
+ rowCount: u32,
641
+ options: *const ExternalGridDrawOptions,
642
+ ) void {
643
+ bufferPtr.drawGrid(
644
+ borderChars,
645
+ utils.f32PtrToRGBA(borderFg),
646
+ utils.f32PtrToRGBA(borderBg),
647
+ columnOffsets,
648
+ columnCount,
649
+ rowOffsets,
650
+ rowCount,
651
+ options.draw_inner,
652
+ options.draw_outer,
653
+ );
654
+ }
655
+
656
+ export fn bufferDrawBox(
657
+ bufferPtr: *buffer.OptimizedBuffer,
658
+ x: i32,
659
+ y: i32,
660
+ width: u32,
661
+ height: u32,
662
+ borderChars: [*]const u32,
663
+ packedOptions: u32,
664
+ borderColor: [*]const f32,
665
+ backgroundColor: [*]const f32,
666
+ title: ?[*]const u8,
667
+ titleLen: u32,
668
+ bottomTitle: ?[*]const u8,
669
+ bottomTitleLen: u32,
670
+ ) void {
671
+ const borderSides = buffer.BorderSides{
672
+ .top = (packedOptions & 0b1000) != 0,
673
+ .right = (packedOptions & 0b0100) != 0,
674
+ .bottom = (packedOptions & 0b0010) != 0,
675
+ .left = (packedOptions & 0b0001) != 0,
676
+ };
677
+
678
+ const shouldFill = ((packedOptions >> 4) & 1) != 0;
679
+ const titleAlignment = @as(u8, @intCast((packedOptions >> 5) & 0b11));
680
+ const bottomTitleAlignment = @as(u8, @intCast((packedOptions >> 7) & 0b11));
681
+ const titleSlice = if (title) |t| t[0..titleLen] else null;
682
+
683
+ const bottomTitleSlice = if (bottomTitle) |bt| bt[0..bottomTitleLen] else null;
684
+
685
+ bufferPtr.drawBox(
686
+ x,
687
+ y,
688
+ width,
689
+ height,
690
+ borderChars,
691
+ borderSides,
692
+ utils.f32PtrToRGBA(borderColor),
693
+ utils.f32PtrToRGBA(backgroundColor),
694
+ shouldFill,
695
+ titleSlice,
696
+ titleAlignment,
697
+ bottomTitleSlice,
698
+ bottomTitleAlignment,
699
+ ) catch {};
700
+ }
701
+
702
+ export fn bufferResize(bufferPtr: *buffer.OptimizedBuffer, width: u32, height: u32) void {
703
+ bufferPtr.resize(width, height) catch {};
704
+ }
705
+
706
+ export fn resizeRenderer(rendererPtr: *renderer.CliRenderer, width: u32, height: u32) void {
707
+ rendererPtr.resize(width, height) catch {};
708
+ }
709
+
710
+ export fn addToHitGrid(rendererPtr: *renderer.CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
711
+ rendererPtr.addToHitGrid(x, y, width, height, id);
712
+ }
713
+
714
+ export fn clearCurrentHitGrid(rendererPtr: *renderer.CliRenderer) void {
715
+ rendererPtr.clearCurrentHitGrid();
716
+ }
717
+
718
+ export fn hitGridPushScissorRect(rendererPtr: *renderer.CliRenderer, x: i32, y: i32, width: u32, height: u32) void {
719
+ rendererPtr.hitGridPushScissorRect(x, y, width, height);
720
+ }
721
+
722
+ export fn hitGridPopScissorRect(rendererPtr: *renderer.CliRenderer) void {
723
+ rendererPtr.hitGridPopScissorRect();
724
+ }
725
+
726
+ export fn hitGridClearScissorRects(rendererPtr: *renderer.CliRenderer) void {
727
+ rendererPtr.hitGridClearScissorRects();
728
+ }
729
+
730
+ export fn addToCurrentHitGridClipped(rendererPtr: *renderer.CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
731
+ rendererPtr.addToCurrentHitGridClipped(x, y, width, height, id);
732
+ }
733
+
734
+ export fn checkHit(rendererPtr: *renderer.CliRenderer, x: u32, y: u32) u32 {
735
+ return rendererPtr.checkHit(x, y);
736
+ }
737
+
738
+ export fn getHitGridDirty(rendererPtr: *renderer.CliRenderer) bool {
739
+ return rendererPtr.getHitGridDirty();
740
+ }
741
+
742
+ export fn dumpHitGrid(rendererPtr: *renderer.CliRenderer) void {
743
+ rendererPtr.dumpHitGrid();
744
+ }
745
+
746
+ export fn dumpBuffers(rendererPtr: *renderer.CliRenderer, timestamp: i64) void {
747
+ rendererPtr.dumpBuffers(timestamp);
748
+ }
749
+
750
+ export fn dumpStdoutBuffer(rendererPtr: *renderer.CliRenderer, timestamp: i64) void {
751
+ rendererPtr.dumpStdoutBuffer(timestamp);
752
+ }
753
+
754
+ export fn restoreTerminalModes(rendererPtr: *renderer.CliRenderer) void {
755
+ rendererPtr.restoreTerminalModes();
756
+ }
757
+
758
+ export fn enableMouse(rendererPtr: *renderer.CliRenderer, enableMovement: bool) void {
759
+ rendererPtr.enableMouse(enableMovement);
760
+ }
761
+
762
+ export fn disableMouse(rendererPtr: *renderer.CliRenderer) void {
763
+ rendererPtr.disableMouse();
764
+ }
765
+
766
+ export fn queryPixelResolution(rendererPtr: *renderer.CliRenderer) void {
767
+ rendererPtr.queryPixelResolution();
768
+ }
769
+
770
+ export fn enableKittyKeyboard(rendererPtr: *renderer.CliRenderer, flags: u8) void {
771
+ rendererPtr.enableKittyKeyboard(flags);
772
+ }
773
+
774
+ export fn disableKittyKeyboard(rendererPtr: *renderer.CliRenderer) void {
775
+ rendererPtr.disableKittyKeyboard();
776
+ }
777
+
778
+ export fn setKittyKeyboardFlags(rendererPtr: *renderer.CliRenderer, flags: u8) void {
779
+ rendererPtr.setKittyKeyboardFlags(flags);
780
+ }
781
+
782
+ export fn getKittyKeyboardFlags(rendererPtr: *renderer.CliRenderer) u8 {
783
+ return rendererPtr.getKittyKeyboardFlags();
784
+ }
785
+
786
+ export fn setupTerminal(rendererPtr: *renderer.CliRenderer, useAlternateScreen: bool) void {
787
+ rendererPtr.setupTerminal(useAlternateScreen);
788
+ }
789
+
790
+ export fn suspendRenderer(rendererPtr: *renderer.CliRenderer) void {
791
+ rendererPtr.suspendRenderer();
792
+ }
793
+
794
+ export fn resumeRenderer(rendererPtr: *renderer.CliRenderer) void {
795
+ rendererPtr.resumeRenderer();
796
+ }
797
+
798
+ export fn writeOut(rendererPtr: *renderer.CliRenderer, dataPtr: [*]const u8, dataLen: usize) void {
799
+ if (dataLen == 0) return;
800
+ const data = dataPtr[0..dataLen];
801
+ rendererPtr.writeOut(data);
802
+ }
803
+
804
+ export fn createTextBuffer(widthMethod: u8) ?*text_buffer.UnifiedTextBuffer {
805
+ const pool = gp.initGlobalPool(globalArena);
806
+ const link_pool = link.initGlobalLinkPool(globalArena);
807
+ const wMethod: utf8.WidthMethod = if (widthMethod == 0) .wcwidth else .unicode;
808
+
809
+ return text_buffer.UnifiedTextBuffer.init(globalAllocator, pool, link_pool, wMethod) catch {
810
+ return null;
811
+ };
812
+ }
813
+
814
+ export fn destroyTextBuffer(tb: *text_buffer.UnifiedTextBuffer) void {
815
+ tb.deinit();
816
+ }
817
+
818
+ export fn textBufferGetLength(tb: *text_buffer.UnifiedTextBuffer) u32 {
819
+ return tb.getLength();
820
+ }
821
+
822
+ export fn textBufferGetByteSize(tb: *text_buffer.UnifiedTextBuffer) u32 {
823
+ return tb.getByteSize();
824
+ }
825
+
826
+ export fn textBufferReset(tb: *text_buffer.UnifiedTextBuffer) void {
827
+ tb.reset();
828
+ }
829
+
830
+ export fn textBufferClear(tb: *text_buffer.UnifiedTextBuffer) void {
831
+ tb.clear();
832
+ }
833
+
834
+ export fn textBufferSetDefaultFg(tb: *text_buffer.UnifiedTextBuffer, fg: ?[*]const f32) void {
835
+ const fgColor = if (fg) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
836
+ tb.setDefaultFg(fgColor);
837
+ }
838
+
839
+ export fn textBufferSetDefaultBg(tb: *text_buffer.UnifiedTextBuffer, bg: ?[*]const f32) void {
840
+ const bgColor = if (bg) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
841
+ tb.setDefaultBg(bgColor);
842
+ }
843
+
844
+ export fn textBufferSetDefaultAttributes(tb: *text_buffer.UnifiedTextBuffer, attr: ?[*]const u32) void {
845
+ const attributes = if (attr) |a| a[0] else null;
846
+ tb.setDefaultAttributes(attributes);
847
+ }
848
+
849
+ export fn textBufferResetDefaults(tb: *text_buffer.UnifiedTextBuffer) void {
850
+ tb.resetDefaults();
851
+ }
852
+
853
+ export fn textBufferGetTabWidth(tb: *text_buffer.UnifiedTextBuffer) u8 {
854
+ return tb.tabWidth();
855
+ }
856
+
857
+ export fn textBufferSetTabWidth(tb: *text_buffer.UnifiedTextBuffer, width: u8) void {
858
+ tb.setTabWidth(width);
859
+ }
860
+
861
+ export fn textBufferRegisterMemBuffer(tb: *text_buffer.UnifiedTextBuffer, dataPtr: [*]const u8, dataLen: usize, owned: bool) u16 {
862
+ const data = dataPtr[0..dataLen];
863
+ const mem_id = tb.registerMemBuffer(data, owned) catch return 0xFFFF;
864
+ return @intCast(mem_id);
865
+ }
866
+
867
+ export fn textBufferReplaceMemBuffer(tb: *text_buffer.UnifiedTextBuffer, id: u8, dataPtr: [*]const u8, dataLen: usize, owned: bool) bool {
868
+ const data = dataPtr[0..dataLen];
869
+ tb.replaceMemBuffer(id, data, owned) catch return false;
870
+ return true;
871
+ }
872
+
873
+ export fn textBufferClearMemRegistry(tb: *text_buffer.UnifiedTextBuffer) void {
874
+ tb.clearMemRegistry();
875
+ }
876
+
877
+ export fn textBufferSetTextFromMem(tb: *text_buffer.UnifiedTextBuffer, id: u8) void {
878
+ tb.setTextFromMemId(id) catch {};
879
+ }
880
+
881
+ export fn textBufferAppend(tb: *text_buffer.UnifiedTextBuffer, dataPtr: [*]const u8, dataLen: usize) void {
882
+ const data = dataPtr[0..dataLen];
883
+ tb.append(data) catch {};
884
+ }
885
+
886
+ export fn textBufferAppendFromMemId(tb: *text_buffer.UnifiedTextBuffer, id: u8) void {
887
+ tb.appendFromMemId(id) catch {};
888
+ }
889
+
890
+ export fn textBufferLoadFile(tb: *text_buffer.UnifiedTextBuffer, pathPtr: [*]const u8, pathLen: usize) bool {
891
+ const path = pathPtr[0..pathLen];
892
+ tb.loadFile(path) catch return false;
893
+ return true;
894
+ }
895
+
896
+ export fn textBufferSetStyledText(
897
+ tb: *text_buffer.UnifiedTextBuffer,
898
+ chunksPtr: [*]const text_buffer.StyledChunk,
899
+ chunkCount: usize,
900
+ ) void {
901
+ if (chunkCount == 0) return;
902
+ const chunks = chunksPtr[0..chunkCount];
903
+ tb.setStyledText(chunks) catch {};
904
+ }
905
+
906
+ export fn textBufferGetLineCount(tb: *text_buffer.UnifiedTextBuffer) u32 {
907
+ return tb.getLineCount();
908
+ }
909
+
910
+ export fn textBufferGetPlainText(tb: *text_buffer.UnifiedTextBuffer, outPtr: [*]u8, maxLen: usize) usize {
911
+ const outBuffer = outPtr[0..maxLen];
912
+ return tb.getPlainTextIntoBuffer(outBuffer);
913
+ }
914
+
915
+ // TextBufferView functions (Array-based for backward compatibility)
916
+ export fn createTextBufferView(tb: *text_buffer.UnifiedTextBuffer) ?*text_buffer_view.UnifiedTextBufferView {
917
+ return text_buffer_view.UnifiedTextBufferView.init(globalAllocator, tb) catch {
918
+ return null;
919
+ };
920
+ }
921
+
922
+ export fn destroyTextBufferView(view: *text_buffer_view.UnifiedTextBufferView) void {
923
+ view.deinit();
924
+ }
925
+
926
+ export fn textBufferViewSetSelection(view: *text_buffer_view.UnifiedTextBufferView, start: u32, end: u32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) void {
927
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
928
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
929
+ view.setSelection(start, end, bg, fg);
930
+ }
931
+
932
+ export fn textBufferViewResetSelection(view: *text_buffer_view.UnifiedTextBufferView) void {
933
+ view.resetSelection();
934
+ }
935
+
936
+ export fn textBufferViewGetSelectionInfo(view: *text_buffer_view.UnifiedTextBufferView) u64 {
937
+ return view.packSelectionInfo();
938
+ }
939
+
940
+ export fn textBufferViewSetLocalSelection(view: *text_buffer_view.UnifiedTextBufferView, anchorX: i32, anchorY: i32, focusX: i32, focusY: i32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) bool {
941
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
942
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
943
+ return view.setLocalSelection(anchorX, anchorY, focusX, focusY, bg, fg);
944
+ }
945
+
946
+ export fn textBufferViewUpdateSelection(view: *text_buffer_view.UnifiedTextBufferView, end: u32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) void {
947
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
948
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
949
+ view.updateSelection(end, bg, fg);
950
+ }
951
+
952
+ export fn textBufferViewUpdateLocalSelection(view: *text_buffer_view.UnifiedTextBufferView, anchorX: i32, anchorY: i32, focusX: i32, focusY: i32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) bool {
953
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
954
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
955
+ return view.updateLocalSelection(anchorX, anchorY, focusX, focusY, bg, fg);
956
+ }
957
+
958
+ export fn textBufferViewResetLocalSelection(view: *text_buffer_view.UnifiedTextBufferView) void {
959
+ view.resetLocalSelection();
960
+ }
961
+
962
+ export fn textBufferViewSetWrapWidth(view: *text_buffer_view.UnifiedTextBufferView, width: u32) void {
963
+ view.setWrapWidth(if (width == 0) null else width);
964
+ }
965
+
966
+ export fn textBufferViewSetWrapMode(view: *text_buffer_view.UnifiedTextBufferView, mode: u8) void {
967
+ const wrapMode: text_buffer.WrapMode = switch (mode) {
968
+ 0 => .none,
969
+ 1 => .char,
970
+ 2 => .word,
971
+ else => .none,
972
+ };
973
+ view.setWrapMode(wrapMode);
974
+ }
975
+
976
+ export fn textBufferViewSetViewportSize(view: *text_buffer_view.UnifiedTextBufferView, width: u32, height: u32) void {
977
+ view.setViewportSize(width, height);
978
+ }
979
+
980
+ export fn textBufferViewSetViewport(view: *text_buffer_view.UnifiedTextBufferView, x: u32, y: u32, width: u32, height: u32) void {
981
+ view.setViewport(text_buffer_view.Viewport{
982
+ .x = x,
983
+ .y = y,
984
+ .width = width,
985
+ .height = height,
986
+ });
987
+ }
988
+
989
+ export fn textBufferViewGetVirtualLineCount(view: *text_buffer_view.UnifiedTextBufferView) u32 {
990
+ return view.getVirtualLineCount();
991
+ }
992
+
993
+ export fn textBufferViewGetLineInfoDirect(view: *text_buffer_view.UnifiedTextBufferView, outPtr: *ExternalLineInfo) void {
994
+ const line_info = view.getCachedLineInfo();
995
+
996
+ outPtr.* = .{
997
+ .start_cols_ptr = line_info.line_start_cols.ptr,
998
+ .start_cols_len = @intCast(line_info.line_start_cols.len),
999
+ .width_cols_ptr = line_info.line_width_cols.ptr,
1000
+ .width_cols_len = @intCast(line_info.line_width_cols.len),
1001
+ .sources_ptr = line_info.line_sources.ptr,
1002
+ .sources_len = @intCast(line_info.line_sources.len),
1003
+ .wraps_ptr = line_info.line_wraps.ptr,
1004
+ .wraps_len = @intCast(line_info.line_wraps.len),
1005
+ .width_cols_max = line_info.line_width_cols_max,
1006
+ };
1007
+ }
1008
+
1009
+ export fn textBufferViewGetLogicalLineInfoDirect(view: *text_buffer_view.UnifiedTextBufferView, outPtr: *ExternalLineInfo) void {
1010
+ const line_info = view.getLogicalLineInfo();
1011
+
1012
+ outPtr.* = .{
1013
+ .start_cols_ptr = line_info.line_start_cols.ptr,
1014
+ .start_cols_len = @intCast(line_info.line_start_cols.len),
1015
+ .width_cols_ptr = line_info.line_width_cols.ptr,
1016
+ .width_cols_len = @intCast(line_info.line_width_cols.len),
1017
+ .sources_ptr = line_info.line_sources.ptr,
1018
+ .sources_len = @intCast(line_info.line_sources.len),
1019
+ .wraps_ptr = line_info.line_wraps.ptr,
1020
+ .wraps_len = @intCast(line_info.line_wraps.len),
1021
+ .width_cols_max = line_info.line_width_cols_max,
1022
+ };
1023
+ }
1024
+
1025
+ export fn textBufferViewGetSelectedText(view: *text_buffer_view.UnifiedTextBufferView, outPtr: [*]u8, maxLen: usize) usize {
1026
+ const outBuffer = outPtr[0..maxLen];
1027
+ return view.getSelectedTextIntoBuffer(outBuffer);
1028
+ }
1029
+
1030
+ export fn textBufferViewGetPlainText(view: *text_buffer_view.UnifiedTextBufferView, outPtr: [*]u8, maxLen: usize) usize {
1031
+ const outBuffer = outPtr[0..maxLen];
1032
+ return view.getPlainTextIntoBuffer(outBuffer);
1033
+ }
1034
+
1035
+ export fn textBufferViewSetTabIndicator(view: *text_buffer_view.UnifiedTextBufferView, indicator: u32) void {
1036
+ view.setTabIndicator(indicator);
1037
+ }
1038
+
1039
+ export fn textBufferViewSetTabIndicatorColor(view: *text_buffer_view.UnifiedTextBufferView, color: [*]const f32) void {
1040
+ view.setTabIndicatorColor(utils.f32PtrToRGBA(color));
1041
+ }
1042
+
1043
+ export fn textBufferViewSetTruncate(view: *text_buffer_view.UnifiedTextBufferView, truncate: bool) void {
1044
+ view.setTruncate(truncate);
1045
+ }
1046
+
1047
+ pub const ExternalMeasureResult = extern struct {
1048
+ line_count: u32,
1049
+ width_cols_max: u32,
1050
+ };
1051
+
1052
+ export fn textBufferViewMeasureForDimensions(view: *text_buffer_view.UnifiedTextBufferView, width: u32, height: u32, outPtr: *ExternalMeasureResult) bool {
1053
+ const result = view.measureForDimensions(width, height) catch return false;
1054
+ outPtr.* = .{
1055
+ .line_count = result.line_count,
1056
+ .width_cols_max = result.width_cols_max,
1057
+ };
1058
+ return true;
1059
+ }
1060
+
1061
+ // ===== EditBuffer Exports =====
1062
+
1063
+ export fn createEditBuffer(widthMethod: u8) ?*edit_buffer_mod.EditBuffer {
1064
+ const pool = gp.initGlobalPool(globalArena);
1065
+ const link_pool = link.initGlobalLinkPool(globalArena);
1066
+ const wMethod: utf8.WidthMethod = if (widthMethod == 0) .wcwidth else .unicode;
1067
+
1068
+ return edit_buffer_mod.EditBuffer.init(
1069
+ globalAllocator,
1070
+ pool,
1071
+ link_pool,
1072
+ wMethod,
1073
+ ) catch null;
1074
+ }
1075
+
1076
+ export fn destroyEditBuffer(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1077
+ edit_buffer.deinit();
1078
+ }
1079
+
1080
+ export fn editBufferGetTextBuffer(edit_buffer: *edit_buffer_mod.EditBuffer) *text_buffer.UnifiedTextBuffer {
1081
+ return edit_buffer.getTextBuffer();
1082
+ }
1083
+
1084
+ export fn editBufferInsertText(edit_buffer: *edit_buffer_mod.EditBuffer, textPtr: [*]const u8, textLen: usize) void {
1085
+ const text = textPtr[0..textLen];
1086
+ edit_buffer.insertText(text) catch {};
1087
+ }
1088
+
1089
+ export fn editBufferDeleteRange(edit_buffer: *edit_buffer_mod.EditBuffer, start_row: u32, start_col: u32, end_row: u32, end_col: u32) void {
1090
+ const start = edit_buffer_mod.Cursor{ .row = start_row, .col = start_col };
1091
+ const end = edit_buffer_mod.Cursor{ .row = end_row, .col = end_col };
1092
+ edit_buffer.deleteRange(start, end) catch {};
1093
+ }
1094
+
1095
+ export fn editBufferDeleteCharBackward(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1096
+ edit_buffer.backspace() catch {};
1097
+ }
1098
+
1099
+ export fn editBufferDeleteChar(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1100
+ edit_buffer.deleteForward() catch {};
1101
+ }
1102
+
1103
+ export fn editBufferMoveCursorLeft(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1104
+ edit_buffer.moveLeft();
1105
+ }
1106
+
1107
+ export fn editBufferMoveCursorRight(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1108
+ edit_buffer.moveRight();
1109
+ }
1110
+
1111
+ export fn editBufferMoveCursorUp(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1112
+ edit_buffer.moveUp();
1113
+ }
1114
+
1115
+ export fn editBufferMoveCursorDown(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1116
+ edit_buffer.moveDown();
1117
+ }
1118
+
1119
+ export fn editBufferGetCursor(edit_buffer: *edit_buffer_mod.EditBuffer, outRow: *u32, outCol: *u32) void {
1120
+ const cursor = edit_buffer.getPrimaryCursor();
1121
+ outRow.* = cursor.row;
1122
+ outCol.* = cursor.col;
1123
+ }
1124
+
1125
+ export fn editBufferSetCursor(edit_buffer: *edit_buffer_mod.EditBuffer, row: u32, col: u32) void {
1126
+ edit_buffer.setCursor(row, col) catch {};
1127
+ }
1128
+
1129
+ export fn editBufferSetCursorToLineCol(edit_buffer: *edit_buffer_mod.EditBuffer, row: u32, col: u32) void {
1130
+ edit_buffer.setCursor(row, col) catch {};
1131
+ }
1132
+
1133
+ export fn editBufferSetCursorByOffset(edit_buffer: *edit_buffer_mod.EditBuffer, offset: u32) void {
1134
+ edit_buffer.setCursorByOffset(offset) catch {};
1135
+ }
1136
+
1137
+ export fn editBufferGetNextWordBoundary(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: *ExternalLogicalCursor) void {
1138
+ const cursor = edit_buffer.getNextWordBoundary();
1139
+ outPtr.* = .{
1140
+ .row = cursor.row,
1141
+ .col = cursor.col,
1142
+ .offset = cursor.offset,
1143
+ };
1144
+ }
1145
+
1146
+ export fn editBufferGetPrevWordBoundary(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: *ExternalLogicalCursor) void {
1147
+ const cursor = edit_buffer.getPrevWordBoundary();
1148
+ outPtr.* = .{
1149
+ .row = cursor.row,
1150
+ .col = cursor.col,
1151
+ .offset = cursor.offset,
1152
+ };
1153
+ }
1154
+
1155
+ export fn editBufferGetEOL(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: *ExternalLogicalCursor) void {
1156
+ const cursor = edit_buffer.getEOL();
1157
+ outPtr.* = .{
1158
+ .row = cursor.row,
1159
+ .col = cursor.col,
1160
+ .offset = cursor.offset,
1161
+ };
1162
+ }
1163
+
1164
+ export fn editBufferOffsetToPosition(edit_buffer: *edit_buffer_mod.EditBuffer, offset: u32, outPtr: *ExternalLogicalCursor) bool {
1165
+ const iter_mod = @import("text-buffer-iterators.zig");
1166
+ const coords = iter_mod.offsetToCoords(edit_buffer.tb.rope(), offset) orelse return false;
1167
+ outPtr.* = .{
1168
+ .row = coords.row,
1169
+ .col = coords.col,
1170
+ .offset = offset,
1171
+ };
1172
+ return true;
1173
+ }
1174
+
1175
+ export fn editBufferPositionToOffset(edit_buffer: *edit_buffer_mod.EditBuffer, row: u32, col: u32) u32 {
1176
+ const iter_mod = @import("text-buffer-iterators.zig");
1177
+ return iter_mod.coordsToOffset(edit_buffer.tb.rope(), row, col) orelse 0;
1178
+ }
1179
+
1180
+ export fn editBufferGetLineStartOffset(edit_buffer: *edit_buffer_mod.EditBuffer, row: u32) u32 {
1181
+ const iter_mod = @import("text-buffer-iterators.zig");
1182
+ return iter_mod.coordsToOffset(edit_buffer.tb.rope(), row, 0) orelse 0;
1183
+ }
1184
+
1185
+ export fn editBufferGetTextRange(edit_buffer: *edit_buffer_mod.EditBuffer, start_offset: u32, end_offset: u32, outPtr: [*]u8, maxLen: usize) usize {
1186
+ const outBuffer = outPtr[0..maxLen];
1187
+ return edit_buffer.getTextRange(start_offset, end_offset, outBuffer) catch 0;
1188
+ }
1189
+
1190
+ export fn editBufferGetTextRangeByCoords(edit_buffer: *edit_buffer_mod.EditBuffer, start_row: u32, start_col: u32, end_row: u32, end_col: u32, outPtr: [*]u8, maxLen: usize) usize {
1191
+ const outBuffer = outPtr[0..maxLen];
1192
+ return edit_buffer.getTextRangeByCoords(start_row, start_col, end_row, end_col, outBuffer);
1193
+ }
1194
+
1195
+ export fn editBufferSetText(edit_buffer: *edit_buffer_mod.EditBuffer, textPtr: [*]const u8, textLen: usize) void {
1196
+ const text = textPtr[0..textLen];
1197
+ edit_buffer.setText(text) catch {};
1198
+ }
1199
+
1200
+ export fn editBufferSetTextFromMem(edit_buffer: *edit_buffer_mod.EditBuffer, mem_id: u8) void {
1201
+ edit_buffer.setTextFromMemId(mem_id) catch {};
1202
+ }
1203
+
1204
+ export fn editBufferReplaceText(edit_buffer: *edit_buffer_mod.EditBuffer, textPtr: [*]const u8, textLen: usize) void {
1205
+ const text = textPtr[0..textLen];
1206
+ edit_buffer.replaceText(text) catch {};
1207
+ }
1208
+
1209
+ export fn editBufferReplaceTextFromMem(edit_buffer: *edit_buffer_mod.EditBuffer, mem_id: u8) void {
1210
+ edit_buffer.replaceTextFromMemId(mem_id) catch {};
1211
+ }
1212
+
1213
+ export fn editBufferGetText(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: [*]u8, maxLen: usize) usize {
1214
+ const outBuffer = outPtr[0..maxLen];
1215
+ return edit_buffer.getText(outBuffer);
1216
+ }
1217
+
1218
+ export fn editBufferInsertChar(edit_buffer: *edit_buffer_mod.EditBuffer, charPtr: [*]const u8, charLen: usize) void {
1219
+ const text = charPtr[0..charLen];
1220
+ edit_buffer.insertText(text) catch {};
1221
+ }
1222
+
1223
+ export fn editBufferNewLine(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1224
+ edit_buffer.insertText("\n") catch {};
1225
+ }
1226
+
1227
+ export fn editBufferDeleteLine(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1228
+ edit_buffer.deleteLine() catch {};
1229
+ }
1230
+
1231
+ export fn editBufferGotoLine(edit_buffer: *edit_buffer_mod.EditBuffer, line: u32) void {
1232
+ edit_buffer.gotoLine(line) catch {};
1233
+ }
1234
+
1235
+ export fn editBufferGetCursorPosition(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: *ExternalLogicalCursor) void {
1236
+ const pos = edit_buffer.getCursorPosition();
1237
+ outPtr.* = .{
1238
+ .row = pos.line,
1239
+ .col = pos.visual_col,
1240
+ .offset = pos.offset,
1241
+ };
1242
+ }
1243
+
1244
+ export fn editBufferGetId(edit_buffer: *edit_buffer_mod.EditBuffer) u16 {
1245
+ return edit_buffer.getId();
1246
+ }
1247
+
1248
+ export fn editBufferDebugLogRope(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1249
+ edit_buffer.debugLogRope();
1250
+ }
1251
+
1252
+ export fn editBufferUndo(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: [*]u8, maxLen: usize) usize {
1253
+ const prev_meta = edit_buffer.undo() catch return 0;
1254
+ const copyLen = @min(prev_meta.len, maxLen);
1255
+ @memcpy(outPtr[0..copyLen], prev_meta[0..copyLen]);
1256
+ return copyLen;
1257
+ }
1258
+
1259
+ export fn editBufferRedo(edit_buffer: *edit_buffer_mod.EditBuffer, outPtr: [*]u8, maxLen: usize) usize {
1260
+ const next_meta = edit_buffer.redo() catch return 0;
1261
+ const copyLen = @min(next_meta.len, maxLen);
1262
+ @memcpy(outPtr[0..copyLen], next_meta[0..copyLen]);
1263
+ return copyLen;
1264
+ }
1265
+
1266
+ export fn editBufferCanUndo(edit_buffer: *edit_buffer_mod.EditBuffer) bool {
1267
+ return edit_buffer.canUndo();
1268
+ }
1269
+
1270
+ export fn editBufferCanRedo(edit_buffer: *edit_buffer_mod.EditBuffer) bool {
1271
+ return edit_buffer.canRedo();
1272
+ }
1273
+
1274
+ export fn editBufferClearHistory(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1275
+ edit_buffer.clearHistory();
1276
+ }
1277
+
1278
+ export fn editBufferClear(edit_buffer: *edit_buffer_mod.EditBuffer) void {
1279
+ edit_buffer.clear() catch {};
1280
+ }
1281
+
1282
+ // ===== EditorView Exports =====
1283
+
1284
+ export fn createEditorView(edit_buffer: *edit_buffer_mod.EditBuffer, viewport_width: u32, viewport_height: u32) ?*editor_view.EditorView {
1285
+ return editor_view.EditorView.init(globalArena, edit_buffer, viewport_width, viewport_height) catch null;
1286
+ }
1287
+
1288
+ export fn destroyEditorView(view: *editor_view.EditorView) void {
1289
+ view.deinit();
1290
+ }
1291
+
1292
+ export fn editorViewSetViewport(view: *editor_view.EditorView, x: u32, y: u32, width: u32, height: u32, moveCursor: bool) void {
1293
+ view.setViewport(text_buffer_view.Viewport{ .x = x, .y = y, .width = width, .height = height }, moveCursor);
1294
+ }
1295
+
1296
+ export fn editorViewClearViewport(view: *editor_view.EditorView) void {
1297
+ view.setViewport(null, false);
1298
+ }
1299
+
1300
+ export fn editorViewGetViewport(view: *editor_view.EditorView, outX: *u32, outY: *u32, outWidth: *u32, outHeight: *u32) bool {
1301
+ if (view.getViewport()) |vp| {
1302
+ outX.* = vp.x;
1303
+ outY.* = vp.y;
1304
+ outWidth.* = vp.width;
1305
+ outHeight.* = vp.height;
1306
+ return true;
1307
+ }
1308
+ return false;
1309
+ }
1310
+
1311
+ export fn editorViewSetScrollMargin(view: *editor_view.EditorView, margin: f32) void {
1312
+ view.setScrollMargin(margin);
1313
+ }
1314
+
1315
+ export fn editorViewGetVirtualLineCount(view: *editor_view.EditorView) u32 {
1316
+ // TODO: There is a getter for that directly, no?
1317
+ return @intCast(view.getVirtualLines().len);
1318
+ }
1319
+
1320
+ export fn editorViewGetTotalVirtualLineCount(view: *editor_view.EditorView) u32 {
1321
+ return view.getTotalVirtualLineCount();
1322
+ }
1323
+
1324
+ export fn editorViewGetLineInfoDirect(view: *editor_view.EditorView, outPtr: *ExternalLineInfo) void {
1325
+ const line_info = view.getCachedLineInfo();
1326
+ outPtr.* = .{
1327
+ .start_cols_ptr = line_info.line_start_cols.ptr,
1328
+ .start_cols_len = @intCast(line_info.line_start_cols.len),
1329
+ .width_cols_ptr = line_info.line_width_cols.ptr,
1330
+ .width_cols_len = @intCast(line_info.line_width_cols.len),
1331
+ .sources_ptr = line_info.line_sources.ptr,
1332
+ .sources_len = @intCast(line_info.line_sources.len),
1333
+ .wraps_ptr = line_info.line_wraps.ptr,
1334
+ .wraps_len = @intCast(line_info.line_wraps.len),
1335
+ .width_cols_max = line_info.line_width_cols_max,
1336
+ };
1337
+ }
1338
+
1339
+ export fn editorViewGetTextBufferView(view: *editor_view.EditorView) *text_buffer_view.UnifiedTextBufferView {
1340
+ return view.getTextBufferView();
1341
+ }
1342
+
1343
+ export fn editorViewGetLogicalLineInfoDirect(view: *editor_view.EditorView, outPtr: *ExternalLineInfo) void {
1344
+ const line_info = view.getLogicalLineInfo();
1345
+ outPtr.* = .{
1346
+ .start_cols_ptr = line_info.line_start_cols.ptr,
1347
+ .start_cols_len = @intCast(line_info.line_start_cols.len),
1348
+ .width_cols_ptr = line_info.line_width_cols.ptr,
1349
+ .width_cols_len = @intCast(line_info.line_width_cols.len),
1350
+ .sources_ptr = line_info.line_sources.ptr,
1351
+ .sources_len = @intCast(line_info.line_sources.len),
1352
+ .wraps_ptr = line_info.line_wraps.ptr,
1353
+ .wraps_len = @intCast(line_info.line_wraps.len),
1354
+ .width_cols_max = line_info.line_width_cols_max,
1355
+ };
1356
+ }
1357
+
1358
+ export fn editorViewSetViewportSize(view: *editor_view.EditorView, width: u32, height: u32) void {
1359
+ view.setViewportSize(width, height);
1360
+ }
1361
+
1362
+ export fn editorViewSetWrapMode(view: *editor_view.EditorView, mode: u8) void {
1363
+ const wrapMode: text_buffer.WrapMode = switch (mode) {
1364
+ 0 => .none,
1365
+ 1 => .char,
1366
+ 2 => .word,
1367
+ else => .none,
1368
+ };
1369
+ view.setWrapMode(wrapMode);
1370
+ }
1371
+
1372
+ // EditorView selection methods - delegate to TextBufferView
1373
+ export fn editorViewSetSelection(view: *editor_view.EditorView, start: u32, end: u32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) void {
1374
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
1375
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
1376
+ view.text_buffer_view.setSelection(start, end, bg, fg);
1377
+ }
1378
+
1379
+ export fn editorViewResetSelection(view: *editor_view.EditorView) void {
1380
+ view.text_buffer_view.resetSelection();
1381
+ }
1382
+
1383
+ export fn editorViewGetSelection(view: *editor_view.EditorView) u64 {
1384
+ return view.text_buffer_view.packSelectionInfo();
1385
+ }
1386
+
1387
+ export fn editorViewSetLocalSelection(view: *editor_view.EditorView, anchorX: i32, anchorY: i32, focusX: i32, focusY: i32, bgColor: ?[*]const f32, fgColor: ?[*]const f32, updateCursor: bool, followCursor: bool) bool {
1388
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
1389
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
1390
+ view.setSelectionFollowCursor(followCursor);
1391
+ return view.setLocalSelection(anchorX, anchorY, focusX, focusY, bg, fg, updateCursor);
1392
+ }
1393
+
1394
+ export fn editorViewUpdateSelection(view: *editor_view.EditorView, end: u32, bgColor: ?[*]const f32, fgColor: ?[*]const f32) void {
1395
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
1396
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
1397
+ view.updateSelection(end, bg, fg);
1398
+ }
1399
+
1400
+ export fn editorViewUpdateLocalSelection(view: *editor_view.EditorView, anchorX: i32, anchorY: i32, focusX: i32, focusY: i32, bgColor: ?[*]const f32, fgColor: ?[*]const f32, updateCursor: bool, followCursor: bool) bool {
1401
+ const bg = if (bgColor) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
1402
+ const fg = if (fgColor) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
1403
+ view.setSelectionFollowCursor(followCursor);
1404
+ return view.updateLocalSelection(anchorX, anchorY, focusX, focusY, bg, fg, updateCursor);
1405
+ }
1406
+
1407
+ export fn editorViewResetLocalSelection(view: *editor_view.EditorView) void {
1408
+ view.setSelectionFollowCursor(false);
1409
+ view.text_buffer_view.resetLocalSelection();
1410
+ }
1411
+
1412
+ export fn editorViewGetSelectedTextBytes(view: *editor_view.EditorView, outPtr: [*]u8, maxLen: usize) usize {
1413
+ const outBuffer = outPtr[0..maxLen];
1414
+ return view.text_buffer_view.getSelectedTextIntoBuffer(outBuffer);
1415
+ }
1416
+
1417
+ // EditorView cursor and text methods
1418
+ export fn editorViewGetCursor(view: *editor_view.EditorView, outRow: *u32, outCol: *u32) void {
1419
+ const cursor = view.getPrimaryCursor();
1420
+ outRow.* = cursor.row;
1421
+ outCol.* = cursor.col;
1422
+ }
1423
+
1424
+ export fn editorViewGetText(view: *editor_view.EditorView, outPtr: [*]u8, maxLen: usize) usize {
1425
+ const outBuffer = outPtr[0..maxLen];
1426
+ return view.getText(outBuffer);
1427
+ }
1428
+
1429
+ // ===== EditorView VisualCursor Exports =====
1430
+
1431
+ export fn editorViewGetVisualCursor(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1432
+ const vcursor = view.getVisualCursor();
1433
+ outPtr.* = .{
1434
+ .visual_row = vcursor.visual_row,
1435
+ .visual_col = vcursor.visual_col,
1436
+ .logical_row = vcursor.logical_row,
1437
+ .logical_col = vcursor.logical_col,
1438
+ .offset = vcursor.offset,
1439
+ };
1440
+ }
1441
+
1442
+ export fn editorViewMoveUpVisual(view: *editor_view.EditorView) void {
1443
+ view.moveUpVisual();
1444
+ }
1445
+
1446
+ export fn editorViewMoveDownVisual(view: *editor_view.EditorView) void {
1447
+ view.moveDownVisual();
1448
+ }
1449
+
1450
+ export fn editorViewDeleteSelectedText(view: *editor_view.EditorView) void {
1451
+ view.deleteSelectedText() catch {};
1452
+ }
1453
+
1454
+ export fn editorViewSetCursorByOffset(view: *editor_view.EditorView, offset: u32) void {
1455
+ view.setCursorByOffset(offset) catch {};
1456
+ }
1457
+
1458
+ export fn editorViewGetNextWordBoundary(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1459
+ const vcursor = view.getNextWordBoundary();
1460
+ outPtr.* = .{
1461
+ .visual_row = vcursor.visual_row,
1462
+ .visual_col = vcursor.visual_col,
1463
+ .logical_row = vcursor.logical_row,
1464
+ .logical_col = vcursor.logical_col,
1465
+ .offset = vcursor.offset,
1466
+ };
1467
+ }
1468
+
1469
+ export fn editorViewGetPrevWordBoundary(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1470
+ const vcursor = view.getPrevWordBoundary();
1471
+ outPtr.* = .{
1472
+ .visual_row = vcursor.visual_row,
1473
+ .visual_col = vcursor.visual_col,
1474
+ .logical_row = vcursor.logical_row,
1475
+ .logical_col = vcursor.logical_col,
1476
+ .offset = vcursor.offset,
1477
+ };
1478
+ }
1479
+
1480
+ export fn editorViewGetEOL(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1481
+ const vcursor = view.getEOL();
1482
+ outPtr.* = .{
1483
+ .visual_row = vcursor.visual_row,
1484
+ .visual_col = vcursor.visual_col,
1485
+ .logical_row = vcursor.logical_row,
1486
+ .logical_col = vcursor.logical_col,
1487
+ .offset = vcursor.offset,
1488
+ };
1489
+ }
1490
+
1491
+ export fn editorViewGetVisualSOL(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1492
+ const vcursor = view.getVisualSOL();
1493
+ outPtr.* = .{
1494
+ .visual_row = vcursor.visual_row,
1495
+ .visual_col = vcursor.visual_col,
1496
+ .logical_row = vcursor.logical_row,
1497
+ .logical_col = vcursor.logical_col,
1498
+ .offset = vcursor.offset,
1499
+ };
1500
+ }
1501
+
1502
+ export fn editorViewGetVisualEOL(view: *editor_view.EditorView, outPtr: *ExternalVisualCursor) void {
1503
+ const vcursor = view.getVisualEOL();
1504
+ outPtr.* = .{
1505
+ .visual_row = vcursor.visual_row,
1506
+ .visual_col = vcursor.visual_col,
1507
+ .logical_row = vcursor.logical_row,
1508
+ .logical_col = vcursor.logical_col,
1509
+ .offset = vcursor.offset,
1510
+ };
1511
+ }
1512
+
1513
+ export fn editorViewSetPlaceholderStyledText(
1514
+ view: *editor_view.EditorView,
1515
+ chunksPtr: [*]const text_buffer.StyledChunk,
1516
+ chunkCount: usize,
1517
+ ) void {
1518
+ if (chunkCount == 0) {
1519
+ view.setPlaceholderStyledText(&[_]text_buffer.StyledChunk{}) catch {};
1520
+ return;
1521
+ }
1522
+ const chunks = chunksPtr[0..chunkCount];
1523
+ view.setPlaceholderStyledText(chunks) catch {};
1524
+ }
1525
+
1526
+ export fn editorViewSetTabIndicator(view: *editor_view.EditorView, indicator: u32) void {
1527
+ view.setTabIndicator(indicator);
1528
+ }
1529
+
1530
+ export fn editorViewSetTabIndicatorColor(view: *editor_view.EditorView, color: [*]const f32) void {
1531
+ view.setTabIndicatorColor(utils.f32PtrToRGBA(color));
1532
+ }
1533
+
1534
+ export fn bufferDrawEditorView(
1535
+ bufferPtr: *buffer.OptimizedBuffer,
1536
+ viewPtr: *editor_view.EditorView,
1537
+ x: i32,
1538
+ y: i32,
1539
+ ) void {
1540
+ bufferPtr.drawEditorView(viewPtr, x, y) catch {};
1541
+ }
1542
+
1543
+ export fn bufferDrawTextBufferView(
1544
+ bufferPtr: *buffer.OptimizedBuffer,
1545
+ viewPtr: *text_buffer_view.UnifiedTextBufferView,
1546
+ x: i32,
1547
+ y: i32,
1548
+ ) void {
1549
+ bufferPtr.drawTextBuffer(viewPtr, x, y) catch {};
1550
+ }
1551
+
1552
+ pub const ExternalHighlight = extern struct {
1553
+ start: u32,
1554
+ end: u32,
1555
+ style_id: u32,
1556
+ priority: u8,
1557
+ hl_ref: u16,
1558
+ };
1559
+
1560
+ pub const ExternalLogicalCursor = extern struct {
1561
+ row: u32,
1562
+ col: u32,
1563
+ offset: u32,
1564
+ };
1565
+
1566
+ pub const ExternalVisualCursor = extern struct {
1567
+ visual_row: u32,
1568
+ visual_col: u32,
1569
+ logical_row: u32,
1570
+ logical_col: u32,
1571
+ offset: u32,
1572
+ };
1573
+
1574
+ pub const ExternalLineInfo = extern struct {
1575
+ start_cols_ptr: [*]const u32,
1576
+ start_cols_len: u32,
1577
+ width_cols_ptr: [*]const u32,
1578
+ width_cols_len: u32,
1579
+ sources_ptr: [*]const u32,
1580
+ sources_len: u32,
1581
+ wraps_ptr: [*]const u32,
1582
+ wraps_len: u32,
1583
+ width_cols_max: u32,
1584
+ };
1585
+
1586
+ export fn textBufferAddHighlightByCharRange(
1587
+ tb: *text_buffer.UnifiedTextBuffer,
1588
+ hl_ptr: [*]const ExternalHighlight,
1589
+ ) void {
1590
+ const hl = hl_ptr[0];
1591
+ // For char-range highlights, start/end in the struct are unused (passed as char_start/char_end)
1592
+ tb.addHighlightByCharRange(hl.start, hl.end, hl.style_id, hl.priority, hl.hl_ref) catch {};
1593
+ }
1594
+
1595
+ export fn textBufferAddHighlight(
1596
+ tb: *text_buffer.UnifiedTextBuffer,
1597
+ line_idx: u32,
1598
+ hl_ptr: [*]const ExternalHighlight,
1599
+ ) void {
1600
+ const hl = hl_ptr[0];
1601
+ // For line-based highlights, start/end are column offsets
1602
+ tb.addHighlight(line_idx, hl.start, hl.end, hl.style_id, hl.priority, hl.hl_ref) catch {};
1603
+ }
1604
+
1605
+ export fn textBufferRemoveHighlightsByRef(tb: *text_buffer.UnifiedTextBuffer, hl_ref: u16) void {
1606
+ tb.removeHighlightsByRef(hl_ref);
1607
+ }
1608
+
1609
+ export fn textBufferClearLineHighlights(tb: *text_buffer.UnifiedTextBuffer, line_idx: u32) void {
1610
+ tb.clearLineHighlights(line_idx);
1611
+ }
1612
+
1613
+ export fn textBufferClearAllHighlights(tb: *text_buffer.UnifiedTextBuffer) void {
1614
+ tb.clearAllHighlights();
1615
+ }
1616
+
1617
+ export fn textBufferSetSyntaxStyle(tb: *text_buffer.UnifiedTextBuffer, style: ?*syntax_style.SyntaxStyle) void {
1618
+ tb.setSyntaxStyle(style);
1619
+ }
1620
+
1621
+ export fn textBufferGetLineHighlightsPtr(
1622
+ tb: *text_buffer.UnifiedTextBuffer,
1623
+ line_idx: u32,
1624
+ out_count: *usize,
1625
+ ) ?[*]const ExternalHighlight {
1626
+ const highs = tb.getLineHighlightsSlice(@intCast(line_idx));
1627
+
1628
+ if (highs.len == 0) {
1629
+ out_count.* = 0;
1630
+ return null;
1631
+ }
1632
+
1633
+ var slice = globalAllocator.alloc(ExternalHighlight, highs.len) catch return null;
1634
+
1635
+ for (highs, 0..) |hl, i| {
1636
+ slice[i] = .{
1637
+ .start = hl.col_start,
1638
+ .end = hl.col_end,
1639
+ .style_id = hl.style_id,
1640
+ .priority = hl.priority,
1641
+ .hl_ref = hl.hl_ref,
1642
+ };
1643
+ }
1644
+
1645
+ out_count.* = highs.len;
1646
+ return slice.ptr;
1647
+ }
1648
+
1649
+ export fn textBufferFreeLineHighlights(ptr: [*]const ExternalHighlight, count: usize) void {
1650
+ globalAllocator.free(@constCast(ptr)[0..count]);
1651
+ }
1652
+
1653
+ export fn textBufferGetHighlightCount(tb: *text_buffer.UnifiedTextBuffer) u32 {
1654
+ return tb.getHighlightCount();
1655
+ }
1656
+
1657
+ export fn textBufferGetTextRange(tb: *text_buffer.UnifiedTextBuffer, start_offset: u32, end_offset: u32, outPtr: [*]u8, maxLen: usize) usize {
1658
+ const outBuffer = outPtr[0..maxLen];
1659
+ return tb.getTextRange(start_offset, end_offset, outBuffer);
1660
+ }
1661
+
1662
+ export fn textBufferGetTextRangeByCoords(tb: *text_buffer.UnifiedTextBuffer, start_row: u32, start_col: u32, end_row: u32, end_col: u32, outPtr: [*]u8, maxLen: usize) usize {
1663
+ const outBuffer = outPtr[0..maxLen];
1664
+ return tb.getTextRangeByCoords(start_row, start_col, end_row, end_col, outBuffer);
1665
+ }
1666
+
1667
+ // SyntaxStyle functions
1668
+ export fn createSyntaxStyle() ?*syntax_style.SyntaxStyle {
1669
+ return syntax_style.SyntaxStyle.init(globalAllocator) catch |err| {
1670
+ logger.err("Failed to create SyntaxStyle: {}", .{err});
1671
+ return null;
1672
+ };
1673
+ }
1674
+
1675
+ export fn destroySyntaxStyle(style: *syntax_style.SyntaxStyle) void {
1676
+ style.deinit();
1677
+ }
1678
+
1679
+ export fn syntaxStyleRegister(style: *syntax_style.SyntaxStyle, namePtr: [*]const u8, nameLen: usize, fg: ?[*]const f32, bg: ?[*]const f32, attributes: u32) u32 {
1680
+ const name = namePtr[0..nameLen];
1681
+ const fgColor = if (fg) |fgPtr| utils.f32PtrToRGBA(fgPtr) else null;
1682
+ const bgColor = if (bg) |bgPtr| utils.f32PtrToRGBA(bgPtr) else null;
1683
+ return style.registerStyle(name, fgColor, bgColor, attributes) catch 0;
1684
+ }
1685
+
1686
+ export fn syntaxStyleResolveByName(style: *syntax_style.SyntaxStyle, namePtr: [*]const u8, nameLen: usize) u32 {
1687
+ const name = namePtr[0..nameLen];
1688
+ return style.resolveByName(name) orelse 0;
1689
+ }
1690
+
1691
+ export fn syntaxStyleGetStyleCount(style: *syntax_style.SyntaxStyle) usize {
1692
+ return style.getStyleCount();
1693
+ }
1694
+
1695
+ // Unicode encoding API
1696
+
1697
+ pub const EncodedChar = extern struct {
1698
+ width: u8,
1699
+ char: u32,
1700
+ };
1701
+
1702
+ export fn encodeUnicode(
1703
+ textPtr: [*]const u8,
1704
+ textLen: usize,
1705
+ outPtr: *[*]EncodedChar,
1706
+ outLenPtr: *usize,
1707
+ widthMethod: u8,
1708
+ ) bool {
1709
+ const text = textPtr[0..textLen];
1710
+ const pool = gp.initGlobalPool(globalArena);
1711
+ const wMethod: utf8.WidthMethod = if (widthMethod == 0) .wcwidth else .unicode;
1712
+
1713
+ // Check if ASCII only for optimization
1714
+ const is_ascii_only = utf8.isAsciiOnly(text);
1715
+
1716
+ // Find grapheme info
1717
+ var grapheme_list: std.ArrayListUnmanaged(utf8.GraphemeInfo) = .{};
1718
+ defer grapheme_list.deinit(globalAllocator);
1719
+
1720
+ const tab_width: u8 = 2;
1721
+ utf8.findGraphemeInfo(text, tab_width, is_ascii_only, wMethod, globalAllocator, &grapheme_list) catch return false;
1722
+ const specials = grapheme_list.items;
1723
+
1724
+ // Allocate output array
1725
+ const estimated_count = if (is_ascii_only) text.len else text.len * 2;
1726
+ var result = globalAllocator.alloc(EncodedChar, estimated_count) catch return false;
1727
+ var result_idx: usize = 0;
1728
+ var success = false;
1729
+ var pending_gid: ?u32 = null; // Track grapheme allocated but not yet stored in result
1730
+
1731
+ // Clean up result array and any allocated grapheme IDs on failure
1732
+ defer {
1733
+ if (!success) {
1734
+ // Clean up pending grapheme that wasn't stored yet
1735
+ if (pending_gid) |gid| {
1736
+ // Try decref first (works if incref was called, refcount >= 1)
1737
+ // If that fails (refcount was 0), use freeUnreferenced
1738
+ pool.decref(gid) catch {
1739
+ pool.freeUnreferenced(gid) catch {};
1740
+ };
1741
+ }
1742
+ // Decref any grapheme IDs we allocated before the failure
1743
+ for (result[0..result_idx]) |encoded_char| {
1744
+ if (gp.isGraphemeChar(encoded_char.char)) {
1745
+ const gid = gp.graphemeIdFromChar(encoded_char.char);
1746
+ pool.decref(gid) catch {};
1747
+ }
1748
+ }
1749
+ globalAllocator.free(result);
1750
+ }
1751
+ }
1752
+
1753
+ var byte_offset: u32 = 0;
1754
+ var col: u32 = 0;
1755
+ var special_idx: usize = 0;
1756
+
1757
+ while (byte_offset < text.len) {
1758
+ const at_special = special_idx < specials.len and specials[special_idx].col_offset == col;
1759
+
1760
+ var grapheme_bytes: []const u8 = undefined;
1761
+ var g_width: u8 = undefined;
1762
+
1763
+ if (at_special) {
1764
+ const g = specials[special_idx];
1765
+ grapheme_bytes = text[g.byte_offset .. g.byte_offset + g.byte_len];
1766
+ g_width = g.width;
1767
+ byte_offset = g.byte_offset + g.byte_len;
1768
+ special_idx += 1;
1769
+ } else {
1770
+ if (byte_offset >= text.len) break;
1771
+ grapheme_bytes = text[byte_offset .. byte_offset + 1];
1772
+ g_width = 1;
1773
+ byte_offset += 1;
1774
+ }
1775
+
1776
+ const cell_width = utf8.getWidthAt(text, if (at_special) specials[special_idx - 1].byte_offset else byte_offset - 1, tab_width, wMethod);
1777
+ if (cell_width == 0) {
1778
+ col += g_width;
1779
+ continue;
1780
+ }
1781
+
1782
+ // Encode the character
1783
+ var encoded_char: u32 = 0;
1784
+ if (grapheme_bytes.len == 1 and cell_width == 1 and grapheme_bytes[0] >= 32) {
1785
+ // Simple ASCII character
1786
+ encoded_char = @as(u32, grapheme_bytes[0]);
1787
+ } else {
1788
+ // Multi-byte or special character - allocate in pool
1789
+ const gid = pool.alloc(grapheme_bytes) catch return false;
1790
+ pending_gid = gid; // Track until stored in result
1791
+ encoded_char = gp.packGraphemeStart(gid & gp.GRAPHEME_ID_MASK, cell_width);
1792
+
1793
+ // Incref since we're handing this off to the caller
1794
+ // Note: incref can only fail if gid is invalid, which shouldn't happen
1795
+ // for a freshly allocated gid. If it does fail, the slot leaks but
1796
+ // this is an edge case that indicates a bug elsewhere.
1797
+ pool.incref(gid) catch return false;
1798
+ }
1799
+
1800
+ // Ensure we have space
1801
+ if (result_idx >= result.len) {
1802
+ const new_len = result.len * 2;
1803
+ result = globalAllocator.realloc(result, new_len) catch return false;
1804
+ }
1805
+
1806
+ result[result_idx] = EncodedChar{
1807
+ .width = @intCast(cell_width),
1808
+ .char = encoded_char,
1809
+ };
1810
+ pending_gid = null; // Successfully stored, no longer pending
1811
+ result_idx += 1;
1812
+ col += g_width;
1813
+ }
1814
+
1815
+ // Trim to actual size
1816
+ result = globalAllocator.realloc(result, result_idx) catch result;
1817
+
1818
+ outPtr.* = result.ptr;
1819
+ outLenPtr.* = result_idx;
1820
+ success = true;
1821
+ return true;
1822
+ }
1823
+
1824
+ export fn freeUnicode(charsPtr: [*]const EncodedChar, charsLen: usize) void {
1825
+ const chars = charsPtr[0..charsLen];
1826
+ const pool = gp.initGlobalPool(globalArena);
1827
+
1828
+ for (chars) |encoded_char| {
1829
+ const char = encoded_char.char;
1830
+
1831
+ // Check if this is a packed grapheme
1832
+ if (gp.isGraphemeChar(char)) {
1833
+ const gid = gp.graphemeIdFromChar(char);
1834
+ pool.decref(gid) catch {};
1835
+ }
1836
+ }
1837
+
1838
+ // Free the array itself
1839
+ globalAllocator.free(chars);
1840
+ }
1841
+
1842
+ export fn bufferDrawChar(
1843
+ bufferPtr: *buffer.OptimizedBuffer,
1844
+ char: u32,
1845
+ x: u32,
1846
+ y: u32,
1847
+ fg: [*]const f32,
1848
+ bg: [*]const f32,
1849
+ attributes: u32,
1850
+ ) void {
1851
+ const rgbaFg = utils.f32PtrToRGBA(fg);
1852
+ const rgbaBg = utils.f32PtrToRGBA(bg);
1853
+ bufferPtr.drawChar(char, x, y, rgbaFg, rgbaBg, attributes) catch {};
1854
+ }