@fairyhunter13/opentui-core 0.1.112 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +63 -51
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-8fks7yv1.js +0 -411
  472. package/index-8fks7yv1.js.map +0 -10
  473. package/index-egy5e2rs.js +0 -12267
  474. package/index-egy5e2rs.js.map +0 -42
  475. package/index-tse8gzh0.js +0 -20614
  476. package/index-tse8gzh0.js.map +0 -67
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,1042 @@
1
+ // Use type-only imports for type safety; values loaded dynamically to prevent WASM OOB crash
2
+ import type { Parser, Query, Tree, Language, Edit, QueryCapture, Range } from "web-tree-sitter"
3
+ import { mkdir } from "fs/promises"
4
+ import * as path from "path"
5
+ import type {
6
+ HighlightRange,
7
+ HighlightResponse,
8
+ SimpleHighlight,
9
+ FiletypeParserOptions,
10
+ PerformanceStats,
11
+ InjectionMapping,
12
+ } from "./types.js"
13
+ import { DownloadUtils } from "./download-utils.js"
14
+ import { isMainThread } from "worker_threads"
15
+ import { isBunfsPath, normalizeBunfsPath } from "../bunfs.js"
16
+
17
+ // Runtime references loaded dynamically in initialize() to prevent WASM heap
18
+ // initialization before Parser.init() is called (causes OOB memory access crash)
19
+ let TreeSitter: { Parser: typeof Parser; Query: typeof Query; Language: typeof Language }
20
+
21
+ const self = globalThis
22
+
23
+ type ParserState = {
24
+ parser: Parser
25
+ tree: Tree
26
+ queries: {
27
+ highlights: Query
28
+ injections?: Query
29
+ }
30
+ filetype: string
31
+ content: string
32
+ injectionMapping?: InjectionMapping
33
+ }
34
+
35
+ interface FiletypeParser {
36
+ filetype: string
37
+ queries: {
38
+ highlights: Query
39
+ injections?: Query
40
+ }
41
+ language: Language
42
+ injectionMapping?: InjectionMapping
43
+ }
44
+
45
+ interface ReusableParserState {
46
+ parser: Parser
47
+ filetypeParser: FiletypeParser
48
+ queries: {
49
+ highlights: Query
50
+ injections?: Query
51
+ }
52
+ }
53
+
54
+ class ParserWorker {
55
+ private bufferParsers: Map<number, ParserState> = new Map()
56
+ private filetypeParserOptions: Map<string, FiletypeParserOptions> = new Map()
57
+ private filetypeAliases: Map<string, string> = new Map()
58
+ private filetypeParsers: Map<string, FiletypeParser> = new Map()
59
+ private filetypeParserPromises: Map<string, Promise<FiletypeParser | undefined>> = new Map()
60
+ private reusableParsers: Map<string, ReusableParserState> = new Map()
61
+ private reusableParserPromises: Map<string, Promise<ReusableParserState | undefined>> = new Map()
62
+ private initializePromise: Promise<void> | undefined
63
+ public performance: PerformanceStats
64
+ private dataPath: string | undefined
65
+ private tsDataPath: string | undefined
66
+ private initialized: boolean = false
67
+
68
+ constructor() {
69
+ this.performance = {
70
+ averageParseTime: 0,
71
+ parseTimes: [],
72
+ averageQueryTime: 0,
73
+ queryTimes: [],
74
+ }
75
+ }
76
+
77
+ private async fetchQueries(sources: string[], filetype: string): Promise<string> {
78
+ if (!this.tsDataPath) {
79
+ return ""
80
+ }
81
+ return DownloadUtils.fetchHighlightQueries(sources, this.tsDataPath, filetype)
82
+ }
83
+
84
+ async initialize({ dataPath }: { dataPath: string }) {
85
+ if (this.initializePromise) {
86
+ return this.initializePromise
87
+ }
88
+ this.initializePromise = (async () => {
89
+ this.dataPath = dataPath
90
+ this.tsDataPath = path.join(dataPath, "tree-sitter")
91
+
92
+ await mkdir(path.join(this.tsDataPath, "languages"), { recursive: true })
93
+ await mkdir(path.join(this.tsDataPath, "queries"), { recursive: true })
94
+
95
+ // Dynamic import: load web-tree-sitter at runtime to prevent WASM OOB crash
96
+ const mod = await import("web-tree-sitter")
97
+ const ns = (mod.default ?? mod) as Record<string, unknown>
98
+ const P = (ns.Parser ?? ns) as unknown as typeof Parser
99
+ TreeSitter = {
100
+ Parser: P,
101
+ Query: (ns.Query ?? (P as unknown as Record<string, unknown>).Query) as unknown as typeof Query,
102
+ Language: (ns.Language ?? (P as unknown as Record<string, unknown>).Language) as unknown as typeof Language,
103
+ }
104
+
105
+ let { default: treeWasm } = await import("web-tree-sitter/tree-sitter.wasm" as string, {
106
+ with: { type: "wasm" },
107
+ })
108
+
109
+ if (isBunfsPath(treeWasm)) {
110
+ treeWasm = normalizeBunfsPath(path.parse(treeWasm).base)
111
+ }
112
+
113
+ await TreeSitter.Parser.init({
114
+ locateFile() {
115
+ return treeWasm
116
+ },
117
+ })
118
+
119
+ this.initialized = true
120
+ })()
121
+ return this.initializePromise
122
+ }
123
+
124
+ public addFiletypeParser(filetypeParser: FiletypeParserOptions) {
125
+ const previousAliases = this.filetypeParserOptions.get(filetypeParser.filetype)?.aliases ?? []
126
+ for (const alias of previousAliases) {
127
+ if (this.filetypeAliases.get(alias) === filetypeParser.filetype) {
128
+ this.filetypeAliases.delete(alias)
129
+ }
130
+ }
131
+
132
+ const aliases = [...new Set((filetypeParser.aliases ?? []).filter((alias) => alias !== filetypeParser.filetype))]
133
+
134
+ this.filetypeAliases.delete(filetypeParser.filetype)
135
+ this.filetypeParserOptions.set(filetypeParser.filetype, {
136
+ ...filetypeParser,
137
+ aliases,
138
+ })
139
+
140
+ for (const alias of aliases) {
141
+ this.filetypeAliases.set(alias, filetypeParser.filetype)
142
+ }
143
+
144
+ this.invalidateParserCaches(filetypeParser.filetype)
145
+ }
146
+
147
+ private resolveCanonicalFiletype(filetype: string): string {
148
+ if (this.filetypeParserOptions.has(filetype)) {
149
+ return filetype
150
+ }
151
+
152
+ return this.filetypeAliases.get(filetype) ?? filetype
153
+ }
154
+
155
+ private invalidateParserCaches(filetype: string): void {
156
+ this.filetypeParsers.delete(filetype)
157
+ this.filetypeParserPromises.delete(filetype)
158
+
159
+ const reusableParser = this.reusableParsers.get(filetype)
160
+ if (reusableParser) {
161
+ reusableParser.parser.delete()
162
+ this.reusableParsers.delete(filetype)
163
+ }
164
+
165
+ this.reusableParserPromises.delete(filetype)
166
+ }
167
+
168
+ private async createQueries(
169
+ filetypeParser: FiletypeParserOptions,
170
+ language: Language,
171
+ ): Promise<
172
+ | {
173
+ highlights: Query
174
+ injections?: Query
175
+ }
176
+ | undefined
177
+ > {
178
+ try {
179
+ const highlightQueryContent = await this.fetchQueries(filetypeParser.queries.highlights, filetypeParser.filetype)
180
+ if (!highlightQueryContent) {
181
+ console.error("Failed to fetch highlight queries for:", filetypeParser.filetype)
182
+ return undefined
183
+ }
184
+
185
+ const highlightsQuery = new TreeSitter.Query(language, highlightQueryContent)
186
+ const result: { highlights: Query; injections?: Query } = {
187
+ highlights: highlightsQuery,
188
+ }
189
+
190
+ if (filetypeParser.queries.injections && filetypeParser.queries.injections.length > 0) {
191
+ const injectionQueryContent = await this.fetchQueries(
192
+ filetypeParser.queries.injections,
193
+ filetypeParser.filetype,
194
+ )
195
+ if (injectionQueryContent) {
196
+ result.injections = new TreeSitter.Query(language, injectionQueryContent)
197
+ }
198
+ }
199
+
200
+ return result
201
+ } catch (error) {
202
+ console.error("Error creating queries for", filetypeParser.filetype, filetypeParser.queries)
203
+ console.error(error)
204
+ return undefined
205
+ }
206
+ }
207
+
208
+ private async loadLanguage(languageSource: string): Promise<Language | undefined> {
209
+ if (!this.initialized || !this.tsDataPath) {
210
+ return undefined
211
+ }
212
+
213
+ const result = await DownloadUtils.downloadOrLoad(languageSource, this.tsDataPath, "languages", ".wasm", false)
214
+
215
+ if (result.error) {
216
+ console.error(`Error loading language ${languageSource}:`, result.error)
217
+ return undefined
218
+ }
219
+
220
+ if (!result.filePath) {
221
+ return undefined
222
+ }
223
+
224
+ // Normalize path for Windows compatibility - tree-sitter expects forward slashes
225
+ const normalizedPath = result.filePath.replaceAll("\\", "/")
226
+
227
+ try {
228
+ const language = await TreeSitter.Language.load(normalizedPath)
229
+ return language
230
+ } catch (error) {
231
+ console.error(`Error loading language from ${normalizedPath}:`, error)
232
+ return undefined
233
+ }
234
+ }
235
+
236
+ private async resolveFiletypeParser(filetype: string): Promise<FiletypeParser | undefined> {
237
+ const canonicalFiletype = this.resolveCanonicalFiletype(filetype)
238
+
239
+ if (this.filetypeParsers.has(canonicalFiletype)) {
240
+ return this.filetypeParsers.get(canonicalFiletype)
241
+ }
242
+
243
+ if (this.filetypeParserPromises.has(canonicalFiletype)) {
244
+ return this.filetypeParserPromises.get(canonicalFiletype)
245
+ }
246
+
247
+ const loadingPromise = this.loadFiletypeParser(canonicalFiletype)
248
+ this.filetypeParserPromises.set(canonicalFiletype, loadingPromise)
249
+
250
+ try {
251
+ const result = await loadingPromise
252
+ if (result) {
253
+ this.filetypeParsers.set(canonicalFiletype, result)
254
+ }
255
+ return result
256
+ } finally {
257
+ this.filetypeParserPromises.delete(canonicalFiletype)
258
+ }
259
+ }
260
+
261
+ private async loadFiletypeParser(filetype: string): Promise<FiletypeParser | undefined> {
262
+ const filetypeParserOptions = this.filetypeParserOptions.get(filetype)
263
+ if (!filetypeParserOptions) {
264
+ return undefined
265
+ }
266
+ const language = await this.loadLanguage(filetypeParserOptions.wasm)
267
+ if (!language) {
268
+ return undefined
269
+ }
270
+ const queries = await this.createQueries(filetypeParserOptions, language)
271
+ if (!queries) {
272
+ console.error("Failed to create queries for:", filetype)
273
+ return undefined
274
+ }
275
+ const filetypeParser: FiletypeParser = {
276
+ ...filetypeParserOptions,
277
+ queries,
278
+ language,
279
+ }
280
+ return filetypeParser
281
+ }
282
+
283
+ public async preloadParser(filetype: string) {
284
+ return this.resolveFiletypeParser(filetype)
285
+ }
286
+
287
+ private async getReusableParser(filetype: string): Promise<ReusableParserState | undefined> {
288
+ const canonicalFiletype = this.resolveCanonicalFiletype(filetype)
289
+
290
+ if (this.reusableParsers.has(canonicalFiletype)) {
291
+ return this.reusableParsers.get(canonicalFiletype)
292
+ }
293
+
294
+ if (this.reusableParserPromises.has(canonicalFiletype)) {
295
+ return this.reusableParserPromises.get(canonicalFiletype)
296
+ }
297
+
298
+ const creationPromise = this.createReusableParser(canonicalFiletype)
299
+ this.reusableParserPromises.set(canonicalFiletype, creationPromise)
300
+
301
+ try {
302
+ const result = await creationPromise
303
+ if (result) {
304
+ this.reusableParsers.set(canonicalFiletype, result)
305
+ }
306
+ return result
307
+ } finally {
308
+ this.reusableParserPromises.delete(canonicalFiletype)
309
+ }
310
+ }
311
+
312
+ private async createReusableParser(filetype: string): Promise<ReusableParserState | undefined> {
313
+ const filetypeParser = await this.resolveFiletypeParser(filetype)
314
+ if (!filetypeParser) {
315
+ return undefined
316
+ }
317
+
318
+ const parser = new TreeSitter.Parser()
319
+ parser.setLanguage(filetypeParser.language)
320
+
321
+ const reusableState: ReusableParserState = {
322
+ parser,
323
+ filetypeParser,
324
+ queries: filetypeParser.queries,
325
+ }
326
+
327
+ return reusableState
328
+ }
329
+
330
+ async handleInitializeParser(
331
+ bufferId: number,
332
+ version: number,
333
+ content: string,
334
+ filetype: string,
335
+ messageId: string,
336
+ ) {
337
+ const filetypeParser = await this.resolveFiletypeParser(filetype)
338
+
339
+ if (!filetypeParser) {
340
+ self.postMessage({
341
+ type: "PARSER_INIT_RESPONSE",
342
+ bufferId,
343
+ messageId,
344
+ hasParser: false,
345
+ warning: `No parser available for filetype ${filetype}`,
346
+ })
347
+ return
348
+ }
349
+
350
+ const parser = new TreeSitter.Parser()
351
+ parser.setLanguage(filetypeParser.language)
352
+ const tree = parser.parse(content)
353
+ if (!tree) {
354
+ self.postMessage({
355
+ type: "PARSER_INIT_RESPONSE",
356
+ bufferId,
357
+ messageId,
358
+ hasParser: false,
359
+ error: "Failed to parse buffer",
360
+ })
361
+ return
362
+ }
363
+
364
+ const parserState: ParserState = {
365
+ parser,
366
+ tree,
367
+ queries: filetypeParser.queries,
368
+ filetype,
369
+ content,
370
+ injectionMapping: filetypeParser.injectionMapping,
371
+ }
372
+ this.bufferParsers.set(bufferId, parserState)
373
+
374
+ self.postMessage({
375
+ type: "PARSER_INIT_RESPONSE",
376
+ bufferId,
377
+ messageId,
378
+ hasParser: true,
379
+ })
380
+ const highlights = await this.initialQuery(parserState)
381
+ self.postMessage({
382
+ type: "HIGHLIGHT_RESPONSE",
383
+ bufferId,
384
+ version,
385
+ ...highlights,
386
+ })
387
+ }
388
+
389
+ private async initialQuery(parserState: ParserState) {
390
+ const query = parserState.queries.highlights
391
+ const matches: QueryCapture[] = query.captures(parserState.tree.rootNode)
392
+ let injectionRanges = new Map<string, Array<{ start: number; end: number }>>()
393
+
394
+ if (parserState.queries.injections) {
395
+ const injectionResult = await this.processInjections(parserState)
396
+ matches.push(...injectionResult.captures)
397
+ injectionRanges = injectionResult.injectionRanges
398
+ }
399
+
400
+ return this.getHighlights(parserState, matches, injectionRanges)
401
+ }
402
+
403
+ private getNodeText(node: any, content: string): string {
404
+ return content.substring(node.startIndex, node.endIndex)
405
+ }
406
+
407
+ private async processInjections(
408
+ parserState: ParserState,
409
+ ): Promise<{ captures: QueryCapture[]; injectionRanges: Map<string, Array<{ start: number; end: number }>> }> {
410
+ const injectionMatches: QueryCapture[] = []
411
+ const injectionRanges = new Map<string, Array<{ start: number; end: number }>>()
412
+
413
+ if (!parserState.queries.injections) {
414
+ return { captures: injectionMatches, injectionRanges }
415
+ }
416
+
417
+ const content = parserState.content
418
+ const injectionCaptures = parserState.queries.injections.captures(parserState.tree.rootNode)
419
+ const languageGroups = new Map<string, Array<{ node: any; name: string }>>()
420
+
421
+ // Use the injection mapping stored in the parser state
422
+ const injectionMapping = parserState.injectionMapping
423
+
424
+ for (const capture of injectionCaptures) {
425
+ const captureName = capture.name
426
+
427
+ if (captureName === "injection.content" || captureName.includes("injection")) {
428
+ const nodeType = capture.node.type
429
+ let targetLanguage: string | undefined
430
+
431
+ // First, check if there's a direct node type mapping
432
+ if (injectionMapping?.nodeTypes && injectionMapping.nodeTypes[nodeType]) {
433
+ targetLanguage = injectionMapping.nodeTypes[nodeType]
434
+ } else if (nodeType === "code_fence_content") {
435
+ // For code fence content, try to extract language from info_string
436
+ const parent = capture.node.parent
437
+ if (parent) {
438
+ const infoString = parent.children.find((child: any) => child.type === "info_string")
439
+ if (infoString) {
440
+ const languageNode = infoString.children.find((child: any) => child.type === "language")
441
+ if (languageNode) {
442
+ const languageName = this.getNodeText(languageNode, content)
443
+
444
+ if (injectionMapping?.infoStringMap && injectionMapping.infoStringMap[languageName]) {
445
+ targetLanguage = injectionMapping.infoStringMap[languageName]
446
+ } else {
447
+ targetLanguage = languageName
448
+ }
449
+ }
450
+ }
451
+ }
452
+ }
453
+
454
+ if (targetLanguage) {
455
+ if (!languageGroups.has(targetLanguage)) {
456
+ languageGroups.set(targetLanguage, [])
457
+ }
458
+ languageGroups.get(targetLanguage)!.push({ node: capture.node, name: capture.name })
459
+ }
460
+ }
461
+ }
462
+
463
+ // Process each language group
464
+ for (const [language, captures] of languageGroups.entries()) {
465
+ const injectedParser = await this.getReusableParser(language)
466
+
467
+ if (!injectedParser) {
468
+ console.warn(`No parser found for injection language: ${language}`)
469
+ continue
470
+ }
471
+
472
+ // Track injection ranges for this language
473
+ if (!injectionRanges.has(language)) {
474
+ injectionRanges.set(language, [])
475
+ }
476
+
477
+ const parser = injectedParser.parser
478
+ for (const { node: injectionNode } of captures) {
479
+ try {
480
+ // Record the injection range
481
+ injectionRanges.get(language)!.push({
482
+ start: injectionNode.startIndex,
483
+ end: injectionNode.endIndex,
484
+ })
485
+
486
+ const injectionContent = this.getNodeText(injectionNode, content)
487
+ const tree = parser.parse(injectionContent)
488
+
489
+ if (tree) {
490
+ const matches = injectedParser.queries.highlights.captures(tree.rootNode)
491
+
492
+ // Create new QueryCapture objects with offset positions
493
+ for (const match of matches) {
494
+ // Calculate offset positions by creating a new capture with adjusted node properties
495
+ // Store the injected query reference so we can look up properties correctly
496
+ const offsetCapture: QueryCapture & { _injectedQuery?: Query } = {
497
+ name: match.name,
498
+ patternIndex: match.patternIndex,
499
+ _injectedQuery: injectedParser.queries.highlights, // Store the correct query reference
500
+ node: {
501
+ ...match.node,
502
+ startPosition: {
503
+ row: match.node.startPosition.row + injectionNode.startPosition.row,
504
+ column:
505
+ match.node.startPosition.row === 0
506
+ ? match.node.startPosition.column + injectionNode.startPosition.column
507
+ : match.node.startPosition.column,
508
+ },
509
+ endPosition: {
510
+ row: match.node.endPosition.row + injectionNode.startPosition.row,
511
+ column:
512
+ match.node.endPosition.row === 0
513
+ ? match.node.endPosition.column + injectionNode.startPosition.column
514
+ : match.node.endPosition.column,
515
+ },
516
+ startIndex: match.node.startIndex + injectionNode.startIndex,
517
+ endIndex: match.node.endIndex + injectionNode.startIndex,
518
+ } as any, // Cast to any since we're creating a pseudo-node
519
+ }
520
+
521
+ injectionMatches.push(offsetCapture)
522
+ }
523
+
524
+ tree.delete()
525
+ }
526
+ } catch (error) {
527
+ console.error(`Error processing injection for language ${language}:`, error)
528
+ }
529
+ }
530
+
531
+ // NOTE: Do NOT call parser.delete() here - this is a reusable parser!
532
+ }
533
+
534
+ return { captures: injectionMatches, injectionRanges }
535
+ }
536
+
537
+ private editToRange(edit: Edit): Range {
538
+ return {
539
+ startPosition: {
540
+ column: edit.startPosition.column,
541
+ row: edit.startPosition.row,
542
+ },
543
+ endPosition: {
544
+ column: edit.newEndPosition.column,
545
+ row: edit.newEndPosition.row,
546
+ },
547
+ startIndex: edit.startIndex,
548
+ endIndex: edit.newEndIndex,
549
+ }
550
+ }
551
+
552
+ async handleEdits(
553
+ bufferId: number,
554
+ content: string,
555
+ edits: Edit[],
556
+ ): Promise<{ highlights?: HighlightResponse[]; warning?: string; error?: string }> {
557
+ const parserState = this.bufferParsers.get(bufferId)
558
+ if (!parserState) {
559
+ return { warning: "No parser state found for buffer" }
560
+ }
561
+
562
+ parserState.content = content
563
+
564
+ for (const edit of edits) {
565
+ parserState.tree.edit(edit)
566
+ }
567
+
568
+ const startParse = performance.now()
569
+
570
+ const newTree = parserState.parser.parse(content, parserState.tree)
571
+
572
+ const endParse = performance.now()
573
+ const parseTime = endParse - startParse
574
+ this.performance.parseTimes.push(parseTime)
575
+ if (this.performance.parseTimes.length > 10) {
576
+ this.performance.parseTimes.shift()
577
+ }
578
+ this.performance.averageParseTime =
579
+ this.performance.parseTimes.reduce((acc, time) => acc + time, 0) / this.performance.parseTimes.length
580
+
581
+ if (!newTree) {
582
+ return { error: "Failed to parse buffer" }
583
+ }
584
+
585
+ const changedRanges = parserState.tree.getChangedRanges(newTree)
586
+ parserState.tree = newTree
587
+
588
+ const startQuery = performance.now()
589
+ const matches: QueryCapture[] = []
590
+
591
+ if (changedRanges.length === 0) {
592
+ edits.forEach((edit) => {
593
+ const range = this.editToRange(edit)
594
+ changedRanges.push(range)
595
+ })
596
+ }
597
+
598
+ for (const range of changedRanges) {
599
+ let node = parserState.tree.rootNode.descendantForPosition(range.startPosition, range.endPosition)
600
+
601
+ if (!node) {
602
+ continue
603
+ }
604
+
605
+ // If we got the root node, query with range to limit scope
606
+ if (node.equals(parserState.tree.rootNode)) {
607
+ // WHY ARE RANGES NOT WORKING!?
608
+ // The changed ranges are not returning anything in some cases
609
+ // Even this shit somehow returns many lines before the actual range,
610
+ // and even though expanded by 1000 bytes it does not capture much beyond the actual range.
611
+ // So freaking weird.
612
+ const rangeCaptures = parserState.queries.highlights.captures(
613
+ node,
614
+ // WTF!?
615
+ {
616
+ startIndex: range.startIndex - 100,
617
+ endIndex: range.endIndex + 1000,
618
+ },
619
+ )
620
+ matches.push(...rangeCaptures)
621
+ continue
622
+ }
623
+
624
+ while (node && !this.nodeContainsRange(node, range)) {
625
+ node = node.parent
626
+ }
627
+
628
+ if (!node) {
629
+ node = parserState.tree.rootNode
630
+ }
631
+
632
+ const nodeCaptures = parserState.queries.highlights.captures(node)
633
+ matches.push(...nodeCaptures)
634
+ }
635
+
636
+ let injectionRanges = new Map<string, Array<{ start: number; end: number }>>()
637
+ if (parserState.queries.injections) {
638
+ const injectionResult = await this.processInjections(parserState)
639
+ // Only add injection matches that are in the changed ranges
640
+ // This is a simplification - ideally we'd only process injections in changed ranges
641
+ matches.push(...injectionResult.captures)
642
+ injectionRanges = injectionResult.injectionRanges
643
+ }
644
+
645
+ const endQuery = performance.now()
646
+ const queryTime = endQuery - startQuery
647
+ this.performance.queryTimes.push(queryTime)
648
+ if (this.performance.queryTimes.length > 10) {
649
+ this.performance.queryTimes.shift()
650
+ }
651
+ this.performance.averageQueryTime =
652
+ this.performance.queryTimes.reduce((acc, time) => acc + time, 0) / this.performance.queryTimes.length
653
+
654
+ return this.getHighlights(parserState, matches, injectionRanges)
655
+ }
656
+
657
+ private nodeContainsRange(node: any, range: any): boolean {
658
+ return (
659
+ node.startPosition.row <= range.startPosition.row &&
660
+ node.endPosition.row >= range.endPosition.row &&
661
+ (node.startPosition.row < range.startPosition.row || node.startPosition.column <= range.startPosition.column) &&
662
+ (node.endPosition.row > range.endPosition.row || node.endPosition.column >= range.endPosition.column)
663
+ )
664
+ }
665
+
666
+ private getHighlights(
667
+ parserState: ParserState,
668
+ matches: QueryCapture[],
669
+ injectionRanges?: Map<string, Array<{ start: number; end: number }>>,
670
+ ): { highlights: HighlightResponse[] } {
671
+ const lineHighlights: Map<number, Map<number, HighlightRange>> = new Map()
672
+ const droppedHighlights: Map<number, Map<number, HighlightRange>> = new Map()
673
+
674
+ for (const match of matches) {
675
+ const node = match.node
676
+ const startLine = node.startPosition.row
677
+ const endLine = node.endPosition.row
678
+
679
+ const highlight = {
680
+ startCol: node.startPosition.column,
681
+ endCol: node.endPosition.column,
682
+ group: match.name,
683
+ }
684
+
685
+ if (!lineHighlights.has(startLine)) {
686
+ lineHighlights.set(startLine, new Map())
687
+ droppedHighlights.set(startLine, new Map())
688
+ }
689
+ if (lineHighlights.get(startLine)?.has(node.id)) {
690
+ droppedHighlights.get(startLine)?.set(node.id, lineHighlights.get(startLine)?.get(node.id)!)
691
+ }
692
+ lineHighlights.get(startLine)?.set(node.id, highlight)
693
+
694
+ if (startLine !== endLine) {
695
+ for (let line = startLine + 1; line <= endLine; line++) {
696
+ if (!lineHighlights.has(line)) {
697
+ lineHighlights.set(line, new Map())
698
+ }
699
+ const hl: HighlightRange = {
700
+ startCol: 0,
701
+ endCol: node.endPosition.column,
702
+ group: match.name,
703
+ }
704
+ lineHighlights.get(line)?.set(node.id, hl)
705
+ }
706
+ }
707
+ }
708
+
709
+ return {
710
+ highlights: Array.from(lineHighlights.entries()).map(([line, lineHighlights]) => ({
711
+ line,
712
+ highlights: Array.from(lineHighlights.values()),
713
+ droppedHighlights: droppedHighlights.get(line) ? Array.from(droppedHighlights.get(line)!.values()) : [],
714
+ })),
715
+ }
716
+ }
717
+
718
+ private getSimpleHighlights(
719
+ matches: QueryCapture[],
720
+ injectionRanges: Map<string, Array<{ start: number; end: number }>>,
721
+ ): SimpleHighlight[] {
722
+ const highlights: SimpleHighlight[] = []
723
+
724
+ const flatInjectionRanges: Array<{ start: number; end: number; lang: string }> = []
725
+ for (const [lang, ranges] of injectionRanges.entries()) {
726
+ for (const range of ranges) {
727
+ flatInjectionRanges.push({ ...range, lang })
728
+ }
729
+ }
730
+
731
+ for (const match of matches) {
732
+ const node = match.node
733
+
734
+ let isInjection = false
735
+ let injectionLang: string | undefined
736
+ let containsInjection = false
737
+ for (const injRange of flatInjectionRanges) {
738
+ if (node.startIndex >= injRange.start && node.endIndex <= injRange.end) {
739
+ isInjection = true
740
+ injectionLang = injRange.lang
741
+ break
742
+ } else if (node.startIndex <= injRange.start && node.endIndex >= injRange.end) {
743
+ containsInjection = true
744
+ break
745
+ }
746
+ }
747
+
748
+ const matchQuery = (match as any)._injectedQuery
749
+ const patternProperties = matchQuery?.setProperties?.[match.patternIndex]
750
+
751
+ const concealValue = patternProperties?.conceal ?? match.setProperties?.conceal
752
+ const concealLines = patternProperties?.conceal_lines ?? match.setProperties?.conceal_lines
753
+
754
+ const meta: any = {}
755
+ if (isInjection && injectionLang) {
756
+ meta.isInjection = true
757
+ meta.injectionLang = injectionLang
758
+ }
759
+ if (containsInjection) {
760
+ meta.containsInjection = true
761
+ }
762
+ if (concealValue !== undefined) {
763
+ meta.conceal = concealValue
764
+ }
765
+ if (concealLines !== undefined) {
766
+ meta.concealLines = concealLines
767
+ }
768
+
769
+ if (Object.keys(meta).length > 0) {
770
+ highlights.push([node.startIndex, node.endIndex, match.name, meta])
771
+ } else {
772
+ highlights.push([node.startIndex, node.endIndex, match.name])
773
+ }
774
+ }
775
+
776
+ highlights.sort((a, b) => a[0] - b[0])
777
+
778
+ return highlights
779
+ }
780
+
781
+ async handleResetBuffer(
782
+ bufferId: number,
783
+ version: number,
784
+ content: string,
785
+ ): Promise<{ highlights?: HighlightResponse[]; warning?: string; error?: string }> {
786
+ const parserState = this.bufferParsers.get(bufferId)
787
+ if (!parserState) {
788
+ return { warning: "No parser state found for buffer" }
789
+ }
790
+
791
+ parserState.content = content
792
+
793
+ const newTree = parserState.parser.parse(content)
794
+
795
+ if (!newTree) {
796
+ return { error: "Failed to parse buffer during reset" }
797
+ }
798
+
799
+ parserState.tree = newTree
800
+ const matches = parserState.queries.highlights.captures(parserState.tree.rootNode)
801
+
802
+ let injectionRanges = new Map<string, Array<{ start: number; end: number }>>()
803
+ if (parserState.queries.injections) {
804
+ const injectionResult = await this.processInjections(parserState)
805
+ matches.push(...injectionResult.captures)
806
+ injectionRanges = injectionResult.injectionRanges
807
+ }
808
+
809
+ return this.getHighlights(parserState, matches, injectionRanges)
810
+ }
811
+
812
+ disposeBuffer(bufferId: number): void {
813
+ const parserState = this.bufferParsers.get(bufferId)
814
+ if (!parserState) {
815
+ return
816
+ }
817
+
818
+ parserState.tree.delete()
819
+ parserState.parser.delete()
820
+
821
+ this.bufferParsers.delete(bufferId)
822
+ }
823
+
824
+ async handleOneShotHighlight(content: string, filetype: string, messageId: string): Promise<void> {
825
+ const reusableState = await this.getReusableParser(filetype)
826
+
827
+ if (!reusableState) {
828
+ self.postMessage({
829
+ type: "ONESHOT_HIGHLIGHT_RESPONSE",
830
+ messageId,
831
+ hasParser: false,
832
+ warning: `No parser available for filetype ${filetype}`,
833
+ })
834
+ return
835
+ }
836
+
837
+ // Markdown Parser BUG: For markdown, ensure content ends with newline so closing delimiters are parsed correctly
838
+ // The tree-sitter markdown parser only creates closing delimiter nodes when followed by newline
839
+ const parseContent = filetype === "markdown" && content.endsWith("```") ? content + "\n" : content
840
+
841
+ const tree = reusableState.parser.parse(parseContent)
842
+
843
+ if (!tree) {
844
+ self.postMessage({
845
+ type: "ONESHOT_HIGHLIGHT_RESPONSE",
846
+ messageId,
847
+ hasParser: false,
848
+ error: "Failed to parse content",
849
+ })
850
+ return
851
+ }
852
+
853
+ try {
854
+ const matches = reusableState.filetypeParser.queries.highlights.captures(tree.rootNode)
855
+
856
+ let injectionRanges = new Map<string, Array<{ start: number; end: number }>>()
857
+ if (reusableState.filetypeParser.queries.injections) {
858
+ const parserState: ParserState = {
859
+ parser: reusableState.parser,
860
+ tree,
861
+ queries: reusableState.filetypeParser.queries,
862
+ filetype,
863
+ content,
864
+ injectionMapping: reusableState.filetypeParser.injectionMapping,
865
+ }
866
+ const injectionResult = await this.processInjections(parserState)
867
+
868
+ matches.push(...injectionResult.captures)
869
+ injectionRanges = injectionResult.injectionRanges
870
+ }
871
+
872
+ const highlights = this.getSimpleHighlights(matches, injectionRanges)
873
+
874
+ self.postMessage({
875
+ type: "ONESHOT_HIGHLIGHT_RESPONSE",
876
+ messageId,
877
+ hasParser: true,
878
+ highlights,
879
+ })
880
+ } finally {
881
+ tree.delete()
882
+ }
883
+ }
884
+
885
+ async updateDataPath(dataPath: string): Promise<void> {
886
+ this.dataPath = dataPath
887
+ this.tsDataPath = path.join(dataPath, "tree-sitter")
888
+
889
+ try {
890
+ await mkdir(path.join(this.tsDataPath, "languages"), { recursive: true })
891
+ await mkdir(path.join(this.tsDataPath, "queries"), { recursive: true })
892
+ } catch (error) {
893
+ throw new Error(`Failed to update data path: ${error}`)
894
+ }
895
+ }
896
+
897
+ async clearCache(): Promise<void> {
898
+ if (!this.dataPath || !this.tsDataPath) {
899
+ throw new Error("No data path configured")
900
+ }
901
+
902
+ const { rm } = await import("fs/promises")
903
+
904
+ try {
905
+ const treeSitterPath = path.join(this.dataPath, "tree-sitter")
906
+
907
+ await rm(treeSitterPath, { recursive: true, force: true })
908
+
909
+ await mkdir(path.join(treeSitterPath, "languages"), { recursive: true })
910
+ await mkdir(path.join(treeSitterPath, "queries"), { recursive: true })
911
+
912
+ this.filetypeParsers.clear()
913
+ this.filetypeParserPromises.clear()
914
+ this.reusableParsers.clear()
915
+ this.reusableParserPromises.clear()
916
+ } catch (error) {
917
+ throw new Error(`Failed to clear cache: ${error}`)
918
+ }
919
+ }
920
+ }
921
+ if (!isMainThread) {
922
+ const worker = new ParserWorker()
923
+
924
+ function logMessage(type: "log" | "error" | "warn", ...args: any[]) {
925
+ self.postMessage({
926
+ type: "WORKER_LOG",
927
+ logType: type,
928
+ data: args,
929
+ })
930
+ }
931
+ console.log = (...args) => logMessage("log", ...args)
932
+ console.error = (...args) => logMessage("error", ...args)
933
+ console.warn = (...args) => logMessage("warn", ...args)
934
+
935
+ // @ts-ignore - we'll fix this in the future for sure
936
+ self.onmessage = async (e: MessageEvent) => {
937
+ const { type, bufferId, version, content, filetype, edits, filetypeParser, messageId, dataPath } = e.data
938
+
939
+ try {
940
+ switch (type) {
941
+ case "INIT":
942
+ try {
943
+ await worker.initialize({ dataPath })
944
+ self.postMessage({ type: "INIT_RESPONSE" })
945
+ } catch (error) {
946
+ self.postMessage({
947
+ type: "INIT_RESPONSE",
948
+ error: error instanceof Error ? error.stack || error.message : String(error),
949
+ })
950
+ }
951
+ break
952
+
953
+ case "ADD_FILETYPE_PARSER":
954
+ worker.addFiletypeParser(filetypeParser)
955
+ break
956
+
957
+ case "PRELOAD_PARSER":
958
+ const maybeParser = await worker.preloadParser(filetype)
959
+ self.postMessage({ type: "PRELOAD_PARSER_RESPONSE", messageId, hasParser: !!maybeParser })
960
+ break
961
+
962
+ case "INITIALIZE_PARSER":
963
+ await worker.handleInitializeParser(bufferId, version, content, filetype, messageId)
964
+ break
965
+
966
+ case "HANDLE_EDITS":
967
+ const response = await worker.handleEdits(bufferId, content, edits)
968
+ if (response.highlights && response.highlights.length > 0) {
969
+ self.postMessage({ type: "HIGHLIGHT_RESPONSE", bufferId, version, ...response })
970
+ } else if (response.warning) {
971
+ self.postMessage({ type: "WARNING", bufferId, warning: response.warning })
972
+ } else if (response.error) {
973
+ self.postMessage({ type: "ERROR", bufferId, error: response.error })
974
+ }
975
+ break
976
+
977
+ case "GET_PERFORMANCE":
978
+ self.postMessage({ type: "PERFORMANCE_RESPONSE", performance: worker.performance, messageId })
979
+ break
980
+
981
+ case "RESET_BUFFER":
982
+ const resetResponse = await worker.handleResetBuffer(bufferId, version, content)
983
+ if (resetResponse.highlights && resetResponse.highlights.length > 0) {
984
+ self.postMessage({ type: "HIGHLIGHT_RESPONSE", bufferId, version, ...resetResponse })
985
+ } else if (resetResponse.warning) {
986
+ self.postMessage({ type: "WARNING", bufferId, warning: resetResponse.warning })
987
+ } else if (resetResponse.error) {
988
+ self.postMessage({ type: "ERROR", bufferId, error: resetResponse.error })
989
+ }
990
+ break
991
+
992
+ case "DISPOSE_BUFFER":
993
+ worker.disposeBuffer(bufferId)
994
+ self.postMessage({ type: "BUFFER_DISPOSED", bufferId })
995
+ break
996
+
997
+ case "ONESHOT_HIGHLIGHT":
998
+ await worker.handleOneShotHighlight(content, filetype, messageId)
999
+ break
1000
+
1001
+ case "UPDATE_DATA_PATH":
1002
+ try {
1003
+ await worker.updateDataPath(dataPath)
1004
+ self.postMessage({ type: "UPDATE_DATA_PATH_RESPONSE", messageId })
1005
+ } catch (error) {
1006
+ self.postMessage({
1007
+ type: "UPDATE_DATA_PATH_RESPONSE",
1008
+ messageId,
1009
+ error: error instanceof Error ? error.message : String(error),
1010
+ })
1011
+ }
1012
+ break
1013
+
1014
+ case "CLEAR_CACHE":
1015
+ try {
1016
+ await worker.clearCache()
1017
+ self.postMessage({ type: "CLEAR_CACHE_RESPONSE", messageId })
1018
+ } catch (error) {
1019
+ self.postMessage({
1020
+ type: "CLEAR_CACHE_RESPONSE",
1021
+ messageId,
1022
+ error: error instanceof Error ? error.message : String(error),
1023
+ })
1024
+ }
1025
+ break
1026
+
1027
+ default:
1028
+ self.postMessage({
1029
+ type: "ERROR",
1030
+ bufferId,
1031
+ error: `Unknown message type: ${type}`,
1032
+ })
1033
+ }
1034
+ } catch (error) {
1035
+ self.postMessage({
1036
+ type: "ERROR",
1037
+ bufferId,
1038
+ error: error instanceof Error ? error.stack || error.message : String(error),
1039
+ })
1040
+ }
1041
+ }
1042
+ }