@fairyhunter13/opentui-core 0.1.112 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +63 -51
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-8fks7yv1.js +0 -411
  472. package/index-8fks7yv1.js.map +0 -10
  473. package/index-egy5e2rs.js +0 -12267
  474. package/index-egy5e2rs.js.map +0 -42
  475. package/index-tse8gzh0.js +0 -20614
  476. package/index-tse8gzh0.js.map +0 -67
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,1393 @@
1
+ const std = @import("std");
2
+ const Allocator = std.mem.Allocator;
3
+ const ansi = @import("ansi.zig");
4
+ const buf = @import("buffer.zig");
5
+ const gp = @import("grapheme.zig");
6
+ const link = @import("link.zig");
7
+ const Terminal = @import("terminal.zig");
8
+ const logger = @import("logger.zig");
9
+
10
+ pub const RGBA = ansi.RGBA;
11
+ pub const OptimizedBuffer = buf.OptimizedBuffer;
12
+ pub const TextAttributes = ansi.TextAttributes;
13
+ pub const CursorStyle = Terminal.CursorStyle;
14
+
15
+ const CLEAR_CHAR = '\u{0a00}';
16
+ const MAX_STAT_SAMPLES = 30;
17
+ const STAT_SAMPLE_CAPACITY = 30;
18
+
19
+ const COLOR_EPSILON_DEFAULT: f32 = 0.00001;
20
+ const OUTPUT_BUFFER_SIZE = 1024 * 1024 * 2; // 2MB
21
+
22
+ pub const RendererError = error{
23
+ OutOfMemory,
24
+ InvalidDimensions,
25
+ ThreadingFailed,
26
+ WriteFailed,
27
+ };
28
+
29
+ fn rgbaComponentToU8(component: f32) u8 {
30
+ if (!std.math.isFinite(component)) return 0;
31
+
32
+ const clamped = std.math.clamp(component, 0.0, 1.0);
33
+ return @intFromFloat(@round(clamped * 255.0));
34
+ }
35
+
36
+ pub const DebugOverlayCorner = enum {
37
+ topLeft,
38
+ topRight,
39
+ bottomLeft,
40
+ bottomRight,
41
+ };
42
+
43
+ pub const CliRenderer = struct {
44
+ width: u32,
45
+ height: u32,
46
+ currentRenderBuffer: *OptimizedBuffer,
47
+ nextRenderBuffer: *OptimizedBuffer,
48
+ pool: *gp.GraphemePool,
49
+ backgroundColor: RGBA,
50
+ renderOffset: u32,
51
+ terminal: Terminal,
52
+ testing: bool = false,
53
+ useAlternateScreen: bool = true,
54
+ terminalSetup: bool = false,
55
+
56
+ renderStats: struct {
57
+ lastFrameTime: f64,
58
+ averageFrameTime: f64,
59
+ frameCount: u64,
60
+ fps: u32,
61
+ cellsUpdated: u32,
62
+ renderTime: ?f64,
63
+ overallFrameTime: ?f64,
64
+ bufferResetTime: ?f64,
65
+ stdoutWriteTime: ?f64,
66
+ heapUsed: u32,
67
+ heapTotal: u32,
68
+ arrayBuffers: u32,
69
+ frameCallbackTime: ?f64,
70
+ },
71
+ statSamples: struct {
72
+ lastFrameTime: std.ArrayListUnmanaged(f64),
73
+ renderTime: std.ArrayListUnmanaged(f64),
74
+ overallFrameTime: std.ArrayListUnmanaged(f64),
75
+ bufferResetTime: std.ArrayListUnmanaged(f64),
76
+ stdoutWriteTime: std.ArrayListUnmanaged(f64),
77
+ cellsUpdated: std.ArrayListUnmanaged(u32),
78
+ frameCallbackTime: std.ArrayListUnmanaged(f64),
79
+ },
80
+ lastRenderTime: i64,
81
+ allocator: Allocator,
82
+ renderThread: ?std.Thread = null,
83
+ stdoutBuffer: [4096]u8,
84
+ writeOutBuf: [1024]u8 = undefined,
85
+ debugOverlay: struct {
86
+ enabled: bool,
87
+ corner: DebugOverlayCorner,
88
+ } = .{
89
+ .enabled = false,
90
+ .corner = .bottomRight,
91
+ },
92
+ // Threading
93
+ useThread: bool = false,
94
+ renderMutex: std.Thread.Mutex = .{},
95
+ renderCondition: std.Thread.Condition = .{},
96
+ renderRequested: bool = false,
97
+ shouldTerminate: bool = false,
98
+ renderInProgress: bool = false,
99
+ currentOutputBuffer: []u8 = &[_]u8{},
100
+ currentOutputLen: usize = 0,
101
+
102
+ // Hit grid for mouse event dispatch.
103
+ //
104
+ // The hit grid is a screen-sized array where each cell stores the renderable ID
105
+ // at that position. Mouse events query checkHit(x, y) to find which element to
106
+ // dispatch to.
107
+ //
108
+ // Double buffering: During render, addToHitGrid writes to nextHitGrid. After
109
+ // render completes, the buffers swap. This keeps hit testing consistent during
110
+ // a frame. Queries see the previous frame's state, not a half-built grid.
111
+ //
112
+ // On-demand sync: When scroll/translate changes between renders, the TypeScript
113
+ // layer can rebuild currentHitGrid directly via addToCurrentHitGridClipped. This
114
+ // updates hover states immediately rather than waiting for the next render.
115
+ //
116
+ // Scissor clipping: The hitScissorStack mirrors overflow:hidden regions. Elements
117
+ // outside their parent's visible area are excluded from hit testing. The stack
118
+ // uses screen coordinates. Buffered renderables need getHitGridScissorRect() to
119
+ // convert from buffer-local (0,0) to their actual screen position.
120
+ currentHitGrid: []u32,
121
+ nextHitGrid: []u32,
122
+ hitGridWidth: u32,
123
+ hitGridHeight: u32,
124
+ hitScissorStack: std.ArrayListUnmanaged(buf.ClipRect),
125
+ hitGridDirty: bool = false,
126
+
127
+ lastCursorStyleTag: ?u8 = null,
128
+ lastCursorBlinking: ?bool = null,
129
+ lastCursorColorRGB: ?[3]u8 = null,
130
+ lastMousePointerStyle: Terminal.MousePointerStyle = .default,
131
+
132
+ // Preallocated output buffer
133
+ var outputBuffer: [OUTPUT_BUFFER_SIZE]u8 = undefined;
134
+ var outputBufferLen: usize = 0;
135
+ var outputBufferB: [OUTPUT_BUFFER_SIZE]u8 = undefined;
136
+ var outputBufferBLen: usize = 0;
137
+ var activeBuffer: enum { A, B } = .A;
138
+
139
+ const OutputBufferWriter = struct {
140
+ pub fn write(_: void, data: []const u8) !usize {
141
+ const bufferLen = if (activeBuffer == .A) &outputBufferLen else &outputBufferBLen;
142
+ const buffer = if (activeBuffer == .A) &outputBuffer else &outputBufferB;
143
+
144
+ if (bufferLen.* + data.len > buffer.len) {
145
+ // TODO: Resize buffer when necessary
146
+ return error.BufferFull;
147
+ }
148
+
149
+ @memcpy(buffer.*[bufferLen.*..][0..data.len], data);
150
+ bufferLen.* += data.len;
151
+
152
+ return data.len;
153
+ }
154
+
155
+ // TODO: std.io.GenericWriter is deprecated, however the "correct" option seems to be much more involved
156
+ // So I have simply used GenericWriter here, and then the proper migration can be done later
157
+ pub fn writer() std.io.GenericWriter(void, error{BufferFull}, write) {
158
+ return .{ .context = {} };
159
+ }
160
+ };
161
+
162
+ pub fn create(allocator: Allocator, width: u32, height: u32, pool: *gp.GraphemePool, testing: bool) !*CliRenderer {
163
+ return createWithOptions(allocator, width, height, pool, testing, false);
164
+ }
165
+
166
+ pub fn createWithOptions(
167
+ allocator: Allocator,
168
+ width: u32,
169
+ height: u32,
170
+ pool: *gp.GraphemePool,
171
+ testing: bool,
172
+ remote: bool,
173
+ ) !*CliRenderer {
174
+ const self = try allocator.create(CliRenderer);
175
+ errdefer allocator.destroy(self);
176
+
177
+ const currentBuffer = try OptimizedBuffer.init(allocator, width, height, .{ .pool = pool, .width_method = .unicode, .id = "current buffer" });
178
+ errdefer currentBuffer.deinit();
179
+ const nextBuffer = try OptimizedBuffer.init(allocator, width, height, .{ .pool = pool, .width_method = .unicode, .id = "next buffer" });
180
+ errdefer nextBuffer.deinit();
181
+
182
+ // stat sample arrays
183
+ var lastFrameTime: std.ArrayListUnmanaged(f64) = .{};
184
+ errdefer lastFrameTime.deinit(allocator);
185
+ var renderTime: std.ArrayListUnmanaged(f64) = .{};
186
+ errdefer renderTime.deinit(allocator);
187
+ var overallFrameTime: std.ArrayListUnmanaged(f64) = .{};
188
+ errdefer overallFrameTime.deinit(allocator);
189
+ var bufferResetTime: std.ArrayListUnmanaged(f64) = .{};
190
+ errdefer bufferResetTime.deinit(allocator);
191
+ var stdoutWriteTime: std.ArrayListUnmanaged(f64) = .{};
192
+ errdefer stdoutWriteTime.deinit(allocator);
193
+ var cellsUpdated: std.ArrayListUnmanaged(u32) = .{};
194
+ errdefer cellsUpdated.deinit(allocator);
195
+ var frameCallbackTimes: std.ArrayListUnmanaged(f64) = .{};
196
+ errdefer frameCallbackTimes.deinit(allocator);
197
+
198
+ try lastFrameTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
199
+ try renderTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
200
+ try overallFrameTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
201
+ try bufferResetTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
202
+ try stdoutWriteTime.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
203
+ try cellsUpdated.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
204
+ try frameCallbackTimes.ensureTotalCapacity(allocator, STAT_SAMPLE_CAPACITY);
205
+
206
+ const hitGridSize = width * height;
207
+ const currentHitGrid = try allocator.alloc(u32, hitGridSize);
208
+ errdefer allocator.free(currentHitGrid);
209
+ const nextHitGrid = try allocator.alloc(u32, hitGridSize);
210
+ errdefer allocator.free(nextHitGrid);
211
+ @memset(currentHitGrid, 0); // Initialize with 0 (no renderable)
212
+ @memset(nextHitGrid, 0);
213
+ const hitScissorStack: std.ArrayListUnmanaged(buf.ClipRect) = .{};
214
+
215
+ self.* = .{
216
+ .width = width,
217
+ .height = height,
218
+ .currentRenderBuffer = currentBuffer,
219
+ .nextRenderBuffer = nextBuffer,
220
+ .pool = pool,
221
+ .backgroundColor = .{ 0.0, 0.0, 0.0, 0.0 },
222
+ .renderOffset = 0,
223
+ .terminal = Terminal.init(.{ .remote = remote }),
224
+ .testing = testing,
225
+ .lastCursorStyleTag = null,
226
+ .lastCursorBlinking = null,
227
+ .lastCursorColorRGB = null,
228
+
229
+ .renderStats = .{
230
+ .lastFrameTime = 0,
231
+ .averageFrameTime = 0,
232
+ .frameCount = 0,
233
+ .fps = 0,
234
+ .cellsUpdated = 0,
235
+ .renderTime = null,
236
+ .overallFrameTime = null,
237
+ .bufferResetTime = null,
238
+ .stdoutWriteTime = null,
239
+ .heapUsed = 0,
240
+ .heapTotal = 0,
241
+ .arrayBuffers = 0,
242
+ .frameCallbackTime = null,
243
+ },
244
+ .statSamples = .{
245
+ .lastFrameTime = lastFrameTime,
246
+ .renderTime = renderTime,
247
+ .overallFrameTime = overallFrameTime,
248
+ .bufferResetTime = bufferResetTime,
249
+ .stdoutWriteTime = stdoutWriteTime,
250
+ .cellsUpdated = cellsUpdated,
251
+ .frameCallbackTime = frameCallbackTimes,
252
+ },
253
+ .lastRenderTime = std.time.microTimestamp(),
254
+ .allocator = allocator,
255
+ .stdoutBuffer = undefined,
256
+ .currentHitGrid = currentHitGrid,
257
+ .nextHitGrid = nextHitGrid,
258
+ .hitGridWidth = width,
259
+ .hitGridHeight = height,
260
+ .hitScissorStack = hitScissorStack,
261
+ };
262
+
263
+ nextBuffer.setBlendBackdropColor(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], 1.0 });
264
+
265
+ try currentBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, CLEAR_CHAR);
266
+ try nextBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null);
267
+
268
+ return self;
269
+ }
270
+
271
+ pub fn destroy(self: *CliRenderer) void {
272
+ self.renderMutex.lock();
273
+ while (self.renderInProgress) {
274
+ self.renderCondition.wait(&self.renderMutex);
275
+ }
276
+
277
+ self.shouldTerminate = true;
278
+ self.renderRequested = true;
279
+ self.renderCondition.signal();
280
+ self.renderMutex.unlock();
281
+
282
+ if (self.renderThread) |thread| {
283
+ thread.join();
284
+ }
285
+
286
+ self.performShutdownSequence();
287
+ self.terminal.deinit();
288
+
289
+ self.currentRenderBuffer.deinit();
290
+ self.nextRenderBuffer.deinit();
291
+
292
+ // Free stat sample arrays
293
+ self.statSamples.lastFrameTime.deinit(self.allocator);
294
+ self.statSamples.renderTime.deinit(self.allocator);
295
+ self.statSamples.overallFrameTime.deinit(self.allocator);
296
+ self.statSamples.bufferResetTime.deinit(self.allocator);
297
+ self.statSamples.stdoutWriteTime.deinit(self.allocator);
298
+ self.statSamples.cellsUpdated.deinit(self.allocator);
299
+ self.statSamples.frameCallbackTime.deinit(self.allocator);
300
+
301
+ self.allocator.free(self.currentHitGrid);
302
+ self.allocator.free(self.nextHitGrid);
303
+ self.hitScissorStack.deinit(self.allocator);
304
+
305
+ self.allocator.destroy(self);
306
+ }
307
+
308
+ pub fn setupTerminal(self: *CliRenderer, useAlternateScreen: bool) void {
309
+ self.useAlternateScreen = useAlternateScreen;
310
+ self.terminalSetup = true;
311
+
312
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
313
+ const writer = &stdoutWriter.interface;
314
+
315
+ self.terminal.queryTerminalSend(writer) catch {
316
+ logger.warn("Failed to query terminal capabilities", .{});
317
+ };
318
+ writer.flush() catch {};
319
+
320
+ self.setupTerminalWithoutDetection(useAlternateScreen);
321
+ }
322
+
323
+ fn setupTerminalWithoutDetection(self: *CliRenderer, useAlternateScreen: bool) void {
324
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
325
+ const writer = &stdoutWriter.interface;
326
+
327
+ writer.writeAll(ansi.ANSI.saveCursorState) catch {};
328
+
329
+ if (useAlternateScreen) {
330
+ self.terminal.enterAltScreen(writer) catch {};
331
+ } else {
332
+ ansi.ANSI.makeRoomForRendererOutput(writer, @max(self.height, 1)) catch {};
333
+ }
334
+
335
+ self.terminal.setCursorPosition(1, 1, false);
336
+ const useKitty = self.terminal.opts.kitty_keyboard_flags > 0;
337
+ self.terminal.enableDetectedFeatures(writer, useKitty) catch {};
338
+
339
+ writer.flush() catch {};
340
+ }
341
+
342
+ pub fn suspendRenderer(self: *CliRenderer) void {
343
+ if (!self.terminalSetup) return;
344
+ self.performShutdownSequence();
345
+ }
346
+
347
+ pub fn resumeRenderer(self: *CliRenderer) void {
348
+ if (!self.terminalSetup) return;
349
+ self.setupTerminalWithoutDetection(self.useAlternateScreen);
350
+ }
351
+
352
+ pub fn performShutdownSequence(self: *CliRenderer) void {
353
+ if (!self.terminalSetup) return;
354
+
355
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
356
+ const direct = &stdoutWriter.interface;
357
+ self.terminal.resetState(direct) catch {
358
+ logger.warn("Failed to reset terminal state", .{});
359
+ };
360
+
361
+ if (self.useAlternateScreen) {
362
+ direct.flush() catch {};
363
+ } else if (self.renderOffset == 0) {
364
+ direct.writeAll("\x1b[H\x1b[J") catch {};
365
+ direct.flush() catch {};
366
+ } else if (self.renderOffset > 0) {
367
+ // Currently still handled in typescript
368
+ // const consoleEndLine = self.height - self.renderOffset;
369
+ // ansi.ANSI.moveToOutput(direct, 1, consoleEndLine) catch {};
370
+ }
371
+
372
+ // NOTE: This messes up state after shutdown, but might be necessary for windows?
373
+ // direct.writeAll(ansi.ANSI.restoreCursorState) catch {};
374
+
375
+ direct.writeAll(ansi.ANSI.resetCursorColorFallback) catch {};
376
+ direct.writeAll(ansi.ANSI.resetCursorColor) catch {};
377
+ direct.writeAll(ansi.ANSI.defaultCursorStyle) catch {};
378
+ // Workaround for Ghostty not showing the cursor after shutdown for some reason
379
+ direct.writeAll(ansi.ANSI.showCursor) catch {};
380
+ direct.flush() catch {};
381
+ std.Thread.sleep(10 * std.time.ns_per_ms);
382
+ direct.writeAll(ansi.ANSI.showCursor) catch {};
383
+ direct.flush() catch {};
384
+ std.Thread.sleep(10 * std.time.ns_per_ms);
385
+ }
386
+
387
+ fn addStatSample(self: *CliRenderer, comptime T: type, samples: *std.ArrayListUnmanaged(T), value: T) void {
388
+ samples.append(self.allocator, value) catch return;
389
+
390
+ if (samples.items.len > MAX_STAT_SAMPLES) {
391
+ _ = samples.orderedRemove(0);
392
+ }
393
+ }
394
+
395
+ fn getStatAverage(comptime T: type, samples: *const std.ArrayListUnmanaged(T)) T {
396
+ if (samples.items.len == 0) {
397
+ return 0;
398
+ }
399
+
400
+ var sum: T = 0;
401
+ for (samples.items) |value| {
402
+ sum += value;
403
+ }
404
+
405
+ if (@typeInfo(T) == .float) {
406
+ return sum / @as(T, @floatFromInt(samples.items.len));
407
+ } else {
408
+ return sum / @as(T, @intCast(samples.items.len));
409
+ }
410
+ }
411
+
412
+ pub fn setUseThread(self: *CliRenderer, useThread: bool) void {
413
+ if (self.useThread == useThread) return;
414
+
415
+ if (useThread) {
416
+ if (self.renderThread == null) {
417
+ self.renderThread = std.Thread.spawn(.{}, renderThreadFn, .{self}) catch |err| {
418
+ std.log.warn("Failed to spawn render thread: {}, falling back to non-threaded mode", .{err});
419
+ self.useThread = false;
420
+ return;
421
+ };
422
+ }
423
+ } else {
424
+ if (self.renderThread) |thread| {
425
+ // Signal the render thread to terminate (same pattern as destroy)
426
+ self.renderMutex.lock();
427
+ while (self.renderInProgress) {
428
+ self.renderCondition.wait(&self.renderMutex);
429
+ }
430
+ self.shouldTerminate = true;
431
+ self.renderRequested = true;
432
+ self.renderCondition.signal();
433
+ self.renderMutex.unlock();
434
+
435
+ thread.join();
436
+ self.renderThread = null;
437
+
438
+ // Reset termination flag so thread can be re-enabled later
439
+ self.shouldTerminate = false;
440
+ }
441
+ }
442
+
443
+ self.useThread = useThread;
444
+ }
445
+
446
+ pub fn updateStats(self: *CliRenderer, time: f64, fps: u32, frameCallbackTime: f64) void {
447
+ self.renderStats.overallFrameTime = time;
448
+ self.renderStats.fps = fps;
449
+ self.renderStats.frameCallbackTime = frameCallbackTime;
450
+
451
+ self.addStatSample(f64, &self.statSamples.overallFrameTime, time);
452
+ self.addStatSample(f64, &self.statSamples.frameCallbackTime, frameCallbackTime);
453
+ }
454
+
455
+ pub fn updateMemoryStats(self: *CliRenderer, heapUsed: u32, heapTotal: u32, arrayBuffers: u32) void {
456
+ self.renderStats.heapUsed = heapUsed;
457
+ self.renderStats.heapTotal = heapTotal;
458
+ self.renderStats.arrayBuffers = arrayBuffers;
459
+ }
460
+
461
+ pub fn resize(self: *CliRenderer, width: u32, height: u32) !void {
462
+ if (self.width == width and self.height == height) return;
463
+
464
+ self.width = width;
465
+ self.height = height;
466
+
467
+ try self.currentRenderBuffer.resize(width, height);
468
+ try self.nextRenderBuffer.resize(width, height);
469
+ self.nextRenderBuffer.setBlendBackdropColor(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], 1.0 });
470
+
471
+ try self.currentRenderBuffer.clear(.{ 0.0, 0.0, 0.0, 1.0 }, CLEAR_CHAR);
472
+ try self.nextRenderBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null);
473
+
474
+ const newHitGridSize = width * height;
475
+ const currentHitGridSize = self.hitGridWidth * self.hitGridHeight;
476
+ if (newHitGridSize > currentHitGridSize) {
477
+ const newCurrentHitGrid = try self.allocator.alloc(u32, newHitGridSize);
478
+ errdefer self.allocator.free(newCurrentHitGrid);
479
+ const newNextHitGrid = try self.allocator.alloc(u32, newHitGridSize);
480
+ @memset(newCurrentHitGrid, 0);
481
+ @memset(newNextHitGrid, 0);
482
+
483
+ self.allocator.free(self.currentHitGrid);
484
+ self.allocator.free(self.nextHitGrid);
485
+ self.currentHitGrid = newCurrentHitGrid;
486
+ self.nextHitGrid = newNextHitGrid;
487
+ }
488
+
489
+ // Always update dimensions. The backing buffer is at least as large as
490
+ // width*height, so this is safe even when the terminal shrinks. Without
491
+ // this, checkHit keeps using stale dimensions after a shrink and returns
492
+ // 0 for any coordinate beyond the old bounds.
493
+ self.hitGridWidth = width;
494
+ self.hitGridHeight = height;
495
+
496
+ const cursor = self.terminal.getCursorPosition();
497
+ self.terminal.setCursorPosition(@min(cursor.x, width), @min(cursor.y, height), cursor.visible);
498
+ }
499
+
500
+ pub fn setBackgroundColor(self: *CliRenderer, rgba: RGBA) void {
501
+ self.backgroundColor = rgba;
502
+ self.nextRenderBuffer.setBlendBackdropColor(.{ rgba[0], rgba[1], rgba[2], 1.0 });
503
+ }
504
+
505
+ pub fn setRenderOffset(self: *CliRenderer, offset: u32) void {
506
+ self.renderOffset = offset;
507
+ }
508
+
509
+ fn renderThreadFn(self: *CliRenderer) void {
510
+ while (true) {
511
+ self.renderMutex.lock();
512
+ while (!self.renderRequested and !self.shouldTerminate) {
513
+ self.renderCondition.wait(&self.renderMutex);
514
+ }
515
+
516
+ if (self.shouldTerminate and !self.renderRequested) {
517
+ self.renderMutex.unlock();
518
+ break;
519
+ }
520
+
521
+ self.renderRequested = false;
522
+
523
+ const outputData = self.currentOutputBuffer;
524
+ const outputLen = self.currentOutputLen;
525
+
526
+ const writeStart = std.time.microTimestamp();
527
+
528
+ if (outputLen > 0 and !self.testing) {
529
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
530
+ const w = &stdoutWriter.interface;
531
+ w.writeAll(outputData[0..outputLen]) catch {};
532
+ w.flush() catch {};
533
+ }
534
+
535
+ // Signal that rendering is complete
536
+ self.renderStats.stdoutWriteTime = @as(f64, @floatFromInt(std.time.microTimestamp() - writeStart));
537
+ self.renderInProgress = false;
538
+ self.renderCondition.signal();
539
+ self.renderMutex.unlock();
540
+ }
541
+ }
542
+
543
+ // Render once with current state
544
+ pub fn render(self: *CliRenderer, force: bool) void {
545
+ const now = std.time.microTimestamp();
546
+ const deltaTimeMs = @as(f64, @floatFromInt(now - self.lastRenderTime));
547
+ const deltaTime = deltaTimeMs / 1000.0; // Convert to seconds
548
+
549
+ self.lastRenderTime = now;
550
+ self.renderDebugOverlay();
551
+
552
+ self.prepareRenderFrame(force);
553
+
554
+ if (self.useThread) {
555
+ self.renderMutex.lock();
556
+ while (self.renderInProgress) {
557
+ self.renderCondition.wait(&self.renderMutex);
558
+ }
559
+
560
+ if (activeBuffer == .A) {
561
+ activeBuffer = .B;
562
+ self.currentOutputBuffer = &outputBuffer;
563
+ self.currentOutputLen = outputBufferLen;
564
+ } else {
565
+ activeBuffer = .A;
566
+ self.currentOutputBuffer = &outputBufferB;
567
+ self.currentOutputLen = outputBufferBLen;
568
+ }
569
+
570
+ self.renderRequested = true;
571
+ self.renderInProgress = true;
572
+ self.renderCondition.signal();
573
+ self.renderMutex.unlock();
574
+ } else {
575
+ const writeStart = std.time.microTimestamp();
576
+ if (!self.testing) {
577
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
578
+ const w = &stdoutWriter.interface;
579
+ w.writeAll(outputBuffer[0..outputBufferLen]) catch {};
580
+ w.flush() catch {};
581
+ }
582
+ self.renderStats.stdoutWriteTime = @as(f64, @floatFromInt(std.time.microTimestamp() - writeStart));
583
+ }
584
+
585
+ self.renderStats.lastFrameTime = deltaTime * 1000.0;
586
+ self.renderStats.frameCount += 1;
587
+
588
+ self.addStatSample(f64, &self.statSamples.lastFrameTime, deltaTime * 1000.0);
589
+ if (self.renderStats.renderTime) |rt| {
590
+ self.addStatSample(f64, &self.statSamples.renderTime, rt);
591
+ }
592
+ if (self.renderStats.bufferResetTime) |brt| {
593
+ self.addStatSample(f64, &self.statSamples.bufferResetTime, brt);
594
+ }
595
+ if (self.renderStats.stdoutWriteTime) |swt| {
596
+ self.addStatSample(f64, &self.statSamples.stdoutWriteTime, swt);
597
+ }
598
+ self.addStatSample(u32, &self.statSamples.cellsUpdated, self.renderStats.cellsUpdated);
599
+ }
600
+
601
+ pub fn getNextBuffer(self: *CliRenderer) *OptimizedBuffer {
602
+ return self.nextRenderBuffer;
603
+ }
604
+
605
+ pub fn getCurrentBuffer(self: *CliRenderer) *OptimizedBuffer {
606
+ return self.currentRenderBuffer;
607
+ }
608
+
609
+ fn prepareRenderFrame(self: *CliRenderer, force: bool) void {
610
+ const renderStartTime = std.time.microTimestamp();
611
+ var cellsUpdated: u32 = 0;
612
+
613
+ if (activeBuffer == .A) {
614
+ outputBufferLen = 0;
615
+ } else {
616
+ outputBufferBLen = 0;
617
+ }
618
+
619
+ var writer = OutputBufferWriter.writer();
620
+
621
+ writer.writeAll(ansi.ANSI.syncSet) catch {};
622
+ writer.writeAll(ansi.ANSI.hideCursor) catch {};
623
+
624
+ var currentFg: ?RGBA = null;
625
+ var currentBg: ?RGBA = null;
626
+ var currentAttributes: i32 = -1;
627
+ var currentLinkId: u32 = 0;
628
+ var utf8Buf: [4]u8 = undefined;
629
+
630
+ const colorEpsilon: f32 = COLOR_EPSILON_DEFAULT;
631
+ const hyperlinksEnabled = self.terminal.getCapabilities().hyperlinks;
632
+
633
+ for (0..self.height) |uy| {
634
+ const y = @as(u32, @intCast(uy));
635
+
636
+ var runStart: i64 = -1;
637
+ var runLength: u32 = 0;
638
+
639
+ for (0..self.width) |ux| {
640
+ const x = @as(u32, @intCast(ux));
641
+ const currentCell = self.currentRenderBuffer.get(x, y);
642
+ const nextCell = self.nextRenderBuffer.get(x, y);
643
+
644
+ if (currentCell == null or nextCell == null) continue;
645
+
646
+ if (!force) {
647
+ const charEqual = currentCell.?.char == nextCell.?.char;
648
+ const attrEqual = currentCell.?.attributes == nextCell.?.attributes;
649
+
650
+ if (charEqual and attrEqual and
651
+ buf.rgbaEqual(currentCell.?.fg, nextCell.?.fg, colorEpsilon) and
652
+ buf.rgbaEqual(currentCell.?.bg, nextCell.?.bg, colorEpsilon))
653
+ {
654
+ if (runLength > 0) {
655
+ writer.writeAll(ansi.ANSI.reset) catch {};
656
+ runStart = -1;
657
+ runLength = 0;
658
+ }
659
+ continue;
660
+ }
661
+ }
662
+
663
+ const cell = nextCell.?;
664
+
665
+ const fgMatch = currentFg != null and buf.rgbaEqual(currentFg.?, cell.fg, colorEpsilon);
666
+ const bgMatch = currentBg != null and buf.rgbaEqual(currentBg.?, cell.bg, colorEpsilon);
667
+ const sameAttributes = fgMatch and bgMatch and @as(i32, @intCast(cell.attributes)) == currentAttributes;
668
+
669
+ const linkId = if (hyperlinksEnabled) ansi.TextAttributes.getLinkId(cell.attributes) else 0;
670
+
671
+ if (hyperlinksEnabled and linkId != currentLinkId) {
672
+ if (currentLinkId != 0) {
673
+ writer.writeAll("\x1b]8;;\x1b\\") catch {};
674
+ }
675
+ currentLinkId = linkId;
676
+ if (currentLinkId != 0) {
677
+ const lp = link.initGlobalLinkPool(self.allocator);
678
+ if (lp.get(currentLinkId)) |url_bytes| {
679
+ writer.print("\x1b]8;id={d};{s}\x1b\\", .{ currentLinkId, url_bytes }) catch {};
680
+ } else |_| {
681
+ // Link not found, treat as no link
682
+ currentLinkId = 0;
683
+ }
684
+ }
685
+ }
686
+
687
+ if (!sameAttributes or runStart == -1) {
688
+ if (runLength > 0) {
689
+ writer.writeAll(ansi.ANSI.reset) catch {};
690
+ }
691
+
692
+ runStart = @intCast(x);
693
+ runLength = 0;
694
+
695
+ currentFg = cell.fg;
696
+ currentBg = cell.bg;
697
+ currentAttributes = @as(i32, @intCast(cell.attributes));
698
+
699
+ ansi.ANSI.moveToOutput(writer, x + 1, y + 1 + self.renderOffset) catch {};
700
+
701
+ const fgR = rgbaComponentToU8(cell.fg[0]);
702
+ const fgG = rgbaComponentToU8(cell.fg[1]);
703
+ const fgB = rgbaComponentToU8(cell.fg[2]);
704
+
705
+ const bgR = rgbaComponentToU8(cell.bg[0]);
706
+ const bgG = rgbaComponentToU8(cell.bg[1]);
707
+ const bgB = rgbaComponentToU8(cell.bg[2]);
708
+ const bgA = cell.bg[3];
709
+
710
+ ansi.ANSI.fgColorOutput(writer, fgR, fgG, fgB) catch {};
711
+
712
+ // If alpha is 0 (transparent), use terminal default background instead of black
713
+ if (bgA < 0.001) {
714
+ writer.writeAll("\x1b[49m") catch {};
715
+ } else {
716
+ ansi.ANSI.bgColorOutput(writer, bgR, bgG, bgB) catch {};
717
+ }
718
+
719
+ ansi.TextAttributes.applyAttributesOutputWriter(writer, cell.attributes) catch {};
720
+ }
721
+
722
+ // Handle grapheme characters
723
+ if (gp.isGraphemeChar(cell.char)) {
724
+ const gid: u32 = gp.graphemeIdFromChar(cell.char);
725
+ const bytes = self.pool.get(gid) catch |err| {
726
+ self.performShutdownSequence();
727
+ std.debug.panic("Fatal: no grapheme bytes in pool for gid {d}: {}", .{ gid, err });
728
+ };
729
+ if (bytes.len > 0) {
730
+ const capabilities = self.terminal.getCapabilities();
731
+ const graphemeWidth = gp.charRightExtent(cell.char) + 1;
732
+ if (capabilities.explicit_width) {
733
+ ansi.ANSI.explicitWidthOutput(writer, graphemeWidth, bytes) catch {};
734
+ } else {
735
+ writer.writeAll(bytes) catch {};
736
+ if (capabilities.explicit_cursor_positioning) {
737
+ const nextX = x + graphemeWidth;
738
+ if (nextX < self.width) {
739
+ ansi.ANSI.moveToOutput(writer, nextX + 1, y + 1 + self.renderOffset) catch {};
740
+ }
741
+ }
742
+ }
743
+ }
744
+ } else if (gp.isContinuationChar(cell.char)) {
745
+ // Intentionally do not write a space for continuation cells.
746
+ // NOTE: disabled to fix 2-cell emoji rendering when the two
747
+ // cells have distinct colors (space overwrite can break glyph output)
748
+
749
+ // writer.writeByte(' ') catch {};
750
+ } else {
751
+ const len = std.unicode.utf8Encode(@intCast(cell.char), &utf8Buf) catch 1;
752
+ writer.writeAll(utf8Buf[0..len]) catch {};
753
+ }
754
+ runLength += 1;
755
+
756
+ // Sync this cell to the current buffer so the next frame's diff
757
+ // is correct. Use syncCell (set without span cleanup) because
758
+ // span cleanup would destroy continuation cells written by an
759
+ // earlier iteration of this same left-to-right pass (#723).
760
+ self.currentRenderBuffer.syncCell(x, y, nextCell.?);
761
+
762
+ cellsUpdated += 1;
763
+ }
764
+ }
765
+
766
+ if (hyperlinksEnabled and currentLinkId != 0) {
767
+ writer.writeAll("\x1b]8;;\x1b\\") catch {};
768
+ }
769
+
770
+ writer.writeAll(ansi.ANSI.reset) catch {};
771
+
772
+ const cursorPos = self.terminal.getCursorPosition();
773
+ const cursorStyle = self.terminal.getCursorStyle();
774
+ const cursorColor = self.terminal.getCursorColor();
775
+
776
+ if (cursorPos.visible) {
777
+ var cursorStyleCode: []const u8 = undefined;
778
+
779
+ switch (cursorStyle.style) {
780
+ .block => {
781
+ cursorStyleCode = if (cursorStyle.blinking)
782
+ ansi.ANSI.cursorBlockBlink
783
+ else
784
+ ansi.ANSI.cursorBlock;
785
+ },
786
+ .line => {
787
+ cursorStyleCode = if (cursorStyle.blinking)
788
+ ansi.ANSI.cursorLineBlink
789
+ else
790
+ ansi.ANSI.cursorLine;
791
+ },
792
+ .underline => {
793
+ cursorStyleCode = if (cursorStyle.blinking)
794
+ ansi.ANSI.cursorUnderlineBlink
795
+ else
796
+ ansi.ANSI.cursorUnderline;
797
+ },
798
+ .default => {
799
+ cursorStyleCode = ansi.ANSI.defaultCursorStyle;
800
+ },
801
+ }
802
+
803
+ const cursorR = rgbaComponentToU8(cursorColor[0]);
804
+ const cursorG = rgbaComponentToU8(cursorColor[1]);
805
+ const cursorB = rgbaComponentToU8(cursorColor[2]);
806
+
807
+ const styleTag: u8 = @intFromEnum(cursorStyle.style);
808
+ const styleChanged = (self.lastCursorStyleTag == null or self.lastCursorStyleTag.? != styleTag) or
809
+ (self.lastCursorBlinking == null or self.lastCursorBlinking.? != cursorStyle.blinking);
810
+ const colorChanged = (self.lastCursorColorRGB == null or self.lastCursorColorRGB.?[0] != cursorR or self.lastCursorColorRGB.?[1] != cursorG or self.lastCursorColorRGB.?[2] != cursorB);
811
+
812
+ if (colorChanged) {
813
+ ansi.ANSI.cursorColorOutputWriter(writer, cursorR, cursorG, cursorB) catch {};
814
+ self.lastCursorColorRGB = .{ cursorR, cursorG, cursorB };
815
+ }
816
+ if (styleChanged) {
817
+ writer.writeAll(cursorStyleCode) catch {};
818
+ self.lastCursorStyleTag = styleTag;
819
+ self.lastCursorBlinking = cursorStyle.blinking;
820
+ }
821
+ ansi.ANSI.moveToOutput(writer, cursorPos.x, cursorPos.y + self.renderOffset) catch {};
822
+ writer.writeAll(ansi.ANSI.showCursor) catch {};
823
+ } else {
824
+ writer.writeAll(ansi.ANSI.hideCursor) catch {};
825
+ self.lastCursorStyleTag = null;
826
+ self.lastCursorBlinking = null;
827
+ self.lastCursorColorRGB = null;
828
+ }
829
+
830
+ const mousePointer = self.terminal.getMousePointer();
831
+ if (mousePointer != self.lastMousePointerStyle) {
832
+ ansi.ANSI.setMousePointerOutput(writer, mousePointer.toName()) catch {};
833
+ self.lastMousePointerStyle = mousePointer;
834
+ }
835
+
836
+ writer.writeAll(ansi.ANSI.syncReset) catch {};
837
+
838
+ const renderEndTime = std.time.microTimestamp();
839
+ const renderTime = @as(f64, @floatFromInt(renderEndTime - renderStartTime));
840
+
841
+ self.renderStats.cellsUpdated = cellsUpdated;
842
+ self.renderStats.renderTime = renderTime;
843
+
844
+ self.nextRenderBuffer.clear(.{ self.backgroundColor[0], self.backgroundColor[1], self.backgroundColor[2], self.backgroundColor[3] }, null) catch {};
845
+
846
+ // Compare hit grids before swap to detect changes. This allows TypeScript to
847
+ // know if hover state needs rechecking without manually tracking dirty state.
848
+ self.hitGridDirty = !std.mem.eql(u32, self.currentHitGrid, self.nextHitGrid);
849
+
850
+ // Swap hit grids: nextHitGrid (built this frame) becomes the active grid for
851
+ // hit testing. The old currentHitGrid becomes nextHitGrid and is cleared for
852
+ // the next frame.
853
+ const temp = self.currentHitGrid;
854
+ self.currentHitGrid = self.nextHitGrid;
855
+ self.nextHitGrid = temp;
856
+ @memset(self.nextHitGrid, 0);
857
+ }
858
+
859
+ pub fn setDebugOverlay(self: *CliRenderer, enabled: bool, corner: DebugOverlayCorner) void {
860
+ self.debugOverlay.enabled = enabled;
861
+ self.debugOverlay.corner = corner;
862
+ }
863
+
864
+ pub fn clearTerminal(self: *CliRenderer) void {
865
+ self.writeOut(ansi.ANSI.clearAndHome);
866
+ }
867
+
868
+ pub fn writeOut(self: *CliRenderer, data: []const u8) void {
869
+ if (data.len == 0) return;
870
+ if (self.testing) return;
871
+
872
+ if (self.useThread) {
873
+ self.renderMutex.lock();
874
+ while (self.renderInProgress) {
875
+ self.renderCondition.wait(&self.renderMutex);
876
+ }
877
+ self.renderMutex.unlock();
878
+ }
879
+
880
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
881
+ const w = &stdoutWriter.interface;
882
+ w.writeAll(data) catch {};
883
+ w.flush() catch {};
884
+ }
885
+
886
+ pub fn writeOutMultiple(self: *CliRenderer, data_slices: []const []const u8) void {
887
+ if (self.testing) return;
888
+
889
+ if (self.useThread) {
890
+ self.renderMutex.lock();
891
+ while (self.renderInProgress) {
892
+ self.renderCondition.wait(&self.renderMutex);
893
+ }
894
+ self.renderMutex.unlock();
895
+ }
896
+
897
+ var totalLen: usize = 0;
898
+ for (data_slices) |slice| {
899
+ totalLen += slice.len;
900
+ }
901
+
902
+ if (totalLen == 0) return;
903
+
904
+ var stdoutWriter = std.fs.File.stdout().writer(&self.stdoutBuffer);
905
+ const w = &stdoutWriter.interface;
906
+ for (data_slices) |slice| {
907
+ w.writeAll(slice) catch {};
908
+ }
909
+ w.flush() catch {};
910
+ }
911
+
912
+ /// Write a renderable's bounds to nextHitGrid for the upcoming frame.
913
+ ///
914
+ /// Called during render for each visible renderable. The rect is clipped to
915
+ /// the current hit scissor stack, so elements inside overflow:hidden parents
916
+ /// only register hits within the visible region. Later renderables overwrite
917
+ /// earlier ones. Z-order is determined by render order.
918
+ pub fn addToHitGrid(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
919
+ const clipped = self.clipRectToHitScissor(x, y, width, height) orelse return;
920
+ const startX = @max(0, clipped.x);
921
+ const startY = @max(0, clipped.y);
922
+ const endX = @min(
923
+ @as(i32, @intCast(self.hitGridWidth)),
924
+ clipped.x + @as(i32, @intCast(clipped.width)),
925
+ );
926
+ const endY = @min(
927
+ @as(i32, @intCast(self.hitGridHeight)),
928
+ clipped.y + @as(i32, @intCast(clipped.height)),
929
+ );
930
+
931
+ if (startX >= endX or startY >= endY) return;
932
+
933
+ const uStartX: u32 = @intCast(startX);
934
+ const uStartY: u32 = @intCast(startY);
935
+ const uEndX: u32 = @intCast(endX);
936
+ const uEndY: u32 = @intCast(endY);
937
+
938
+ for (uStartY..uEndY) |row| {
939
+ const rowStart = row * self.hitGridWidth;
940
+ const startIdx = rowStart + uStartX;
941
+ const endIdx = rowStart + uEndX;
942
+
943
+ @memset(self.nextHitGrid[startIdx..endIdx], id);
944
+ }
945
+ }
946
+
947
+ /// Clear currentHitGrid before an immediate rebuild.
948
+ ///
949
+ /// Used by syncHitGridIfNeeded in TypeScript when scroll/translate changes
950
+ /// require updating hit targets without waiting for the next render.
951
+ pub fn clearCurrentHitGrid(self: *CliRenderer) void {
952
+ @memset(self.currentHitGrid, 0);
953
+ }
954
+
955
+ /// Return whether the hit grid changed during the last render.
956
+ /// This is set by comparing the previous and current hit grids after render.
957
+ /// TypeScript can use this to decide if hover state needs rechecking.
958
+ pub fn getHitGridDirty(self: *CliRenderer) bool {
959
+ return self.hitGridDirty;
960
+ }
961
+
962
+ /// Return the renderable ID at screen position (x, y), or 0 if none.
963
+ pub fn checkHit(self: *CliRenderer, x: u32, y: u32) u32 {
964
+ if (x >= self.hitGridWidth or y >= self.hitGridHeight) {
965
+ return 0;
966
+ }
967
+
968
+ const index = y * self.hitGridWidth + x;
969
+ return self.currentHitGrid[index];
970
+ }
971
+
972
+ /// Return the current (topmost) hit scissor rect, or null if the stack is empty.
973
+ fn getCurrentHitScissorRect(self: *const CliRenderer) ?buf.ClipRect {
974
+ if (self.hitScissorStack.items.len == 0) return null;
975
+ return self.hitScissorStack.items[self.hitScissorStack.items.len - 1];
976
+ }
977
+
978
+ /// Intersect a rect with the current hit scissor. Returns null if fully clipped.
979
+ fn clipRectToHitScissor(self: *const CliRenderer, x: i32, y: i32, width: u32, height: u32) ?buf.ClipRect {
980
+ const scissor = self.getCurrentHitScissorRect() orelse return buf.ClipRect{
981
+ .x = x,
982
+ .y = y,
983
+ .width = width,
984
+ .height = height,
985
+ };
986
+
987
+ const rect_end_x = x + @as(i32, @intCast(width));
988
+ const rect_end_y = y + @as(i32, @intCast(height));
989
+ const scissor_end_x = scissor.x + @as(i32, @intCast(scissor.width));
990
+ const scissor_end_y = scissor.y + @as(i32, @intCast(scissor.height));
991
+
992
+ const intersect_x = @max(x, scissor.x);
993
+ const intersect_y = @max(y, scissor.y);
994
+ const intersect_end_x = @min(rect_end_x, scissor_end_x);
995
+ const intersect_end_y = @min(rect_end_y, scissor_end_y);
996
+
997
+ if (intersect_x >= intersect_end_x or intersect_y >= intersect_end_y) {
998
+ return null;
999
+ }
1000
+
1001
+ return buf.ClipRect{
1002
+ .x = intersect_x,
1003
+ .y = intersect_y,
1004
+ .width = @intCast(intersect_end_x - intersect_x),
1005
+ .height = @intCast(intersect_end_y - intersect_y),
1006
+ };
1007
+ }
1008
+
1009
+ /// Push a scissor rect for hit grid clipping.
1010
+ ///
1011
+ /// The rect is intersected with any existing scissor, so nested overflow:hidden
1012
+ /// containers compound correctly. All coordinates are in screen space.
1013
+ pub fn hitGridPushScissorRect(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32) void {
1014
+ var rect = buf.ClipRect{
1015
+ .x = x,
1016
+ .y = y,
1017
+ .width = width,
1018
+ .height = height,
1019
+ };
1020
+
1021
+ if (self.getCurrentHitScissorRect() != null) {
1022
+ const intersect = self.clipRectToHitScissor(rect.x, rect.y, rect.width, rect.height);
1023
+ if (intersect) |clipped| {
1024
+ rect = clipped;
1025
+ } else {
1026
+ rect = buf.ClipRect{ .x = 0, .y = 0, .width = 0, .height = 0 };
1027
+ }
1028
+ }
1029
+
1030
+ self.hitScissorStack.append(self.allocator, rect) catch |err| {
1031
+ logger.warn("Failed to push hit-grid scissor rect: {}", .{err});
1032
+ };
1033
+ }
1034
+
1035
+ pub fn hitGridPopScissorRect(self: *CliRenderer) void {
1036
+ if (self.hitScissorStack.items.len > 0) {
1037
+ _ = self.hitScissorStack.pop();
1038
+ }
1039
+ }
1040
+
1041
+ /// Clear all hit grid scissors. Called at start of render to reset state.
1042
+ pub fn hitGridClearScissorRects(self: *CliRenderer) void {
1043
+ self.hitScissorStack.clearRetainingCapacity();
1044
+ }
1045
+
1046
+ /// Write directly to currentHitGrid with scissor clipping.
1047
+ ///
1048
+ /// Used for immediate hit grid sync when scroll/translate changes. Unlike
1049
+ /// addToHitGrid (which writes to nextHitGrid for the upcoming frame), this
1050
+ /// updates the grid that checkHit reads right now. Lets hover states update
1051
+ /// without waiting for the next render.
1052
+ pub fn addToCurrentHitGridClipped(self: *CliRenderer, x: i32, y: i32, width: u32, height: u32, id: u32) void {
1053
+ const clipped = self.clipRectToHitScissor(x, y, width, height) orelse return;
1054
+
1055
+ const startX = @max(0, clipped.x);
1056
+ const startY = @max(0, clipped.y);
1057
+ const endX = @min(@as(i32, @intCast(self.hitGridWidth)), clipped.x + @as(i32, @intCast(clipped.width)));
1058
+ const endY = @min(@as(i32, @intCast(self.hitGridHeight)), clipped.y + @as(i32, @intCast(clipped.height)));
1059
+
1060
+ if (startX >= endX or startY >= endY) return;
1061
+
1062
+ const uStartX: u32 = @intCast(startX);
1063
+ const uStartY: u32 = @intCast(startY);
1064
+ const uEndX: u32 = @intCast(endX);
1065
+ const uEndY: u32 = @intCast(endY);
1066
+
1067
+ for (uStartY..uEndY) |row| {
1068
+ const rowStart = row * self.hitGridWidth;
1069
+ const startIdx = rowStart + uStartX;
1070
+ const endIdx = rowStart + uEndX;
1071
+
1072
+ @memset(self.currentHitGrid[startIdx..endIdx], id);
1073
+ }
1074
+ }
1075
+
1076
+ pub fn dumpHitGrid(self: *CliRenderer) void {
1077
+ const timestamp = std.time.timestamp();
1078
+ var filename_buf: [64]u8 = undefined;
1079
+ const filename = std.fmt.bufPrint(&filename_buf, "hitgrid_{d}.txt", .{timestamp}) catch return;
1080
+
1081
+ const file = std.fs.cwd().createFile(filename, .{}) catch return;
1082
+ defer file.close();
1083
+
1084
+ var fileBuffer: [4096]u8 = undefined;
1085
+ var fileWriter = file.writer(&fileBuffer);
1086
+ const writer = &fileWriter.interface;
1087
+
1088
+ for (0..self.hitGridHeight) |y| {
1089
+ for (0..self.hitGridWidth) |x| {
1090
+ const index = y * self.hitGridWidth + x;
1091
+ const id = self.currentHitGrid[index];
1092
+
1093
+ const char = if (id == 0) '.' else ('0' + @as(u8, @intCast(id % 10)));
1094
+ writer.writeByte(char) catch return;
1095
+ }
1096
+ writer.writeByte('\n') catch return;
1097
+ }
1098
+ writer.flush() catch {};
1099
+ }
1100
+
1101
+ fn dumpSingleBuffer(self: *CliRenderer, buffer: *OptimizedBuffer, buffer_name: []const u8, timestamp: i64) void {
1102
+ std.fs.cwd().makeDir("buffer_dump") catch |err| switch (err) {
1103
+ error.PathAlreadyExists => {},
1104
+ else => return,
1105
+ };
1106
+
1107
+ var filename_buf: [128]u8 = undefined;
1108
+ const filename = std.fmt.bufPrint(&filename_buf, "buffer_dump/{s}_buffer_{d}.txt", .{ buffer_name, timestamp }) catch return;
1109
+
1110
+ const file = std.fs.cwd().createFile(filename, .{}) catch return;
1111
+ defer file.close();
1112
+
1113
+ var fileBuffer: [4096]u8 = undefined;
1114
+ var fileWriter = file.writer(&fileBuffer);
1115
+ const writer = &fileWriter.interface;
1116
+
1117
+ writer.print("{s} Buffer ({d}x{d}):\n", .{ buffer_name, self.width, self.height }) catch return;
1118
+ writer.writeAll("Characters:\n") catch return;
1119
+
1120
+ for (0..self.height) |y| {
1121
+ for (0..self.width) |x| {
1122
+ const cell = buffer.get(@intCast(x), @intCast(y));
1123
+ if (cell) |c| {
1124
+ if (gp.isContinuationChar(c.char)) {
1125
+ // skip
1126
+ } else if (gp.isGraphemeChar(c.char)) {
1127
+ const gid: u32 = gp.graphemeIdFromChar(c.char);
1128
+ const bytes = self.pool.get(gid) catch &[_]u8{};
1129
+ if (bytes.len > 0) writer.writeAll(bytes) catch return;
1130
+ } else {
1131
+ var utf8Buf: [4]u8 = undefined;
1132
+ const len = std.unicode.utf8Encode(@intCast(c.char), &utf8Buf) catch 1;
1133
+ writer.writeAll(utf8Buf[0..len]) catch return;
1134
+ }
1135
+ } else {
1136
+ writer.writeByte(' ') catch return;
1137
+ }
1138
+ }
1139
+ writer.writeByte('\n') catch return;
1140
+ }
1141
+ writer.flush() catch {};
1142
+ }
1143
+
1144
+ pub fn getLastOutputForTest(_: *CliRenderer) []const u8 {
1145
+ // In non-threaded mode, we want the current active buffer
1146
+ // In threaded mode, we want the previously rendered buffer
1147
+ const currentBuffer = if (activeBuffer == .A) &outputBuffer else &outputBufferB;
1148
+ const currentLen = if (activeBuffer == .A) outputBufferLen else outputBufferBLen;
1149
+ return currentBuffer.*[0..currentLen];
1150
+ }
1151
+
1152
+ pub fn dumpStdoutBuffer(self: *CliRenderer, timestamp: i64) void {
1153
+ _ = self;
1154
+ std.fs.cwd().makeDir("buffer_dump") catch |err| switch (err) {
1155
+ error.PathAlreadyExists => {},
1156
+ else => return,
1157
+ };
1158
+
1159
+ var filename_buf: [128]u8 = undefined;
1160
+ const filename = std.fmt.bufPrint(&filename_buf, "buffer_dump/stdout_buffer_{d}.txt", .{timestamp}) catch return;
1161
+
1162
+ const file = std.fs.cwd().createFile(filename, .{}) catch return;
1163
+ defer file.close();
1164
+
1165
+ var fileBuffer: [4096]u8 = undefined;
1166
+ var fileWriter = file.writer(&fileBuffer);
1167
+ const writer = &fileWriter.interface;
1168
+
1169
+ writer.print("Stdout Buffer Output (timestamp: {d}):\n", .{timestamp}) catch return;
1170
+ writer.writeAll("Last Rendered ANSI Output:\n") catch return;
1171
+ writer.writeAll("================\n") catch return;
1172
+
1173
+ const lastBuffer = if (activeBuffer == .A) &outputBufferB else &outputBuffer;
1174
+ const lastLen = if (activeBuffer == .A) outputBufferBLen else outputBufferLen;
1175
+
1176
+ if (lastLen > 0) {
1177
+ writer.writeAll(lastBuffer.*[0..lastLen]) catch return;
1178
+ } else {
1179
+ writer.writeAll("(no output rendered yet)\n") catch return;
1180
+ }
1181
+
1182
+ writer.writeAll("\n================\n") catch return;
1183
+ writer.print("Buffer size: {d} bytes\n", .{lastLen}) catch return;
1184
+ writer.print("Active buffer: {s}\n", .{if (activeBuffer == .A) "A" else "B"}) catch return;
1185
+ writer.flush() catch {};
1186
+ }
1187
+
1188
+ pub fn dumpBuffers(self: *CliRenderer, timestamp: i64) void {
1189
+ self.dumpSingleBuffer(self.currentRenderBuffer, "current", timestamp);
1190
+ self.dumpSingleBuffer(self.nextRenderBuffer, "next", timestamp);
1191
+ self.dumpStdoutBuffer(timestamp);
1192
+ }
1193
+
1194
+ pub fn restoreTerminalModes(self: *CliRenderer) void {
1195
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1196
+ self.terminal.restoreTerminalModes(stream.writer()) catch {};
1197
+ self.writeOut(stream.getWritten());
1198
+ }
1199
+
1200
+ pub fn enableMouse(self: *CliRenderer, enableMovement: bool) void {
1201
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1202
+ self.terminal.setMouseMode(stream.writer(), true, enableMovement) catch {};
1203
+ self.writeOut(stream.getWritten());
1204
+ }
1205
+
1206
+ pub fn queryPixelResolution(self: *CliRenderer) void {
1207
+ self.writeOut(ansi.ANSI.queryPixelSize);
1208
+ }
1209
+
1210
+ pub fn disableMouse(self: *CliRenderer) void {
1211
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1212
+ self.terminal.setMouseMode(stream.writer(), false, self.terminal.state.mouse_movement) catch {};
1213
+ self.writeOut(stream.getWritten());
1214
+ }
1215
+
1216
+ pub fn enableKittyKeyboard(self: *CliRenderer, flags: u8) void {
1217
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1218
+ self.terminal.setKittyKeyboard(stream.writer(), true, flags) catch {};
1219
+ self.writeOut(stream.getWritten());
1220
+ }
1221
+
1222
+ pub fn disableKittyKeyboard(self: *CliRenderer) void {
1223
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1224
+ self.terminal.setKittyKeyboard(stream.writer(), false, 0) catch {};
1225
+ self.writeOut(stream.getWritten());
1226
+ }
1227
+
1228
+ pub fn getTerminalCapabilities(self: *CliRenderer) Terminal.Capabilities {
1229
+ return self.terminal.getCapabilities();
1230
+ }
1231
+
1232
+ pub fn setTerminalEnvVar(self: *CliRenderer, key: []const u8, value: []const u8) bool {
1233
+ self.terminal.setHostEnvVar(self.allocator, key, value) catch return false;
1234
+ return true;
1235
+ }
1236
+
1237
+ pub fn processCapabilityResponse(self: *CliRenderer, response: []const u8) void {
1238
+ self.terminal.processCapabilityResponse(response);
1239
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1240
+ _ = self.terminal.sendPendingQueries(stream.writer()) catch |err| blk: {
1241
+ logger.warn("Failed to send pending queries: {}", .{err});
1242
+ break :blk false;
1243
+ };
1244
+ const useKitty = self.terminal.opts.kitty_keyboard_flags > 0;
1245
+ self.terminal.enableDetectedFeatures(stream.writer(), useKitty) catch {};
1246
+ self.writeOut(stream.getWritten());
1247
+ }
1248
+
1249
+ pub fn setKittyKeyboardFlags(self: *CliRenderer, flags: u8) void {
1250
+ self.terminal.setKittyKeyboardFlags(flags);
1251
+ }
1252
+
1253
+ pub fn getKittyKeyboardFlags(self: *CliRenderer) u8 {
1254
+ return self.terminal.opts.kitty_keyboard_flags;
1255
+ }
1256
+
1257
+ pub fn setTerminalTitle(self: *CliRenderer, title: []const u8) void {
1258
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1259
+ self.terminal.setTerminalTitle(stream.writer(), title);
1260
+ self.writeOut(stream.getWritten());
1261
+ }
1262
+
1263
+ pub fn copyToClipboardOSC52(self: *CliRenderer, target: Terminal.ClipboardTarget, payload: []const u8) bool {
1264
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1265
+ self.terminal.writeClipboard(stream.writer(), target, payload) catch return false;
1266
+ self.writeOut(stream.getWritten());
1267
+ return true;
1268
+ }
1269
+
1270
+ pub fn clearClipboardOSC52(self: *CliRenderer, target: Terminal.ClipboardTarget) bool {
1271
+ var stream = std.io.fixedBufferStream(&self.writeOutBuf);
1272
+ self.terminal.writeClipboard(stream.writer(), target, "") catch return false;
1273
+ self.writeOut(stream.getWritten());
1274
+ return true;
1275
+ }
1276
+
1277
+ fn renderDebugOverlay(self: *CliRenderer) void {
1278
+ if (!self.debugOverlay.enabled) return;
1279
+
1280
+ const width: u32 = 40;
1281
+ const height: u32 = 11;
1282
+ var x: u32 = 0;
1283
+ var y: u32 = 0;
1284
+
1285
+ if (self.width < width + 2 or self.height < height + 2) return;
1286
+
1287
+ switch (self.debugOverlay.corner) {
1288
+ .topLeft => {
1289
+ x = 1;
1290
+ y = 1;
1291
+ },
1292
+ .topRight => {
1293
+ x = self.width - width - 1;
1294
+ y = 1;
1295
+ },
1296
+ .bottomLeft => {
1297
+ x = 1;
1298
+ y = self.height - height - 1;
1299
+ },
1300
+ .bottomRight => {
1301
+ x = self.width - width - 1;
1302
+ y = self.height - height - 1;
1303
+ },
1304
+ }
1305
+
1306
+ self.nextRenderBuffer.fillRect(x, y, width, height, .{ 20.0 / 255.0, 20.0 / 255.0, 40.0 / 255.0, 1.0 }) catch {};
1307
+ self.nextRenderBuffer.drawText("Debug Information", x + 1, y + 1, .{ 1.0, 1.0, 100.0 / 255.0, 1.0 }, .{ 0.0, 0.0, 0.0, 0.0 }, ansi.TextAttributes.BOLD) catch {};
1308
+
1309
+ var row: u32 = 2;
1310
+ const bg: RGBA = .{ 0.0, 0.0, 0.0, 0.0 };
1311
+ const fg: RGBA = .{ 200.0 / 255.0, 200.0 / 255.0, 200.0 / 255.0, 1.0 };
1312
+
1313
+ // Calculate averages
1314
+ const lastFrameTimeAvg = getStatAverage(f64, &self.statSamples.lastFrameTime);
1315
+ const renderTimeAvg = getStatAverage(f64, &self.statSamples.renderTime);
1316
+ const overallFrameTimeAvg = getStatAverage(f64, &self.statSamples.overallFrameTime);
1317
+ const bufferResetTimeAvg = getStatAverage(f64, &self.statSamples.bufferResetTime);
1318
+ const stdoutWriteTimeAvg = getStatAverage(f64, &self.statSamples.stdoutWriteTime);
1319
+ const cellsUpdatedAvg = getStatAverage(u32, &self.statSamples.cellsUpdated);
1320
+ const frameCallbackTimeAvg = getStatAverage(f64, &self.statSamples.frameCallbackTime);
1321
+
1322
+ // FPS
1323
+ var fpsText: [32]u8 = undefined;
1324
+ const fpsLen = std.fmt.bufPrint(&fpsText, "FPS: {d}", .{self.renderStats.fps}) catch return;
1325
+ self.nextRenderBuffer.drawText(fpsLen, x + 1, y + row, fg, bg, 0) catch {};
1326
+ row += 1;
1327
+
1328
+ // Frame Time
1329
+ var frameTimeText: [64]u8 = undefined;
1330
+ const frameTimeLen = std.fmt.bufPrint(&frameTimeText, "Frame: {d:.3}ms (avg: {d:.3}ms)", .{ self.renderStats.lastFrameTime / 1000.0, lastFrameTimeAvg / 1000.0 }) catch return;
1331
+ self.nextRenderBuffer.drawText(frameTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1332
+ row += 1;
1333
+
1334
+ // Frame Callback Time
1335
+ if (self.renderStats.frameCallbackTime) |frameCallbackTime| {
1336
+ var frameCallbackTimeText: [64]u8 = undefined;
1337
+ const frameCallbackTimeLen = std.fmt.bufPrint(&frameCallbackTimeText, "Frame Callback: {d:.3}ms (avg: {d:.3}ms)", .{ frameCallbackTime, frameCallbackTimeAvg }) catch return;
1338
+ self.nextRenderBuffer.drawText(frameCallbackTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1339
+ row += 1;
1340
+ }
1341
+
1342
+ // Overall Time
1343
+ if (self.renderStats.overallFrameTime) |overallTime| {
1344
+ var overallTimeText: [64]u8 = undefined;
1345
+ const overallTimeLen = std.fmt.bufPrint(&overallTimeText, "Overall: {d:.3}ms (avg: {d:.3}ms)", .{ overallTime, overallFrameTimeAvg }) catch return;
1346
+ self.nextRenderBuffer.drawText(overallTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1347
+ row += 1;
1348
+ }
1349
+
1350
+ // Render Time
1351
+ if (self.renderStats.renderTime) |renderTime| {
1352
+ var renderTimeText: [64]u8 = undefined;
1353
+ const renderTimeLen = std.fmt.bufPrint(&renderTimeText, "Render: {d:.3}ms (avg: {d:.3}ms)", .{ renderTime / 1000.0, renderTimeAvg / 1000.0 }) catch return;
1354
+ self.nextRenderBuffer.drawText(renderTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1355
+ row += 1;
1356
+ }
1357
+
1358
+ // Buffer Reset Time
1359
+ if (self.renderStats.bufferResetTime) |resetTime| {
1360
+ var resetTimeText: [64]u8 = undefined;
1361
+ const resetTimeLen = std.fmt.bufPrint(&resetTimeText, "Reset: {d:.3}ms (avg: {d:.3}ms)", .{ resetTime / 1000.0, bufferResetTimeAvg / 1000.0 }) catch return;
1362
+ self.nextRenderBuffer.drawText(resetTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1363
+ row += 1;
1364
+ }
1365
+
1366
+ // Stdout Write Time
1367
+ if (self.renderStats.stdoutWriteTime) |writeTime| {
1368
+ var writeTimeText: [64]u8 = undefined;
1369
+ const writeTimeLen = std.fmt.bufPrint(&writeTimeText, "Stdout: {d:.3}ms (avg: {d:.3}ms)", .{ writeTime / 1000.0, stdoutWriteTimeAvg / 1000.0 }) catch return;
1370
+ self.nextRenderBuffer.drawText(writeTimeLen, x + 1, y + row, fg, bg, 0) catch {};
1371
+ row += 1;
1372
+ }
1373
+
1374
+ // Cells Updated
1375
+ var cellsText: [64]u8 = undefined;
1376
+ const cellsLen = std.fmt.bufPrint(&cellsText, "Cells: {d} (avg: {d})", .{ self.renderStats.cellsUpdated, cellsUpdatedAvg }) catch return;
1377
+ self.nextRenderBuffer.drawText(cellsLen, x + 1, y + row, fg, bg, 0) catch {};
1378
+ row += 1;
1379
+
1380
+ if (self.renderStats.heapUsed > 0 or self.renderStats.heapTotal > 0) {
1381
+ var memoryText: [64]u8 = undefined;
1382
+ const memoryLen = std.fmt.bufPrint(&memoryText, "Memory: {d:.2}MB / {d:.2}MB / {d:.2}MB", .{ @as(f64, @floatFromInt(self.renderStats.heapUsed)) / 1024.0 / 1024.0, @as(f64, @floatFromInt(self.renderStats.heapTotal)) / 1024.0 / 1024.0, @as(f64, @floatFromInt(self.renderStats.arrayBuffers)) / 1024.0 / 1024.0 }) catch return;
1383
+ self.nextRenderBuffer.drawText(memoryLen, x + 1, y + row, fg, bg, 0) catch {};
1384
+ row += 1;
1385
+ }
1386
+
1387
+ // Is threaded?
1388
+ var isThreadedText: [64]u8 = undefined;
1389
+ const isThreadedLen = std.fmt.bufPrint(&isThreadedText, "Threaded: {s}", .{if (self.useThread) "Yes" else "No"}) catch return;
1390
+ self.nextRenderBuffer.drawText(isThreadedLen, x + 1, y + row, fg, bg, 0) catch {};
1391
+ row += 1;
1392
+ }
1393
+ };