@fairyhunter13/opentui-core 0.1.88 → 0.1.90

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 (570) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +141 -0
  7. package/docs/env-vars.md +140 -0
  8. package/docs/getting-started.md +353 -0
  9. package/docs/renderables-vs-constructs.md +159 -0
  10. package/docs/tree-sitter.md +311 -0
  11. package/package.json +61 -52
  12. package/scripts/build.ts +400 -0
  13. package/scripts/publish.ts +60 -0
  14. package/src/3d/SpriteResourceManager.ts +286 -0
  15. package/src/3d/SpriteUtils.ts +71 -0
  16. package/src/3d/TextureUtils.ts +196 -0
  17. package/src/3d/ThreeRenderable.ts +197 -0
  18. package/src/3d/WGPURenderer.ts +294 -0
  19. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  20. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  21. package/src/3d/animation/SpriteAnimator.ts +633 -0
  22. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  23. package/src/3d/canvas.ts +464 -0
  24. package/src/3d/index.ts +12 -0
  25. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  26. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  27. package/src/3d/physics/physics-interface.ts +31 -0
  28. package/src/3d/shaders/supersampling.wgsl +201 -0
  29. package/src/3d.ts +3 -0
  30. package/src/NativeSpanFeed.ts +300 -0
  31. package/src/Renderable.ts +1698 -0
  32. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  33. package/src/animation/Timeline.test.ts +2709 -0
  34. package/src/animation/Timeline.ts +598 -0
  35. package/src/ansi.ts +18 -0
  36. package/src/benchmark/latest-all-bench-run.json +707 -0
  37. package/src/benchmark/latest-async-bench-run.json +336 -0
  38. package/src/benchmark/latest-default-bench-run.json +657 -0
  39. package/src/benchmark/latest-large-bench-run.json +707 -0
  40. package/src/benchmark/latest-quick-bench-run.json +207 -0
  41. package/src/benchmark/markdown-benchmark.ts +1804 -0
  42. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  43. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  44. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  45. package/src/benchmark/native-span-feed-compare.ts +280 -0
  46. package/src/benchmark/renderer-benchmark.ts +754 -0
  47. package/src/benchmark/text-table-benchmark.ts +947 -0
  48. package/src/buffer.test.ts +291 -0
  49. package/src/buffer.ts +519 -0
  50. package/src/console.test.ts +612 -0
  51. package/src/console.ts +1255 -0
  52. package/src/edit-buffer.test.ts +1769 -0
  53. package/src/edit-buffer.ts +411 -0
  54. package/src/editor-view.test.ts +1032 -0
  55. package/src/editor-view.ts +284 -0
  56. package/src/examples/ascii-font-selection-demo.ts +245 -0
  57. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  58. package/src/examples/assets/concrete.png +0 -0
  59. package/src/examples/assets/crate.png +0 -0
  60. package/src/examples/assets/crate_emissive.png +0 -0
  61. package/src/examples/assets/forrest_background.png +0 -0
  62. package/src/examples/assets/hast-example.json +1018 -0
  63. package/src/examples/assets/heart.png +0 -0
  64. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  65. package/src/examples/assets/main_char_idle.png +0 -0
  66. package/src/examples/assets/main_char_jump_end.png +0 -0
  67. package/src/examples/assets/main_char_jump_landing.png +0 -0
  68. package/src/examples/assets/main_char_jump_start.png +0 -0
  69. package/src/examples/assets/main_char_run_loop.png +0 -0
  70. package/src/examples/assets/roughness_map.jpg +0 -0
  71. package/src/examples/build.ts +115 -0
  72. package/src/examples/code-demo.ts +584 -0
  73. package/src/examples/console-demo.ts +358 -0
  74. package/src/examples/core-plugin-slots-demo.ts +759 -0
  75. package/src/examples/diff-demo.ts +699 -0
  76. package/src/examples/draggable-three-demo.ts +259 -0
  77. package/src/examples/editor-demo.ts +322 -0
  78. package/src/examples/extmarks-demo.ts +204 -0
  79. package/src/examples/focus-restore-demo.ts +310 -0
  80. package/src/examples/fonts.ts +245 -0
  81. package/src/examples/fractal-shader-demo.ts +268 -0
  82. package/src/examples/framebuffer-demo.ts +674 -0
  83. package/src/examples/full-unicode-demo.ts +181 -0
  84. package/src/examples/golden-star-demo.ts +933 -0
  85. package/src/examples/grayscale-buffer-demo.ts +249 -0
  86. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  87. package/src/examples/index.ts +925 -0
  88. package/src/examples/input-demo.ts +377 -0
  89. package/src/examples/input-select-layout-demo.ts +425 -0
  90. package/src/examples/install.sh +143 -0
  91. package/src/examples/keypress-debug-demo.ts +452 -0
  92. package/src/examples/lib/HexList.ts +122 -0
  93. package/src/examples/lib/PaletteGrid.ts +125 -0
  94. package/src/examples/lib/standalone-keys.ts +25 -0
  95. package/src/examples/lib/tab-controller.ts +243 -0
  96. package/src/examples/lights-phong-demo.ts +290 -0
  97. package/src/examples/link-demo.ts +220 -0
  98. package/src/examples/live-state-demo.ts +480 -0
  99. package/src/examples/markdown-demo.ts +620 -0
  100. package/src/examples/mouse-interaction-demo.ts +428 -0
  101. package/src/examples/nested-zindex-demo.ts +357 -0
  102. package/src/examples/opacity-example.ts +235 -0
  103. package/src/examples/opentui-demo.ts +1057 -0
  104. package/src/examples/physx-planck-2d-demo.ts +507 -0
  105. package/src/examples/physx-rapier-2d-demo.ts +526 -0
  106. package/src/examples/relative-positioning-demo.ts +323 -0
  107. package/src/examples/scroll-example.ts +214 -0
  108. package/src/examples/scrollbox-mouse-test.ts +112 -0
  109. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  110. package/src/examples/select-demo.ts +237 -0
  111. package/src/examples/shader-cube-demo.ts +772 -0
  112. package/src/examples/simple-layout-example.ts +591 -0
  113. package/src/examples/slider-demo.ts +617 -0
  114. package/src/examples/split-mode-demo.ts +445 -0
  115. package/src/examples/sprite-animation-demo.ts +443 -0
  116. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  117. package/src/examples/static-sprite-demo.ts +193 -0
  118. package/src/examples/sticky-scroll-example.ts +308 -0
  119. package/src/examples/styled-text-demo.ts +282 -0
  120. package/src/examples/tab-select-demo.ts +219 -0
  121. package/src/examples/terminal-title.ts +29 -0
  122. package/src/examples/terminal.ts +305 -0
  123. package/src/examples/text-node-demo.ts +416 -0
  124. package/src/examples/text-selection-demo.ts +377 -0
  125. package/src/examples/text-table-demo.ts +503 -0
  126. package/src/examples/text-truncation-demo.ts +481 -0
  127. package/src/examples/text-wrap.ts +757 -0
  128. package/src/examples/texture-loading-demo.ts +259 -0
  129. package/src/examples/timeline-example.ts +670 -0
  130. package/src/examples/transparency-demo.ts +241 -0
  131. package/src/examples/vnode-composition-demo.ts +404 -0
  132. package/src/index.ts +22 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +168 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +31 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +280 -0
  168. package/src/lib/keymapping.ts +87 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +1676 -0
  187. package/src/lib/stdin-parser.ts +1248 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +331 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +270 -0
  197. package/src/lib/tree-sitter/client.test.ts +1061 -0
  198. package/src/lib/tree-sitter/client.ts +615 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +80 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1001 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +75 -0
  204. package/src/lib/tree-sitter/resolve-ft.ts +62 -0
  205. package/src/lib/tree-sitter/types.ts +81 -0
  206. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  207. package/src/lib/tree-sitter-styled-text.ts +306 -0
  208. package/src/lib/validate-dir-name.ts +55 -0
  209. package/src/lib/yoga.options.test.ts +628 -0
  210. package/src/lib/yoga.options.ts +346 -0
  211. package/src/plugins/core-slot.ts +579 -0
  212. package/src/plugins/registry.ts +377 -0
  213. package/src/plugins/types.ts +46 -0
  214. package/src/post/filters.ts +888 -0
  215. package/src/renderables/ASCIIFont.ts +219 -0
  216. package/src/renderables/Box.test.ts +160 -0
  217. package/src/renderables/Box.ts +295 -0
  218. package/src/renderables/Code.test.ts +2062 -0
  219. package/src/renderables/Code.ts +357 -0
  220. package/src/renderables/Diff.regression.test.ts +226 -0
  221. package/src/renderables/Diff.test.ts +3027 -0
  222. package/src/renderables/Diff.ts +1209 -0
  223. package/src/renderables/EditBufferRenderable.ts +764 -0
  224. package/src/renderables/FrameBuffer.ts +47 -0
  225. package/src/renderables/Input.test.ts +1228 -0
  226. package/src/renderables/Input.ts +245 -0
  227. package/src/renderables/LineNumberRenderable.ts +675 -0
  228. package/src/renderables/Markdown.ts +1106 -0
  229. package/src/renderables/ScrollBar.ts +422 -0
  230. package/src/renderables/ScrollBox.ts +883 -0
  231. package/src/renderables/Select.test.ts +1010 -0
  232. package/src/renderables/Select.ts +523 -0
  233. package/src/renderables/Slider.test.ts +456 -0
  234. package/src/renderables/Slider.ts +347 -0
  235. package/src/renderables/TabSelect.test.ts +197 -0
  236. package/src/renderables/TabSelect.ts +455 -0
  237. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  238. package/src/renderables/Text.test.ts +2660 -0
  239. package/src/renderables/Text.ts +147 -0
  240. package/src/renderables/TextBufferRenderable.ts +518 -0
  241. package/src/renderables/TextNode.test.ts +1058 -0
  242. package/src/renderables/TextNode.ts +325 -0
  243. package/src/renderables/TextTable.test.ts +1421 -0
  244. package/src/renderables/TextTable.ts +1344 -0
  245. package/src/renderables/Textarea.ts +732 -0
  246. package/src/renderables/TimeToFirstDraw.ts +89 -0
  247. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  248. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  249. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  250. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  251. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  252. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  253. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1787 -0
  254. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  255. package/src/renderables/__tests__/Markdown.test.ts +2287 -0
  256. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  257. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  258. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  259. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  260. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  261. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  262. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  263. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  264. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  265. package/src/renderables/__tests__/Textarea.rendering.test.ts +1864 -0
  266. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  267. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  268. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  269. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  270. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  271. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  272. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  273. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  274. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  275. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  276. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  277. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  278. package/src/renderables/composition/README.md +8 -0
  279. package/src/renderables/composition/VRenderable.ts +32 -0
  280. package/src/renderables/composition/constructs.ts +127 -0
  281. package/src/renderables/composition/vnode.ts +289 -0
  282. package/src/renderables/index.ts +22 -0
  283. package/src/renderables/markdown-parser.ts +66 -0
  284. package/src/renderer.ts +2363 -0
  285. package/src/runtime-plugin-support.ts +39 -0
  286. package/src/runtime-plugin.ts +144 -0
  287. package/src/syntax-style.test.ts +841 -0
  288. package/src/syntax-style.ts +264 -0
  289. package/src/testing/README.md +210 -0
  290. package/src/testing/capture-spans.test.ts +194 -0
  291. package/src/testing/integration.test.ts +276 -0
  292. package/src/testing/manual-clock.ts +106 -0
  293. package/src/testing/mock-keys.test.ts +1356 -0
  294. package/src/testing/mock-keys.ts +449 -0
  295. package/src/testing/mock-mouse.test.ts +218 -0
  296. package/src/testing/mock-mouse.ts +247 -0
  297. package/src/testing/mock-tree-sitter-client.ts +73 -0
  298. package/src/testing/spy.ts +13 -0
  299. package/src/testing/test-recorder.test.ts +415 -0
  300. package/src/testing/test-recorder.ts +145 -0
  301. package/src/testing/test-renderer.ts +116 -0
  302. package/src/testing.ts +7 -0
  303. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  304. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  305. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  306. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  307. package/src/tests/allocator-stats.test.ts +38 -0
  308. package/src/tests/destroy-during-render.test.ts +200 -0
  309. package/src/tests/hover-cursor.test.ts +98 -0
  310. package/src/tests/native-span-feed-async.test.ts +173 -0
  311. package/src/tests/native-span-feed-close.test.ts +120 -0
  312. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  313. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  314. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  315. package/src/tests/opacity.test.ts +123 -0
  316. package/src/tests/renderable.snapshot.test.ts +524 -0
  317. package/src/tests/renderable.test.ts +1281 -0
  318. package/src/tests/renderer.console-startup.test.ts +65 -0
  319. package/src/tests/renderer.control.test.ts +364 -0
  320. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  321. package/src/tests/renderer.cursor.test.ts +26 -0
  322. package/src/tests/renderer.destroy-during-render.test.ts +110 -0
  323. package/src/tests/renderer.focus-restore.test.ts +228 -0
  324. package/src/tests/renderer.focus.test.ts +251 -0
  325. package/src/tests/renderer.idle.test.ts +219 -0
  326. package/src/tests/renderer.input.test.ts +2145 -0
  327. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  328. package/src/tests/renderer.mouse.test.ts +1269 -0
  329. package/src/tests/renderer.palette.test.ts +629 -0
  330. package/src/tests/renderer.selection.test.ts +49 -0
  331. package/src/tests/renderer.slot-registry.test.ts +649 -0
  332. package/src/tests/renderer.useMouse.test.ts +50 -0
  333. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  334. package/src/tests/runtime-plugin-support.test.ts +28 -0
  335. package/src/tests/runtime-plugin.fixture.ts +40 -0
  336. package/src/tests/runtime-plugin.test.ts +190 -0
  337. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  338. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  339. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  340. package/src/tests/scrollbox.test.ts +1530 -0
  341. package/src/tests/wrap-resize-perf.test.ts +229 -0
  342. package/src/tests/yoga-setters.test.ts +921 -0
  343. package/src/text-buffer-view.test.ts +705 -0
  344. package/src/text-buffer-view.ts +189 -0
  345. package/src/text-buffer.test.ts +347 -0
  346. package/src/text-buffer.ts +250 -0
  347. package/src/types.ts +152 -0
  348. package/src/utils.ts +88 -0
  349. package/src/zig/ansi.zig +268 -0
  350. package/src/zig/bench/README.md +50 -0
  351. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  352. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  353. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  354. package/src/zig/bench/rope-markers_bench.zig +713 -0
  355. package/src/zig/bench/rope_bench.zig +514 -0
  356. package/src/zig/bench/styled-text_bench.zig +470 -0
  357. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  358. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  359. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  360. package/src/zig/bench/utf8_bench.zig +799 -0
  361. package/src/zig/bench-utils.zig +431 -0
  362. package/src/zig/bench.zig +217 -0
  363. package/src/zig/buffer.zig +2223 -0
  364. package/src/zig/build.zig +289 -0
  365. package/src/zig/build.zig.zon +16 -0
  366. package/src/zig/edit-buffer.zig +825 -0
  367. package/src/zig/editor-view.zig +802 -0
  368. package/src/zig/event-bus.zig +13 -0
  369. package/src/zig/event-emitter.zig +65 -0
  370. package/src/zig/file-logger.zig +92 -0
  371. package/src/zig/grapheme.zig +599 -0
  372. package/src/zig/lib.zig +1834 -0
  373. package/src/zig/link.zig +333 -0
  374. package/src/zig/logger.zig +43 -0
  375. package/src/zig/mem-registry.zig +125 -0
  376. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  377. package/src/zig/native-span-feed.zig +708 -0
  378. package/src/zig/renderer.zig +1386 -0
  379. package/src/zig/rope.zig +1220 -0
  380. package/src/zig/syntax-style.zig +161 -0
  381. package/src/zig/terminal.zig +975 -0
  382. package/src/zig/test.zig +70 -0
  383. package/src/zig/tests/README.md +18 -0
  384. package/src/zig/tests/buffer_test.zig +2526 -0
  385. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  386. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  387. package/src/zig/tests/editor-view_test.zig +3299 -0
  388. package/src/zig/tests/event-emitter_test.zig +249 -0
  389. package/src/zig/tests/grapheme_test.zig +1304 -0
  390. package/src/zig/tests/link_test.zig +190 -0
  391. package/src/zig/tests/mem-registry_test.zig +473 -0
  392. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  393. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  394. package/src/zig/tests/renderer_test.zig +1010 -0
  395. package/src/zig/tests/rope-nested_test.zig +712 -0
  396. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  397. package/src/zig/tests/rope_test.zig +2362 -0
  398. package/src/zig/tests/segment-merge.test.zig +148 -0
  399. package/src/zig/tests/syntax-style_test.zig +557 -0
  400. package/src/zig/tests/terminal_test.zig +719 -0
  401. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  402. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  403. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  404. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  405. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  406. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  407. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  408. package/src/zig/tests/text-buffer_test.zig +2191 -0
  409. package/src/zig/tests/unicode-width-map.zon +3909 -0
  410. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  411. package/src/zig/tests/utf8_test.zig +4057 -0
  412. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  413. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  414. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  415. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  416. package/src/zig/text-buffer-iterators.zig +499 -0
  417. package/src/zig/text-buffer-segment.zig +404 -0
  418. package/src/zig/text-buffer-view.zig +1371 -0
  419. package/src/zig/text-buffer.zig +1180 -0
  420. package/src/zig/utf8.zig +1948 -0
  421. package/src/zig/utils.zig +9 -0
  422. package/src/zig-structs.ts +261 -0
  423. package/src/zig.ts +3843 -0
  424. package/tsconfig.build.json +22 -0
  425. package/tsconfig.json +28 -0
  426. package/3d/SpriteResourceManager.d.ts +0 -74
  427. package/3d/SpriteUtils.d.ts +0 -13
  428. package/3d/TextureUtils.d.ts +0 -24
  429. package/3d/ThreeRenderable.d.ts +0 -40
  430. package/3d/WGPURenderer.d.ts +0 -61
  431. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  432. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  433. package/3d/animation/SpriteAnimator.d.ts +0 -124
  434. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  435. package/3d/canvas.d.ts +0 -44
  436. package/3d/index.d.ts +0 -12
  437. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  438. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  439. package/3d/physics/physics-interface.d.ts +0 -27
  440. package/3d.d.ts +0 -2
  441. package/3d.js +0 -34042
  442. package/3d.js.map +0 -155
  443. package/LICENSE +0 -21
  444. package/NativeSpanFeed.d.ts +0 -41
  445. package/Renderable.d.ts +0 -334
  446. package/animation/Timeline.d.ts +0 -126
  447. package/ansi.d.ts +0 -13
  448. package/buffer.d.ts +0 -107
  449. package/console.d.ts +0 -143
  450. package/edit-buffer.d.ts +0 -98
  451. package/editor-view.d.ts +0 -73
  452. package/index-e4hzc2j2.js +0 -113
  453. package/index-e4hzc2j2.js.map +0 -10
  454. package/index-nkrr8a4c.js +0 -18415
  455. package/index-nkrr8a4c.js.map +0 -64
  456. package/index-nyw5p3ep.js +0 -12619
  457. package/index-nyw5p3ep.js.map +0 -43
  458. package/index.d.ts +0 -21
  459. package/index.js +0 -430
  460. package/index.js.map +0 -9
  461. package/lib/KeyHandler.d.ts +0 -61
  462. package/lib/RGBA.d.ts +0 -25
  463. package/lib/ascii.font.d.ts +0 -508
  464. package/lib/border.d.ts +0 -49
  465. package/lib/bunfs.d.ts +0 -7
  466. package/lib/clipboard.d.ts +0 -17
  467. package/lib/clock.d.ts +0 -15
  468. package/lib/data-paths.d.ts +0 -26
  469. package/lib/debounce.d.ts +0 -42
  470. package/lib/detect-links.d.ts +0 -6
  471. package/lib/env.d.ts +0 -42
  472. package/lib/extmarks-history.d.ts +0 -17
  473. package/lib/extmarks.d.ts +0 -89
  474. package/lib/hast-styled-text.d.ts +0 -17
  475. package/lib/index.d.ts +0 -21
  476. package/lib/keymapping.d.ts +0 -25
  477. package/lib/objects-in-viewport.d.ts +0 -24
  478. package/lib/output.capture.d.ts +0 -24
  479. package/lib/parse.keypress-kitty.d.ts +0 -2
  480. package/lib/parse.keypress.d.ts +0 -26
  481. package/lib/parse.mouse.d.ts +0 -30
  482. package/lib/paste.d.ts +0 -7
  483. package/lib/queue.d.ts +0 -15
  484. package/lib/renderable.validations.d.ts +0 -12
  485. package/lib/scroll-acceleration.d.ts +0 -43
  486. package/lib/selection.d.ts +0 -63
  487. package/lib/singleton.d.ts +0 -7
  488. package/lib/stdin-parser.d.ts +0 -76
  489. package/lib/styled-text.d.ts +0 -63
  490. package/lib/terminal-capability-detection.d.ts +0 -30
  491. package/lib/terminal-palette.d.ts +0 -50
  492. package/lib/tree-sitter/assets/update.d.ts +0 -11
  493. package/lib/tree-sitter/client.d.ts +0 -47
  494. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  495. package/lib/tree-sitter/download-utils.d.ts +0 -21
  496. package/lib/tree-sitter/index.d.ts +0 -8
  497. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  498. package/lib/tree-sitter/parsers-config.d.ts +0 -38
  499. package/lib/tree-sitter/resolve-ft.d.ts +0 -2
  500. package/lib/tree-sitter/types.d.ts +0 -81
  501. package/lib/tree-sitter-styled-text.d.ts +0 -14
  502. package/lib/validate-dir-name.d.ts +0 -1
  503. package/lib/yoga.options.d.ts +0 -32
  504. package/parser.worker.js +0 -869
  505. package/parser.worker.js.map +0 -12
  506. package/plugins/core-slot.d.ts +0 -72
  507. package/plugins/registry.d.ts +0 -38
  508. package/plugins/types.d.ts +0 -34
  509. package/post/filters.d.ts +0 -105
  510. package/renderables/ASCIIFont.d.ts +0 -52
  511. package/renderables/Box.d.ts +0 -72
  512. package/renderables/Code.d.ts +0 -78
  513. package/renderables/Diff.d.ts +0 -142
  514. package/renderables/EditBufferRenderable.d.ts +0 -162
  515. package/renderables/FrameBuffer.d.ts +0 -16
  516. package/renderables/Input.d.ts +0 -67
  517. package/renderables/LineNumberRenderable.d.ts +0 -74
  518. package/renderables/Markdown.d.ts +0 -173
  519. package/renderables/ScrollBar.d.ts +0 -77
  520. package/renderables/ScrollBox.d.ts +0 -124
  521. package/renderables/Select.d.ts +0 -115
  522. package/renderables/Slider.d.ts +0 -44
  523. package/renderables/TabSelect.d.ts +0 -96
  524. package/renderables/Text.d.ts +0 -36
  525. package/renderables/TextBufferRenderable.d.ts +0 -105
  526. package/renderables/TextNode.d.ts +0 -91
  527. package/renderables/TextTable.d.ts +0 -140
  528. package/renderables/Textarea.d.ts +0 -114
  529. package/renderables/TimeToFirstDraw.d.ts +0 -24
  530. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  531. package/renderables/composition/VRenderable.d.ts +0 -16
  532. package/renderables/composition/constructs.d.ts +0 -35
  533. package/renderables/composition/vnode.d.ts +0 -46
  534. package/renderables/index.d.ts +0 -22
  535. package/renderables/markdown-parser.d.ts +0 -10
  536. package/renderer.d.ts +0 -388
  537. package/runtime-plugin-support.d.ts +0 -3
  538. package/runtime-plugin-support.js +0 -29
  539. package/runtime-plugin-support.js.map +0 -10
  540. package/runtime-plugin.d.ts +0 -11
  541. package/runtime-plugin.js +0 -16
  542. package/runtime-plugin.js.map +0 -9
  543. package/syntax-style.d.ts +0 -54
  544. package/testing/manual-clock.d.ts +0 -16
  545. package/testing/mock-keys.d.ts +0 -81
  546. package/testing/mock-mouse.d.ts +0 -38
  547. package/testing/mock-tree-sitter-client.d.ts +0 -23
  548. package/testing/spy.d.ts +0 -7
  549. package/testing/test-recorder.d.ts +0 -61
  550. package/testing/test-renderer.d.ts +0 -23
  551. package/testing.d.ts +0 -6
  552. package/testing.js +0 -675
  553. package/testing.js.map +0 -15
  554. package/text-buffer-view.d.ts +0 -42
  555. package/text-buffer.d.ts +0 -67
  556. package/types.d.ts +0 -131
  557. package/utils.d.ts +0 -14
  558. package/zig-structs.d.ts +0 -155
  559. package/zig.d.ts +0 -351
  560. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  561. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  562. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  563. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  564. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  565. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  566. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  567. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  568. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  569. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  570. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,1061 @@
