@fairyhunter13/opentui-core 0.1.114 → 0.1.115

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/3d/SpriteResourceManager.d.ts +74 -0
  2. package/3d/SpriteUtils.d.ts +13 -0
  3. package/3d/TextureUtils.d.ts +24 -0
  4. package/3d/ThreeRenderable.d.ts +40 -0
  5. package/3d/WGPURenderer.d.ts +61 -0
  6. package/3d/animation/ExplodingSpriteEffect.d.ts +71 -0
  7. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +76 -0
  8. package/3d/animation/SpriteAnimator.d.ts +124 -0
  9. package/3d/animation/SpriteParticleGenerator.d.ts +62 -0
  10. package/3d/canvas.d.ts +44 -0
  11. package/3d/index.d.ts +12 -0
  12. package/3d/physics/PlanckPhysicsAdapter.d.ts +19 -0
  13. package/3d/physics/RapierPhysicsAdapter.d.ts +19 -0
  14. package/3d/physics/physics-interface.d.ts +27 -0
  15. package/3d.d.ts +2 -0
  16. package/3d.js +34041 -0
  17. package/3d.js.map +155 -0
  18. package/LICENSE +21 -0
  19. package/NativeSpanFeed.d.ts +41 -0
  20. package/Renderable.d.ts +334 -0
  21. package/animation/Timeline.d.ts +126 -0
  22. package/ansi.d.ts +13 -0
  23. package/buffer.d.ts +111 -0
  24. package/console.d.ts +144 -0
  25. package/edit-buffer.d.ts +98 -0
  26. package/editor-view.d.ts +73 -0
  27. package/index-j4m38kjn.js +411 -0
  28. package/index-j4m38kjn.js.map +10 -0
  29. package/index-tse8gzh0.js +20614 -0
  30. package/index-tse8gzh0.js.map +67 -0
  31. package/index-vv2jcd4r.js +12299 -0
  32. package/index-vv2jcd4r.js.map +42 -0
  33. package/index.d.ts +23 -0
  34. package/index.js +478 -0
  35. package/index.js.map +9 -0
  36. package/lib/KeyHandler.d.ts +61 -0
  37. package/lib/RGBA.d.ts +25 -0
  38. package/lib/ascii.font.d.ts +508 -0
  39. package/lib/border.d.ts +51 -0
  40. package/lib/bunfs.d.ts +7 -0
  41. package/lib/clipboard.d.ts +17 -0
  42. package/lib/clock.d.ts +15 -0
  43. package/lib/data-paths.d.ts +26 -0
  44. package/lib/debounce.d.ts +42 -0
  45. package/lib/detect-links.d.ts +6 -0
  46. package/lib/env.d.ts +42 -0
  47. package/lib/extmarks-history.d.ts +17 -0
  48. package/lib/extmarks.d.ts +89 -0
  49. package/lib/hast-styled-text.d.ts +17 -0
  50. package/lib/index.d.ts +21 -0
  51. package/lib/keymapping.d.ts +25 -0
  52. package/lib/objects-in-viewport.d.ts +24 -0
  53. package/lib/output.capture.d.ts +24 -0
  54. package/lib/parse.keypress-kitty.d.ts +2 -0
  55. package/lib/parse.keypress.d.ts +26 -0
  56. package/lib/parse.mouse.d.ts +30 -0
  57. package/lib/paste.d.ts +7 -0
  58. package/lib/queue.d.ts +15 -0
  59. package/lib/renderable.validations.d.ts +12 -0
  60. package/lib/scroll-acceleration.d.ts +43 -0
  61. package/lib/selection.d.ts +63 -0
  62. package/lib/singleton.d.ts +7 -0
  63. package/lib/stdin-parser.d.ts +87 -0
  64. package/lib/styled-text.d.ts +63 -0
  65. package/lib/terminal-capability-detection.d.ts +30 -0
  66. package/lib/terminal-palette.d.ts +50 -0
  67. package/lib/tree-sitter/assets/update.d.ts +11 -0
  68. package/lib/tree-sitter/client.d.ts +47 -0
  69. package/lib/tree-sitter/default-parsers.d.ts +2 -0
  70. package/lib/tree-sitter/download-utils.d.ts +21 -0
  71. package/lib/tree-sitter/index.d.ts +8 -0
  72. package/lib/tree-sitter/parser.worker.d.ts +1 -0
  73. package/lib/tree-sitter/parsers-config.d.ts +53 -0
  74. package/lib/tree-sitter/resolve-ft.d.ts +5 -0
  75. package/lib/tree-sitter/types.d.ts +82 -0
  76. package/lib/tree-sitter-styled-text.d.ts +14 -0
  77. package/lib/validate-dir-name.d.ts +1 -0
  78. package/lib/yoga.options.d.ts +32 -0
  79. package/package.json +50 -62
  80. package/parser.worker.js +899 -0
  81. package/parser.worker.js.map +12 -0
  82. package/plugins/core-slot.d.ts +72 -0
  83. package/plugins/registry.d.ts +42 -0
  84. package/plugins/types.d.ts +34 -0
  85. package/post/effects.d.ts +147 -0
  86. package/post/filters.d.ts +65 -0
  87. package/post/matrices.d.ts +20 -0
  88. package/renderables/ASCIIFont.d.ts +52 -0
  89. package/renderables/Box.d.ts +81 -0
  90. package/renderables/Code.d.ts +78 -0
  91. package/renderables/Diff.d.ts +142 -0
  92. package/renderables/EditBufferRenderable.d.ts +237 -0
  93. package/renderables/FrameBuffer.d.ts +16 -0
  94. package/renderables/Input.d.ts +67 -0
  95. package/renderables/LineNumberRenderable.d.ts +78 -0
  96. package/renderables/Markdown.d.ts +185 -0
  97. package/renderables/ScrollBar.d.ts +77 -0
  98. package/renderables/ScrollBox.d.ts +124 -0
  99. package/renderables/Select.d.ts +115 -0
  100. package/renderables/Slider.d.ts +47 -0
  101. package/renderables/TabSelect.d.ts +96 -0
  102. package/renderables/Text.d.ts +36 -0
  103. package/renderables/TextBufferRenderable.d.ts +105 -0
  104. package/renderables/TextNode.d.ts +91 -0
  105. package/renderables/TextTable.d.ts +140 -0
  106. package/renderables/Textarea.d.ts +63 -0
  107. package/renderables/TimeToFirstDraw.d.ts +24 -0
  108. package/renderables/__tests__/renderable-test-utils.d.ts +12 -0
  109. package/renderables/composition/VRenderable.d.ts +16 -0
  110. package/renderables/composition/constructs.d.ts +35 -0
  111. package/renderables/composition/vnode.d.ts +46 -0
  112. package/renderables/index.d.ts +23 -0
  113. package/renderables/markdown-parser.d.ts +10 -0
  114. package/renderer.d.ts +419 -0
  115. package/runtime-plugin-support.d.ts +3 -0
  116. package/runtime-plugin-support.js +29 -0
  117. package/runtime-plugin-support.js.map +10 -0
  118. package/runtime-plugin.d.ts +16 -0
  119. package/runtime-plugin.js +16 -0
  120. package/runtime-plugin.js.map +9 -0
  121. package/syntax-style.d.ts +54 -0
  122. package/testing/manual-clock.d.ts +17 -0
  123. package/testing/mock-keys.d.ts +81 -0
  124. package/testing/mock-mouse.d.ts +38 -0
  125. package/testing/mock-tree-sitter-client.d.ts +23 -0
  126. package/testing/spy.d.ts +7 -0
  127. package/testing/test-recorder.d.ts +61 -0
  128. package/testing/test-renderer.d.ts +23 -0
  129. package/testing.d.ts +6 -0
  130. package/testing.js +697 -0
  131. package/testing.js.map +15 -0
  132. package/text-buffer-view.d.ts +42 -0
  133. package/text-buffer.d.ts +67 -0
  134. package/types.d.ts +139 -0
  135. package/utils.d.ts +14 -0
  136. package/zig-structs.d.ts +155 -0
  137. package/zig.d.ts +353 -0
  138. package/dev/keypress-debug-renderer.ts +0 -148
  139. package/dev/keypress-debug.ts +0 -43
  140. package/dev/print-env-vars.ts +0 -32
  141. package/dev/test-tmux-graphics-334.sh +0 -68
  142. package/dev/thai-debug-test.ts +0 -68
  143. package/docs/development.md +0 -144
  144. package/scripts/build.ts +0 -400
  145. package/scripts/publish.ts +0 -60
  146. package/src/3d/SpriteResourceManager.ts +0 -286
  147. package/src/3d/SpriteUtils.ts +0 -70
  148. package/src/3d/TextureUtils.ts +0 -196
  149. package/src/3d/ThreeRenderable.ts +0 -197
  150. package/src/3d/WGPURenderer.ts +0 -294
  151. package/src/3d/animation/ExplodingSpriteEffect.ts +0 -513
  152. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +0 -429
  153. package/src/3d/animation/SpriteAnimator.ts +0 -633
  154. package/src/3d/animation/SpriteParticleGenerator.ts +0 -435
  155. package/src/3d/canvas.ts +0 -464
  156. package/src/3d/index.ts +0 -12
  157. package/src/3d/physics/PlanckPhysicsAdapter.ts +0 -72
  158. package/src/3d/physics/RapierPhysicsAdapter.ts +0 -66
  159. package/src/3d/physics/physics-interface.ts +0 -31
  160. package/src/3d/shaders/supersampling.wgsl +0 -201
  161. package/src/3d.ts +0 -3
  162. package/src/NativeSpanFeed.ts +0 -300
  163. package/src/Renderable.ts +0 -1704
  164. package/src/__snapshots__/buffer.test.ts.snap +0 -28
  165. package/src/animation/Timeline.test.ts +0 -2709
  166. package/src/animation/Timeline.ts +0 -598
  167. package/src/ansi.ts +0 -18
  168. package/src/benchmark/attenuation-benchmark.ts +0 -81
  169. package/src/benchmark/colormatrix-benchmark.ts +0 -128
  170. package/src/benchmark/gain-benchmark.ts +0 -80
  171. package/src/benchmark/latest-all-bench-run.json +0 -707
  172. package/src/benchmark/latest-async-bench-run.json +0 -336
  173. package/src/benchmark/latest-default-bench-run.json +0 -657
  174. package/src/benchmark/latest-large-bench-run.json +0 -707
  175. package/src/benchmark/latest-quick-bench-run.json +0 -207
  176. package/src/benchmark/markdown-benchmark.ts +0 -1796
  177. package/src/benchmark/native-span-feed-async-benchmark.ts +0 -355
  178. package/src/benchmark/native-span-feed-benchmark.md +0 -56
  179. package/src/benchmark/native-span-feed-benchmark.ts +0 -596
  180. package/src/benchmark/native-span-feed-compare.ts +0 -280
  181. package/src/benchmark/renderer-benchmark.ts +0 -754
  182. package/src/benchmark/text-table-benchmark.ts +0 -948
  183. package/src/buffer.test.ts +0 -291
  184. package/src/buffer.ts +0 -554
  185. package/src/console.test.ts +0 -612
  186. package/src/console.ts +0 -1254
  187. package/src/edit-buffer.test.ts +0 -1769
  188. package/src/edit-buffer.ts +0 -411
  189. package/src/editor-view.test.ts +0 -1032
  190. package/src/editor-view.ts +0 -284
  191. package/src/examples/ascii-font-selection-demo.ts +0 -245
  192. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  193. package/src/examples/assets/concrete.png +0 -0
  194. package/src/examples/assets/crate.png +0 -0
  195. package/src/examples/assets/crate_emissive.png +0 -0
  196. package/src/examples/assets/forrest_background.png +0 -0
  197. package/src/examples/assets/hast-example.json +0 -1018
  198. package/src/examples/assets/heart.png +0 -0
  199. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  200. package/src/examples/assets/main_char_idle.png +0 -0
  201. package/src/examples/assets/main_char_jump_end.png +0 -0
  202. package/src/examples/assets/main_char_jump_landing.png +0 -0
  203. package/src/examples/assets/main_char_jump_start.png +0 -0
  204. package/src/examples/assets/main_char_run_loop.png +0 -0
  205. package/src/examples/assets/roughness_map.jpg +0 -0
  206. package/src/examples/build.ts +0 -115
  207. package/src/examples/code-demo.ts +0 -924
  208. package/src/examples/console-demo.ts +0 -358
  209. package/src/examples/core-plugin-slots-demo.ts +0 -759
  210. package/src/examples/diff-demo.ts +0 -701
  211. package/src/examples/draggable-three-demo.ts +0 -259
  212. package/src/examples/editor-demo.ts +0 -322
  213. package/src/examples/extmarks-demo.ts +0 -196
  214. package/src/examples/focus-restore-demo.ts +0 -310
  215. package/src/examples/fonts.ts +0 -245
  216. package/src/examples/fractal-shader-demo.ts +0 -268
  217. package/src/examples/framebuffer-demo.ts +0 -674
  218. package/src/examples/full-unicode-demo.ts +0 -241
  219. package/src/examples/golden-star-demo.ts +0 -933
  220. package/src/examples/grayscale-buffer-demo.ts +0 -249
  221. package/src/examples/hast-syntax-highlighting-demo.ts +0 -129
  222. package/src/examples/index.ts +0 -926
  223. package/src/examples/input-demo.ts +0 -377
  224. package/src/examples/input-select-layout-demo.ts +0 -425
  225. package/src/examples/install.sh +0 -143
  226. package/src/examples/keypress-debug-demo.ts +0 -452
  227. package/src/examples/lib/HexList.ts +0 -122
  228. package/src/examples/lib/PaletteGrid.ts +0 -125
  229. package/src/examples/lib/standalone-keys.ts +0 -25
  230. package/src/examples/lib/tab-controller.ts +0 -243
  231. package/src/examples/lights-phong-demo.ts +0 -290
  232. package/src/examples/link-demo.ts +0 -220
  233. package/src/examples/live-state-demo.ts +0 -480
  234. package/src/examples/markdown-demo.ts +0 -725
  235. package/src/examples/mouse-interaction-demo.ts +0 -428
  236. package/src/examples/nested-zindex-demo.ts +0 -357
  237. package/src/examples/opacity-example.ts +0 -235
  238. package/src/examples/opentui-demo.ts +0 -1057
  239. package/src/examples/physx-planck-2d-demo.ts +0 -623
  240. package/src/examples/physx-rapier-2d-demo.ts +0 -655
  241. package/src/examples/relative-positioning-demo.ts +0 -323
  242. package/src/examples/scroll-example.ts +0 -214
  243. package/src/examples/scrollbox-mouse-test.ts +0 -112
  244. package/src/examples/scrollbox-overlay-hit-test.ts +0 -206
  245. package/src/examples/select-demo.ts +0 -237
  246. package/src/examples/shader-cube-demo.ts +0 -1015
  247. package/src/examples/simple-layout-example.ts +0 -591
  248. package/src/examples/slider-demo.ts +0 -617
  249. package/src/examples/split-mode-demo.ts +0 -453
  250. package/src/examples/sprite-animation-demo.ts +0 -443
  251. package/src/examples/sprite-particle-generator-demo.ts +0 -486
  252. package/src/examples/static-sprite-demo.ts +0 -193
  253. package/src/examples/sticky-scroll-example.ts +0 -308
  254. package/src/examples/styled-text-demo.ts +0 -282
  255. package/src/examples/tab-select-demo.ts +0 -219
  256. package/src/examples/terminal-title.ts +0 -29
  257. package/src/examples/terminal.ts +0 -305
  258. package/src/examples/text-node-demo.ts +0 -416
  259. package/src/examples/text-selection-demo.ts +0 -377
  260. package/src/examples/text-table-demo.ts +0 -503
  261. package/src/examples/text-truncation-demo.ts +0 -481
  262. package/src/examples/text-wrap.ts +0 -757
  263. package/src/examples/texture-loading-demo.ts +0 -259
  264. package/src/examples/timeline-example.ts +0 -670
  265. package/src/examples/transparency-demo.ts +0 -400
  266. package/src/examples/vnode-composition-demo.ts +0 -404
  267. package/src/examples/wide-grapheme-overlay-demo.ts +0 -280
  268. package/src/index.ts +0 -24
  269. package/src/lib/KeyHandler.integration.test.ts +0 -292
  270. package/src/lib/KeyHandler.stopPropagation.test.ts +0 -289
  271. package/src/lib/KeyHandler.test.ts +0 -662
  272. package/src/lib/KeyHandler.ts +0 -222
  273. package/src/lib/RGBA.test.ts +0 -984
  274. package/src/lib/RGBA.ts +0 -204
  275. package/src/lib/ascii.font.ts +0 -330
  276. package/src/lib/border.test.ts +0 -83
  277. package/src/lib/border.ts +0 -170
  278. package/src/lib/bunfs.test.ts +0 -27
  279. package/src/lib/bunfs.ts +0 -18
  280. package/src/lib/clipboard.test.ts +0 -41
  281. package/src/lib/clipboard.ts +0 -47
  282. package/src/lib/clock.ts +0 -35
  283. package/src/lib/data-paths.test.ts +0 -133
  284. package/src/lib/data-paths.ts +0 -109
  285. package/src/lib/debounce.ts +0 -106
  286. package/src/lib/detect-links.test.ts +0 -98
  287. package/src/lib/detect-links.ts +0 -56
  288. package/src/lib/env.test.ts +0 -228
  289. package/src/lib/env.ts +0 -209
  290. package/src/lib/extmarks-history.ts +0 -51
  291. package/src/lib/extmarks-multiwidth.test.ts +0 -322
  292. package/src/lib/extmarks.test.ts +0 -3457
  293. package/src/lib/extmarks.ts +0 -843
  294. package/src/lib/fonts/block.json +0 -405
  295. package/src/lib/fonts/grid.json +0 -265
  296. package/src/lib/fonts/huge.json +0 -741
  297. package/src/lib/fonts/pallet.json +0 -314
  298. package/src/lib/fonts/shade.json +0 -591
  299. package/src/lib/fonts/slick.json +0 -321
  300. package/src/lib/fonts/tiny.json +0 -69
  301. package/src/lib/hast-styled-text.ts +0 -59
  302. package/src/lib/index.ts +0 -21
  303. package/src/lib/keymapping.test.ts +0 -317
  304. package/src/lib/keymapping.ts +0 -115
  305. package/src/lib/objects-in-viewport.test.ts +0 -787
  306. package/src/lib/objects-in-viewport.ts +0 -153
  307. package/src/lib/output.capture.ts +0 -58
  308. package/src/lib/parse.keypress-kitty.protocol.test.ts +0 -340
  309. package/src/lib/parse.keypress-kitty.test.ts +0 -663
  310. package/src/lib/parse.keypress-kitty.ts +0 -439
  311. package/src/lib/parse.keypress.test.ts +0 -1849
  312. package/src/lib/parse.keypress.ts +0 -397
  313. package/src/lib/parse.mouse.test.ts +0 -552
  314. package/src/lib/parse.mouse.ts +0 -232
  315. package/src/lib/paste.ts +0 -16
  316. package/src/lib/queue.ts +0 -65
  317. package/src/lib/renderable.validations.test.ts +0 -87
  318. package/src/lib/renderable.validations.ts +0 -83
  319. package/src/lib/scroll-acceleration.ts +0 -98
  320. package/src/lib/selection.ts +0 -240
  321. package/src/lib/singleton.ts +0 -28
  322. package/src/lib/stdin-parser.test.ts +0 -2290
  323. package/src/lib/stdin-parser.ts +0 -1810
  324. package/src/lib/styled-text.ts +0 -178
  325. package/src/lib/terminal-capability-detection.test.ts +0 -202
  326. package/src/lib/terminal-capability-detection.ts +0 -79
  327. package/src/lib/terminal-palette.test.ts +0 -878
  328. package/src/lib/terminal-palette.ts +0 -383
  329. package/src/lib/tree-sitter/assets/README.md +0 -118
  330. package/src/lib/tree-sitter/assets/update.ts +0 -334
  331. package/src/lib/tree-sitter/assets.d.ts +0 -9
  332. package/src/lib/tree-sitter/cache.test.ts +0 -273
  333. package/src/lib/tree-sitter/client.test.ts +0 -1165
  334. package/src/lib/tree-sitter/client.ts +0 -607
  335. package/src/lib/tree-sitter/default-parsers.ts +0 -86
  336. package/src/lib/tree-sitter/download-utils.ts +0 -148
  337. package/src/lib/tree-sitter/index.ts +0 -28
  338. package/src/lib/tree-sitter/parser.worker.ts +0 -1042
  339. package/src/lib/tree-sitter/parsers-config.ts +0 -81
  340. package/src/lib/tree-sitter/resolve-ft.test.ts +0 -55
  341. package/src/lib/tree-sitter/resolve-ft.ts +0 -189
  342. package/src/lib/tree-sitter/types.ts +0 -82
  343. package/src/lib/tree-sitter-styled-text.test.ts +0 -1253
  344. package/src/lib/tree-sitter-styled-text.ts +0 -306
  345. package/src/lib/validate-dir-name.ts +0 -55
  346. package/src/lib/yoga.options.test.ts +0 -628
  347. package/src/lib/yoga.options.ts +0 -346
  348. package/src/plugins/core-slot.ts +0 -579
  349. package/src/plugins/registry.ts +0 -402
  350. package/src/plugins/types.ts +0 -46
  351. package/src/post/effects.ts +0 -930
  352. package/src/post/filters.ts +0 -489
  353. package/src/post/matrices.ts +0 -288
  354. package/src/renderables/ASCIIFont.ts +0 -219
  355. package/src/renderables/Box.test.ts +0 -205
  356. package/src/renderables/Box.ts +0 -326
  357. package/src/renderables/Code.test.ts +0 -2062
  358. package/src/renderables/Code.ts +0 -357
  359. package/src/renderables/Diff.regression.test.ts +0 -226
  360. package/src/renderables/Diff.test.ts +0 -3101
  361. package/src/renderables/Diff.ts +0 -1211
  362. package/src/renderables/EditBufferRenderable.test.ts +0 -288
  363. package/src/renderables/EditBufferRenderable.ts +0 -1166
  364. package/src/renderables/FrameBuffer.ts +0 -47
  365. package/src/renderables/Input.test.ts +0 -1228
  366. package/src/renderables/Input.ts +0 -247
  367. package/src/renderables/LineNumberRenderable.ts +0 -724
  368. package/src/renderables/Markdown.ts +0 -1393
  369. package/src/renderables/ScrollBar.ts +0 -422
  370. package/src/renderables/ScrollBox.ts +0 -883
  371. package/src/renderables/Select.test.ts +0 -1033
  372. package/src/renderables/Select.ts +0 -524
  373. package/src/renderables/Slider.test.ts +0 -456
  374. package/src/renderables/Slider.ts +0 -342
  375. package/src/renderables/TabSelect.test.ts +0 -197
  376. package/src/renderables/TabSelect.ts +0 -455
  377. package/src/renderables/Text.selection-buffer.test.ts +0 -123
  378. package/src/renderables/Text.test.ts +0 -2660
  379. package/src/renderables/Text.ts +0 -147
  380. package/src/renderables/TextBufferRenderable.ts +0 -518
  381. package/src/renderables/TextNode.test.ts +0 -1058
  382. package/src/renderables/TextNode.ts +0 -325
  383. package/src/renderables/TextTable.test.ts +0 -1421
  384. package/src/renderables/TextTable.ts +0 -1344
  385. package/src/renderables/Textarea.ts +0 -430
  386. package/src/renderables/TimeToFirstDraw.ts +0 -89
  387. package/src/renderables/__snapshots__/Code.test.ts.snap +0 -13
  388. package/src/renderables/__snapshots__/Diff.test.ts.snap +0 -785
  389. package/src/renderables/__snapshots__/Text.test.ts.snap +0 -421
  390. package/src/renderables/__snapshots__/TextTable.test.ts.snap +0 -215
  391. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +0 -144
  392. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +0 -816
  393. package/src/renderables/__tests__/LineNumberRenderable.test.ts +0 -1865
  394. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +0 -85
  395. package/src/renderables/__tests__/Markdown.code-colors.test.ts +0 -242
  396. package/src/renderables/__tests__/Markdown.test.ts +0 -2518
  397. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +0 -87
  398. package/src/renderables/__tests__/Textarea.buffer.test.ts +0 -682
  399. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +0 -675
  400. package/src/renderables/__tests__/Textarea.editing.test.ts +0 -2041
  401. package/src/renderables/__tests__/Textarea.error-handling.test.ts +0 -35
  402. package/src/renderables/__tests__/Textarea.events.test.ts +0 -738
  403. package/src/renderables/__tests__/Textarea.highlights.test.ts +0 -590
  404. package/src/renderables/__tests__/Textarea.keybinding.test.ts +0 -3149
  405. package/src/renderables/__tests__/Textarea.paste.test.ts +0 -357
  406. package/src/renderables/__tests__/Textarea.rendering.test.ts +0 -1866
  407. package/src/renderables/__tests__/Textarea.scroll.test.ts +0 -733
  408. package/src/renderables/__tests__/Textarea.selection.test.ts +0 -1590
  409. package/src/renderables/__tests__/Textarea.stress.test.ts +0 -670
  410. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +0 -383
  411. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +0 -310
  412. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +0 -221
  413. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +0 -89
  414. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +0 -457
  415. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +0 -158
  416. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +0 -387
  417. package/src/renderables/__tests__/markdown-parser.test.ts +0 -217
  418. package/src/renderables/__tests__/renderable-test-utils.ts +0 -60
  419. package/src/renderables/composition/README.md +0 -8
  420. package/src/renderables/composition/VRenderable.ts +0 -32
  421. package/src/renderables/composition/constructs.ts +0 -127
  422. package/src/renderables/composition/vnode.ts +0 -289
  423. package/src/renderables/index.ts +0 -23
  424. package/src/renderables/markdown-parser.ts +0 -66
  425. package/src/renderer.ts +0 -2681
  426. package/src/runtime-plugin-support.ts +0 -39
  427. package/src/runtime-plugin.ts +0 -615
  428. package/src/syntax-style.test.ts +0 -841
  429. package/src/syntax-style.ts +0 -257
  430. package/src/testing/README.md +0 -210
  431. package/src/testing/capture-spans.test.ts +0 -194
  432. package/src/testing/integration.test.ts +0 -276
  433. package/src/testing/manual-clock.ts +0 -117
  434. package/src/testing/mock-keys.test.ts +0 -1378
  435. package/src/testing/mock-keys.ts +0 -457
  436. package/src/testing/mock-mouse.test.ts +0 -218
  437. package/src/testing/mock-mouse.ts +0 -247
  438. package/src/testing/mock-tree-sitter-client.ts +0 -73
  439. package/src/testing/spy.ts +0 -13
  440. package/src/testing/test-recorder.test.ts +0 -415
  441. package/src/testing/test-recorder.ts +0 -145
  442. package/src/testing/test-renderer.ts +0 -132
  443. package/src/testing.ts +0 -7
  444. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +0 -481
  445. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +0 -19
  446. package/src/tests/__snapshots__/scrollbox.test.ts.snap +0 -29
  447. package/src/tests/absolute-positioning.snapshot.test.ts +0 -638
  448. package/src/tests/allocator-stats.test.ts +0 -38
  449. package/src/tests/destroy-during-render.test.ts +0 -200
  450. package/src/tests/destroy-on-exit.fixture.ts +0 -36
  451. package/src/tests/destroy-on-exit.test.ts +0 -41
  452. package/src/tests/hover-cursor.test.ts +0 -98
  453. package/src/tests/native-span-feed-async.test.ts +0 -173
  454. package/src/tests/native-span-feed-close.test.ts +0 -120
  455. package/src/tests/native-span-feed-coverage.test.ts +0 -227
  456. package/src/tests/native-span-feed-edge-cases.test.ts +0 -352
  457. package/src/tests/native-span-feed-use-after-free.test.ts +0 -45
  458. package/src/tests/opacity.test.ts +0 -123
  459. package/src/tests/renderable.snapshot.test.ts +0 -524
  460. package/src/tests/renderable.test.ts +0 -1281
  461. package/src/tests/renderer.clock.test.ts +0 -158
  462. package/src/tests/renderer.console-startup.test.ts +0 -185
  463. package/src/tests/renderer.control.test.ts +0 -425
  464. package/src/tests/renderer.core-slot-binding.test.ts +0 -952
  465. package/src/tests/renderer.cursor.test.ts +0 -26
  466. package/src/tests/renderer.destroy-during-render.test.ts +0 -147
  467. package/src/tests/renderer.focus-restore.test.ts +0 -257
  468. package/src/tests/renderer.focus.test.ts +0 -294
  469. package/src/tests/renderer.idle.test.ts +0 -219
  470. package/src/tests/renderer.input.test.ts +0 -2237
  471. package/src/tests/renderer.kitty-flags.test.ts +0 -195
  472. package/src/tests/renderer.mouse.test.ts +0 -1274
  473. package/src/tests/renderer.palette.test.ts +0 -629
  474. package/src/tests/renderer.selection.test.ts +0 -49
  475. package/src/tests/renderer.slot-registry.test.ts +0 -684
  476. package/src/tests/renderer.useMouse.test.ts +0 -47
  477. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +0 -76
  478. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +0 -43
  479. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +0 -67
  480. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +0 -72
  481. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +0 -44
  482. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +0 -85
  483. package/src/tests/runtime-plugin-path-alias.fixture.ts +0 -43
  484. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +0 -65
  485. package/src/tests/runtime-plugin-support.fixture.ts +0 -11
  486. package/src/tests/runtime-plugin-support.test.ts +0 -19
  487. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +0 -30
  488. package/src/tests/runtime-plugin.fixture.ts +0 -40
  489. package/src/tests/runtime-plugin.test.ts +0 -354
  490. package/src/tests/scrollbox-culling-bug.test.ts +0 -114
  491. package/src/tests/scrollbox-hitgrid-resize.test.ts +0 -136
  492. package/src/tests/scrollbox-hitgrid.test.ts +0 -909
  493. package/src/tests/scrollbox.test.ts +0 -1530
  494. package/src/tests/wrap-resize-perf.test.ts +0 -276
  495. package/src/tests/yoga-setters.test.ts +0 -921
  496. package/src/text-buffer-view.test.ts +0 -705
  497. package/src/text-buffer-view.ts +0 -189
  498. package/src/text-buffer.test.ts +0 -347
  499. package/src/text-buffer.ts +0 -250
  500. package/src/types.ts +0 -161
  501. package/src/utils.ts +0 -88
  502. package/src/zig/ansi.zig +0 -268
  503. package/src/zig/bench/README.md +0 -50
  504. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +0 -887
  505. package/src/zig/bench/edit-buffer_bench.zig +0 -476
  506. package/src/zig/bench/native-span-feed_bench.zig +0 -100
  507. package/src/zig/bench/rope-markers_bench.zig +0 -713
  508. package/src/zig/bench/rope_bench.zig +0 -514
  509. package/src/zig/bench/styled-text_bench.zig +0 -470
  510. package/src/zig/bench/text-buffer-coords_bench.zig +0 -362
  511. package/src/zig/bench/text-buffer-view_bench.zig +0 -459
  512. package/src/zig/bench/text-chunk-graphemes_bench.zig +0 -273
  513. package/src/zig/bench/utf8_bench.zig +0 -799
  514. package/src/zig/bench-utils.zig +0 -431
  515. package/src/zig/bench.zig +0 -217
  516. package/src/zig/buffer-methods.zig +0 -211
  517. package/src/zig/buffer.zig +0 -2281
  518. package/src/zig/build.zig +0 -289
  519. package/src/zig/build.zig.zon +0 -16
  520. package/src/zig/edit-buffer.zig +0 -825
  521. package/src/zig/editor-view.zig +0 -802
  522. package/src/zig/event-bus.zig +0 -13
  523. package/src/zig/event-emitter.zig +0 -65
  524. package/src/zig/file-logger.zig +0 -92
  525. package/src/zig/grapheme.zig +0 -599
  526. package/src/zig/lib.zig +0 -1854
  527. package/src/zig/link.zig +0 -333
  528. package/src/zig/logger.zig +0 -43
  529. package/src/zig/mem-registry.zig +0 -125
  530. package/src/zig/native-span-feed-bench-lib.zig +0 -7
  531. package/src/zig/native-span-feed.zig +0 -708
  532. package/src/zig/renderer.zig +0 -1393
  533. package/src/zig/rope.zig +0 -1220
  534. package/src/zig/syntax-style.zig +0 -161
  535. package/src/zig/terminal.zig +0 -987
  536. package/src/zig/test.zig +0 -72
  537. package/src/zig/tests/README.md +0 -18
  538. package/src/zig/tests/buffer-methods_test.zig +0 -1109
  539. package/src/zig/tests/buffer_test.zig +0 -2557
  540. package/src/zig/tests/edit-buffer-history_test.zig +0 -271
  541. package/src/zig/tests/edit-buffer_test.zig +0 -1689
  542. package/src/zig/tests/editor-view_test.zig +0 -3299
  543. package/src/zig/tests/event-emitter_test.zig +0 -249
  544. package/src/zig/tests/grapheme_test.zig +0 -1304
  545. package/src/zig/tests/link_test.zig +0 -190
  546. package/src/zig/tests/mem-registry_test.zig +0 -473
  547. package/src/zig/tests/memory_leak_regression_test.zig +0 -159
  548. package/src/zig/tests/native-span-feed_test.zig +0 -1264
  549. package/src/zig/tests/renderer_test.zig +0 -1017
  550. package/src/zig/tests/rope-nested_test.zig +0 -712
  551. package/src/zig/tests/rope_fuzz_test.zig +0 -238
  552. package/src/zig/tests/rope_test.zig +0 -2362
  553. package/src/zig/tests/segment-merge.test.zig +0 -148
  554. package/src/zig/tests/syntax-style_test.zig +0 -557
  555. package/src/zig/tests/terminal_test.zig +0 -754
  556. package/src/zig/tests/text-buffer-drawing_test.zig +0 -3237
  557. package/src/zig/tests/text-buffer-highlights_test.zig +0 -666
  558. package/src/zig/tests/text-buffer-iterators_test.zig +0 -776
  559. package/src/zig/tests/text-buffer-segment_test.zig +0 -320
  560. package/src/zig/tests/text-buffer-selection_test.zig +0 -1035
  561. package/src/zig/tests/text-buffer-selection_viewport_test.zig +0 -358
  562. package/src/zig/tests/text-buffer-view_test.zig +0 -3649
  563. package/src/zig/tests/text-buffer_test.zig +0 -2191
  564. package/src/zig/tests/unicode-width-map.zon +0 -3909
  565. package/src/zig/tests/utf8_no_zwj_test.zig +0 -260
  566. package/src/zig/tests/utf8_test.zig +0 -4057
  567. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +0 -267
  568. package/src/zig/tests/utf8_wcwidth_test.zig +0 -357
  569. package/src/zig/tests/word-wrap-editing_test.zig +0 -498
  570. package/src/zig/tests/wrap-cache-perf_test.zig +0 -113
  571. package/src/zig/text-buffer-iterators.zig +0 -499
  572. package/src/zig/text-buffer-segment.zig +0 -404
  573. package/src/zig/text-buffer-view.zig +0 -1371
  574. package/src/zig/text-buffer.zig +0 -1180
  575. package/src/zig/utf8.zig +0 -1948
  576. package/src/zig/utils.zig +0 -9
  577. package/src/zig-structs.ts +0 -261
  578. package/src/zig.ts +0 -3884
  579. package/tsconfig.build.json +0 -24
  580. package/tsconfig.json +0 -27
  581. /package/{src/lib/tree-sitter/assets → assets}/javascript/highlights.scm +0 -0
  582. /package/{src/lib/tree-sitter/assets → assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{src/lib/tree-sitter/assets → assets}/markdown/highlights.scm +0 -0
  584. /package/{src/lib/tree-sitter/assets → assets}/markdown/injections.scm +0 -0
  585. /package/{src/lib/tree-sitter/assets → assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{src/lib/tree-sitter/assets → assets}/typescript/highlights.scm +0 -0
  589. /package/{src/lib/tree-sitter/assets → assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{src/lib/tree-sitter/assets → assets}/zig/highlights.scm +0 -0
  591. /package/{src/lib/tree-sitter/assets → assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -1,2062 +0,0 @@
1
- import { test, expect, beforeEach, afterEach } from "bun:test"
2
- import { CodeRenderable } from "./Code.js"
3
- import { SyntaxStyle } from "../syntax-style.js"
4
- import { RGBA } from "../lib/RGBA.js"
5
- import { createTestRenderer, type TestRenderer, MockTreeSitterClient, type MockMouse } from "../testing.js"
6
- import { TreeSitterClient } from "../lib/tree-sitter/index.js"
7
- import type { SimpleHighlight } from "../lib/tree-sitter/types.js"
8
- import { BoxRenderable } from "./Box.js"
9
-
10
- let currentRenderer: TestRenderer
11
- let renderOnce: () => Promise<void>
12
- let captureFrame: () => string
13
- let mockMouse: MockMouse
14
- let resize: (width: number, height: number) => void
15
-
16
- beforeEach(async () => {
17
- const testRenderer = await createTestRenderer({ width: 80, height: 24 })
18
- currentRenderer = testRenderer.renderer
19
- renderOnce = testRenderer.renderOnce
20
- captureFrame = testRenderer.captureCharFrame
21
- mockMouse = testRenderer.mockMouse
22
- resize = testRenderer.resize
23
- })
24
-
25
- afterEach(async () => {
26
- if (currentRenderer) {
27
- currentRenderer.destroy()
28
- }
29
- })
30
-
31
- test("CodeRenderable - basic construction", async () => {
32
- const syntaxStyle = SyntaxStyle.fromStyles({
33
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
34
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
35
- string: { fg: RGBA.fromValues(0, 1, 0, 1) },
36
- })
37
-
38
- const codeRenderable = new CodeRenderable(currentRenderer, {
39
- id: "test-code",
40
- content: 'const message = "Hello, world!";',
41
- filetype: "javascript",
42
- syntaxStyle,
43
- conceal: false,
44
- })
45
-
46
- expect(codeRenderable.content).toBe('const message = "Hello, world!";')
47
- expect(codeRenderable.filetype).toBe("javascript")
48
- expect(codeRenderable.syntaxStyle).toBe(syntaxStyle)
49
- })
50
-
51
- test("CodeRenderable - content updates", async () => {
52
- const syntaxStyle = SyntaxStyle.fromStyles({
53
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
54
- })
55
-
56
- const codeRenderable = new CodeRenderable(currentRenderer, {
57
- id: "test-code",
58
- content: "original content",
59
- filetype: "javascript",
60
- syntaxStyle,
61
- conceal: false,
62
- })
63
-
64
- expect(codeRenderable.content).toBe("original content")
65
-
66
- codeRenderable.content = "updated content"
67
- expect(codeRenderable.content).toBe("updated content")
68
- })
69
-
70
- test("CodeRenderable - filetype updates", async () => {
71
- const syntaxStyle = SyntaxStyle.fromStyles({
72
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
73
- })
74
-
75
- const codeRenderable = new CodeRenderable(currentRenderer, {
76
- id: "test-code",
77
- content: "console.log('test');",
78
- filetype: "javascript",
79
- syntaxStyle,
80
- conceal: false,
81
- })
82
-
83
- expect(codeRenderable.filetype).toBe("javascript")
84
-
85
- codeRenderable.filetype = "typescript"
86
- expect(codeRenderable.filetype).toBe("typescript")
87
- })
88
-
89
- test("CodeRenderable - re-highlights when content changes during active highlighting", async () => {
90
- const syntaxStyle = SyntaxStyle.fromStyles({
91
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
92
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
93
- })
94
-
95
- const mockClient = new MockTreeSitterClient()
96
- mockClient.setMockResult({
97
- highlights: [
98
- [0, 5, "keyword"],
99
- [6, 13, "identifier"],
100
- ] as SimpleHighlight[],
101
- })
102
-
103
- const codeRenderable = new CodeRenderable(currentRenderer, {
104
- id: "test-code",
105
- content: "const message = 'hello';",
106
- filetype: "javascript",
107
- syntaxStyle,
108
- treeSitterClient: mockClient,
109
- conceal: false,
110
- })
111
-
112
- currentRenderer.root.add(codeRenderable)
113
- await renderOnce()
114
-
115
- expect(mockClient.isHighlighting()).toBe(true)
116
-
117
- codeRenderable.content = "let newMessage = 'world';"
118
-
119
- expect(codeRenderable.content).toBe("let newMessage = 'world';")
120
-
121
- await renderOnce()
122
- expect(mockClient.isHighlighting()).toBe(true)
123
-
124
- mockClient.resolveHighlightOnce(0)
125
- await new Promise((resolve) => setTimeout(resolve, 10))
126
-
127
- expect(mockClient.isHighlighting()).toBe(true)
128
-
129
- mockClient.resolveHighlightOnce(0)
130
- await new Promise((resolve) => setTimeout(resolve, 10))
131
-
132
- expect(mockClient.isHighlighting()).toBe(false)
133
- })
134
-
135
- test("CodeRenderable - multiple content changes during highlighting", async () => {
136
- const syntaxStyle = SyntaxStyle.fromStyles({
137
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
138
- })
139
-
140
- const mockClient = new MockTreeSitterClient()
141
- mockClient.setMockResult({ highlights: [] })
142
-
143
- const codeRenderable = new CodeRenderable(currentRenderer, {
144
- id: "test-code",
145
- content: "original content",
146
- filetype: "javascript",
147
- syntaxStyle,
148
- treeSitterClient: mockClient,
149
- conceal: false,
150
- })
151
-
152
- currentRenderer.root.add(codeRenderable)
153
- await renderOnce()
154
-
155
- expect(mockClient.isHighlighting()).toBe(true)
156
-
157
- codeRenderable.content = "first change"
158
- codeRenderable.content = "second change"
159
- codeRenderable.content = "final content"
160
-
161
- expect(codeRenderable.content).toBe("final content")
162
-
163
- await renderOnce()
164
- expect(mockClient.isHighlighting()).toBe(true)
165
-
166
- mockClient.resolveHighlightOnce(0)
167
-
168
- await new Promise((resolve) => setTimeout(resolve, 10))
169
-
170
- expect(mockClient.isHighlighting()).toBe(true)
171
-
172
- mockClient.resolveHighlightOnce(0)
173
- await new Promise((resolve) => setTimeout(resolve, 10))
174
-
175
- expect(mockClient.isHighlighting()).toBe(false)
176
- })
177
-
178
- test("CodeRenderable - uses fallback rendering when no filetype provided", async () => {
179
- const syntaxStyle = SyntaxStyle.fromStyles({
180
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
181
- })
182
-
183
- const codeRenderable = new CodeRenderable(currentRenderer, {
184
- id: "test-code",
185
- content: "const message = 'hello world';",
186
- syntaxStyle,
187
- conceal: false,
188
- })
189
-
190
- currentRenderer.root.add(codeRenderable)
191
- await renderOnce()
192
-
193
- expect(codeRenderable.content).toBe("const message = 'hello world';")
194
- expect(codeRenderable.filetype).toBeUndefined()
195
- expect(codeRenderable.plainText).toBe("const message = 'hello world';")
196
- })
197
-
198
- test("CodeRenderable - uses fallback rendering when highlighting throws error", async () => {
199
- const syntaxStyle = SyntaxStyle.fromStyles({
200
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
201
- })
202
-
203
- const mockClient = new MockTreeSitterClient()
204
-
205
- mockClient.highlightOnce = async () => {
206
- throw new Error("Highlighting failed")
207
- }
208
-
209
- const codeRenderable = new CodeRenderable(currentRenderer, {
210
- id: "test-code",
211
- content: "const message = 'hello world';",
212
- filetype: "javascript",
213
- syntaxStyle,
214
- treeSitterClient: mockClient,
215
- conceal: false,
216
- })
217
-
218
- currentRenderer.root.add(codeRenderable)
219
- await renderOnce()
220
-
221
- await new Promise((resolve) => setTimeout(resolve, 20))
222
- await renderOnce()
223
-
224
- expect(codeRenderable.content).toBe("const message = 'hello world';")
225
- expect(codeRenderable.filetype).toBe("javascript")
226
- expect(codeRenderable.plainText).toBe("const message = 'hello world';")
227
- })
228
-
229
- test("CodeRenderable - handles empty content", async () => {
230
- const syntaxStyle = SyntaxStyle.fromStyles({
231
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
232
- })
233
-
234
- const codeRenderable = new CodeRenderable(currentRenderer, {
235
- id: "test-code",
236
- content: "",
237
- filetype: "javascript",
238
- syntaxStyle,
239
- conceal: false,
240
- })
241
-
242
- await renderOnce()
243
-
244
- expect(codeRenderable.content).toBe("")
245
- expect(codeRenderable.filetype).toBe("javascript")
246
- expect(codeRenderable.plainText).toBe("")
247
- })
248
-
249
- test("CodeRenderable - empty content does not trigger highlighting", async () => {
250
- const syntaxStyle = SyntaxStyle.fromStyles({
251
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
252
- })
253
-
254
- const mockClient = new MockTreeSitterClient()
255
- mockClient.setMockResult({ highlights: [] })
256
-
257
- const codeRenderable = new CodeRenderable(currentRenderer, {
258
- id: "test-code",
259
- content: "const message = 'hello';",
260
- filetype: "javascript",
261
- syntaxStyle,
262
- treeSitterClient: mockClient,
263
- conceal: false,
264
- })
265
-
266
- currentRenderer.root.add(codeRenderable)
267
- await renderOnce()
268
-
269
- mockClient.resolveHighlightOnce(0)
270
- await new Promise((resolve) => setTimeout(resolve, 10))
271
- await renderOnce()
272
-
273
- expect(codeRenderable.content).toBe("const message = 'hello';")
274
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
275
-
276
- codeRenderable.content = ""
277
- await renderOnce()
278
-
279
- expect(mockClient.isHighlighting()).toBe(false)
280
- expect(codeRenderable.content).toBe("")
281
- })
282
-
283
- test("CodeRenderable - text renders immediately before highlighting completes", async () => {
284
- resize(32, 2)
285
-
286
- const syntaxStyle = SyntaxStyle.fromStyles({
287
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
288
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
289
- })
290
-
291
- const mockClient = new MockTreeSitterClient()
292
- mockClient.setMockResult({
293
- highlights: [
294
- [0, 5, "keyword"],
295
- [6, 13, "identifier"],
296
- ] as SimpleHighlight[],
297
- })
298
-
299
- const codeRenderable = new CodeRenderable(currentRenderer, {
300
- id: "test-code",
301
- content: "const message = 'hello world';",
302
- filetype: "javascript",
303
- syntaxStyle,
304
- treeSitterClient: mockClient,
305
- conceal: false,
306
- left: 0,
307
- top: 0,
308
- })
309
-
310
- currentRenderer.root.add(codeRenderable)
311
- await renderOnce()
312
-
313
- expect(mockClient.isHighlighting()).toBe(true)
314
-
315
- const frameBeforeHighlighting = captureFrame()
316
- expect(frameBeforeHighlighting).toMatchSnapshot("text visible before highlighting completes")
317
-
318
- mockClient.resolveHighlightOnce(0)
319
- await new Promise((resolve) => setTimeout(resolve, 10))
320
- await renderOnce()
321
-
322
- const frameAfterHighlighting = captureFrame()
323
- expect(frameAfterHighlighting).toMatchSnapshot("text visible after highlighting completes")
324
- })
325
-
326
- test("CodeRenderable - batches concurrent content and filetype updates", async () => {
327
- const syntaxStyle = SyntaxStyle.fromStyles({
328
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
329
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
330
- })
331
-
332
- let highlightCount = 0
333
- const mockClient = new MockTreeSitterClient()
334
- const originalHighlightOnce = mockClient.highlightOnce.bind(mockClient)
335
-
336
- mockClient.highlightOnce = async (content: string, filetype: string) => {
337
- highlightCount++
338
- return originalHighlightOnce(content, filetype)
339
- }
340
-
341
- mockClient.setMockResult({
342
- highlights: [[0, 3, "keyword"]] as SimpleHighlight[],
343
- })
344
-
345
- const codeRenderable = new CodeRenderable(currentRenderer, {
346
- id: "test-code",
347
- content: "const message = 'hello';",
348
- filetype: "javascript",
349
- syntaxStyle,
350
- treeSitterClient: mockClient,
351
- conceal: false,
352
- })
353
-
354
- currentRenderer.root.add(codeRenderable)
355
- await renderOnce()
356
-
357
- mockClient.resolveHighlightOnce(0)
358
- await new Promise((resolve) => setTimeout(resolve, 10))
359
-
360
- highlightCount = 0
361
-
362
- codeRenderable.content = "let newMessage = 'world';"
363
- codeRenderable.filetype = "typescript"
364
-
365
- await renderOnce()
366
-
367
- mockClient.resolveAllHighlightOnce()
368
- await new Promise((resolve) => setTimeout(resolve, 10))
369
-
370
- expect(highlightCount).toBe(1)
371
- expect(codeRenderable.content).toBe("let newMessage = 'world';")
372
- expect(codeRenderable.filetype).toBe("typescript")
373
- })
374
-
375
- test("CodeRenderable - batches multiple updates in same tick into single highlight", async () => {
376
- const syntaxStyle = SyntaxStyle.fromStyles({
377
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
378
- })
379
-
380
- let highlightCount = 0
381
- const highlightCalls: Array<{ content: string; filetype: string }> = []
382
- const mockClient = new MockTreeSitterClient()
383
- const originalHighlightOnce = mockClient.highlightOnce.bind(mockClient)
384
-
385
- mockClient.highlightOnce = async (content: string, filetype: string) => {
386
- highlightCount++
387
- highlightCalls.push({ content, filetype })
388
- return originalHighlightOnce(content, filetype)
389
- }
390
-
391
- mockClient.setMockResult({ highlights: [] })
392
-
393
- const codeRenderable = new CodeRenderable(currentRenderer, {
394
- id: "test-code",
395
- content: "initial",
396
- filetype: "javascript",
397
- syntaxStyle,
398
- treeSitterClient: mockClient,
399
- conceal: false,
400
- })
401
-
402
- currentRenderer.root.add(codeRenderable)
403
- await renderOnce()
404
-
405
- mockClient.resolveHighlightOnce(0)
406
- await new Promise((resolve) => setTimeout(resolve, 10))
407
-
408
- highlightCount = 0
409
- highlightCalls.length = 0
410
-
411
- codeRenderable.content = "first content change"
412
- codeRenderable.filetype = "typescript"
413
- codeRenderable.content = "second content change"
414
-
415
- await renderOnce()
416
-
417
- mockClient.resolveAllHighlightOnce()
418
- await new Promise((resolve) => setTimeout(resolve, 10))
419
-
420
- expect(highlightCount).toBe(1)
421
- expect(highlightCalls[0]?.content).toBe("second content change")
422
- expect(highlightCalls[0]?.filetype).toBe("typescript")
423
- })
424
-
425
- test("CodeRenderable - renders markdown with TypeScript injection correctly", async () => {
426
- const syntaxStyle = SyntaxStyle.fromStyles({
427
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
428
- keyword: { fg: RGBA.fromValues(1, 0, 0, 1) }, // Red
429
- string: { fg: RGBA.fromValues(0, 1, 0, 1) }, // Green
430
- "markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) }, // Blue
431
- })
432
-
433
- const markdownCode = `# Hello\n\n\`\`\`typescript\nconst msg: string = "hi";\n\`\`\``
434
-
435
- const codeRenderable = new CodeRenderable(currentRenderer, {
436
- id: "test-markdown",
437
- content: markdownCode,
438
- filetype: "markdown",
439
- syntaxStyle,
440
- conceal: false,
441
- left: 0,
442
- top: 0,
443
- })
444
-
445
- currentRenderer.root.add(codeRenderable)
446
- await renderOnce()
447
-
448
- await new Promise((resolve) => setTimeout(resolve, 100))
449
- await renderOnce()
450
-
451
- expect(codeRenderable.plainText).toContain("# Hello")
452
- expect(codeRenderable.plainText).toContain("const msg")
453
- expect(codeRenderable.plainText).toContain("typescript")
454
- })
455
-
456
- test("CodeRenderable - continues highlighting after unresolved promise", async () => {
457
- const syntaxStyle = SyntaxStyle.fromStyles({
458
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
459
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
460
- })
461
-
462
- let highlightCount = 0
463
- const pendingPromises: Array<{ content: string; filetype: string; never: boolean }> = []
464
-
465
- class HangingMockClient extends TreeSitterClient {
466
- constructor() {
467
- super({ dataPath: "/tmp/mock" })
468
- }
469
-
470
- override async highlightOnce(
471
- content: string,
472
- filetype: string,
473
- ): Promise<{ highlights?: SimpleHighlight[]; warning?: string; error?: string }> {
474
- highlightCount++
475
-
476
- const shouldHang = highlightCount === 4 && filetype === "typescript"
477
-
478
- pendingPromises.push({ content, filetype, never: shouldHang })
479
-
480
- if (shouldHang) {
481
- return new Promise(() => {})
482
- }
483
-
484
- return Promise.resolve({ highlights: [] })
485
- }
486
- }
487
-
488
- const mockClient = new HangingMockClient()
489
-
490
- const codeRenderable = new CodeRenderable(currentRenderer, {
491
- id: "test-code",
492
- content: "interface User { name: string; }",
493
- filetype: "typescript",
494
- syntaxStyle,
495
- treeSitterClient: mockClient,
496
- conceal: false,
497
- })
498
-
499
- currentRenderer.root.add(codeRenderable)
500
- await renderOnce()
501
- await new Promise((resolve) => setTimeout(resolve, 20))
502
-
503
- highlightCount = 0
504
- pendingPromises.length = 0
505
-
506
- codeRenderable.content = "const message = 'hello';"
507
- codeRenderable.filetype = "javascript"
508
- await renderOnce()
509
- await new Promise((resolve) => setTimeout(resolve, 20))
510
-
511
- codeRenderable.content = "# Documentation"
512
- codeRenderable.filetype = "markdown"
513
- await renderOnce()
514
- await new Promise((resolve) => setTimeout(resolve, 20))
515
-
516
- codeRenderable.content = "const message = 'world';"
517
- codeRenderable.filetype = "javascript"
518
- await renderOnce()
519
- await new Promise((resolve) => setTimeout(resolve, 20))
520
-
521
- codeRenderable.content = "interface User { name: string; }"
522
- codeRenderable.filetype = "typescript"
523
- await renderOnce()
524
- await new Promise((resolve) => setTimeout(resolve, 20))
525
-
526
- codeRenderable.content = "# New Documentation"
527
- codeRenderable.filetype = "markdown"
528
- await renderOnce()
529
- await new Promise((resolve) => setTimeout(resolve, 20))
530
-
531
- const markdownHighlightHappened = pendingPromises.some(
532
- (p) => p.content === "# New Documentation" && p.filetype === "markdown",
533
- )
534
-
535
- expect(codeRenderable.content).toBe("# New Documentation")
536
- expect(codeRenderable.filetype).toBe("markdown")
537
- expect(markdownHighlightHappened).toBe(true)
538
- expect(highlightCount).toBe(5)
539
- })
540
-
541
- test("CodeRenderable - concealment is enabled by default", async () => {
542
- const syntaxStyle = SyntaxStyle.fromStyles({
543
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
544
- })
545
-
546
- const codeRenderable = new CodeRenderable(currentRenderer, {
547
- id: "test-code",
548
- content: "const message = 'hello';",
549
- filetype: "javascript",
550
- syntaxStyle,
551
- })
552
-
553
- expect(codeRenderable.conceal).toBe(true)
554
- })
555
-
556
- test("CodeRenderable - concealment can be disabled explicitly", async () => {
557
- const syntaxStyle = SyntaxStyle.fromStyles({
558
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
559
- })
560
-
561
- const codeRenderable = new CodeRenderable(currentRenderer, {
562
- id: "test-code",
563
- content: "const message = 'hello';",
564
- filetype: "javascript",
565
- syntaxStyle,
566
- conceal: false,
567
- })
568
-
569
- expect(codeRenderable.conceal).toBe(false)
570
- })
571
-
572
- test("CodeRenderable - applies concealment to styled text", async () => {
573
- const syntaxStyle = SyntaxStyle.fromStyles({
574
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
575
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
576
- })
577
-
578
- const mockClient = new MockTreeSitterClient()
579
- mockClient.setMockResult({
580
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
581
- })
582
-
583
- const codeRenderable = new CodeRenderable(currentRenderer, {
584
- id: "test-code",
585
- content: "const message = 'hello';",
586
- filetype: "javascript",
587
- syntaxStyle,
588
- treeSitterClient: mockClient,
589
- conceal: true,
590
- left: 0,
591
- top: 0,
592
- })
593
-
594
- currentRenderer.root.add(codeRenderable)
595
-
596
- expect(codeRenderable.conceal).toBe(true)
597
-
598
- mockClient.resolveHighlightOnce(0)
599
- await new Promise((resolve) => setTimeout(resolve, 10))
600
- await renderOnce()
601
-
602
- expect(codeRenderable.content).toBe("const message = 'hello';")
603
- })
604
-
605
- test("CodeRenderable - updating conceal triggers re-highlighting", async () => {
606
- const syntaxStyle = SyntaxStyle.fromStyles({
607
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
608
- })
609
-
610
- const mockClient = new MockTreeSitterClient()
611
- mockClient.setMockResult({ highlights: [] })
612
-
613
- const codeRenderable = new CodeRenderable(currentRenderer, {
614
- id: "test-code",
615
- content: "const message = 'hello';",
616
- filetype: "javascript",
617
- syntaxStyle,
618
- treeSitterClient: mockClient,
619
- conceal: true,
620
- })
621
-
622
- currentRenderer.root.add(codeRenderable)
623
- await renderOnce()
624
-
625
- expect(codeRenderable.conceal).toBe(true)
626
-
627
- mockClient.resolveHighlightOnce(0)
628
- await new Promise((resolve) => setTimeout(resolve, 10))
629
-
630
- codeRenderable.conceal = false
631
- expect(codeRenderable.conceal).toBe(false)
632
-
633
- await renderOnce()
634
-
635
- expect(mockClient.isHighlighting()).toBe(true)
636
- mockClient.resolveHighlightOnce(0)
637
- await new Promise((resolve) => setTimeout(resolve, 10))
638
- })
639
-
640
- test("CodeRenderable - drawUnstyledText is true by default", async () => {
641
- const syntaxStyle = SyntaxStyle.fromStyles({
642
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
643
- })
644
-
645
- const codeRenderable = new CodeRenderable(currentRenderer, {
646
- id: "test-code",
647
- content: "const message = 'hello';",
648
- filetype: "javascript",
649
- syntaxStyle,
650
- })
651
-
652
- expect(codeRenderable.drawUnstyledText).toBe(true)
653
- })
654
-
655
- test("CodeRenderable - drawUnstyledText can be set to false", async () => {
656
- const syntaxStyle = SyntaxStyle.fromStyles({
657
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
658
- })
659
-
660
- const codeRenderable = new CodeRenderable(currentRenderer, {
661
- id: "test-code",
662
- content: "const message = 'hello';",
663
- filetype: "javascript",
664
- syntaxStyle,
665
- drawUnstyledText: false,
666
- })
667
-
668
- expect(codeRenderable.drawUnstyledText).toBe(false)
669
- })
670
-
671
- test("CodeRenderable - with drawUnstyledText=true, text renders before highlighting", async () => {
672
- const syntaxStyle = SyntaxStyle.fromStyles({
673
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
674
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
675
- })
676
-
677
- const mockClient = new MockTreeSitterClient()
678
- mockClient.setMockResult({
679
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
680
- })
681
-
682
- const codeRenderable = new CodeRenderable(currentRenderer, {
683
- id: "test-code",
684
- content: "const message = 'hello';",
685
- filetype: "javascript",
686
- syntaxStyle,
687
- treeSitterClient: mockClient,
688
- drawUnstyledText: true,
689
- left: 0,
690
- top: 0,
691
- })
692
-
693
- currentRenderer.root.add(codeRenderable)
694
- await renderOnce()
695
-
696
- expect(mockClient.isHighlighting()).toBe(true)
697
-
698
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
699
-
700
- mockClient.resolveHighlightOnce(0)
701
- await new Promise((resolve) => setTimeout(resolve, 10))
702
- await renderOnce()
703
-
704
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
705
- })
706
-
707
- test("CodeRenderable - with drawUnstyledText=false, text does not render before highlighting but lineCount is correct", async () => {
708
- const syntaxStyle = SyntaxStyle.fromStyles({
709
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
710
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
711
- })
712
-
713
- const mockClient = new MockTreeSitterClient()
714
- mockClient.setMockResult({
715
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
716
- })
717
-
718
- const codeRenderable = new CodeRenderable(currentRenderer, {
719
- id: "test-code",
720
- content: "const message = 'hello';",
721
- filetype: "javascript",
722
- syntaxStyle,
723
- treeSitterClient: mockClient,
724
- drawUnstyledText: false,
725
- left: 0,
726
- top: 0,
727
- })
728
-
729
- currentRenderer.root.add(codeRenderable)
730
- await renderOnce()
731
-
732
- expect(mockClient.isHighlighting()).toBe(true)
733
-
734
- // Text buffer has content (for lineCount), but nothing renders yet
735
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
736
- expect(codeRenderable.lineCount).toBe(1)
737
- const frameBeforeHighlighting = captureFrame()
738
- expect(frameBeforeHighlighting.trim()).toBe("")
739
-
740
- mockClient.resolveHighlightOnce(0)
741
- await new Promise((resolve) => setTimeout(resolve, 10))
742
- await renderOnce()
743
-
744
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
745
- const frameAfterHighlighting = captureFrame()
746
- expect(frameAfterHighlighting).toContain("const message")
747
- })
748
-
749
- test("CodeRenderable - updating drawUnstyledText from false to true triggers re-highlighting", async () => {
750
- const syntaxStyle = SyntaxStyle.fromStyles({
751
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
752
- })
753
-
754
- const mockClient = new MockTreeSitterClient()
755
- mockClient.setMockResult({ highlights: [] })
756
-
757
- const codeRenderable = new CodeRenderable(currentRenderer, {
758
- id: "test-code",
759
- content: "const message = 'hello';",
760
- filetype: "javascript",
761
- syntaxStyle,
762
- treeSitterClient: mockClient,
763
- drawUnstyledText: false,
764
- left: 0,
765
- top: 0,
766
- })
767
-
768
- currentRenderer.root.add(codeRenderable)
769
-
770
- expect(codeRenderable.drawUnstyledText).toBe(false)
771
-
772
- await renderOnce()
773
- // Text buffer has content for lineCount, but we can verify nothing renders
774
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
775
- expect(codeRenderable.lineCount).toBe(1)
776
-
777
- mockClient.resolveHighlightOnce(0)
778
- await new Promise((resolve) => setTimeout(resolve, 10))
779
-
780
- codeRenderable.drawUnstyledText = true
781
- expect(codeRenderable.drawUnstyledText).toBe(true)
782
-
783
- await renderOnce()
784
-
785
- expect(mockClient.isHighlighting()).toBe(true)
786
-
787
- mockClient.resolveHighlightOnce(0)
788
- await new Promise((resolve) => setTimeout(resolve, 10))
789
- await renderOnce()
790
-
791
- expect(mockClient.isHighlighting()).toBe(false)
792
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
793
- })
794
-
795
- test("CodeRenderable - updating drawUnstyledText from true to false triggers re-highlighting", async () => {
796
- const syntaxStyle = SyntaxStyle.fromStyles({
797
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
798
- })
799
-
800
- const mockClient = new MockTreeSitterClient()
801
- mockClient.setMockResult({ highlights: [] })
802
-
803
- const codeRenderable = new CodeRenderable(currentRenderer, {
804
- id: "test-code",
805
- content: "const message = 'hello';",
806
- filetype: "javascript",
807
- syntaxStyle,
808
- treeSitterClient: mockClient,
809
- drawUnstyledText: true,
810
- })
811
-
812
- currentRenderer.root.add(codeRenderable)
813
- await renderOnce()
814
-
815
- expect(codeRenderable.drawUnstyledText).toBe(true)
816
-
817
- mockClient.resolveHighlightOnce(0)
818
- await new Promise((resolve) => setTimeout(resolve, 10))
819
-
820
- codeRenderable.drawUnstyledText = false
821
- expect(codeRenderable.drawUnstyledText).toBe(false)
822
-
823
- await renderOnce()
824
-
825
- expect(mockClient.isHighlighting()).toBe(true)
826
- mockClient.resolveHighlightOnce(0)
827
- await new Promise((resolve) => setTimeout(resolve, 10))
828
- })
829
-
830
- test("CodeRenderable - uses fallback rendering on error even with drawUnstyledText=false", async () => {
831
- const syntaxStyle = SyntaxStyle.fromStyles({
832
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
833
- })
834
-
835
- const mockClient = new MockTreeSitterClient()
836
-
837
- mockClient.highlightOnce = async () => {
838
- throw new Error("Highlighting failed")
839
- }
840
-
841
- const codeRenderable = new CodeRenderable(currentRenderer, {
842
- id: "test-code",
843
- content: "const message = 'hello world';",
844
- filetype: "javascript",
845
- syntaxStyle,
846
- treeSitterClient: mockClient,
847
- drawUnstyledText: false,
848
- left: 0,
849
- top: 0,
850
- })
851
-
852
- currentRenderer.root.add(codeRenderable)
853
-
854
- await new Promise((resolve) => setTimeout(resolve, 20))
855
- await renderOnce()
856
-
857
- expect(codeRenderable.plainText).toBe("const message = 'hello world';")
858
- })
859
-
860
- test("CodeRenderable - with drawUnstyledText=false and no filetype, fallback is used", async () => {
861
- const syntaxStyle = SyntaxStyle.fromStyles({
862
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
863
- })
864
-
865
- const codeRenderable = new CodeRenderable(currentRenderer, {
866
- id: "test-code",
867
- content: "const message = 'hello world';",
868
- syntaxStyle,
869
- drawUnstyledText: false,
870
- left: 0,
871
- top: 0,
872
- })
873
-
874
- currentRenderer.root.add(codeRenderable)
875
-
876
- await renderOnce()
877
-
878
- expect(codeRenderable.filetype).toBeUndefined()
879
- expect(codeRenderable.plainText).toBe("const message = 'hello world';")
880
- })
881
-
882
- test("CodeRenderable - with drawUnstyledText=false, multiple updates only render final highlighted text", async () => {
883
- const syntaxStyle = SyntaxStyle.fromStyles({
884
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
885
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
886
- })
887
-
888
- const mockClient = new MockTreeSitterClient()
889
- mockClient.setMockResult({
890
- highlights: [[0, 3, "keyword"]] as SimpleHighlight[],
891
- })
892
-
893
- const codeRenderable = new CodeRenderable(currentRenderer, {
894
- id: "test-code",
895
- content: "const message = 'hello';",
896
- filetype: "javascript",
897
- syntaxStyle,
898
- treeSitterClient: mockClient,
899
- drawUnstyledText: false,
900
- left: 0,
901
- top: 0,
902
- })
903
-
904
- currentRenderer.root.add(codeRenderable)
905
- await renderOnce()
906
-
907
- expect(mockClient.isHighlighting()).toBe(true)
908
-
909
- // Text buffer has content (for lineCount), but nothing renders yet
910
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
911
- expect(codeRenderable.lineCount).toBe(1)
912
- const frameBeforeHighlighting = captureFrame()
913
- expect(frameBeforeHighlighting.trim()).toBe("")
914
-
915
- codeRenderable.content = "let newMessage = 'world';"
916
- await renderOnce()
917
-
918
- // Text buffer updated immediately, but still no rendering
919
- expect(codeRenderable.plainText).toBe("let newMessage = 'world';")
920
- expect(codeRenderable.lineCount).toBe(1)
921
- const frameAfterUpdate = captureFrame()
922
- expect(frameAfterUpdate.trim()).toBe("")
923
-
924
- mockClient.resolveAllHighlightOnce()
925
- await new Promise((resolve) => setTimeout(resolve, 10))
926
- await renderOnce()
927
- await new Promise((resolve) => setTimeout(resolve, 10))
928
-
929
- expect(mockClient.isHighlighting()).toBe(false)
930
- expect(codeRenderable.plainText).toBe("let newMessage = 'world';")
931
- const frameAfterHighlighting = captureFrame()
932
- expect(frameAfterHighlighting).toContain("let newMessage")
933
- })
934
-
935
- // TODO: flaky in CI because it needs to finish in time
936
- // lib/tree-sitter/client.ts needs a way to check if the queue is empty
937
- // then this can wait for all tree-sitter operations to complete
938
- // instead of the arbitrary 500ms wait
939
- // it worked before because text was set anyway for drawUnstyledText=false
940
- test.skip("CodeRenderable - simulates markdown stream from LLM with async updates", async () => {
941
- const syntaxStyle = SyntaxStyle.fromStyles({
942
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
943
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
944
- string: { fg: RGBA.fromValues(0, 1, 0, 1) },
945
- "markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) },
946
- })
947
-
948
- // Base markdown content that we'll repeat to grow to ~1MB
949
- const baseMarkdownContent = `# Code Example
950
-
951
- Here's a simple TypeScript function:
952
-
953
- \`\`\`typescript
954
- function greet(name: string): string {
955
- return \`Hello, \${name}!\`;
956
- }
957
-
958
- const message = greet("World");
959
- console.log(message);
960
- \`\`\`
961
- `
962
-
963
- const targetSize = 64 * 128
964
- let fullMarkdownContent = ""
965
- let iteration = 0
966
- while (fullMarkdownContent.length < targetSize) {
967
- fullMarkdownContent += `\n--- Iteration ${iteration} ---\n\n` + baseMarkdownContent
968
- iteration++
969
- }
970
-
971
- const codeRenderable = new CodeRenderable(currentRenderer, {
972
- id: "test-markdown-stream",
973
- content: "",
974
- filetype: "markdown",
975
- syntaxStyle,
976
- conceal: false,
977
- left: 0,
978
- top: 0,
979
- drawUnstyledText: false,
980
- })
981
- await codeRenderable.treeSitterClient.initialize()
982
- await codeRenderable.treeSitterClient.preloadParser("markdown")
983
-
984
- currentRenderer.root.add(codeRenderable)
985
- currentRenderer.start()
986
-
987
- let currentContent = ""
988
-
989
- const chunkSize = 64
990
- const chunks: string[] = []
991
- for (let i = 0; i < fullMarkdownContent.length; i += chunkSize) {
992
- chunks.push(fullMarkdownContent.slice(i, Math.min(i + chunkSize, fullMarkdownContent.length)))
993
- }
994
-
995
- for (let i = 0; i < chunks.length; i++) {
996
- const chunk = chunks[i]
997
- currentContent += chunk
998
- codeRenderable.content = currentContent
999
- await new Promise((resolve) => setTimeout(resolve, Math.floor(Math.random() * 25) + 1))
1000
- }
1001
-
1002
- // wait for highlighting to complete (long for slow machines/CI)
1003
- await new Promise((resolve) => setTimeout(resolve, 500))
1004
-
1005
- expect(codeRenderable.content).toBe(fullMarkdownContent)
1006
- expect(codeRenderable.content.length).toBeGreaterThanOrEqual(targetSize)
1007
- expect(codeRenderable.plainText).toContain("# Code Example")
1008
- expect(codeRenderable.plainText).toContain("function greet")
1009
- expect(codeRenderable.plainText).toContain("typescript")
1010
- expect(codeRenderable.plainText).toContain("Hello")
1011
-
1012
- const plainText = codeRenderable.plainText
1013
- expect(plainText.length).toBeGreaterThan(targetSize * 0.9)
1014
- expect(plainText).toContain("Code Example")
1015
- expect(plainText).toContain("const message = greet")
1016
- })
1017
-
1018
- test("CodeRenderable - streaming option is false by default", async () => {
1019
- const syntaxStyle = SyntaxStyle.fromStyles({
1020
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1021
- })
1022
-
1023
- const codeRenderable = new CodeRenderable(currentRenderer, {
1024
- id: "test-code",
1025
- content: "const message = 'hello';",
1026
- filetype: "javascript",
1027
- syntaxStyle,
1028
- })
1029
-
1030
- expect(codeRenderable.streaming).toBe(false)
1031
- })
1032
-
1033
- test("CodeRenderable - streaming can be enabled", async () => {
1034
- const syntaxStyle = SyntaxStyle.fromStyles({
1035
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1036
- })
1037
-
1038
- const codeRenderable = new CodeRenderable(currentRenderer, {
1039
- id: "test-code",
1040
- content: "const message = 'hello';",
1041
- filetype: "javascript",
1042
- syntaxStyle,
1043
- streaming: true,
1044
- })
1045
-
1046
- expect(codeRenderable.streaming).toBe(true)
1047
- })
1048
-
1049
- test("CodeRenderable - streaming mode respects drawUnstyledText only for initial content", async () => {
1050
- const syntaxStyle = SyntaxStyle.fromStyles({
1051
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1052
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1053
- })
1054
-
1055
- const mockClient = new MockTreeSitterClient()
1056
- mockClient.setMockResult({
1057
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1058
- })
1059
-
1060
- const codeRenderable = new CodeRenderable(currentRenderer, {
1061
- id: "test-code",
1062
- content: "const initial = 'hello';",
1063
- filetype: "javascript",
1064
- syntaxStyle,
1065
- treeSitterClient: mockClient,
1066
- streaming: true,
1067
- drawUnstyledText: true,
1068
- left: 0,
1069
- top: 0,
1070
- })
1071
-
1072
- currentRenderer.root.add(codeRenderable)
1073
-
1074
- await renderOnce()
1075
- expect(codeRenderable.plainText).toBe("const initial = 'hello';")
1076
-
1077
- mockClient.resolveHighlightOnce(0)
1078
- await new Promise((resolve) => setTimeout(resolve, 10))
1079
-
1080
- codeRenderable.content = "const updated = 'world';"
1081
- await new Promise((resolve) => queueMicrotask(resolve))
1082
-
1083
- expect(codeRenderable.content).toBe("const updated = 'world';")
1084
- })
1085
-
1086
- test("CodeRenderable - streaming mode with drawUnstyledText=false waits for new highlights", async () => {
1087
- const syntaxStyle = SyntaxStyle.fromStyles({
1088
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1089
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1090
- })
1091
-
1092
- const mockClient = new MockTreeSitterClient({ autoResolveTimeout: 10 })
1093
- mockClient.setMockResult({
1094
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1095
- })
1096
-
1097
- const codeRenderable = new CodeRenderable(currentRenderer, {
1098
- id: "test-code",
1099
- content: "const initial = 'hello';",
1100
- filetype: "javascript",
1101
- syntaxStyle,
1102
- treeSitterClient: mockClient,
1103
- streaming: true,
1104
- drawUnstyledText: false,
1105
- left: 0,
1106
- top: 0,
1107
- })
1108
-
1109
- currentRenderer.root.add(codeRenderable)
1110
- currentRenderer.start()
1111
-
1112
- await Bun.sleep(30)
1113
-
1114
- expect(codeRenderable.plainText).toBe("const initial = 'hello';")
1115
-
1116
- codeRenderable.content = "const updated = 'world';"
1117
- expect(codeRenderable.plainText).toBe("const initial = 'hello';")
1118
-
1119
- await Bun.sleep(30)
1120
-
1121
- expect(codeRenderable.plainText).toBe("const updated = 'world';")
1122
-
1123
- currentRenderer.stop()
1124
- })
1125
-
1126
- test("CodeRenderable - onChunks callback can transform chunks when highlights are empty", async () => {
1127
- const syntaxStyle = SyntaxStyle.fromStyles({
1128
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1129
- })
1130
-
1131
- const mockClient = new MockTreeSitterClient()
1132
- mockClient.setMockResult({ highlights: [] })
1133
-
1134
- let callbackInvoked = false
1135
-
1136
- const codeRenderable = new CodeRenderable(currentRenderer, {
1137
- id: "test-code",
1138
- content: "hello",
1139
- filetype: "plaintext",
1140
- syntaxStyle,
1141
- treeSitterClient: mockClient,
1142
- onChunks: (chunks) => {
1143
- callbackInvoked = true
1144
- return chunks.map((chunk) => ({
1145
- ...chunk,
1146
- text: chunk.text.toUpperCase(),
1147
- }))
1148
- },
1149
- })
1150
-
1151
- currentRenderer.root.add(codeRenderable)
1152
- await renderOnce()
1153
-
1154
- mockClient.resolveHighlightOnce(0)
1155
- await new Promise((resolve) => setTimeout(resolve, 10))
1156
- await renderOnce()
1157
-
1158
- expect(callbackInvoked).toBe(true)
1159
- expect(codeRenderable.plainText).toBe("HELLO")
1160
- })
1161
-
1162
- test("CodeRenderable - onHighlight callback receives highlights and context", async () => {
1163
- const syntaxStyle = SyntaxStyle.fromStyles({
1164
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1165
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1166
- })
1167
-
1168
- const mockClient = new MockTreeSitterClient()
1169
- mockClient.setMockResult({
1170
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1171
- })
1172
-
1173
- let callbackInvoked = false
1174
- let receivedHighlights: SimpleHighlight[] | null = null
1175
- let receivedContext: { content: string; filetype: string | undefined; syntaxStyle: SyntaxStyle } | null = null
1176
-
1177
- const codeRenderable = new CodeRenderable(currentRenderer, {
1178
- id: "test-code",
1179
- content: "const message = 'hello';",
1180
- filetype: "javascript",
1181
- syntaxStyle,
1182
- treeSitterClient: mockClient,
1183
- onHighlight: (highlights, context) => {
1184
- callbackInvoked = true
1185
- receivedHighlights = [...highlights]
1186
- receivedContext = { ...context }
1187
- return highlights
1188
- },
1189
- })
1190
-
1191
- currentRenderer.root.add(codeRenderable)
1192
- await renderOnce()
1193
-
1194
- mockClient.resolveHighlightOnce(0)
1195
- await new Promise((resolve) => setTimeout(resolve, 10))
1196
- await renderOnce()
1197
-
1198
- expect(callbackInvoked).toBe(true)
1199
- expect(receivedHighlights).not.toBeNull()
1200
- expect(receivedHighlights?.length).toBe(1)
1201
- expect(receivedHighlights?.[0]).toEqual([0, 5, "keyword"])
1202
- expect(receivedContext?.content).toBe("const message = 'hello';")
1203
- expect(receivedContext?.filetype).toBe("javascript")
1204
- expect(receivedContext?.syntaxStyle).toBe(syntaxStyle)
1205
- })
1206
-
1207
- test("CodeRenderable - onHighlight callback can add custom highlights", async () => {
1208
- const syntaxStyle = SyntaxStyle.fromStyles({
1209
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1210
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1211
- "custom.highlight": { fg: RGBA.fromValues(1, 0, 0, 1) },
1212
- })
1213
-
1214
- const mockClient = new MockTreeSitterClient()
1215
- mockClient.setMockResult({
1216
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1217
- })
1218
-
1219
- const codeRenderable = new CodeRenderable(currentRenderer, {
1220
- id: "test-code",
1221
- content: "const message = 'hello';",
1222
- filetype: "javascript",
1223
- syntaxStyle,
1224
- treeSitterClient: mockClient,
1225
- onHighlight: (highlights) => {
1226
- highlights.push([6, 13, "custom.highlight", {}])
1227
- return highlights
1228
- },
1229
- })
1230
-
1231
- currentRenderer.root.add(codeRenderable)
1232
- await renderOnce()
1233
-
1234
- mockClient.resolveHighlightOnce(0)
1235
- await new Promise((resolve) => setTimeout(resolve, 10))
1236
- await renderOnce()
1237
-
1238
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
1239
-
1240
- // Verify both the original keyword highlight and the custom highlight are applied
1241
- const lineHighlights = codeRenderable.getLineHighlights(0)
1242
- expect(lineHighlights.length).toBeGreaterThanOrEqual(2)
1243
-
1244
- // Check keyword highlight exists with the correct styleId
1245
- const keywordStyleId = syntaxStyle.getStyleId("keyword")
1246
- const keywordHighlight = lineHighlights.find((h) => h.styleId === keywordStyleId)
1247
- expect(keywordHighlight).toBeDefined()
1248
-
1249
- // Check custom highlight exists with the correct styleId
1250
- const customStyleId = syntaxStyle.getStyleId("custom.highlight")
1251
- const customHighlight = lineHighlights.find((h) => h.styleId === customStyleId)
1252
- expect(customHighlight).toBeDefined()
1253
- })
1254
-
1255
- test("CodeRenderable - onHighlight callback returning undefined uses original highlights", async () => {
1256
- const syntaxStyle = SyntaxStyle.fromStyles({
1257
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1258
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1259
- })
1260
-
1261
- const mockClient = new MockTreeSitterClient()
1262
- mockClient.setMockResult({
1263
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1264
- })
1265
-
1266
- let callbackInvoked = false
1267
-
1268
- const codeRenderable = new CodeRenderable(currentRenderer, {
1269
- id: "test-code",
1270
- content: "const message = 'hello';",
1271
- filetype: "javascript",
1272
- syntaxStyle,
1273
- treeSitterClient: mockClient,
1274
- onHighlight: (highlights) => {
1275
- callbackInvoked = true
1276
- return undefined as unknown as SimpleHighlight[]
1277
- },
1278
- })
1279
-
1280
- currentRenderer.root.add(codeRenderable)
1281
- await renderOnce()
1282
-
1283
- mockClient.resolveHighlightOnce(0)
1284
- await new Promise((resolve) => setTimeout(resolve, 10))
1285
- await renderOnce()
1286
-
1287
- expect(callbackInvoked).toBe(true)
1288
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
1289
- })
1290
-
1291
- test("CodeRenderable - onHighlight callback is called on re-highlighting when content changes", async () => {
1292
- const syntaxStyle = SyntaxStyle.fromStyles({
1293
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1294
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1295
- })
1296
-
1297
- const mockClient = new MockTreeSitterClient()
1298
- mockClient.setMockResult({
1299
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1300
- })
1301
-
1302
- let callbackCount = 0
1303
-
1304
- const codeRenderable = new CodeRenderable(currentRenderer, {
1305
- id: "test-code",
1306
- content: "const message = 'hello';",
1307
- filetype: "javascript",
1308
- syntaxStyle,
1309
- treeSitterClient: mockClient,
1310
- onHighlight: (highlights) => {
1311
- callbackCount++
1312
- return highlights
1313
- },
1314
- })
1315
-
1316
- currentRenderer.root.add(codeRenderable)
1317
- await renderOnce()
1318
-
1319
- mockClient.resolveHighlightOnce(0)
1320
- await new Promise((resolve) => setTimeout(resolve, 10))
1321
- await renderOnce()
1322
-
1323
- expect(callbackCount).toBe(1)
1324
-
1325
- codeRenderable.content = "let newMessage = 'world';"
1326
- await renderOnce()
1327
-
1328
- mockClient.resolveHighlightOnce(0)
1329
- await new Promise((resolve) => setTimeout(resolve, 10))
1330
- await renderOnce()
1331
-
1332
- expect(callbackCount).toBe(2)
1333
- })
1334
-
1335
- test("CodeRenderable - onHighlight callback supports async functions", async () => {
1336
- const syntaxStyle = SyntaxStyle.fromStyles({
1337
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1338
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1339
- "async.highlight": { fg: RGBA.fromValues(0, 1, 0, 1) },
1340
- })
1341
-
1342
- const mockClient = new MockTreeSitterClient()
1343
- mockClient.setMockResult({
1344
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1345
- })
1346
-
1347
- let asyncCallbackCompleted = false
1348
-
1349
- const codeRenderable = new CodeRenderable(currentRenderer, {
1350
- id: "test-code",
1351
- content: "const message = 'hello';",
1352
- filetype: "javascript",
1353
- syntaxStyle,
1354
- treeSitterClient: mockClient,
1355
- onHighlight: async (highlights) => {
1356
- // Simulate async operation (e.g., fetching additional highlight data)
1357
- await new Promise((resolve) => setTimeout(resolve, 5))
1358
- highlights.push([6, 13, "async.highlight", {}])
1359
- asyncCallbackCompleted = true
1360
- return highlights
1361
- },
1362
- })
1363
-
1364
- currentRenderer.root.add(codeRenderable)
1365
- await renderOnce()
1366
-
1367
- mockClient.resolveHighlightOnce(0)
1368
- await new Promise((resolve) => setTimeout(resolve, 20))
1369
- await renderOnce()
1370
-
1371
- expect(asyncCallbackCompleted).toBe(true)
1372
- expect(codeRenderable.plainText).toBe("const message = 'hello';")
1373
-
1374
- // Verify the async highlight was applied
1375
- const lineHighlights = codeRenderable.getLineHighlights(0)
1376
- expect(lineHighlights.length).toBeGreaterThanOrEqual(2)
1377
-
1378
- const asyncStyleId = syntaxStyle.getStyleId("async.highlight")
1379
- const asyncHighlight = lineHighlights.find((h) => h.styleId === asyncStyleId && h.start === 6 && h.end === 13)
1380
- expect(asyncHighlight).toBeDefined()
1381
- })
1382
-
1383
- test("CodeRenderable - streaming mode caches highlights between updates", async () => {
1384
- const syntaxStyle = SyntaxStyle.fromStyles({
1385
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1386
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1387
- })
1388
-
1389
- const mockClient = new MockTreeSitterClient()
1390
- mockClient.setMockResult({
1391
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1392
- })
1393
-
1394
- const codeRenderable = new CodeRenderable(currentRenderer, {
1395
- id: "test-code",
1396
- content: "const initial = 'hello';",
1397
- filetype: "javascript",
1398
- syntaxStyle,
1399
- treeSitterClient: mockClient,
1400
- streaming: true,
1401
- left: 0,
1402
- top: 0,
1403
- })
1404
-
1405
- currentRenderer.root.add(codeRenderable)
1406
-
1407
- mockClient.resolveHighlightOnce(0)
1408
- await new Promise((resolve) => setTimeout(resolve, 10))
1409
-
1410
- codeRenderable.content = "const updated = 'world';"
1411
- await new Promise((resolve) => queueMicrotask(resolve))
1412
-
1413
- codeRenderable.content = "const updated2 = 'test';"
1414
- await new Promise((resolve) => queueMicrotask(resolve))
1415
-
1416
- codeRenderable.content = "const final = 'done';"
1417
- await new Promise((resolve) => queueMicrotask(resolve))
1418
-
1419
- await renderOnce()
1420
-
1421
- expect(codeRenderable.content).toBe("const final = 'done';")
1422
- expect(codeRenderable.plainText).toBe("const final = 'done';")
1423
- })
1424
-
1425
- test("CodeRenderable - streaming mode works with large content updates", async () => {
1426
- const syntaxStyle = SyntaxStyle.fromStyles({
1427
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1428
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1429
- })
1430
-
1431
- const mockClient = new MockTreeSitterClient()
1432
- mockClient.setMockResult({
1433
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1434
- })
1435
-
1436
- const codeRenderable = new CodeRenderable(currentRenderer, {
1437
- id: "test-code",
1438
- content: "const x = 1;",
1439
- filetype: "javascript",
1440
- syntaxStyle,
1441
- treeSitterClient: mockClient,
1442
- streaming: true,
1443
- drawUnstyledText: true,
1444
- left: 0,
1445
- top: 0,
1446
- })
1447
-
1448
- currentRenderer.root.add(codeRenderable)
1449
- await renderOnce()
1450
-
1451
- // Wait for initial highlighting
1452
- mockClient.resolveHighlightOnce(0)
1453
- await new Promise((resolve) => setTimeout(resolve, 10))
1454
-
1455
- // Simulate streaming with progressively larger content
1456
- let content = "const x = 1;"
1457
- for (let i = 0; i < 10; i++) {
1458
- content += `\nconst var${i} = ${i};`
1459
- codeRenderable.content = content
1460
- await new Promise((resolve) => setTimeout(resolve, 5))
1461
- }
1462
-
1463
- await renderOnce()
1464
- mockClient.resolveAllHighlightOnce()
1465
- await new Promise((resolve) => setTimeout(resolve, 20))
1466
- await renderOnce()
1467
-
1468
- expect(codeRenderable.content).toContain("const var9 = 9;")
1469
- expect(codeRenderable.plainText).toContain("const var9 = 9;")
1470
- })
1471
-
1472
- test("CodeRenderable - disabling streaming clears cached highlights", async () => {
1473
- const syntaxStyle = SyntaxStyle.fromStyles({
1474
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1475
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1476
- })
1477
-
1478
- const mockClient = new MockTreeSitterClient()
1479
- mockClient.setMockResult({
1480
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1481
- })
1482
-
1483
- const codeRenderable = new CodeRenderable(currentRenderer, {
1484
- id: "test-code",
1485
- content: "const initial = 'hello';",
1486
- filetype: "javascript",
1487
- syntaxStyle,
1488
- treeSitterClient: mockClient,
1489
- streaming: true,
1490
- })
1491
-
1492
- currentRenderer.root.add(codeRenderable)
1493
- await renderOnce()
1494
-
1495
- expect(codeRenderable.streaming).toBe(true)
1496
-
1497
- mockClient.resolveHighlightOnce(0)
1498
- await new Promise((resolve) => setTimeout(resolve, 10))
1499
-
1500
- codeRenderable.streaming = false
1501
- expect(codeRenderable.streaming).toBe(false)
1502
-
1503
- await renderOnce()
1504
-
1505
- expect(mockClient.isHighlighting()).toBe(true)
1506
- })
1507
-
1508
- test("CodeRenderable - streaming mode with drawUnstyledText=false shows nothing initially", async () => {
1509
- const syntaxStyle = SyntaxStyle.fromStyles({
1510
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1511
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1512
- })
1513
-
1514
- const mockClient = new MockTreeSitterClient()
1515
- mockClient.setMockResult({
1516
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1517
- })
1518
-
1519
- const codeRenderable = new CodeRenderable(currentRenderer, {
1520
- id: "test-code",
1521
- content: "const initial = 'hello';",
1522
- filetype: "javascript",
1523
- syntaxStyle,
1524
- treeSitterClient: mockClient,
1525
- streaming: true,
1526
- drawUnstyledText: false,
1527
- left: 0,
1528
- top: 0,
1529
- })
1530
-
1531
- currentRenderer.root.add(codeRenderable)
1532
-
1533
- await renderOnce()
1534
- const frameBeforeHighlighting = captureFrame()
1535
- expect(frameBeforeHighlighting.trim()).toBe("")
1536
-
1537
- mockClient.resolveHighlightOnce(0)
1538
- await new Promise((resolve) => setTimeout(resolve, 10))
1539
- await renderOnce()
1540
-
1541
- const frameAfterHighlighting = captureFrame()
1542
- expect(frameAfterHighlighting).toContain("const initial")
1543
- })
1544
-
1545
- test("CodeRenderable - streaming mode handles empty cached highlights gracefully", async () => {
1546
- const syntaxStyle = SyntaxStyle.fromStyles({
1547
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1548
- })
1549
-
1550
- const mockClient = new MockTreeSitterClient()
1551
- mockClient.setMockResult({
1552
- highlights: [],
1553
- })
1554
-
1555
- const codeRenderable = new CodeRenderable(currentRenderer, {
1556
- id: "test-code",
1557
- content: "plain text",
1558
- filetype: "javascript",
1559
- syntaxStyle,
1560
- treeSitterClient: mockClient,
1561
- streaming: true,
1562
- drawUnstyledText: true,
1563
- })
1564
-
1565
- currentRenderer.root.add(codeRenderable)
1566
- await renderOnce()
1567
-
1568
- mockClient.resolveHighlightOnce(0)
1569
- await new Promise((resolve) => setTimeout(resolve, 10))
1570
-
1571
- codeRenderable.content = "more plain text"
1572
- await renderOnce()
1573
-
1574
- expect(codeRenderable.content).toBe("more plain text")
1575
- expect(codeRenderable.plainText).toBe("more plain text")
1576
- })
1577
-
1578
- test("CodeRenderable - selection across two Code renderables in flex row", async () => {
1579
- const syntaxStyle = SyntaxStyle.fromStyles({
1580
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1581
- })
1582
-
1583
- const container = new BoxRenderable(currentRenderer, {
1584
- id: "container",
1585
- width: 80,
1586
- height: 10,
1587
- flexDirection: "row",
1588
- left: 0,
1589
- top: 0,
1590
- })
1591
- currentRenderer.root.add(container)
1592
-
1593
- const leftCode = new CodeRenderable(currentRenderer, {
1594
- id: "left-code",
1595
- content: "line1\nline2\nline3\nline4\nline5",
1596
- syntaxStyle,
1597
- selectable: true,
1598
- wrapMode: "none",
1599
- width: 20,
1600
- height: 5,
1601
- })
1602
-
1603
- const rightCode = new CodeRenderable(currentRenderer, {
1604
- id: "right-code",
1605
- content: "lineA\nlineB\nlineC\nlineD\nlineE",
1606
- syntaxStyle,
1607
- selectable: true,
1608
- wrapMode: "none",
1609
- width: 20,
1610
- height: 5,
1611
- })
1612
-
1613
- container.add(leftCode)
1614
- container.add(rightCode)
1615
-
1616
- await renderOnce()
1617
-
1618
- expect(leftCode.x).toBe(0)
1619
- expect(rightCode.x).toBeGreaterThan(leftCode.x)
1620
-
1621
- const startX = leftCode.x + 2
1622
- const startY = leftCode.y + 2
1623
- const endX = rightCode.x + 3
1624
- const endY = rightCode.y + rightCode.height + 2
1625
-
1626
- await mockMouse.drag(startX, startY, endX, endY)
1627
- await renderOnce()
1628
-
1629
- expect(leftCode.hasSelection()).toBe(true)
1630
- expect(rightCode.hasSelection()).toBe(true)
1631
-
1632
- const leftSelection = leftCode.getSelectedText()
1633
- const rightSelection = rightCode.getSelectedText()
1634
- const leftSelectionObj = leftCode.getSelection()
1635
- const rightSelectionObj = rightCode.getSelection()
1636
-
1637
- expect(leftSelectionObj).not.toBeNull()
1638
- expect(rightSelectionObj).not.toBeNull()
1639
-
1640
- if (leftSelectionObj && rightSelectionObj) {
1641
- expect(leftSelectionObj.start).toBeGreaterThan(0)
1642
- expect(leftSelectionObj.end).toBe(29)
1643
- expect(rightSelectionObj.start).toBe(0)
1644
- expect(rightSelectionObj.end).toBe(29)
1645
- expect(leftSelection).toBe("ne3\nline4\nline5")
1646
- expect(rightSelection).toBe("lineA\nlineB\nlineC\nlineD\nlineE")
1647
- }
1648
- })
1649
-
1650
- test("CodeRenderable - content update during async highlighting does not get overwritten by stale highlight result", async () => {
1651
- const syntaxStyle = SyntaxStyle.fromStyles({
1652
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1653
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1654
- })
1655
-
1656
- const mockClient = new MockTreeSitterClient()
1657
- mockClient.setMockResult({
1658
- highlights: [[0, 5, "keyword"]] as SimpleHighlight[],
1659
- })
1660
-
1661
- const codeRenderable = new CodeRenderable(currentRenderer, {
1662
- id: "test-code",
1663
- content: "line1\nline2\nline3",
1664
- filetype: "javascript",
1665
- syntaxStyle,
1666
- treeSitterClient: mockClient,
1667
- drawUnstyledText: true,
1668
- })
1669
-
1670
- currentRenderer.root.add(codeRenderable)
1671
- await renderOnce()
1672
-
1673
- expect(mockClient.isHighlighting()).toBe(true)
1674
- expect(codeRenderable.lineCount).toBe(3)
1675
-
1676
- codeRenderable.content = "line1\nline2\nline3\nline4\nline5"
1677
- expect(codeRenderable.lineCount).toBe(5)
1678
-
1679
- mockClient.resolveHighlightOnce(0)
1680
- await new Promise((resolve) => setTimeout(resolve, 10))
1681
-
1682
- expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
1683
- expect(codeRenderable.lineCount).toBe(5)
1684
-
1685
- await renderOnce()
1686
- expect(codeRenderable.lineCount).toBe(5)
1687
-
1688
- expect(mockClient.isHighlighting()).toBe(true)
1689
-
1690
- mockClient.resolveHighlightOnce(0)
1691
- await new Promise((resolve) => setTimeout(resolve, 10))
1692
- await renderOnce()
1693
-
1694
- expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
1695
- expect(codeRenderable.lineCount).toBe(5)
1696
- expect(codeRenderable.plainText).toBe("line1\nline2\nline3\nline4\nline5")
1697
- })
1698
-
1699
- test("CodeRenderable - lineCount is correct immediately with drawUnstyledText=false", async () => {
1700
- const syntaxStyle = SyntaxStyle.fromStyles({
1701
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1702
- })
1703
-
1704
- const mockClient = new MockTreeSitterClient()
1705
- mockClient.setMockResult({ highlights: [] })
1706
-
1707
- const codeRenderable = new CodeRenderable(currentRenderer, {
1708
- id: "test-code",
1709
- content: "line1\nline2\nline3\nline4\nline5",
1710
- filetype: "javascript",
1711
- syntaxStyle,
1712
- treeSitterClient: mockClient,
1713
- drawUnstyledText: false,
1714
- })
1715
-
1716
- expect(codeRenderable.lineCount).toBe(5)
1717
- expect(codeRenderable.content).toBe("line1\nline2\nline3\nline4\nline5")
1718
-
1719
- currentRenderer.root.add(codeRenderable)
1720
- await renderOnce()
1721
-
1722
- expect(mockClient.isHighlighting()).toBe(true)
1723
- expect(codeRenderable.lineCount).toBe(5)
1724
-
1725
- const frameBeforeHighlighting = captureFrame()
1726
- expect(frameBeforeHighlighting.trim()).toBe("")
1727
-
1728
- mockClient.resolveHighlightOnce(0)
1729
- await new Promise((resolve) => setTimeout(resolve, 10))
1730
- await renderOnce()
1731
-
1732
- expect(codeRenderable.lineCount).toBe(5)
1733
- const frameAfterHighlighting = captureFrame()
1734
- expect(frameAfterHighlighting).toContain("line1")
1735
- })
1736
-
1737
- test("CodeRenderable - lineCount updates correctly when content changes with drawUnstyledText=false", async () => {
1738
- const syntaxStyle = SyntaxStyle.fromStyles({
1739
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1740
- })
1741
-
1742
- const mockClient = new MockTreeSitterClient()
1743
- mockClient.setMockResult({ highlights: [] })
1744
-
1745
- const codeRenderable = new CodeRenderable(currentRenderer, {
1746
- id: "test-code",
1747
- content: "line1\nline2\nline3",
1748
- filetype: "javascript",
1749
- syntaxStyle,
1750
- treeSitterClient: mockClient,
1751
- drawUnstyledText: false,
1752
- })
1753
-
1754
- expect(codeRenderable.lineCount).toBe(3)
1755
-
1756
- currentRenderer.root.add(codeRenderable)
1757
- await renderOnce()
1758
-
1759
- codeRenderable.content = "line1\nline2\nline3\nline4\nline5\nline6\nline7"
1760
- expect(codeRenderable.lineCount).toBe(7)
1761
-
1762
- await renderOnce()
1763
- expect(codeRenderable.lineCount).toBe(7)
1764
-
1765
- codeRenderable.content = "line1\nline2"
1766
- expect(codeRenderable.lineCount).toBe(2)
1767
-
1768
- await renderOnce()
1769
- expect(codeRenderable.lineCount).toBe(2)
1770
- })
1771
-
1772
- test("CodeRenderable - lineInfo is accessible with drawUnstyledText=false before highlighting", async () => {
1773
- const syntaxStyle = SyntaxStyle.fromStyles({
1774
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1775
- })
1776
-
1777
- const mockClient = new MockTreeSitterClient()
1778
- mockClient.setMockResult({ highlights: [] })
1779
-
1780
- const codeRenderable = new CodeRenderable(currentRenderer, {
1781
- id: "test-code",
1782
- content: "short\nlonger line here\nmed",
1783
- filetype: "javascript",
1784
- syntaxStyle,
1785
- treeSitterClient: mockClient,
1786
- drawUnstyledText: false,
1787
- })
1788
-
1789
- currentRenderer.root.add(codeRenderable)
1790
-
1791
- expect(codeRenderable.lineCount).toBe(3)
1792
- expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
1793
-
1794
- await renderOnce()
1795
-
1796
- expect(mockClient.isHighlighting()).toBe(true)
1797
- expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
1798
- expect(codeRenderable.lineInfo.lineSources.length).toBe(3)
1799
-
1800
- mockClient.resolveHighlightOnce(0)
1801
- await new Promise((resolve) => setTimeout(resolve, 10))
1802
- await renderOnce()
1803
-
1804
- expect(codeRenderable.lineInfo.lineStartCols.length).toBe(3)
1805
- expect(codeRenderable.lineInfo.lineSources.length).toBe(3)
1806
- })
1807
-
1808
- test("CodeRenderable - plainText reflects content immediately with drawUnstyledText=false", async () => {
1809
- const syntaxStyle = SyntaxStyle.fromStyles({
1810
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1811
- })
1812
-
1813
- const mockClient = new MockTreeSitterClient()
1814
- mockClient.setMockResult({ highlights: [] })
1815
-
1816
- const codeRenderable = new CodeRenderable(currentRenderer, {
1817
- id: "test-code",
1818
- content: "initial content",
1819
- filetype: "javascript",
1820
- syntaxStyle,
1821
- treeSitterClient: mockClient,
1822
- drawUnstyledText: false,
1823
- })
1824
-
1825
- expect(codeRenderable.plainText).toBe("initial content")
1826
-
1827
- currentRenderer.root.add(codeRenderable)
1828
- await renderOnce()
1829
-
1830
- expect(mockClient.isHighlighting()).toBe(true)
1831
- expect(codeRenderable.plainText).toBe("initial content")
1832
-
1833
- codeRenderable.content = "updated content"
1834
- expect(codeRenderable.plainText).toBe("updated content")
1835
-
1836
- await renderOnce()
1837
- const frame = captureFrame()
1838
- expect(frame.trim()).toBe("")
1839
-
1840
- mockClient.resolveAllHighlightOnce()
1841
- await new Promise((resolve) => setTimeout(resolve, 10))
1842
- await renderOnce()
1843
-
1844
- expect(codeRenderable.plainText).toBe("updated content")
1845
- const finalFrame = captureFrame()
1846
- expect(finalFrame).toContain("updated content")
1847
- })
1848
-
1849
- test("CodeRenderable - textLength is correct with drawUnstyledText=false", async () => {
1850
- const syntaxStyle = SyntaxStyle.fromStyles({
1851
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1852
- })
1853
-
1854
- const mockClient = new MockTreeSitterClient()
1855
- mockClient.setMockResult({ highlights: [] })
1856
-
1857
- const content = "hello world test"
1858
- const codeRenderable = new CodeRenderable(currentRenderer, {
1859
- id: "test-code",
1860
- content,
1861
- filetype: "javascript",
1862
- syntaxStyle,
1863
- treeSitterClient: mockClient,
1864
- drawUnstyledText: false,
1865
- })
1866
-
1867
- expect(codeRenderable.textLength).toBe(content.length)
1868
-
1869
- currentRenderer.root.add(codeRenderable)
1870
- await renderOnce()
1871
-
1872
- expect(codeRenderable.textLength).toBe(content.length)
1873
-
1874
- const newContent = "longer content here"
1875
- codeRenderable.content = newContent
1876
- expect(codeRenderable.textLength).toBe(newContent.length)
1877
- })
1878
-
1879
- test("CodeRenderable - streaming mode with drawUnstyledText=false has correct lineCount", async () => {
1880
- const syntaxStyle = SyntaxStyle.fromStyles({
1881
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1882
- })
1883
-
1884
- const mockClient = new MockTreeSitterClient()
1885
- mockClient.setMockResult({ highlights: [] })
1886
-
1887
- const codeRenderable = new CodeRenderable(currentRenderer, {
1888
- id: "test-code",
1889
- content: "line1\nline2",
1890
- filetype: "javascript",
1891
- syntaxStyle,
1892
- treeSitterClient: mockClient,
1893
- streaming: true,
1894
- drawUnstyledText: false,
1895
- })
1896
-
1897
- expect(codeRenderable.lineCount).toBe(2)
1898
-
1899
- currentRenderer.root.add(codeRenderable)
1900
- await renderOnce()
1901
-
1902
- const frameBeforeHighlighting = captureFrame()
1903
- expect(frameBeforeHighlighting.trim()).toBe("")
1904
-
1905
- mockClient.resolveHighlightOnce(0)
1906
- await new Promise((resolve) => setTimeout(resolve, 10))
1907
- await renderOnce()
1908
-
1909
- expect(codeRenderable.lineCount).toBe(2)
1910
-
1911
- codeRenderable.content = "line1\nline2\nline3\nline4"
1912
- expect(codeRenderable.lineCount).toBe(2)
1913
-
1914
- codeRenderable.content = "line1\nline2\nline3\nline4\nline5\nline6"
1915
- expect(codeRenderable.lineCount).toBe(2)
1916
-
1917
- await renderOnce()
1918
- mockClient.resolveAllHighlightOnce()
1919
- await new Promise((resolve) => setTimeout(resolve, 10))
1920
- await renderOnce()
1921
-
1922
- expect(codeRenderable.lineCount).toBe(6)
1923
- const finalFrame = captureFrame()
1924
- expect(finalFrame).toContain("line1")
1925
- })
1926
-
1927
- test("CodeRenderable - streaming with conceal and drawUnstyledText=false should not jump when fenced code blocks are concealed", async () => {
1928
- resize(80, 20)
1929
-
1930
- const syntaxStyle = SyntaxStyle.fromStyles({
1931
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
1932
- keyword: { fg: RGBA.fromValues(0, 0, 1, 1) },
1933
- string: { fg: RGBA.fromValues(0, 1, 0, 1) },
1934
- "markup.heading.1": { fg: RGBA.fromValues(0, 0, 1, 1) },
1935
- "markup.raw.block": { fg: RGBA.fromValues(0.5, 0.5, 0.5, 1) },
1936
- })
1937
-
1938
- const codeRenderable = new CodeRenderable(currentRenderer, {
1939
- id: "test-markdown",
1940
- content: "# Example",
1941
- filetype: "markdown",
1942
- syntaxStyle,
1943
- streaming: true,
1944
- conceal: true,
1945
- drawUnstyledText: false,
1946
- left: 0,
1947
- top: 0,
1948
- })
1949
-
1950
- currentRenderer.root.add(codeRenderable)
1951
-
1952
- const waitForHighlightingCycle = async (timeout = 2000) => {
1953
- const start = Date.now()
1954
- await renderOnce()
1955
- await new Promise((resolve) => setTimeout(resolve, 10))
1956
- while (codeRenderable.isHighlighting && Date.now() - start < timeout) {
1957
- await new Promise((resolve) => setTimeout(resolve, 10))
1958
- }
1959
- await renderOnce()
1960
- }
1961
-
1962
- // Use TestRecorder to capture frames
1963
- const { TestRecorder } = await import("../testing/test-recorder.js")
1964
- const recorder = new TestRecorder(currentRenderer)
1965
-
1966
- // Start renderer and recorder
1967
- currentRenderer.start()
1968
- recorder.rec()
1969
-
1970
- // Wait for initial highlighting to complete
1971
- await waitForHighlightingCycle()
1972
-
1973
- // Now simulate streaming: add more content including fenced code block
1974
- codeRenderable.content = `# Example\n\nHere's some code:\n\n\`\`\`typescript\nconst x = 1;\n\`\`\``
1975
-
1976
- // Wait for highlighting to process the update
1977
- await waitForHighlightingCycle()
1978
-
1979
- // Stop everything
1980
- currentRenderer.stop()
1981
- recorder.stop()
1982
-
1983
- const frames = recorder.recordedFrames
1984
-
1985
- // Analyze frames to detect the presence of backticks
1986
- const frameAnalysis: Array<{ hasBackticks: boolean; lineCount: number; isEmpty: boolean }> = []
1987
-
1988
- for (const recordedFrame of frames) {
1989
- const frame = recordedFrame.frame
1990
- const hasBackticks = frame.includes("```")
1991
- const lines = frame.split("\n").filter((line) => line.trim().length > 0)
1992
- const isEmpty = frame.trim().length === 0
1993
-
1994
- frameAnalysis.push({
1995
- hasBackticks,
1996
- lineCount: lines.length,
1997
- isEmpty,
1998
- })
1999
- }
2000
-
2001
- let hasFlickering = false
2002
- for (let i = 2; i < frameAnalysis.length; i++) {
2003
- const prev = frameAnalysis[i - 1]
2004
- const curr = frameAnalysis[i]
2005
- if (!prev.isEmpty && curr.isEmpty) {
2006
- hasFlickering = true
2007
- }
2008
- }
2009
-
2010
- const framesWithBackticks = frameAnalysis.filter((f) => f.hasBackticks && !f.isEmpty)
2011
-
2012
- expect(framesWithBackticks.length).toBe(0)
2013
- expect(hasFlickering).toBe(false)
2014
-
2015
- const finalFrame = frameAnalysis[frameAnalysis.length - 1]
2016
- expect(finalFrame.isEmpty).toBe(false)
2017
- expect(finalFrame.hasBackticks).toBe(false)
2018
- expect(finalFrame.lineCount).toBe(3)
2019
-
2020
- const finalFrameText = frames[frames.length - 1].frame
2021
- expect(finalFrameText).toContain("Example")
2022
- expect(finalFrameText).toContain("Here's some code")
2023
- expect(finalFrameText).toContain("const x = 1")
2024
- expect(finalFrameText).not.toContain("```")
2025
- })
2026
-
2027
- test("CodeRenderable - streaming with drawUnstyledText=false falls back to unstyled text when highlights fail", async () => {
2028
- const syntaxStyle = SyntaxStyle.fromStyles({
2029
- default: { fg: RGBA.fromValues(1, 1, 1, 1) },
2030
- })
2031
-
2032
- const mockClient = new MockTreeSitterClient({ autoResolveTimeout: 10 })
2033
-
2034
- const codeRenderable = new CodeRenderable(currentRenderer, {
2035
- id: "test-code",
2036
- content: "const initial = 'hello';",
2037
- filetype: "javascript",
2038
- syntaxStyle,
2039
- treeSitterClient: mockClient,
2040
- streaming: true,
2041
- drawUnstyledText: false,
2042
- left: 0,
2043
- top: 0,
2044
- })
2045
-
2046
- currentRenderer.root.add(codeRenderable)
2047
- currentRenderer.start()
2048
-
2049
- await Bun.sleep(30)
2050
-
2051
- mockClient.highlightOnce = async () => {
2052
- throw new Error("Highlighting failed")
2053
- }
2054
-
2055
- codeRenderable.content = "const updated = 'world';"
2056
-
2057
- await Bun.sleep(30)
2058
-
2059
- expect(codeRenderable.plainText).toBe("const updated = 'world';")
2060
-
2061
- currentRenderer.stop()
2062
- })