1
+ import { test, expect, beforeEach, afterEach, beforeAll, describe } from "bun:test"
2
+ import { TreeSitterClient } from "./client.js"
3
+ import { tmpdir } from "os"
4
+ import { join } from "path"
5
+ import { mkdir, writeFile, unlink } from "fs/promises"
6
+ import { getDataPaths } from "../data-paths.js"
7
+ import { getTreeSitterClient } from "./index.js"
8
+
9
+ describe("TreeSitterClient", () => {
10
+ let client: TreeSitterClient
11
+ let dataPath: string
12
+
13
+ const sharedDataPath = join(tmpdir(), "tree-sitter-shared-test-data")
14
+
15
+ beforeAll(async () => {
16
+ await mkdir(sharedDataPath, { recursive: true })
17
+ })
18
+
19
+ beforeEach(async () => {
20
+ dataPath = sharedDataPath
21
+ client = new TreeSitterClient({
22
+ dataPath,
23
+ })
24
+ })
25
+
26
+ afterEach(async () => {
27
+ if (client) {
28
+ await client.destroy()
29
+ }
30
+ })
31
+
32
+ test("should initialize successfully", async () => {
33
+ await client.initialize()
34
+ expect(client.isInitialized()).toBe(true)
35
+ })
36
+
37
+ test("should preload parsers for supported filetypes", async () => {
38
+ await client.initialize()
39
+
40
+ const hasJavaScript = await client.preloadParser("javascript")
41
+ expect(hasJavaScript).toBe(true)
42
+
43
+ const hasTypeScript = await client.preloadParser("typescript")
44
+ expect(hasTypeScript).toBe(true)
45
+ })
46
+
47
+ test("should return false for unsupported filetypes", async () => {
48
+ await client.initialize()
49
+
50
+ const hasUnsupported = await client.preloadParser("unsupported-language")
51
+ expect(hasUnsupported).toBe(false)
52
+ })
53
+
54
+ test("should create buffer with supported filetype", async () => {
55
+ await client.initialize()
56
+
57
+ const jsCode = 'const hello = "world";'
58
+ const hasParser = await client.createBuffer(1, jsCode, "javascript")
59
+
60
+ expect(hasParser).toBe(true)
61
+
62
+ const buffer = client.getBuffer(1)
63
+ expect(buffer).toBeDefined()
64
+ expect(buffer?.hasParser).toBe(true)
65
+ expect(buffer?.content).toBe(jsCode)
66
+ expect(buffer?.filetype).toBe("javascript")
67
+ })
68
+
69
+ test("should create buffer without parser for unsupported filetype", async () => {
70
+ await client.initialize()
71
+
72
+ const content = "some random content"
73
+ const hasParser = await client.createBuffer(1, content, "unsupported")
74
+
75
+ expect(hasParser).toBe(false)
76
+
77
+ const buffer = client.getBuffer(1)
78
+ expect(buffer).toBeDefined()
79
+ expect(buffer?.hasParser).toBe(false)
80
+ })
81
+
82
+ test("should emit highlights:response event when buffer is updated", async () => {
83
+ await client.initialize()
84
+
85
+ const jsCode = 'const hello = "world";'
86
+ await client.createBuffer(1, jsCode, "javascript")
87
+
88
+ let highlightReceived = false
89
+ let receivedBufferId: number | undefined
90
+ let receivedVersion: number | undefined
91
+
92
+ client.on("highlights:response", (bufferId, version, highlights) => {
93
+ highlightReceived = true
94
+ receivedBufferId = bufferId
95
+ receivedVersion = version
96
+ })
97
+
98
+ await new Promise((resolve) => setTimeout(resolve, 100))
99
+
100
+ const newCode = 'const hello = "world";\nconst foo = 42;'
101
+ const edits = [
102
+ {
103
+ startIndex: jsCode.length,
104
+ oldEndIndex: jsCode.length,
105
+ newEndIndex: newCode.length,
106
+ startPosition: { row: 0, column: jsCode.length },
107
+ oldEndPosition: { row: 0, column: jsCode.length },
108
+ newEndPosition: { row: 1, column: 14 },
109
+ },
110
+ ]
111
+
112
+ await client.updateBuffer(1, edits, newCode, 2)
113
+
114
+ await new Promise((resolve) => setTimeout(resolve, 200))
115
+
116
+ expect(highlightReceived).toBe(true)
117
+ expect(receivedBufferId).toBe(1)
118
+ expect(receivedVersion).toBe(2)
119
+ })
120
+
121
+ test("should handle buffer removal", async () => {
122
+ await client.initialize()
123
+
124
+ const jsCode = 'const hello = "world";'
125
+ await client.createBuffer(1, jsCode, "javascript")
126
+
127
+ let bufferDisposed = false
128
+ client.on("buffer:disposed", (bufferId) => {
129
+ if (bufferId === 1) {
130
+ bufferDisposed = true
131
+ }
132
+ })
133
+
134
+ await client.removeBuffer(1)
135
+
136
+ expect(bufferDisposed).toBe(true)
137
+ expect(client.getBuffer(1)).toBeUndefined()
138
+ })
139
+
140
+ test("should handle multiple buffers", async () => {
141
+ await client.initialize()
142
+
143
+ const jsCode = 'const hello = "world";'
144
+ const tsCode = "interface Test { value: string }"
145
+
146
+ await client.createBuffer(1, jsCode, "javascript")
147
+ await client.createBuffer(2, tsCode, "typescript")
148
+
149
+ const buffers = client.getAllBuffers()
150
+ expect(buffers).toHaveLength(2)
151
+
152
+ const jsBuffer = client.getBuffer(1)
153
+ const tsBuffer = client.getBuffer(2)
154
+
155
+ expect(jsBuffer?.filetype).toBe("javascript")
156
+ expect(tsBuffer?.filetype).toBe("typescript")
157
+ expect(jsBuffer?.hasParser).toBe(true)
158
+ expect(tsBuffer?.hasParser).toBe(true)
159
+ })
160
+
161
+ test("should handle buffer reset", async () => {
162
+ await client.initialize()
163
+
164
+ const jsCode = 'const hello = "world";'
165
+ await client.createBuffer(1, jsCode, "javascript")
166
+
167
+ const newContent = "function test() { return 42; }"
168
+ await client.resetBuffer(1, 2, newContent)
169
+
170
+ const buffer = client.getBuffer(1)
171
+ expect(buffer?.content).toBe(newContent)
172
+ expect(buffer?.version).toBe(2)
173
+ })
174
+
175
+ test("should emit error events for invalid operations", async () => {
176
+ await client.initialize()
177
+
178
+ let errorReceived = false
179
+ let errorMessage = ""
180
+
181
+ client.on("error", (error, bufferId) => {
182
+ errorReceived = true
183
+ errorMessage = error
184
+ })
185
+
186
+ await client.resetBuffer(999, 1, "test")
187
+
188
+ expect(errorReceived).toBe(true)
189
+ expect(errorMessage).toContain("Cannot reset buffer with no parser")
190
+ })
191
+
192
+ test("should prevent duplicate buffer creation", async () => {
193
+ await client.initialize()
194
+
195
+ const jsCode = 'const hello = "world";'
196
+ await client.createBuffer(1, jsCode, "javascript")
197
+
198
+ await expect(client.createBuffer(1, "other code", "javascript")).rejects.toThrow("Buffer with id 1 already exists")
199
+ })
200
+
201
+ test("should handle performance metrics", async () => {
202
+ await client.initialize()
203
+
204
+ const performance = await client.getPerformance()
205
+ expect(performance).toBeDefined()
206
+ expect(typeof performance.averageParseTime).toBe("number")
207
+ expect(typeof performance.averageQueryTime).toBe("number")
208
+ expect(Array.isArray(performance.parseTimes)).toBe(true)
209
+ expect(Array.isArray(performance.queryTimes)).toBe(true)
210
+ })
211
+
212
+ test("should handle concurrent buffer operations", async () => {
213
+ await client.initialize()
214
+
215
+ const promises = []
216
+
217
+ for (let i = 0; i < 5; i++) {
218
+ const code = `const var${i} = ${i};`
219
+ promises.push(client.createBuffer(i, code, "javascript"))
220
+ }
221
+
222
+ const results = await Promise.all(promises)
223
+ expect(results.every((result) => result === true)).toBe(true)
224
+
225
+ const buffers = client.getAllBuffers()
226
+ expect(buffers).toHaveLength(5)
227
+ })
228
+
229
+ test("should clean up resources on destroy", async () => {
230
+ await client.initialize()
231
+
232
+ const jsCode = 'const hello = "world";'
233
+ await client.createBuffer(1, jsCode, "javascript")
234
+
235
+ expect(client.getAllBuffers()).toHaveLength(1)
236
+
237
+ await client.destroy()
238
+
239
+ expect(client.isInitialized()).toBe(false)
240
+ expect(client.getAllBuffers()).toHaveLength(0)
241
+ })
242
+
243
+ test("should perform one-shot highlighting", async () => {
244
+ await client.initialize()
245
+
246
+ const jsCode = 'const hello = "world";\nfunction test() { return 42; }'
247
+ const result = await client.highlightOnce(jsCode, "javascript")
248
+
249
+ expect(result.highlights).toBeDefined()
250
+ expect(result.highlights!.length).toBeGreaterThan(0)
251
+
252
+ const firstHighlight = result.highlights![0]
253
+ expect(Array.isArray(firstHighlight)).toBe(true)
254
+ expect(firstHighlight).toHaveLength(3)
255
+ expect(typeof firstHighlight[0]).toBe("number")
256
+ expect(typeof firstHighlight[1]).toBe("number")
257
+ expect(typeof firstHighlight[2]).toBe("string")
258
+
259
+ const groups = result.highlights!.map((hl) => hl[2])
260
+ expect(groups.length).toBeGreaterThan(0)
261
+ expect(groups).toContain("keyword")
262
+ })
263
+
264
+ test("should handle one-shot highlighting for unsupported filetype", async () => {
265
+ await client.initialize()
266
+
267
+ const result = await client.highlightOnce("some content", "unsupported-lang")
268
+
269
+ expect(result.highlights).toBeUndefined()
270
+ expect(result.warning).toContain("No parser available for filetype unsupported-lang")
271
+ }, 5000)
272
+
273
+ test("should perform multiple one-shot highlights independently", async () => {
274
+ await client.initialize()
275
+
276
+ const jsCode = 'const hello = "world";'
277
+ const tsCode = "interface Test { value: string }"
278
+
279
+ const [jsResult, tsResult] = await Promise.all([
280
+ client.highlightOnce(jsCode, "javascript"),
281
+ client.highlightOnce(tsCode, "typescript"),
282
+ ])
283
+
284
+ expect(jsResult.highlights).toBeDefined()
285
+ expect(tsResult.highlights).toBeDefined()
286
+ expect(jsResult.highlights!.length).toBeGreaterThan(0)
287
+ expect(tsResult.highlights!.length).toBeGreaterThan(0)
288
+
289
+ jsResult.highlights!.forEach((hl) => {
290
+ expect(Array.isArray(hl)).toBe(true)
291
+ expect(hl).toHaveLength(3)
292
+ })
293
+
294
+ tsResult.highlights!.forEach((hl) => {
295
+ expect(Array.isArray(hl)).toBe(true)
296
+ expect(hl).toHaveLength(3)
297
+ })
298
+
299
+ expect(client.getAllBuffers()).toHaveLength(0)
300
+ })
301
+
302
+ test("should handle Devanagari characters and highlight ranges after them correctly", async () => {
303
+ await client.initialize()
304
+
305
+ const jsCode = 'const greeting = "नमस्ते";\nconst x = 42;'
306
+ const result = await client.highlightOnce(jsCode, "javascript")
307
+
308
+ expect(result.highlights).toBeDefined()
309
+ expect(result.highlights!.length).toBeGreaterThan(0)
310
+
311
+ const keywordHighlights = result.highlights!.filter((hl) => hl[2] === "keyword")
312
+ expect(keywordHighlights.length).toBeGreaterThanOrEqual(2)
313
+
314
+ const constHighlights = keywordHighlights.filter((hl) => {
315
+ const text = jsCode.substring(hl[0], hl[1])
316
+ return text === "const"
317
+ })
318
+
319
+ expect(constHighlights).toHaveLength(2)
320
+
321
+ const firstConst = constHighlights[0]
322
+ const secondConst = constHighlights[1]
323
+
324
+ expect(jsCode.substring(firstConst[0], firstConst[1])).toBe("const")
325
+ expect(jsCode.substring(secondConst[0], secondConst[1])).toBe("const")
326
+
327
+ expect(firstConst[0]).toBe(0)
328
+ expect(firstConst[1]).toBe(5)
329
+
330
+ expect(secondConst[0]).toBeGreaterThan(firstConst[1])
331
+ const textBetween = jsCode.substring(firstConst[1], secondConst[0])
332
+ expect(textBetween).toContain("नमस्ते")
333
+
334
+ const numberHighlight = result.highlights!.find((hl) => {
335
+ const text = jsCode.substring(hl[0], hl[1])
336
+ return text === "42" && hl[2] === "number"
337
+ })
338
+
339
+ expect(numberHighlight).toBeDefined()
340
+ if (numberHighlight) {
341
+ const [start, end] = numberHighlight
342
+ const actualText = jsCode.substring(start, end)
343
+ expect(actualText).toBe("42")
344
+
345
+ const secondLine = jsCode.split("\n")[1]
346
+ const secondLineStart = jsCode.indexOf(secondLine)
347
+ const expectedStart = secondLineStart + secondLine.indexOf("42")
348
+ expect(start).toBe(expectedStart)
349
+ }
350
+ })
351
+
352
+ test("should support local file paths for parser configuration", async () => {
353
+ const testQueryPath = join(dataPath, `test-highlights-${Date.now()}.scm`)
354
+ const simpleQuery = "(identifier) @variable"
355
+ await writeFile(testQueryPath, simpleQuery, "utf8")
356
+
357
+ try {
358
+ client.addFiletypeParser({
359
+ filetype: "test-lang",
360
+ queries: {
361
+ highlights: [testQueryPath],
362
+ },
363
+ wasm: "https://github.com/tree-sitter/tree-sitter-javascript/releases/download/v0.23.1/tree-sitter-javascript.wasm",
364
+ })
365
+
366
+ await client.initialize()
367
+
368
+ const hasParser = await client.preloadParser("test-lang")
369
+ expect(hasParser).toBe(true)
370
+
371
+ const testCode = "const myVariable = 42;"
372
+ const result = await client.highlightOnce(testCode, "test-lang")
373
+
374
+ expect(result.highlights).toBeDefined()
375
+ expect(result.error).toBeUndefined()
376
+ expect(result.warning).toBeUndefined()
377
+ } finally {
378
+ try {
379
+ await unlink(testQueryPath)
380
+ } catch (e) {
381
+ // Ignore cleanup errors
382
+ }
383
+ }
384
+ })
385
+
386
+ test("should handle concurrent highlightOnce calls efficiently (no duplicate parser loading)", async () => {
387
+ const freshClient = new TreeSitterClient({ dataPath })
388
+ const workerLogs: string[] = []
389
+
390
+ freshClient.on("worker:log", (logType, message) => {
391
+ if (message.includes("Loading from local path:")) {
392
+ workerLogs.push(message)
393
+ }
394
+ })
395
+
396
+ try {
397
+ await freshClient.initialize()
398
+
399
+ const jsCode = 'const hello = "world"; function test() { return 42; }'
400
+ const promises = Array.from({ length: 5 }, () => freshClient.highlightOnce(jsCode, "javascript"))
401
+
402
+ const results = await Promise.all(promises)
403
+
404
+ for (const result of results) {
405
+ expect(result.highlights).toBeDefined()
406
+ expect(result.highlights!.length).toBeGreaterThan(0)
407
+ expect(result.error).toBeUndefined()
408
+ }
409
+
410
+ const firstResult = results[0]
411
+ for (let i = 1; i < results.length; i++) {
412
+ expect(results[i].highlights).toEqual(firstResult.highlights)
413
+ }
414
+
415
+ await new Promise((resolve) => setTimeout(resolve, 100))
416
+
417
+ const languageLoadLogs = workerLogs.filter((log) => log.includes("tree-sitter-javascript.wasm"))
418
+ const queryLoadLogs = workerLogs.filter((log) => log.includes("highlights.scm"))
419
+
420
+ expect(languageLoadLogs.length).toBeLessThanOrEqual(1)
421
+ expect(queryLoadLogs.length).toBeLessThanOrEqual(1)
422
+ } finally {
423
+ await freshClient.destroy()
424
+ }
425
+ })
426
+ })
427
+
428
+ describe("TreeSitterClient Injections", () => {
429
+ let dataPath: string
430
+
431
+ const injectionsDataPath = join(tmpdir(), "tree-sitter-injections-test-data")
432
+
433
+ beforeAll(async () => {
434
+ await mkdir(injectionsDataPath, { recursive: true })
435
+ })
436
+
437
+ beforeEach(async () => {
438
+ dataPath = injectionsDataPath
439
+ })
440
+
441
+ test("should highlight inline code in markdown using markdown_inline injection", async () => {
442
+ const client = new TreeSitterClient({ dataPath })
443
+
444
+ try {
445
+ await client.initialize()
446
+
447
+ const markdownCode = `# Hello World
448
+
449
+ The \`CodeRenderable\` component provides syntax highlighting.
450
+
451
+ You can use \`const x = 42\` in your code.`
452
+
453
+ const result = await client.highlightOnce(markdownCode, "markdown")
454
+
455
+ expect(result.highlights).toBeDefined()
456
+ expect(result.highlights!.length).toBeGreaterThan(0)
457
+
458
+ const groups = result.highlights!.map((hl) => hl[2])
459
+ const hasInlineCodeHighlights = groups.some((g) => g.includes("markup.raw"))
460
+
461
+ expect(hasInlineCodeHighlights).toBe(true)
462
+ } finally {
463
+ await client.destroy()
464
+ }
465
+ }, 10000)
466
+
467
+ test("should highlight code blocks in markdown using language-specific injection", async () => {
468
+ const client = new TreeSitterClient({ dataPath })
469
+
470
+ try {
471
+ await client.initialize()
472
+
473
+ const markdownCode = `# Code Example
474
+
475
+ \`\`\`typescript
476
+ const hello: string = "world";
477
+ function test() { return 42; }
478
+ \`\`\`
479
+
480
+ Some text here.`
481
+
482
+ const result = await client.highlightOnce(markdownCode, "markdown")
483
+
484
+ expect(result.highlights).toBeDefined()
485
+ expect(result.highlights!.length).toBeGreaterThan(0)
486
+
487
+ const groups = result.highlights!.map((hl) => hl[2])
488
+ const hasTypeScriptHighlights = groups.some((g) => g === "keyword" || g === "type" || g === "function")
489
+
490
+ expect(hasTypeScriptHighlights).toBe(true)
491
+ } finally {
492
+ await client.destroy()
493
+ }
494
+ }, 10000)
495
+
496
+ test("should return correct offsets for injected code in markdown code blocks", async () => {
497
+ const client = new TreeSitterClient({ dataPath })
498
+
499
+ try {
500
+ await client.initialize()
501
+
502
+ const markdownCode = `# Title\n\n\`\`\`typescript\nconst x = 42;\n\`\`\``
503
+
504
+ const result = await client.highlightOnce(markdownCode, "markdown")
505
+
506
+ expect(result.highlights).toBeDefined()
507
+ expect(result.highlights!.length).toBeGreaterThan(0)
508
+
509
+ const constHighlight = result.highlights!.find((hl) => {
510
+ const text = markdownCode.substring(hl[0], hl[1])
511
+ return text === "const" && hl[2] === "keyword"
512
+ })
513
+
514
+ expect(constHighlight).toBeDefined()
515
+ if (constHighlight) {
516
+ const [start, end, group] = constHighlight
517
+ const text = markdownCode.substring(start, end)
518
+
519
+ expect(text).toBe("const")
520
+ expect(group).toBe("keyword")
521
+ expect(start).toBe(23)
522
+ expect(end).toBe(28)
523
+ }
524
+
525
+ const numberHighlight = result.highlights!.find((hl) => {
526
+ const text = markdownCode.substring(hl[0], hl[1])
527
+ return text === "42" && hl[2] === "number"
528
+ })
529
+
530
+ expect(numberHighlight).toBeDefined()
531
+ if (numberHighlight) {
532
+ const [start, end, group] = numberHighlight
533
+ const text = markdownCode.substring(start, end)
534
+
535
+ expect(text).toBe("42")
536
+ expect(group).toBe("number")
537
+ expect(start).toBe(33)
538
+ expect(end).toBe(35)
539
+ }
540
+ } finally {
541
+ await client.destroy()
542
+ }
543
+ }, 10000)
544
+
545
+ test("should return highlights sorted by start offset for injected code", async () => {
546
+ const client = new TreeSitterClient({ dataPath })
547
+
548
+ try {
549
+ await client.initialize()
550
+
551
+ const markdownCode = `# Documentation
552
+
553
+ Some text with \`inline code\` here.
554
+
555
+ \`\`\`typescript
556
+ const first = 1;
557
+ const second = 2;
558
+ \`\`\`
559
+
560
+ More text with \`another inline\` code.
561
+
562
+ \`\`\`javascript
563
+ function test() {
564
+ return 42;
565
+ }
566
+ \`\`\``
567
+
568
+ const result = await client.highlightOnce(markdownCode, "markdown")
569
+
570
+ expect(result.highlights).toBeDefined()
571
+ expect(result.highlights!.length).toBeGreaterThan(0)
572
+
573
+ for (let i = 1; i < result.highlights!.length; i++) {
574
+ const prevStart = result.highlights![i - 1][0]
575
+ const currStart = result.highlights![i][0]
576
+ expect(currStart).toBeGreaterThanOrEqual(prevStart)
577
+ }
578
+ } finally {
579
+ await client.destroy()
580
+ }
581
+ }, 10000)
582
+
583
+ test("should handle markdown with injections and return valid highlights", async () => {
584
+ const client = new TreeSitterClient({ dataPath })
585
+
586
+ try {
587
+ await client.initialize()
588
+
589
+ const markdownCode = `# Heading
590
+
591
+ Some **bold** text with \`inline code\`.
592
+
593
+ \`\`\`typescript
594
+ const x: string = "hello";
595
+ \`\`\`
596
+
597
+ [Link text](https://example.com)`
598
+
599
+ const result = await client.highlightOnce(markdownCode, "markdown")
600
+
601
+ expect(result.highlights).toBeDefined()
602
+ expect(result.highlights!.length).toBeGreaterThan(0)
603
+
604
+ const overlaps: Array<[number, number]> = []
605
+ for (let i = 0; i < result.highlights!.length; i++) {
606
+ for (let j = i + 1; j < result.highlights!.length; j++) {
607
+ const [start1, end1] = result.highlights![i]
608
+ const [start2, end2] = result.highlights![j]
609
+
610
+ if (start2 < end1) {
611
+ overlaps.push([i, j])
612
+ }
613
+ }
614
+ }
615
+
616
+ expect(overlaps.length).toBeGreaterThanOrEqual(0)
617
+
618
+ const injectionHighlights = result.highlights!.filter((hl) => hl[2].includes("injection"))
619
+ expect(injectionHighlights).toBeDefined()
620
+
621
+ const concealHighlights = result.highlights!.filter((hl) => hl[2] === "conceal")
622
+ expect(concealHighlights).toBeDefined()
623
+
624
+ const blockHighlights = result.highlights!.filter((hl) => hl[2] === "markup.raw.block")
625
+ expect(blockHighlights).toBeDefined()
626
+ } finally {
627
+ await client.destroy()
628
+ }
629
+ }, 10000)
630
+
631
+ test("should handle fast concurrent markdown highlighting requests with injections", async () => {
632
+ const client = new TreeSitterClient({ dataPath })
633
+
634
+ const errors: string[] = []
635
+ client.on("error", (error) => {
636
+ errors.push(error)
637
+ })
638
+
639
+ client.on("worker:log", (logType, message) => {
640
+ if (logType === "error") {
641
+ errors.push(message)
642
+ }
643
+ })
644
+
645
+ try {
646
+ await client.initialize()
647
+
648
+ const markdownCode = `# OpenTUI Documentation
649
+
650
+ ## Getting Started
651
+
652
+ OpenTUI is a modern terminal UI framework built on **tree-sitter** and WebGPU.
653
+
654
+ ### Installation
655
+
656
+ \`\`\`bash
657
+ bun install opentui
658
+ \`\`\`
659
+
660
+ ### Quick Example
661
+
662
+ \`\`\`typescript
663
+ import { createCliRenderer, BoxRenderable } from 'opentui';
664
+
665
+ const renderer = await createCliRenderer();
666
+ const box = new BoxRenderable(renderer, {
667
+ border: true,
668
+ title: "Hello World"
669
+ });
670
+ renderer.root.add(box);
671
+ \`\`\`
672
+
673
+ The \`CodeRenderable\` component provides syntax highlighting.
674
+
675
+ | Property | Type | Description |
676
+ |----------|------|-------------|
677
+ | content | string | Code to display |
678
+ | filetype | string | Language type |`
679
+
680
+ const jsCode = `function test() {
681
+ const hello = "world";
682
+ return hello;
683
+ }`
684
+
685
+ const tsCode = `interface User {
686
+ name: string;
687
+ age: number;
688
+ }
689
+
690
+ const user: User = { name: "Alice", age: 25 };`
691
+
692
+ const promises = []
693
+ for (let i = 0; i < 5; i++) {
694
+ promises.push(client.highlightOnce(markdownCode, "markdown"))
695
+ }
696
+
697
+ const results = await Promise.allSettled(promises)
698
+
699
+ for (let i = 0; i < results.length; i++) {
700
+ const result = results[i]
701
+ if (result.status === "fulfilled") {
702
+ expect(result.value.error).toBeUndefined()
703
+ expect(result.value.highlights).toBeDefined()
704
+ } else {
705
+ throw new Error(`Request ${i} was rejected: ${result.reason}`)
706
+ }
707
+ }
708
+
709
+ await new Promise((resolve) => setTimeout(resolve, 500))
710
+
711
+ const hasMemoryErrors = errors.some((err) => err.includes("Out of bounds memory access"))
712
+ expect(hasMemoryErrors).toBe(false)
713
+ } finally {
714
+ await client.destroy()
715
+ }
716
+ }, 15000)
717
+ })
718
+
719
+ describe("TreeSitterClient Conceal Values", () => {
720
+ let dataPath: string
721
+
722
+ const concealDataPath = join(tmpdir(), "tree-sitter-conceal-test-data")
723
+
724
+ beforeAll(async () => {
725
+ await mkdir(concealDataPath, { recursive: true })
726
+ })
727
+
728
+ beforeEach(async () => {
729
+ dataPath = concealDataPath
730
+ })
731
+
732
+ test("should return conceal values from normal (non-injected) queries", async () => {
733
+ const client = new TreeSitterClient({ dataPath })
734
+
735
+ try {
736
+ await client.initialize()
737
+
738
+ const markdownCode = `![Image Alt Text](https://example.com/image.png)`
739
+
740
+ const result = await client.highlightOnce(markdownCode, "markdown")
741
+
742
+ expect(result.highlights).toBeDefined()
743
+ expect(result.error).toBeUndefined()
744
+
745
+ const concealedHighlights = result.highlights!.filter((hl) => {
746
+ const meta = (hl as any)[3]
747
+ return meta && meta.conceal !== undefined
748
+ })
749
+
750
+ expect(concealedHighlights.length).toBeGreaterThan(0)
751
+
752
+ concealedHighlights.forEach((hl) => {
753
+ const meta = (hl as any)[3]
754
+ expect(meta.conceal).toBeDefined()
755
+ })
756
+ } finally {
757
+ await client.destroy()
758
+ }
759
+ }, 10000)
760
+
761
+ test("should return conceal values from injected queries (markdown_inline)", async () => {
762
+ const client = new TreeSitterClient({ dataPath })
763
+
764
+ try {
765
+ await client.initialize()
766
+
767
+ const markdownCode = `Here is a [link](https://example.com) in text.`
768
+
769
+ const result = await client.highlightOnce(markdownCode, "markdown")
770
+
771
+ expect(result.highlights).toBeDefined()
772
+ expect(result.error).toBeUndefined()
773
+
774
+ const concealedHighlights = result.highlights!.filter((hl) => {
775
+ const meta = (hl as any)[3]
776
+ return meta && meta.conceal !== undefined
777
+ })
778
+
779
+ expect(concealedHighlights.length).toBeGreaterThan(0)
780
+
781
+ concealedHighlights.forEach((hl) => {
782
+ const meta = (hl as any)[3]
783
+ expect(meta.conceal).toBeDefined()
784
+ expect(meta.isInjection).toBeDefined()
785
+ })
786
+
787
+ const closingBracketHighlight = concealedHighlights.find((hl) => {
788
+ const text = markdownCode.substring(hl[0], hl[1])
789
+ const meta = (hl as any)[3]
790
+ return text === "]" && meta.conceal !== ""
791
+ })
792
+
793
+ if (closingBracketHighlight) {
794
+ const meta = (closingBracketHighlight as any)[3]
795
+ expect(meta.conceal).toBeDefined()
796
+ }
797
+ } finally {
798
+ await client.destroy()
799
+ }
800
+ }, 10000)
801
+
802
+ test("should distinguish conceal values between normal and injected queries", async () => {
803
+ const client = new TreeSitterClient({ dataPath })
804
+
805
+ try {
806
+ await client.initialize()
807
+
808
+ const markdownCode = `Here is a [link](https://example.com) and ![image](https://example.com/img.png).`
809
+
810
+ const result = await client.highlightOnce(markdownCode, "markdown")
811
+
812
+ expect(result.highlights).toBeDefined()
813
+ expect(result.error).toBeUndefined()
814
+
815
+ const concealedHighlights = result.highlights!.filter((hl) => {
816
+ const meta = (hl as any)[3]
817
+ return meta && meta.conceal !== undefined
818
+ })
819
+
820
+ expect(concealedHighlights.length).toBeGreaterThan(0)
821
+
822
+ const normalConceal = concealedHighlights.filter((hl) => {
823
+ const meta = (hl as any)[3]
824
+ return !meta.isInjection
825
+ })
826
+
827
+ const injectedConceal = concealedHighlights.filter((hl) => {
828
+ const meta = (hl as any)[3]
829
+ return meta.isInjection
830
+ })
831
+
832
+ expect(injectedConceal.length).toBeGreaterThan(0)
833
+
834
+ injectedConceal.forEach((hl) => {
835
+ const meta = (hl as any)[3]
836
+ expect(meta.conceal).toBeDefined()
837
+ expect(meta.isInjection).toBe(true)
838
+ })
839
+
840
+ concealedHighlights.forEach((hl) => {
841
+ const meta = (hl as any)[3]
842
+ expect(meta.conceal).toBeDefined()
843
+ expect(typeof meta.isInjection).toBe("boolean")
844
+ })
845
+ } finally {
846
+ await client.destroy()
847
+ }
848
+ }, 10000)
849
+
850
+ test("should handle pattern index lookups correctly for injections", async () => {
851
+ const client = new TreeSitterClient({ dataPath })
852
+
853
+ try {
854
+ await client.initialize()
855
+
856
+ const markdownCode = `A [link](url) here.`
857
+
858
+ const result = await client.highlightOnce(markdownCode, "markdown")
859
+
860
+ expect(result.highlights).toBeDefined()
861
+ expect(result.error).toBeUndefined()
862
+
863
+ const concealedHighlights = result.highlights!.filter((hl) => {
864
+ const meta = (hl as any)[3]
865
+ return meta && meta.conceal !== undefined
866
+ })
867
+
868
+ expect(concealedHighlights.length).toBeGreaterThan(0)
869
+
870
+ concealedHighlights.forEach((hl) => {
871
+ const meta = (hl as any)[3]
872
+ expect(meta.conceal).toBeDefined()
873
+ })
874
+ } finally {
875
+ await client.destroy()
876
+ }
877
+ }, 10000)
878
+
879
+ test("should handle multiple injected languages with different conceal patterns", async () => {
880
+ const client = new TreeSitterClient({ dataPath })
881
+
882
+ try {
883
+ await client.initialize()
884
+
885
+ const markdownCode = `# Title
886
+
887
+ Inline \`code\` and a [link](url) here.
888
+
889
+ \`\`\`typescript
890
+ const x = 42;
891
+ \`\`\`
892
+
893
+ More text with ![image](img.png) and **bold**.`
894
+
895
+ const result = await client.highlightOnce(markdownCode, "markdown")
896
+
897
+ expect(result.highlights).toBeDefined()
898
+ expect(result.error).toBeUndefined()
899
+
900
+ const concealedHighlights = result.highlights!.filter((hl) => {
901
+ const meta = (hl as any)[3]
902
+ return meta && meta.conceal !== undefined
903
+ })
904
+
905
+ expect(concealedHighlights.length).toBeGreaterThan(0)
906
+
907
+ const byLang = new Map<string, any[]>()
908
+ concealedHighlights.forEach((hl) => {
909
+ const meta = (hl as any)[3]
910
+ const lang = meta.isInjection ? meta.injectionLang || "injected" : "normal"
911
+ if (!byLang.has(lang)) {
912
+ byLang.set(lang, [])
913
+ }
914
+ byLang.get(lang)!.push(hl)
915
+ })
916
+
917
+ expect(byLang.size).toBeGreaterThan(0)
918
+
919
+ byLang.forEach((highlights) => {
920
+ expect(highlights.length).toBeGreaterThan(0)
921
+ highlights.forEach((hl: any) => {
922
+ const meta = hl[3]
923
+ expect(meta.conceal).toBeDefined()
924
+ })
925
+ })
926
+ } finally {
927
+ await client.destroy()
928
+ }
929
+ }, 10000)
930
+
931
+ test("should preserve non-empty conceal replacements like space character", async () => {
932
+ const client = new TreeSitterClient({ dataPath })
933
+
934
+ try {
935
+ await client.initialize()
936
+
937
+ const markdownCode = `Check [this link](https://example.com) out!`
938
+
939
+ const result = await client.highlightOnce(markdownCode, "markdown")
940
+
941
+ expect(result.highlights).toBeDefined()
942
+ expect(result.error).toBeUndefined()
943
+
944
+ const closingBracket = result.highlights!.find((hl) => {
945
+ const text = markdownCode.substring(hl[0], hl[1])
946
+ const meta = (hl as any)[3]
947
+ return text === "]" && hl[2] === "conceal" && meta?.conceal !== undefined
948
+ })
949
+
950
+ if (closingBracket) {
951
+ const meta = (closingBracket as any)[3]
952
+ expect(meta).toBeDefined()
953
+ expect(meta.conceal).toBeDefined()
954
+ expect(meta.conceal).toBe(" ")
955
+ expect(meta.conceal.length).toBeGreaterThan(0)
956
+ }
957
+ } finally {
958
+ await client.destroy()
959
+ }
960
+ }, 10000)
961
+ })
962
+
963
+ describe("TreeSitterClient Edge Cases", () => {
964
+ let dataPath: string
965
+
966
+ const edgeCaseDataPath = join(tmpdir(), "tree-sitter-edge-case-test-data")
967
+
968
+ beforeAll(async () => {
969
+ await mkdir(edgeCaseDataPath, { recursive: true })
970
+ })
971
+
972
+ beforeEach(async () => {
973
+ dataPath = edgeCaseDataPath
974
+ })
975
+
976
+ test("should handle initialization timeout", async () => {
977
+ const client = new TreeSitterClient({
978
+ dataPath,
979
+ workerPath: "invalid-path",
980
+ initTimeout: 500,
981
+ })
982
+
983
+ await expect(client.initialize()).rejects.toThrow(/Worker error|Worker initialization timed out/)
984
+
985
+ await client.destroy()
986
+ })
987
+
988
+ test("should handle operations before initialization", async () => {
989
+ const client = new TreeSitterClient({ dataPath })
990
+
991
+ expect(client.isInitialized()).toBe(false)
992
+ expect(client.getAllBuffers()).toHaveLength(0)
993
+ expect(client.getBuffer(1)).toBeUndefined()
994
+
995
+ await client.destroy()
996
+ })
997
+
998
+ test("should handle destroy() during pending initialization", async () => {
999
+ const client = new TreeSitterClient({ dataPath })
1000
+
1001
+ // Start init but don't await
1002
+ const initPromise = client.initialize()
1003
+
1004
+ // Immediately destroy
1005
+ await client.destroy()
1006
+
1007
+ // Init promise should reject with specific error
1008
+ await expect(initPromise).rejects.toThrow("Client destroyed during initialization")
1009
+
1010
+ expect(client.isInitialized()).toBe(false)
1011
+ })
1012
+
1013
+ test("should handle worker errors gracefully", async () => {
1014
+ const client = new TreeSitterClient({ dataPath })
1015
+
1016
+ let errorReceived = false
1017
+ client.on("error", () => {
1018
+ errorReceived = true
1019
+ })
1020
+
1021
+ const hasParser = await client.createBuffer(1, "test", "javascript", 1, false)
1022
+ expect(hasParser).toBe(false)
1023
+ expect(errorReceived).toBe(true)
1024
+
1025
+ await client.destroy()
1026
+ })
1027
+
1028
+ test("should handle data path changes with reactive getTreeSitterClient", async () => {
1029
+ const dataPathsManager = getDataPaths()
1030
+ const originalAppName = dataPathsManager.appName
1031
+ let client: any
1032
+
1033
+ try {
1034
+ client = getTreeSitterClient()
1035
+ await client.initialize()
1036
+
1037
+ const initialDataPath = dataPathsManager.globalDataPath
1038
+
1039
+ dataPathsManager.appName = "test-app-changed"
1040
+
1041
+ await new Promise((resolve) => setTimeout(resolve, 100))
1042
+
1043
+ const newDataPath = dataPathsManager.globalDataPath
1044
+ expect(newDataPath).not.toBe(initialDataPath)
1045
+ expect(newDataPath).toContain("test-app-changed")
1046
+
1047
+ if (!client.isInitialized()) {
1048
+ await client.initialize()
1049
+ }
1050
+
1051
+ const hasParser = await client.preloadParser("javascript")
1052
+ expect(hasParser).toBe(true)
1053
+ } finally {
1054
+ if (client) {
1055
+ await client.destroy()
1056
+ }
1057
+
1058
+ dataPathsManager.appName = originalAppName
1059
+ }
1060
+ })
1061
+ })