@fairyhunter13/opentui-core 0.1.113 → 0.1.114

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/dev/keypress-debug-renderer.ts +148 -0
  2. package/dev/keypress-debug.ts +43 -0
  3. package/dev/print-env-vars.ts +32 -0
  4. package/dev/test-tmux-graphics-334.sh +68 -0
  5. package/dev/thai-debug-test.ts +68 -0
  6. package/docs/development.md +144 -0
  7. package/package.json +62 -53
  8. package/scripts/build.ts +400 -0
  9. package/scripts/publish.ts +60 -0
  10. package/src/3d/SpriteResourceManager.ts +286 -0
  11. package/src/3d/SpriteUtils.ts +70 -0
  12. package/src/3d/TextureUtils.ts +196 -0
  13. package/src/3d/ThreeRenderable.ts +197 -0
  14. package/src/3d/WGPURenderer.ts +294 -0
  15. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  16. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  17. package/src/3d/animation/SpriteAnimator.ts +633 -0
  18. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  19. package/src/3d/canvas.ts +464 -0
  20. package/src/3d/index.ts +12 -0
  21. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  22. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  23. package/src/3d/physics/physics-interface.ts +31 -0
  24. package/src/3d/shaders/supersampling.wgsl +201 -0
  25. package/src/3d.ts +3 -0
  26. package/src/NativeSpanFeed.ts +300 -0
  27. package/src/Renderable.ts +1704 -0
  28. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  29. package/src/animation/Timeline.test.ts +2709 -0
  30. package/src/animation/Timeline.ts +598 -0
  31. package/src/ansi.ts +18 -0
  32. package/src/benchmark/attenuation-benchmark.ts +81 -0
  33. package/src/benchmark/colormatrix-benchmark.ts +128 -0
  34. package/src/benchmark/gain-benchmark.ts +80 -0
  35. package/src/benchmark/latest-all-bench-run.json +707 -0
  36. package/src/benchmark/latest-async-bench-run.json +336 -0
  37. package/src/benchmark/latest-default-bench-run.json +657 -0
  38. package/src/benchmark/latest-large-bench-run.json +707 -0
  39. package/src/benchmark/latest-quick-bench-run.json +207 -0
  40. package/src/benchmark/markdown-benchmark.ts +1796 -0
  41. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  42. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  43. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  44. package/src/benchmark/native-span-feed-compare.ts +280 -0
  45. package/src/benchmark/renderer-benchmark.ts +754 -0
  46. package/src/benchmark/text-table-benchmark.ts +948 -0
  47. package/src/buffer.test.ts +291 -0
  48. package/src/buffer.ts +554 -0
  49. package/src/console.test.ts +612 -0
  50. package/src/console.ts +1254 -0
  51. package/src/edit-buffer.test.ts +1769 -0
  52. package/src/edit-buffer.ts +411 -0
  53. package/src/editor-view.test.ts +1032 -0
  54. package/src/editor-view.ts +284 -0
  55. package/src/examples/ascii-font-selection-demo.ts +245 -0
  56. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  57. package/src/examples/assets/concrete.png +0 -0
  58. package/src/examples/assets/crate.png +0 -0
  59. package/src/examples/assets/crate_emissive.png +0 -0
  60. package/src/examples/assets/forrest_background.png +0 -0
  61. package/src/examples/assets/hast-example.json +1018 -0
  62. package/src/examples/assets/heart.png +0 -0
  63. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  64. package/src/examples/assets/main_char_idle.png +0 -0
  65. package/src/examples/assets/main_char_jump_end.png +0 -0
  66. package/src/examples/assets/main_char_jump_landing.png +0 -0
  67. package/src/examples/assets/main_char_jump_start.png +0 -0
  68. package/src/examples/assets/main_char_run_loop.png +0 -0
  69. package/src/examples/assets/roughness_map.jpg +0 -0
  70. package/src/examples/build.ts +115 -0
  71. package/src/examples/code-demo.ts +924 -0
  72. package/src/examples/console-demo.ts +358 -0
  73. package/src/examples/core-plugin-slots-demo.ts +759 -0
  74. package/src/examples/diff-demo.ts +701 -0
  75. package/src/examples/draggable-three-demo.ts +259 -0
  76. package/src/examples/editor-demo.ts +322 -0
  77. package/src/examples/extmarks-demo.ts +196 -0
  78. package/src/examples/focus-restore-demo.ts +310 -0
  79. package/src/examples/fonts.ts +245 -0
  80. package/src/examples/fractal-shader-demo.ts +268 -0
  81. package/src/examples/framebuffer-demo.ts +674 -0
  82. package/src/examples/full-unicode-demo.ts +241 -0
  83. package/src/examples/golden-star-demo.ts +933 -0
  84. package/src/examples/grayscale-buffer-demo.ts +249 -0
  85. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  86. package/src/examples/index.ts +926 -0
  87. package/src/examples/input-demo.ts +377 -0
  88. package/src/examples/input-select-layout-demo.ts +425 -0
  89. package/src/examples/install.sh +143 -0
  90. package/src/examples/keypress-debug-demo.ts +452 -0
  91. package/src/examples/lib/HexList.ts +122 -0
  92. package/src/examples/lib/PaletteGrid.ts +125 -0
  93. package/src/examples/lib/standalone-keys.ts +25 -0
  94. package/src/examples/lib/tab-controller.ts +243 -0
  95. package/src/examples/lights-phong-demo.ts +290 -0
  96. package/src/examples/link-demo.ts +220 -0
  97. package/src/examples/live-state-demo.ts +480 -0
  98. package/src/examples/markdown-demo.ts +725 -0
  99. package/src/examples/mouse-interaction-demo.ts +428 -0
  100. package/src/examples/nested-zindex-demo.ts +357 -0
  101. package/src/examples/opacity-example.ts +235 -0
  102. package/src/examples/opentui-demo.ts +1057 -0
  103. package/src/examples/physx-planck-2d-demo.ts +623 -0
  104. package/src/examples/physx-rapier-2d-demo.ts +655 -0
  105. package/src/examples/relative-positioning-demo.ts +323 -0
  106. package/src/examples/scroll-example.ts +214 -0
  107. package/src/examples/scrollbox-mouse-test.ts +112 -0
  108. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  109. package/src/examples/select-demo.ts +237 -0
  110. package/src/examples/shader-cube-demo.ts +1015 -0
  111. package/src/examples/simple-layout-example.ts +591 -0
  112. package/src/examples/slider-demo.ts +617 -0
  113. package/src/examples/split-mode-demo.ts +453 -0
  114. package/src/examples/sprite-animation-demo.ts +443 -0
  115. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  116. package/src/examples/static-sprite-demo.ts +193 -0
  117. package/src/examples/sticky-scroll-example.ts +308 -0
  118. package/src/examples/styled-text-demo.ts +282 -0
  119. package/src/examples/tab-select-demo.ts +219 -0
  120. package/src/examples/terminal-title.ts +29 -0
  121. package/src/examples/terminal.ts +305 -0
  122. package/src/examples/text-node-demo.ts +416 -0
  123. package/src/examples/text-selection-demo.ts +377 -0
  124. package/src/examples/text-table-demo.ts +503 -0
  125. package/src/examples/text-truncation-demo.ts +481 -0
  126. package/src/examples/text-wrap.ts +757 -0
  127. package/src/examples/texture-loading-demo.ts +259 -0
  128. package/src/examples/timeline-example.ts +670 -0
  129. package/src/examples/transparency-demo.ts +400 -0
  130. package/src/examples/vnode-composition-demo.ts +404 -0
  131. package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
  132. package/src/index.ts +24 -0
  133. package/src/lib/KeyHandler.integration.test.ts +292 -0
  134. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  135. package/src/lib/KeyHandler.test.ts +662 -0
  136. package/src/lib/KeyHandler.ts +222 -0
  137. package/src/lib/RGBA.test.ts +984 -0
  138. package/src/lib/RGBA.ts +204 -0
  139. package/src/lib/ascii.font.ts +330 -0
  140. package/src/lib/border.test.ts +83 -0
  141. package/src/lib/border.ts +170 -0
  142. package/src/lib/bunfs.test.ts +27 -0
  143. package/src/lib/bunfs.ts +18 -0
  144. package/src/lib/clipboard.test.ts +41 -0
  145. package/src/lib/clipboard.ts +47 -0
  146. package/src/lib/clock.ts +35 -0
  147. package/src/lib/data-paths.test.ts +133 -0
  148. package/src/lib/data-paths.ts +109 -0
  149. package/src/lib/debounce.ts +106 -0
  150. package/src/lib/detect-links.test.ts +98 -0
  151. package/src/lib/detect-links.ts +56 -0
  152. package/src/lib/env.test.ts +228 -0
  153. package/src/lib/env.ts +209 -0
  154. package/src/lib/extmarks-history.ts +51 -0
  155. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  156. package/src/lib/extmarks.test.ts +3457 -0
  157. package/src/lib/extmarks.ts +843 -0
  158. package/src/lib/fonts/block.json +405 -0
  159. package/src/lib/fonts/grid.json +265 -0
  160. package/src/lib/fonts/huge.json +741 -0
  161. package/src/lib/fonts/pallet.json +314 -0
  162. package/src/lib/fonts/shade.json +591 -0
  163. package/src/lib/fonts/slick.json +321 -0
  164. package/src/lib/fonts/tiny.json +69 -0
  165. package/src/lib/hast-styled-text.ts +59 -0
  166. package/src/lib/index.ts +21 -0
  167. package/src/lib/keymapping.test.ts +317 -0
  168. package/src/lib/keymapping.ts +115 -0
  169. package/src/lib/objects-in-viewport.test.ts +787 -0
  170. package/src/lib/objects-in-viewport.ts +153 -0
  171. package/src/lib/output.capture.ts +58 -0
  172. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  173. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  174. package/src/lib/parse.keypress-kitty.ts +439 -0
  175. package/src/lib/parse.keypress.test.ts +1849 -0
  176. package/src/lib/parse.keypress.ts +397 -0
  177. package/src/lib/parse.mouse.test.ts +552 -0
  178. package/src/lib/parse.mouse.ts +232 -0
  179. package/src/lib/paste.ts +16 -0
  180. package/src/lib/queue.ts +65 -0
  181. package/src/lib/renderable.validations.test.ts +87 -0
  182. package/src/lib/renderable.validations.ts +83 -0
  183. package/src/lib/scroll-acceleration.ts +98 -0
  184. package/src/lib/selection.ts +240 -0
  185. package/src/lib/singleton.ts +28 -0
  186. package/src/lib/stdin-parser.test.ts +2290 -0
  187. package/src/lib/stdin-parser.ts +1810 -0
  188. package/src/lib/styled-text.ts +178 -0
  189. package/src/lib/terminal-capability-detection.test.ts +202 -0
  190. package/src/lib/terminal-capability-detection.ts +79 -0
  191. package/src/lib/terminal-palette.test.ts +878 -0
  192. package/src/lib/terminal-palette.ts +383 -0
  193. package/src/lib/tree-sitter/assets/README.md +118 -0
  194. package/src/lib/tree-sitter/assets/update.ts +334 -0
  195. package/src/lib/tree-sitter/assets.d.ts +9 -0
  196. package/src/lib/tree-sitter/cache.test.ts +273 -0
  197. package/src/lib/tree-sitter/client.test.ts +1165 -0
  198. package/src/lib/tree-sitter/client.ts +607 -0
  199. package/src/lib/tree-sitter/default-parsers.ts +86 -0
  200. package/src/lib/tree-sitter/download-utils.ts +148 -0
  201. package/src/lib/tree-sitter/index.ts +28 -0
  202. package/src/lib/tree-sitter/parser.worker.ts +1042 -0
  203. package/src/lib/tree-sitter/parsers-config.ts +81 -0
  204. package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +189 -0
  206. package/src/lib/tree-sitter/types.ts +82 -0
  207. package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
  208. package/src/lib/tree-sitter-styled-text.ts +306 -0
  209. package/src/lib/validate-dir-name.ts +55 -0
  210. package/src/lib/yoga.options.test.ts +628 -0
  211. package/src/lib/yoga.options.ts +346 -0
  212. package/src/plugins/core-slot.ts +579 -0
  213. package/src/plugins/registry.ts +402 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/effects.ts +930 -0
  216. package/src/post/filters.ts +489 -0
  217. package/src/post/matrices.ts +288 -0
  218. package/src/renderables/ASCIIFont.ts +219 -0
  219. package/src/renderables/Box.test.ts +205 -0
  220. package/src/renderables/Box.ts +326 -0
  221. package/src/renderables/Code.test.ts +2062 -0
  222. package/src/renderables/Code.ts +357 -0
  223. package/src/renderables/Diff.regression.test.ts +226 -0
  224. package/src/renderables/Diff.test.ts +3101 -0
  225. package/src/renderables/Diff.ts +1211 -0
  226. package/src/renderables/EditBufferRenderable.test.ts +288 -0
  227. package/src/renderables/EditBufferRenderable.ts +1166 -0
  228. package/src/renderables/FrameBuffer.ts +47 -0
  229. package/src/renderables/Input.test.ts +1228 -0
  230. package/src/renderables/Input.ts +247 -0
  231. package/src/renderables/LineNumberRenderable.ts +724 -0
  232. package/src/renderables/Markdown.ts +1393 -0
  233. package/src/renderables/ScrollBar.ts +422 -0
  234. package/src/renderables/ScrollBox.ts +883 -0
  235. package/src/renderables/Select.test.ts +1033 -0
  236. package/src/renderables/Select.ts +524 -0
  237. package/src/renderables/Slider.test.ts +456 -0
  238. package/src/renderables/Slider.ts +342 -0
  239. package/src/renderables/TabSelect.test.ts +197 -0
  240. package/src/renderables/TabSelect.ts +455 -0
  241. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  242. package/src/renderables/Text.test.ts +2660 -0
  243. package/src/renderables/Text.ts +147 -0
  244. package/src/renderables/TextBufferRenderable.ts +518 -0
  245. package/src/renderables/TextNode.test.ts +1058 -0
  246. package/src/renderables/TextNode.ts +325 -0
  247. package/src/renderables/TextTable.test.ts +1421 -0
  248. package/src/renderables/TextTable.ts +1344 -0
  249. package/src/renderables/Textarea.ts +430 -0
  250. package/src/renderables/TimeToFirstDraw.ts +89 -0
  251. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  252. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  253. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  254. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  256. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  257. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
  258. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  259. package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
  260. package/src/renderables/__tests__/Markdown.test.ts +2518 -0
  261. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  262. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  263. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  264. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  265. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  266. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  267. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  268. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  269. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  270. package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
  271. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  272. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  273. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  274. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  275. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  276. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  277. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  278. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  279. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  280. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  281. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  282. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  283. package/src/renderables/composition/README.md +8 -0
  284. package/src/renderables/composition/VRenderable.ts +32 -0
  285. package/src/renderables/composition/constructs.ts +127 -0
  286. package/src/renderables/composition/vnode.ts +289 -0
  287. package/src/renderables/index.ts +23 -0
  288. package/src/renderables/markdown-parser.ts +66 -0
  289. package/src/renderer.ts +2681 -0
  290. package/src/runtime-plugin-support.ts +39 -0
  291. package/src/runtime-plugin.ts +615 -0
  292. package/src/syntax-style.test.ts +841 -0
  293. package/src/syntax-style.ts +257 -0
  294. package/src/testing/README.md +210 -0
  295. package/src/testing/capture-spans.test.ts +194 -0
  296. package/src/testing/integration.test.ts +276 -0
  297. package/src/testing/manual-clock.ts +117 -0
  298. package/src/testing/mock-keys.test.ts +1378 -0
  299. package/src/testing/mock-keys.ts +457 -0
  300. package/src/testing/mock-mouse.test.ts +218 -0
  301. package/src/testing/mock-mouse.ts +247 -0
  302. package/src/testing/mock-tree-sitter-client.ts +73 -0
  303. package/src/testing/spy.ts +13 -0
  304. package/src/testing/test-recorder.test.ts +415 -0
  305. package/src/testing/test-recorder.ts +145 -0
  306. package/src/testing/test-renderer.ts +132 -0
  307. package/src/testing.ts +7 -0
  308. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  309. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  310. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  311. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  312. package/src/tests/allocator-stats.test.ts +38 -0
  313. package/src/tests/destroy-during-render.test.ts +200 -0
  314. package/src/tests/destroy-on-exit.fixture.ts +36 -0
  315. package/src/tests/destroy-on-exit.test.ts +41 -0
  316. package/src/tests/hover-cursor.test.ts +98 -0
  317. package/src/tests/native-span-feed-async.test.ts +173 -0
  318. package/src/tests/native-span-feed-close.test.ts +120 -0
  319. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  320. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  321. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  322. package/src/tests/opacity.test.ts +123 -0
  323. package/src/tests/renderable.snapshot.test.ts +524 -0
  324. package/src/tests/renderable.test.ts +1281 -0
  325. package/src/tests/renderer.clock.test.ts +158 -0
  326. package/src/tests/renderer.console-startup.test.ts +185 -0
  327. package/src/tests/renderer.control.test.ts +425 -0
  328. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  329. package/src/tests/renderer.cursor.test.ts +26 -0
  330. package/src/tests/renderer.destroy-during-render.test.ts +147 -0
  331. package/src/tests/renderer.focus-restore.test.ts +257 -0
  332. package/src/tests/renderer.focus.test.ts +294 -0
  333. package/src/tests/renderer.idle.test.ts +219 -0
  334. package/src/tests/renderer.input.test.ts +2237 -0
  335. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  336. package/src/tests/renderer.mouse.test.ts +1274 -0
  337. package/src/tests/renderer.palette.test.ts +629 -0
  338. package/src/tests/renderer.selection.test.ts +49 -0
  339. package/src/tests/renderer.slot-registry.test.ts +684 -0
  340. package/src/tests/renderer.useMouse.test.ts +47 -0
  341. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
  342. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
  343. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
  344. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
  345. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
  346. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
  347. package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
  348. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
  349. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  350. package/src/tests/runtime-plugin-support.test.ts +19 -0
  351. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
  352. package/src/tests/runtime-plugin.fixture.ts +40 -0
  353. package/src/tests/runtime-plugin.test.ts +354 -0
  354. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  355. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  356. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  357. package/src/tests/scrollbox.test.ts +1530 -0
  358. package/src/tests/wrap-resize-perf.test.ts +276 -0
  359. package/src/tests/yoga-setters.test.ts +921 -0
  360. package/src/text-buffer-view.test.ts +705 -0
  361. package/src/text-buffer-view.ts +189 -0
  362. package/src/text-buffer.test.ts +347 -0
  363. package/src/text-buffer.ts +250 -0
  364. package/src/types.ts +161 -0
  365. package/src/utils.ts +88 -0
  366. package/src/zig/ansi.zig +268 -0
  367. package/src/zig/bench/README.md +50 -0
  368. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  369. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  370. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  371. package/src/zig/bench/rope-markers_bench.zig +713 -0
  372. package/src/zig/bench/rope_bench.zig +514 -0
  373. package/src/zig/bench/styled-text_bench.zig +470 -0
  374. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  375. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  376. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  377. package/src/zig/bench/utf8_bench.zig +799 -0
  378. package/src/zig/bench-utils.zig +431 -0
  379. package/src/zig/bench.zig +217 -0
  380. package/src/zig/buffer-methods.zig +211 -0
  381. package/src/zig/buffer.zig +2281 -0
  382. package/src/zig/build.zig +289 -0
  383. package/src/zig/build.zig.zon +16 -0
  384. package/src/zig/edit-buffer.zig +825 -0
  385. package/src/zig/editor-view.zig +802 -0
  386. package/src/zig/event-bus.zig +13 -0
  387. package/src/zig/event-emitter.zig +65 -0
  388. package/src/zig/file-logger.zig +92 -0
  389. package/src/zig/grapheme.zig +599 -0
  390. package/src/zig/lib.zig +1854 -0
  391. package/src/zig/link.zig +333 -0
  392. package/src/zig/logger.zig +43 -0
  393. package/src/zig/mem-registry.zig +125 -0
  394. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  395. package/src/zig/native-span-feed.zig +708 -0
  396. package/src/zig/renderer.zig +1393 -0
  397. package/src/zig/rope.zig +1220 -0
  398. package/src/zig/syntax-style.zig +161 -0
  399. package/src/zig/terminal.zig +987 -0
  400. package/src/zig/test.zig +72 -0
  401. package/src/zig/tests/README.md +18 -0
  402. package/src/zig/tests/buffer-methods_test.zig +1109 -0
  403. package/src/zig/tests/buffer_test.zig +2557 -0
  404. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  405. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  406. package/src/zig/tests/editor-view_test.zig +3299 -0
  407. package/src/zig/tests/event-emitter_test.zig +249 -0
  408. package/src/zig/tests/grapheme_test.zig +1304 -0
  409. package/src/zig/tests/link_test.zig +190 -0
  410. package/src/zig/tests/mem-registry_test.zig +473 -0
  411. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  412. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  413. package/src/zig/tests/renderer_test.zig +1017 -0
  414. package/src/zig/tests/rope-nested_test.zig +712 -0
  415. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  416. package/src/zig/tests/rope_test.zig +2362 -0
  417. package/src/zig/tests/segment-merge.test.zig +148 -0
  418. package/src/zig/tests/syntax-style_test.zig +557 -0
  419. package/src/zig/tests/terminal_test.zig +754 -0
  420. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  421. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  422. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  423. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  424. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  425. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  426. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  427. package/src/zig/tests/text-buffer_test.zig +2191 -0
  428. package/src/zig/tests/unicode-width-map.zon +3909 -0
  429. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  430. package/src/zig/tests/utf8_test.zig +4057 -0
  431. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  432. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  433. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  434. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  435. package/src/zig/text-buffer-iterators.zig +499 -0
  436. package/src/zig/text-buffer-segment.zig +404 -0
  437. package/src/zig/text-buffer-view.zig +1371 -0
  438. package/src/zig/text-buffer.zig +1180 -0
  439. package/src/zig/utf8.zig +1948 -0
  440. package/src/zig/utils.zig +9 -0
  441. package/src/zig-structs.ts +261 -0
  442. package/src/zig.ts +3884 -0
  443. package/tsconfig.build.json +24 -0
  444. package/tsconfig.json +27 -0
  445. package/3d/SpriteResourceManager.d.ts +0 -74
  446. package/3d/SpriteUtils.d.ts +0 -13
  447. package/3d/TextureUtils.d.ts +0 -24
  448. package/3d/ThreeRenderable.d.ts +0 -40
  449. package/3d/WGPURenderer.d.ts +0 -61
  450. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  451. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  452. package/3d/animation/SpriteAnimator.d.ts +0 -124
  453. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  454. package/3d/canvas.d.ts +0 -44
  455. package/3d/index.d.ts +0 -12
  456. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  457. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  458. package/3d/physics/physics-interface.d.ts +0 -27
  459. package/3d.d.ts +0 -2
  460. package/3d.js +0 -34041
  461. package/3d.js.map +0 -155
  462. package/LICENSE +0 -21
  463. package/NativeSpanFeed.d.ts +0 -41
  464. package/Renderable.d.ts +0 -334
  465. package/animation/Timeline.d.ts +0 -126
  466. package/ansi.d.ts +0 -13
  467. package/buffer.d.ts +0 -111
  468. package/console.d.ts +0 -144
  469. package/edit-buffer.d.ts +0 -98
  470. package/editor-view.d.ts +0 -73
  471. package/index-9vwc3fg6.js +0 -12260
  472. package/index-9vwc3fg6.js.map +0 -42
  473. package/index-dcj62y8t.js +0 -20614
  474. package/index-dcj62y8t.js.map +0 -67
  475. package/index-f7n39gpy.js +0 -411
  476. package/index-f7n39gpy.js.map +0 -10
  477. package/index.d.ts +0 -23
  478. package/index.js +0 -478
  479. package/index.js.map +0 -9
  480. package/lib/KeyHandler.d.ts +0 -61
  481. package/lib/RGBA.d.ts +0 -25
  482. package/lib/ascii.font.d.ts +0 -508
  483. package/lib/border.d.ts +0 -51
  484. package/lib/bunfs.d.ts +0 -7
  485. package/lib/clipboard.d.ts +0 -17
  486. package/lib/clock.d.ts +0 -15
  487. package/lib/data-paths.d.ts +0 -26
  488. package/lib/debounce.d.ts +0 -42
  489. package/lib/detect-links.d.ts +0 -6
  490. package/lib/env.d.ts +0 -42
  491. package/lib/extmarks-history.d.ts +0 -17
  492. package/lib/extmarks.d.ts +0 -89
  493. package/lib/hast-styled-text.d.ts +0 -17
  494. package/lib/index.d.ts +0 -21
  495. package/lib/keymapping.d.ts +0 -25
  496. package/lib/objects-in-viewport.d.ts +0 -24
  497. package/lib/output.capture.d.ts +0 -24
  498. package/lib/parse.keypress-kitty.d.ts +0 -2
  499. package/lib/parse.keypress.d.ts +0 -26
  500. package/lib/parse.mouse.d.ts +0 -30
  501. package/lib/paste.d.ts +0 -7
  502. package/lib/queue.d.ts +0 -15
  503. package/lib/renderable.validations.d.ts +0 -12
  504. package/lib/scroll-acceleration.d.ts +0 -43
  505. package/lib/selection.d.ts +0 -63
  506. package/lib/singleton.d.ts +0 -7
  507. package/lib/stdin-parser.d.ts +0 -87
  508. package/lib/styled-text.d.ts +0 -63
  509. package/lib/terminal-capability-detection.d.ts +0 -30
  510. package/lib/terminal-palette.d.ts +0 -50
  511. package/lib/tree-sitter/assets/update.d.ts +0 -11
  512. package/lib/tree-sitter/client.d.ts +0 -47
  513. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  514. package/lib/tree-sitter/download-utils.d.ts +0 -21
  515. package/lib/tree-sitter/index.d.ts +0 -8
  516. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  517. package/lib/tree-sitter/parsers-config.d.ts +0 -53
  518. package/lib/tree-sitter/resolve-ft.d.ts +0 -5
  519. package/lib/tree-sitter/types.d.ts +0 -82
  520. package/lib/tree-sitter-styled-text.d.ts +0 -14
  521. package/lib/validate-dir-name.d.ts +0 -1
  522. package/lib/yoga.options.d.ts +0 -32
  523. package/parser.worker.js +0 -899
  524. package/parser.worker.js.map +0 -12
  525. package/plugins/core-slot.d.ts +0 -72
  526. package/plugins/registry.d.ts +0 -42
  527. package/plugins/types.d.ts +0 -34
  528. package/post/effects.d.ts +0 -147
  529. package/post/filters.d.ts +0 -65
  530. package/post/matrices.d.ts +0 -20
  531. package/renderables/ASCIIFont.d.ts +0 -52
  532. package/renderables/Box.d.ts +0 -81
  533. package/renderables/Code.d.ts +0 -78
  534. package/renderables/Diff.d.ts +0 -142
  535. package/renderables/EditBufferRenderable.d.ts +0 -237
  536. package/renderables/FrameBuffer.d.ts +0 -16
  537. package/renderables/Input.d.ts +0 -67
  538. package/renderables/LineNumberRenderable.d.ts +0 -78
  539. package/renderables/Markdown.d.ts +0 -185
  540. package/renderables/ScrollBar.d.ts +0 -77
  541. package/renderables/ScrollBox.d.ts +0 -124
  542. package/renderables/Select.d.ts +0 -115
  543. package/renderables/Slider.d.ts +0 -47
  544. package/renderables/TabSelect.d.ts +0 -96
  545. package/renderables/Text.d.ts +0 -36
  546. package/renderables/TextBufferRenderable.d.ts +0 -105
  547. package/renderables/TextNode.d.ts +0 -91
  548. package/renderables/TextTable.d.ts +0 -140
  549. package/renderables/Textarea.d.ts +0 -63
  550. package/renderables/TimeToFirstDraw.d.ts +0 -24
  551. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  552. package/renderables/composition/VRenderable.d.ts +0 -16
  553. package/renderables/composition/constructs.d.ts +0 -35
  554. package/renderables/composition/vnode.d.ts +0 -46
  555. package/renderables/index.d.ts +0 -23
  556. package/renderables/markdown-parser.d.ts +0 -10
  557. package/renderer.d.ts +0 -419
  558. package/runtime-plugin-support.d.ts +0 -3
  559. package/runtime-plugin-support.js +0 -29
  560. package/runtime-plugin-support.js.map +0 -10
  561. package/runtime-plugin.d.ts +0 -16
  562. package/runtime-plugin.js +0 -16
  563. package/runtime-plugin.js.map +0 -9
  564. package/syntax-style.d.ts +0 -54
  565. package/testing/manual-clock.d.ts +0 -17
  566. package/testing/mock-keys.d.ts +0 -81
  567. package/testing/mock-mouse.d.ts +0 -38
  568. package/testing/mock-tree-sitter-client.d.ts +0 -23
  569. package/testing/spy.d.ts +0 -7
  570. package/testing/test-recorder.d.ts +0 -61
  571. package/testing/test-renderer.d.ts +0 -23
  572. package/testing.d.ts +0 -6
  573. package/testing.js +0 -697
  574. package/testing.js.map +0 -15
  575. package/text-buffer-view.d.ts +0 -42
  576. package/text-buffer.d.ts +0 -67
  577. package/types.d.ts +0 -139
  578. package/utils.d.ts +0 -14
  579. package/zig-structs.d.ts +0 -155
  580. package/zig.d.ts +0 -353
  581. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  582. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  584. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  585. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  589. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  591. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
@@ -0,0 +1,987 @@
1
+ const std = @import("std");
2
+ const builtin = @import("builtin");
3
+ const atomic = std.atomic;
4
+ const assert = std.debug.assert;
5
+ const ansi = @import("ansi.zig");
6
+ const utf8 = @import("utf8.zig");
7
+ const logger = @import("logger.zig");
8
+
9
+ const WidthMethod = utf8.WidthMethod;
10
+
11
+ /// Terminal capability detection and management
12
+ pub const Terminal = @This();
13
+
14
+ pub const Capabilities = struct {
15
+ kitty_keyboard: bool = false,
16
+ kitty_graphics: bool = false,
17
+ rgb: bool = false,
18
+ unicode: WidthMethod = .unicode,
19
+ sgr_pixels: bool = false,
20
+ color_scheme_updates: bool = false,
21
+ explicit_width: bool = false,
22
+ scaled_text: bool = false,
23
+ sixel: bool = false,
24
+ focus_tracking: bool = false,
25
+ sync: bool = false,
26
+ bracketed_paste: bool = false,
27
+ hyperlinks: bool = false,
28
+ osc52: bool = false,
29
+ explicit_cursor_positioning: bool = false,
30
+ };
31
+
32
+ pub const MouseLevel = enum {
33
+ none,
34
+ basic, // click only
35
+ drag, // click + drag
36
+ motion, // all motion
37
+ pixels, // pixel coordinates
38
+ };
39
+
40
+ pub const CursorStyle = enum {
41
+ block,
42
+ line,
43
+ underline,
44
+ default,
45
+ };
46
+
47
+ pub const MousePointerStyle = enum(u8) {
48
+ default = 0,
49
+ pointer = 1,
50
+ text = 2,
51
+ crosshair = 3,
52
+ move = 4,
53
+ not_allowed = 5,
54
+
55
+ pub fn toName(self: MousePointerStyle) []const u8 {
56
+ return if (self == .not_allowed) "not-allowed" else @tagName(self);
57
+ }
58
+ };
59
+
60
+ pub const ClipboardTarget = enum {
61
+ clipboard, // "c"
62
+ primary, // "p"
63
+ secondary, // "s"
64
+ query, // "q"
65
+
66
+ pub fn toChar(self: ClipboardTarget) u8 {
67
+ return switch (self) {
68
+ .clipboard => 'c',
69
+ .primary => 'p',
70
+ .secondary => 's',
71
+ .query => 'q',
72
+ };
73
+ }
74
+ };
75
+
76
+ pub const Options = struct {
77
+ // Kitty keyboard protocol flags (progressive enhancement):
78
+ // See: https://sw.kovidgoyal.net/kitty/keyboard-protocol/#progressive-enhancement
79
+ // Bit 0 (0b1): Disambiguate escape codes (fixes ESC timing, alt+key ambiguity, ctrl+c as event)
80
+ // Bit 1 (0b10): Report event types (press/repeat/release)
81
+ // Bit 2 (0b100): Report alternate keys (e.g., numpad vs regular, shifted, base layout)
82
+ // Bit 3 (0b1000): Report all keys as escape codes
83
+ // Bit 4 (0b10000): Report text associated with key events
84
+ // Default 0b00101 (5) = disambiguate + alternate keys
85
+ // Use 0b00111 (7) to also enable event types for key release detection
86
+ kitty_keyboard_flags: u8 = 0b00101,
87
+ remote: bool = false,
88
+ // Optional override for environment lookups. Caller owns the map.
89
+ env_map: ?*const std.process.EnvMap = null,
90
+ };
91
+
92
+ pub const TerminalInfo = struct {
93
+ name: [64]u8 = [_]u8{0} ** 64,
94
+ name_len: usize = 0,
95
+ version: [32]u8 = [_]u8{0} ** 32,
96
+ version_len: usize = 0,
97
+ from_xtversion: bool = false,
98
+ };
99
+
100
+ caps: Capabilities = .{},
101
+ opts: Options = .{},
102
+ host_env_map: ?std.process.EnvMap = null,
103
+
104
+ in_tmux: bool = false,
105
+ skip_graphics_query: bool = false,
106
+ skip_explicit_width_query: bool = false,
107
+ graphics_query_pending: bool = false,
108
+ capability_queries_pending: bool = false,
109
+
110
+ state: struct {
111
+ alt_screen: bool = false,
112
+ kitty_keyboard: bool = false,
113
+ kitty_keyboard_flags: u8 = 0,
114
+ bracketed_paste: bool = false,
115
+ mouse: bool = false,
116
+ mouse_movement: bool = true,
117
+ pixel_mouse: bool = false,
118
+ color_scheme_updates: bool = false,
119
+ focus_tracking: bool = false,
120
+ modify_other_keys: bool = false,
121
+ mouse_pointer: MousePointerStyle = .default,
122
+ cursor: struct {
123
+ row: u16 = 0,
124
+ col: u16 = 0,
125
+ x: u32 = 1, // 1-based for rendering
126
+ y: u32 = 1, // 1-based for rendering
127
+ visible: bool = true,
128
+ style: CursorStyle = .default,
129
+ blinking: bool = false,
130
+ color: [4]f32 = .{ 1.0, 1.0, 1.0, 1.0 }, // RGBA
131
+ } = .{},
132
+ } = .{},
133
+
134
+ term_info: TerminalInfo = .{},
135
+
136
+ pub fn init(opts: Options) Terminal {
137
+ var term: Terminal = .{
138
+ .opts = opts,
139
+ };
140
+
141
+ term.checkEnvironmentOverrides();
142
+ return term;
143
+ }
144
+
145
+ pub fn deinit(self: *Terminal) void {
146
+ if (self.host_env_map) |*env_map| {
147
+ env_map.deinit();
148
+ self.host_env_map = null;
149
+ }
150
+ self.opts.env_map = null;
151
+ }
152
+
153
+ pub fn setHostEnvVar(self: *Terminal, allocator: std.mem.Allocator, key: []const u8, value: []const u8) !void {
154
+ if (self.host_env_map == null) {
155
+ self.host_env_map = std.process.EnvMap.init(allocator);
156
+ }
157
+
158
+ const env_map = &self.host_env_map.?;
159
+ try env_map.put(key, value);
160
+ self.opts.env_map = env_map;
161
+ self.checkEnvironmentOverrides();
162
+ }
163
+
164
+ pub fn resetState(self: *Terminal, tty: anytype) !void {
165
+ try tty.writeAll(ansi.ANSI.showCursor);
166
+ try tty.writeAll(ansi.ANSI.reset);
167
+ try tty.writeAll(ansi.ANSI.resetMousePointer);
168
+ self.state.mouse_pointer = .default;
169
+
170
+ if (self.state.kitty_keyboard) {
171
+ try self.setKittyKeyboard(tty, false, 0);
172
+ }
173
+
174
+ if (self.state.modify_other_keys) {
175
+ try self.setModifyOtherKeys(tty, false);
176
+ }
177
+
178
+ if (self.state.mouse) {
179
+ try self.setMouseMode(tty, false, self.state.mouse_movement);
180
+ }
181
+
182
+ if (self.state.bracketed_paste) {
183
+ try self.setBracketedPaste(tty, false);
184
+ }
185
+
186
+ if (self.state.focus_tracking) {
187
+ try self.setFocusTracking(tty, false);
188
+ }
189
+
190
+ if (self.state.alt_screen) {
191
+ try self.exitAltScreen(tty);
192
+ } else {
193
+ switch (builtin.os.tag) {
194
+ .windows => {
195
+ try tty.writeByte('\r');
196
+ var i: u16 = 0;
197
+ while (i < self.state.cursor.row) : (i += 1) {
198
+ try tty.writeAll(ansi.ANSI.reverseIndex);
199
+ }
200
+ try tty.writeAll(ansi.ANSI.eraseBelowCursor);
201
+ },
202
+ else => {},
203
+ }
204
+ }
205
+
206
+ if (self.state.color_scheme_updates) {
207
+ try self.setColorSchemeUpdates(tty, false);
208
+ }
209
+
210
+ self.setTerminalTitle(tty, "");
211
+ }
212
+
213
+ pub fn enterAltScreen(self: *Terminal, tty: anytype) !void {
214
+ try tty.writeAll(ansi.ANSI.switchToAlternateScreen);
215
+ self.state.alt_screen = true;
216
+ }
217
+
218
+ pub fn exitAltScreen(self: *Terminal, tty: anytype) !void {
219
+ try tty.writeAll(ansi.ANSI.switchToMainScreen);
220
+ self.state.alt_screen = false;
221
+ }
222
+
223
+ pub fn queryTerminalSend(self: *Terminal, tty: anytype) !void {
224
+ self.checkEnvironmentOverrides();
225
+ self.graphics_query_pending = !self.skip_graphics_query;
226
+ self.capability_queries_pending = false;
227
+
228
+ // Send xtversion first (doesn't need DCS wrapping - used for tmux detection)
229
+ try tty.writeAll(ansi.ANSI.xtversion ++
230
+ ansi.ANSI.hideCursor ++
231
+ ansi.ANSI.saveCursorState);
232
+
233
+ if (self.in_tmux) {
234
+ try tty.writeAll(ansi.ANSI.capabilityQueriesTmux);
235
+ } else {
236
+ try tty.writeAll(ansi.ANSI.capabilityQueries);
237
+ self.capability_queries_pending = true;
238
+ }
239
+
240
+ if (!self.skip_explicit_width_query) {
241
+ try tty.writeAll(ansi.ANSI.home ++
242
+ ansi.ANSI.explicitWidthQuery ++
243
+ ansi.ANSI.cursorPositionRequest ++
244
+ ansi.ANSI.home ++
245
+ ansi.ANSI.scaledTextQuery ++
246
+ ansi.ANSI.cursorPositionRequest);
247
+ }
248
+
249
+ try tty.writeAll(ansi.ANSI.restoreCursorState);
250
+ }
251
+
252
+ pub fn sendPendingQueries(self: *Terminal, tty: anytype) !bool {
253
+ var sent = false;
254
+ const is_tmux = self.in_tmux or self.isXtversionTmux();
255
+
256
+ // Re-send capability queries DCS wrapped if tmux detected via xtversion
257
+ // Only needed if we got xtversion response indicating tmux
258
+ if (self.capability_queries_pending) {
259
+ if (self.term_info.from_xtversion and is_tmux) {
260
+ try tty.writeAll(ansi.ANSI.capabilityQueriesTmux);
261
+ sent = true;
262
+ }
263
+ // Clear pending flag regardless - non-tmux terminals already received unwrapped queries
264
+ self.capability_queries_pending = false;
265
+ }
266
+
267
+ if (self.graphics_query_pending and !self.skip_graphics_query) {
268
+ if (is_tmux) {
269
+ try tty.writeAll(ansi.ANSI.kittyGraphicsQueryTmux);
270
+ } else {
271
+ try tty.writeAll(ansi.ANSI.kittyGraphicsQuery);
272
+ }
273
+ self.graphics_query_pending = false;
274
+ sent = true;
275
+ }
276
+
277
+ return sent;
278
+ }
279
+
280
+ pub fn enableDetectedFeatures(self: *Terminal, tty: anytype, use_kitty_keyboard: bool) !void {
281
+ if (builtin.os.tag == .windows) {
282
+ // Windows-specific defaults for ConPTY
283
+ self.caps.rgb = true;
284
+ self.caps.bracketed_paste = true;
285
+ }
286
+
287
+ self.checkEnvironmentOverrides();
288
+
289
+ if (!self.state.modify_other_keys and !self.state.kitty_keyboard) {
290
+ try self.setModifyOtherKeys(tty, true);
291
+ }
292
+
293
+ if (self.caps.kitty_keyboard and use_kitty_keyboard) {
294
+ if (self.state.modify_other_keys) {
295
+ try self.setModifyOtherKeys(tty, false);
296
+ }
297
+ try self.setKittyKeyboard(tty, true, self.opts.kitty_keyboard_flags);
298
+ }
299
+
300
+ if (self.caps.unicode == .unicode and !self.caps.explicit_width) {
301
+ try tty.writeAll(ansi.ANSI.unicodeSet);
302
+ }
303
+
304
+ if (self.caps.bracketed_paste) {
305
+ try self.setBracketedPaste(tty, true);
306
+ }
307
+
308
+ if (self.caps.focus_tracking) {
309
+ try self.setFocusTracking(tty, true);
310
+ }
311
+
312
+ if (!self.state.color_scheme_updates) {
313
+ try self.setColorSchemeUpdates(tty, true);
314
+ try tty.writeAll(ansi.ANSI.colorSchemeRequest);
315
+ }
316
+ }
317
+
318
+ fn checkEnvironmentOverrides(self: *Terminal) void {
319
+ self.in_tmux = false;
320
+ self.skip_graphics_query = false;
321
+ self.skip_explicit_width_query = false;
322
+
323
+ // Always just try to enable bracketed paste, even if it was reported as not supported
324
+ self.caps.bracketed_paste = true;
325
+
326
+ if (self.caps.rgb) {
327
+ self.caps.hyperlinks = true;
328
+ }
329
+
330
+ if (self.opts.remote) {
331
+ return;
332
+ }
333
+
334
+ var env_map_storage: ?std.process.EnvMap = null;
335
+ const env_map: *const std.process.EnvMap = self.opts.env_map orelse blk: {
336
+ env_map_storage = std.process.getEnvMap(std.heap.page_allocator) catch |err| {
337
+ logger.err("Failed to get environment map: {}", .{err});
338
+ return;
339
+ };
340
+ break :blk &env_map_storage.?;
341
+ };
342
+ defer if (env_map_storage) |*map| map.deinit();
343
+
344
+ if (!self.term_info.from_xtversion) {
345
+ if (env_map.get("TMUX")) |_| {
346
+ self.in_tmux = true;
347
+ self.caps.unicode = .wcwidth;
348
+ self.caps.explicit_cursor_positioning = true;
349
+ } else if (env_map.get("TERM")) |term| {
350
+ if (std.mem.startsWith(u8, term, "tmux")) {
351
+ self.in_tmux = true;
352
+ self.caps.unicode = .wcwidth;
353
+ self.caps.explicit_cursor_positioning = true;
354
+ } else if (std.mem.startsWith(u8, term, "screen")) {
355
+ self.skip_graphics_query = true;
356
+ self.caps.unicode = .wcwidth;
357
+ self.caps.explicit_cursor_positioning = true;
358
+ }
359
+ if (std.mem.indexOf(u8, term, "alacritty") != null) {
360
+ self.caps.explicit_cursor_positioning = true;
361
+ }
362
+ }
363
+ }
364
+
365
+ if (env_map.get("OPENTUI_GRAPHICS")) |val| {
366
+ if (std.mem.eql(u8, val, "false") or std.mem.eql(u8, val, "0")) {
367
+ self.skip_graphics_query = true;
368
+ } else if (std.mem.eql(u8, val, "true") or std.mem.eql(u8, val, "1")) {
369
+ self.skip_graphics_query = false;
370
+ }
371
+ }
372
+
373
+ if (!self.term_info.from_xtversion) {
374
+ if (env_map.get("TERM_PROGRAM")) |prog| {
375
+ const copy_len = @min(prog.len, self.term_info.name.len);
376
+ @memcpy(self.term_info.name[0..copy_len], prog[0..copy_len]);
377
+ self.term_info.name_len = copy_len;
378
+
379
+ if (env_map.get("TERM_PROGRAM_VERSION")) |ver| {
380
+ const ver_len = @min(ver.len, self.term_info.version.len);
381
+ @memcpy(self.term_info.version[0..ver_len], ver[0..ver_len]);
382
+ self.term_info.version_len = ver_len;
383
+ }
384
+ }
385
+
386
+ if (env_map.get("TERM_PROGRAM")) |prog| {
387
+ if (std.mem.eql(u8, prog, "vscode")) {
388
+ self.caps.kitty_keyboard = false;
389
+ self.caps.kitty_graphics = false;
390
+ self.caps.unicode = .unicode;
391
+ } else if (std.mem.eql(u8, prog, "Apple_Terminal")) {
392
+ self.caps.unicode = .wcwidth;
393
+ } else if (std.mem.eql(u8, prog, "Alacritty")) {
394
+ self.caps.explicit_cursor_positioning = true;
395
+ }
396
+ }
397
+
398
+ if (env_map.get("ALACRITTY_SOCKET") != null or env_map.get("ALACRITTY_LOG") != null) {
399
+ self.caps.explicit_cursor_positioning = true;
400
+ if (self.term_info.name_len == 0) {
401
+ const name = "Alacritty";
402
+ @memcpy(self.term_info.name[0..name.len], name);
403
+ self.term_info.name_len = name.len;
404
+ }
405
+ }
406
+ }
407
+
408
+ if (env_map.get("COLORTERM")) |colorterm| {
409
+ if (std.mem.eql(u8, colorterm, "truecolor") or
410
+ std.mem.eql(u8, colorterm, "24bit"))
411
+ {
412
+ self.caps.rgb = true;
413
+ }
414
+ }
415
+
416
+ if (!self.term_info.from_xtversion) {
417
+ if (env_map.get("TERMUX_VERSION")) |_| {
418
+ self.caps.unicode = .wcwidth;
419
+ }
420
+
421
+ if (env_map.get("VHS_RECORD")) |_| {
422
+ self.caps.unicode = .wcwidth;
423
+ self.caps.kitty_keyboard = false;
424
+ self.caps.kitty_graphics = false;
425
+ }
426
+ }
427
+
428
+ if (env_map.get("OPENTUI_FORCE_WCWIDTH")) |_| {
429
+ self.caps.unicode = .wcwidth;
430
+ }
431
+ if (env_map.get("OPENTUI_FORCE_UNICODE")) |_| {
432
+ self.caps.unicode = .unicode;
433
+ }
434
+ if (env_map.get("OPENTUI_FORCE_NOZWJ")) |_| {
435
+ self.caps.unicode = .no_zwj;
436
+ }
437
+
438
+ if (env_map.get("OPENTUI_FORCE_EXPLICIT_WIDTH")) |val| {
439
+ if (std.mem.eql(u8, val, "true") or std.mem.eql(u8, val, "1")) {
440
+ self.caps.explicit_width = true;
441
+ } else if (std.mem.eql(u8, val, "false") or std.mem.eql(u8, val, "0")) {
442
+ self.caps.explicit_width = false;
443
+ self.skip_explicit_width_query = true;
444
+ }
445
+ }
446
+
447
+ if (!self.caps.hyperlinks and self.term_info.from_xtversion) {
448
+ if (isHyperlinkTerm(self.getTerminalName())) {
449
+ self.caps.hyperlinks = true;
450
+ }
451
+ }
452
+
453
+ if (!self.caps.hyperlinks and !self.term_info.from_xtversion) {
454
+ if (env_map.get("TERM")) |term| {
455
+ if (isHyperlinkTerm(term)) {
456
+ self.caps.hyperlinks = true;
457
+ }
458
+ }
459
+ }
460
+
461
+ if (!self.caps.hyperlinks and !self.term_info.from_xtversion) {
462
+ const is_wsl = env_map.get("WSL_DISTRO_NAME") != null or env_map.get("WSL_INTEROP") != null;
463
+ const has_wt_session = env_map.get("WT_SESSION") != null;
464
+ if (is_wsl and has_wt_session) {
465
+ if (env_map.get("TERM")) |term| {
466
+ if (std.mem.startsWith(u8, term, "xterm")) {
467
+ self.caps.hyperlinks = true;
468
+ }
469
+ }
470
+ }
471
+ }
472
+
473
+ if (!self.caps.osc52 and !self.term_info.from_xtversion) {
474
+ if (env_map.get("WT_SESSION") != null) {
475
+ self.caps.osc52 = true;
476
+ }
477
+
478
+ if (!self.caps.osc52 and (self.in_tmux or env_map.get("STY") != null)) {
479
+ self.caps.osc52 = true;
480
+ }
481
+
482
+ if (!self.caps.osc52) {
483
+ if (env_map.get("TERM_PROGRAM")) |prog| {
484
+ if (isOsc52Term(prog)) {
485
+ self.caps.osc52 = true;
486
+ }
487
+ }
488
+ }
489
+
490
+ if (!self.caps.osc52) {
491
+ if (env_map.get("TERM")) |term| {
492
+ if (isOsc52Term(term) or std.mem.indexOf(u8, term, "256color") != null or std.mem.indexOf(u8, term, "xterm") != null) {
493
+ self.caps.osc52 = true;
494
+ }
495
+ }
496
+ }
497
+ }
498
+ }
499
+
500
+ // TODO: Allow pixel mouse mode to be enabled,
501
+ // currently does not make sense and is not supported by higher levels
502
+ pub fn setMouseMode(self: *Terminal, tty: anytype, enable: bool, enable_movement: bool) !void {
503
+ if (enable) {
504
+ if (self.state.mouse and self.state.mouse_movement == enable_movement) return;
505
+ } else if (!self.state.mouse) {
506
+ return;
507
+ }
508
+
509
+ if (enable) {
510
+ self.state.mouse = true;
511
+ self.state.mouse_movement = enable_movement;
512
+ if (!enable_movement) {
513
+ // Some terminals treat ?1000/?1002/?1003 as one family and let the
514
+ // last sequence win. Reset any-event tracking first, then enable
515
+ // click/drag modes so they remain active.
516
+ try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
517
+ }
518
+ try tty.writeAll(ansi.ANSI.enableMouseTracking);
519
+ try tty.writeAll(ansi.ANSI.enableButtonEventTracking);
520
+ if (enable_movement) {
521
+ try tty.writeAll(ansi.ANSI.enableAnyEventTracking);
522
+ }
523
+ try tty.writeAll(ansi.ANSI.enableSGRMouseMode);
524
+ } else {
525
+ self.state.mouse = false;
526
+ self.state.pixel_mouse = false;
527
+ try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
528
+ try tty.writeAll(ansi.ANSI.disableButtonEventTracking);
529
+ try tty.writeAll(ansi.ANSI.disableMouseTracking);
530
+ try tty.writeAll(ansi.ANSI.disableSGRMouseMode);
531
+ }
532
+ }
533
+
534
+ pub fn setBracketedPaste(self: *Terminal, tty: anytype, enable: bool) !void {
535
+ const seq = if (enable) ansi.ANSI.bracketedPasteSet else ansi.ANSI.bracketedPasteReset;
536
+ try tty.writeAll(seq);
537
+ self.state.bracketed_paste = enable;
538
+ }
539
+
540
+ pub fn setFocusTracking(self: *Terminal, tty: anytype, enable: bool) !void {
541
+ const seq = if (enable) ansi.ANSI.focusSet else ansi.ANSI.focusReset;
542
+ try tty.writeAll(seq);
543
+ self.state.focus_tracking = enable;
544
+ }
545
+
546
+ pub fn setKittyKeyboard(self: *Terminal, tty: anytype, enable: bool, flags: u8) !void {
547
+ if (enable) {
548
+ if (!self.state.kitty_keyboard) {
549
+ try tty.print(ansi.ANSI.csiUPush, .{flags});
550
+ self.state.kitty_keyboard = true;
551
+ self.state.kitty_keyboard_flags = flags;
552
+ }
553
+ } else {
554
+ if (self.state.kitty_keyboard) {
555
+ try tty.writeAll(ansi.ANSI.csiUPop);
556
+ self.state.kitty_keyboard = false;
557
+ self.state.kitty_keyboard_flags = 0;
558
+ }
559
+ }
560
+ }
561
+
562
+ pub fn setModifyOtherKeys(self: *Terminal, tty: anytype, enable: bool) !void {
563
+ const seq = if (enable) ansi.ANSI.modifyOtherKeysSet else ansi.ANSI.modifyOtherKeysReset;
564
+ try tty.writeAll(seq);
565
+ self.state.modify_other_keys = enable;
566
+ }
567
+
568
+ pub fn setColorSchemeUpdates(self: *Terminal, tty: anytype, enable: bool) !void {
569
+ const seq = if (enable) ansi.ANSI.colorSchemeSet else ansi.ANSI.colorSchemeReset;
570
+ try tty.writeAll(seq);
571
+ self.state.color_scheme_updates = enable;
572
+ }
573
+
574
+ /// Re-send all currently-active terminal mode escape sequences unconditionally.
575
+ ///
576
+ /// When the terminal loses and regains focus (e.g. alt-tab, tab switch, minimize),
577
+ /// some terminal emulators (notably Windows Terminal / ConPTY) strip or reset
578
+ /// DEC private modes like mouse tracking (?1000/?1002/?1003/?1006), focus
579
+ /// tracking (?1004), and bracketed paste (?2004). This function re-emits the
580
+ /// enable sequences for every mode that our state tracking says is currently on,
581
+ /// without checking whether the mode "should" already be enabled — because the
582
+ /// terminal may have silently disabled it.
583
+ ///
584
+ /// This should be called in response to the focus-in event (\x1b[I).
585
+ ///
586
+ /// Per the xterm ctlseqs spec (Patch #401, 2025/06/22) and the Microsoft
587
+ /// Console Virtual Terminal Sequences documentation, the relevant DECSET
588
+ /// private modes are:
589
+ /// ?1000h - Normal mouse tracking (sends button press/release)
590
+ /// ?1002h - Button-event tracking (adds drag reporting)
591
+ /// ?1003h - Any-event tracking (adds all motion reporting)
592
+ /// ?1006h - SGR extended mouse mode (extended coordinate encoding)
593
+ /// ?1004h - Focus event tracking (sends \x1b[I / \x1b[O)
594
+ /// ?2004h - Bracketed paste mode (wraps pasted text in markers)
595
+ /// Kitty keyboard protocol (CSI > flags u) - progressive enhancement
596
+ /// modifyOtherKeys (CSI > 4 ; 1 m) - xterm key modification
597
+ pub fn restoreTerminalModes(self: *Terminal, tty: anytype) !void {
598
+ // Re-enable mouse tracking modes if active
599
+ if (self.state.mouse) {
600
+ if (!self.state.mouse_movement) {
601
+ try tty.writeAll(ansi.ANSI.disableAnyEventTracking);
602
+ }
603
+ try tty.writeAll(ansi.ANSI.enableMouseTracking);
604
+ try tty.writeAll(ansi.ANSI.enableButtonEventTracking);
605
+ if (self.state.mouse_movement) {
606
+ try tty.writeAll(ansi.ANSI.enableAnyEventTracking);
607
+ }
608
+ try tty.writeAll(ansi.ANSI.enableSGRMouseMode);
609
+ }
610
+
611
+ // Re-enable focus tracking if active
612
+ if (self.state.focus_tracking) {
613
+ try tty.writeAll(ansi.ANSI.focusSet);
614
+ }
615
+
616
+ // Re-enable bracketed paste if active
617
+ if (self.state.bracketed_paste) {
618
+ try tty.writeAll(ansi.ANSI.bracketedPasteSet);
619
+ }
620
+
621
+ // Pop stale entry then re-push kitty keyboard protocol to avoid stack growth.
622
+ // Both sequences are in the same write buffer so the terminal processes them atomically.
623
+ if (self.state.kitty_keyboard) {
624
+ try tty.writeAll(ansi.ANSI.csiUPop);
625
+ try tty.print(ansi.ANSI.csiUPush, .{self.state.kitty_keyboard_flags});
626
+ }
627
+
628
+ // Re-enable modifyOtherKeys if active
629
+ if (self.state.modify_other_keys) {
630
+ try tty.writeAll(ansi.ANSI.modifyOtherKeysSet);
631
+ }
632
+ }
633
+
634
+ /// The responses look like these:
635
+ /// kitty - '\x1B[?1016;2$y\x1B[?2027;0$y\x1B[?2031;2$y\x1B[?1004;1$y\x1B[?2026;2$y\x1B[1;2R\x1B[1;3R\x1BP>|kitty(0.40.1)\x1B\\\x1B[?0u\x1B_Gi=1;EINVAL:Zero width/height not allowed\x1B\\\x1B[?62;c'
636
+ /// ghostty - '\x1B[?1016;1$y\x1B[?2027;1$y\x1B[?2031;2$y\x1B[?1004;1$y\x1B[?2004;2$y\x1B[?2026;2$y\x1B[1;1R\x1B[1;1R\x1BP>|ghostty 1.1.3\x1B\\\x1B[?0u\x1B_Gi=1;OK\x1B\\\x1B[?62;22c'
637
+ /// tmux - '\x1B[1;1R\x1B[1;1R\x1BP>|tmux 3.5a\x1B\\\x1B[?1;2;4c\x1B[?2;3;0S'
638
+ /// vscode - '\x1B[?1016;2$y'
639
+ /// alacritty - '\x1B[?1016;0$y\x1B[?2027;0$y\x1B[?2031;0$y\x1B[?1004;2$y\x1B[?2004;2$y\x1B[?2026;2$y\x1B[1;1R\x1B[1;1R\x1B[?0u\x1B[?6c'
640
+ ///
641
+ /// Parsing these is not complete yet
642
+ pub fn processCapabilityResponse(self: *Terminal, response: []const u8) void {
643
+ // DECRPM responses
644
+ if (std.mem.indexOf(u8, response, "1016;2$y")) |_| {
645
+ self.caps.sgr_pixels = true;
646
+ }
647
+ if (std.mem.indexOf(u8, response, "2027;2$y")) |_| {
648
+ self.caps.unicode = .unicode;
649
+ }
650
+ if (std.mem.indexOf(u8, response, "2031;1$y") != null or std.mem.indexOf(u8, response, "2031;2$y") != null) {
651
+ self.caps.color_scheme_updates = true;
652
+ }
653
+ if (std.mem.indexOf(u8, response, "1004;1$y") != null or std.mem.indexOf(u8, response, "1004;2$y") != null) {
654
+ self.caps.focus_tracking = true;
655
+ }
656
+ if (std.mem.indexOf(u8, response, "2026;1$y") != null or std.mem.indexOf(u8, response, "2026;2$y") != null) {
657
+ self.caps.sync = true;
658
+ }
659
+ if (std.mem.indexOf(u8, response, "2004;1$y") != null or std.mem.indexOf(u8, response, "2004;2$y") != null) {
660
+ self.caps.bracketed_paste = true;
661
+ }
662
+
663
+ // Explicit width detection - cursor position report [1;NR where N >= 2 means explicit width supported
664
+ // We look for ESC[1; followed by a digit >= 2
665
+ // This handles cases where the cursor isn't at exact home position when queries are sent
666
+ if (std.mem.indexOf(u8, response, "\x1b[1;")) |pos| {
667
+ const after = response[pos + 4 ..];
668
+ if (after.len > 0) {
669
+ var end: usize = 0;
670
+ while (end < after.len and after[end] >= '0' and after[end] <= '9') : (end += 1) {}
671
+ if (end > 0 and end < after.len and after[end] == 'R') {
672
+ const col = std.fmt.parseInt(u16, after[0..end], 10) catch 0;
673
+ if (col >= 2) {
674
+ self.caps.explicit_width = true;
675
+ }
676
+ if (col >= 3) {
677
+ self.caps.scaled_text = true;
678
+ }
679
+ }
680
+ }
681
+ }
682
+
683
+ // Parse xtversion response: ESC P > | name version ESC \
684
+ // Examples: "\x1BP>|kitty(0.40.1)\x1B\\" or "\x1BP>|ghostty 1.1.3\x1B\\" or "\x1BP>|tmux 3.5a\x1B\\"
685
+ if (std.mem.indexOf(u8, response, "\x1bP>|")) |pos| {
686
+ const start = pos + 4; // Skip past "\x1BP>|"
687
+ if (std.mem.indexOf(u8, response[start..], "\x1b\\")) |end_offset| {
688
+ const term_str = response[start .. start + end_offset];
689
+ self.parseXtversion(term_str);
690
+ }
691
+ }
692
+
693
+ // Kitty detection
694
+ if (std.mem.indexOf(u8, response, "kitty")) |_| {
695
+ self.caps.kitty_keyboard = true;
696
+ self.caps.kitty_graphics = true;
697
+ self.caps.unicode = .unicode;
698
+ self.caps.rgb = true;
699
+ self.caps.sixel = true;
700
+ self.caps.bracketed_paste = true;
701
+ self.caps.hyperlinks = true;
702
+ }
703
+
704
+ // Kitty keyboard protocol detection via CSI ? u response
705
+ // Terminals supporting the protocol respond to CSI ? u with CSI ? <flags> u
706
+ // Examples: \x1b[?0u (ghostty, alacritty), \x1b[?1u, etc.
707
+ if (std.mem.indexOf(u8, response, "\x1b[?") != null and std.mem.indexOf(u8, response, "u") != null) {
708
+ // Look for pattern \x1b[?Nu where N is 0-31
709
+ var i: usize = 0;
710
+ while (i + 4 < response.len) : (i += 1) {
711
+ if (response[i] == '\x1b' and i + 1 < response.len and response[i + 1] == '[' and i + 2 < response.len and response[i + 2] == '?') {
712
+ var num_end = i + 3;
713
+ while (num_end < response.len and response[num_end] >= '0' and response[num_end] <= '9') : (num_end += 1) {}
714
+ if (num_end > i + 3 and num_end < response.len and response[num_end] == 'u') {
715
+ self.caps.kitty_keyboard = true;
716
+ break;
717
+ }
718
+ }
719
+ }
720
+ }
721
+
722
+ if (std.mem.indexOf(u8, response, "tmux")) |_| {
723
+ self.caps.unicode = .wcwidth;
724
+ self.caps.explicit_cursor_positioning = true;
725
+ }
726
+
727
+ if (std.mem.indexOf(u8, response, "alacritty")) |_| {
728
+ self.caps.explicit_cursor_positioning = true;
729
+ }
730
+
731
+ // Sixel detection via device attributes (capability 4 in DA1 response ending with 'c')
732
+ if (std.mem.indexOf(u8, response, ";c")) |pos| {
733
+ var start: usize = 0;
734
+ if (pos >= 4) {
735
+ start = pos;
736
+ while (start > 0 and response[start] != '\x1b') {
737
+ start -= 1;
738
+ }
739
+
740
+ const da_response = response[start .. pos + 2];
741
+
742
+ if (std.mem.indexOf(u8, da_response, "\x1b[?") == 0) {
743
+ if (std.mem.indexOf(u8, da_response, "4;") != null or std.mem.indexOf(u8, da_response, ";4;") != null or std.mem.indexOf(u8, da_response, ";4c") != null) {
744
+ self.caps.sixel = true;
745
+ }
746
+ }
747
+ }
748
+ }
749
+
750
+ // Kitty graphics response: ESC_Gi=31337;OK ESC\ or ESC_Gi=31337;EERROR... ESC\
751
+ // We look for our specific query ID (31337) to avoid false positives
752
+ if (std.mem.indexOf(u8, response, "\x1b_G")) |_| {
753
+ if (std.mem.indexOf(u8, response, "i=31337")) |_| {
754
+ // Got a response to our graphics query with our ID
755
+ // If it contains "OK" or even an error, the protocol is supported
756
+ // (errors mean the query was understood, just parameters were wrong)
757
+ self.caps.kitty_graphics = true;
758
+ }
759
+ }
760
+
761
+ if (!self.caps.osc52 and isOsc52Term(response)) {
762
+ self.caps.osc52 = true;
763
+ }
764
+
765
+ if (!self.caps.hyperlinks and isHyperlinkTerm(response)) {
766
+ self.caps.hyperlinks = true;
767
+ }
768
+ }
769
+
770
+ fn isOsc52Term(value: []const u8) bool {
771
+ return std.ascii.indexOfIgnoreCase(value, "iterm") != null or
772
+ std.ascii.indexOfIgnoreCase(value, "kitty") != null or
773
+ std.ascii.indexOfIgnoreCase(value, "alacritty") != null or
774
+ std.ascii.indexOfIgnoreCase(value, "wezterm") != null or
775
+ std.ascii.indexOfIgnoreCase(value, "contour") != null or
776
+ std.ascii.indexOfIgnoreCase(value, "foot") != null or
777
+ std.ascii.indexOfIgnoreCase(value, "rio") != null or
778
+ std.ascii.indexOfIgnoreCase(value, "ghostty") != null or
779
+ std.ascii.indexOfIgnoreCase(value, "tmux") != null or
780
+ std.ascii.indexOfIgnoreCase(value, "screen") != null;
781
+ }
782
+
783
+ fn isHyperlinkTerm(value: []const u8) bool {
784
+ return std.ascii.indexOfIgnoreCase(value, "ghostty") != null or
785
+ std.ascii.indexOfIgnoreCase(value, "kitty") != null or
786
+ std.ascii.indexOfIgnoreCase(value, "wezterm") != null or
787
+ std.ascii.indexOfIgnoreCase(value, "alacritty") != null or
788
+ std.ascii.indexOfIgnoreCase(value, "iterm") != null;
789
+ }
790
+
791
+ pub fn getCapabilities(self: *Terminal) Capabilities {
792
+ return self.caps;
793
+ }
794
+
795
+ pub fn setMousePointerStyle(self: *Terminal, style: MousePointerStyle) void {
796
+ self.state.mouse_pointer = style;
797
+ }
798
+
799
+ pub fn getMousePointer(self: *Terminal) MousePointerStyle {
800
+ return self.state.mouse_pointer;
801
+ }
802
+
803
+ pub fn setCursorPosition(self: *Terminal, x: u32, y: u32, visible: bool) void {
804
+ self.state.cursor.x = @max(1, x);
805
+ self.state.cursor.y = @max(1, y);
806
+ self.state.cursor.visible = visible;
807
+
808
+ // Update 0-based coordinates for terminal operations
809
+ self.state.cursor.col = @intCast(@max(0, x - 1));
810
+ self.state.cursor.row = @intCast(@max(0, y - 1));
811
+ }
812
+
813
+ pub fn setCursorStyle(self: *Terminal, style: CursorStyle, blinking: bool) void {
814
+ self.state.cursor.style = style;
815
+ self.state.cursor.blinking = blinking;
816
+ }
817
+
818
+ pub fn setCursorColor(self: *Terminal, color: [4]f32) void {
819
+ self.state.cursor.color = color;
820
+ }
821
+
822
+ pub fn getCursorPosition(self: *Terminal) struct { x: u32, y: u32, visible: bool } {
823
+ return .{
824
+ .x = self.state.cursor.x,
825
+ .y = self.state.cursor.y,
826
+ .visible = self.state.cursor.visible,
827
+ };
828
+ }
829
+
830
+ pub fn getCursorStyle(self: *Terminal) struct { style: CursorStyle, blinking: bool } {
831
+ return .{
832
+ .style = self.state.cursor.style,
833
+ .blinking = self.state.cursor.blinking,
834
+ };
835
+ }
836
+
837
+ pub fn getCursorColor(self: *Terminal) [4]f32 {
838
+ return self.state.cursor.color;
839
+ }
840
+
841
+ pub fn setKittyKeyboardFlags(self: *Terminal, flags: u8) void {
842
+ self.opts.kitty_keyboard_flags = flags;
843
+ }
844
+
845
+ pub fn setTerminalTitle(_: *Terminal, tty: anytype, title: []const u8) void {
846
+ // For Windows, we might need to use different approach, but ANSI sequences work in Windows Terminal, ConPTY, etc.
847
+ // For other platforms, ANSI OSC sequences work reliably
848
+ ansi.ANSI.setTerminalTitleOutput(tty, title) catch {};
849
+ }
850
+
851
+ /// Write OSC 52 clipboard sequence to the terminal
852
+ /// Supports tmux/screen passthrough, including nested tmux sessions
853
+ pub fn writeClipboard(self: *Terminal, tty: anytype, target: ClipboardTarget, payload: []const u8) !void {
854
+ if (!self.canWriteClipboard()) {
855
+ return error.NotSupported;
856
+ }
857
+
858
+ var buf: [1024]u8 = undefined;
859
+ var stream = std.io.fixedBufferStream(&buf);
860
+ const writer = stream.writer();
861
+
862
+ // Build OSC 52 sequence: ESC]52;<target>;<payload>ESC\
863
+ try writer.writeAll("\x1b]52;");
864
+ try writer.writeByte(target.toChar());
865
+ try writer.writeByte(';');
866
+ try writer.writeAll(payload);
867
+ try writer.writeAll("\x1b\\");
868
+
869
+ const osc52 = stream.getWritten();
870
+
871
+ // Use self.in_tmux which is set by checkEnvironmentOverrides() considering
872
+ // env vars, xtversion response, and remote option
873
+ const is_tmux = self.in_tmux or self.isXtversionTmux();
874
+
875
+ if (is_tmux) {
876
+ // For nested tmux, we use a fixed level of 1 as we don't have access
877
+ // to env vars here (by design - detection already happened in checkEnvironmentOverrides)
878
+ // In practice, single-level wrapping works for most cases
879
+ var wrapped_buf: [4096]u8 = undefined;
880
+ var wrapped_stream = std.io.fixedBufferStream(&wrapped_buf);
881
+ const wrap_writer = wrapped_stream.writer();
882
+ for (osc52) |c| {
883
+ if (c == '\x1b') {
884
+ try wrap_writer.writeByte('\x1b');
885
+ }
886
+ try wrap_writer.writeByte(c);
887
+ }
888
+ const doubled = wrapped_stream.getWritten();
889
+
890
+ try tty.writeAll(ansi.ANSI.tmuxDcsStart);
891
+ try tty.writeAll(doubled);
892
+ try tty.writeAll(ansi.ANSI.tmuxDcsEnd);
893
+ } else if (self.opts.remote) {
894
+ try tty.writeAll(osc52);
895
+ } else {
896
+ var env_map_storage: ?std.process.EnvMap = null;
897
+ const env_map: *const std.process.EnvMap = self.opts.env_map orelse blk: {
898
+ env_map_storage = std.process.getEnvMap(std.heap.page_allocator) catch |err| {
899
+ logger.err("Failed to get environment map: {}", .{err});
900
+ return;
901
+ };
902
+ break :blk &env_map_storage.?;
903
+ };
904
+ defer if (env_map_storage) |*map| map.deinit();
905
+
906
+ if (env_map.get("STY")) |_| {
907
+ var wrapped_buf: [2048]u8 = undefined;
908
+ var wrapped_stream = std.io.fixedBufferStream(&wrapped_buf);
909
+ const wrapped_writer = wrapped_stream.writer();
910
+
911
+ for (osc52) |c| {
912
+ if (c == '\x1b') {
913
+ try wrapped_writer.writeByte('\x1b');
914
+ }
915
+ try wrapped_writer.writeByte(c);
916
+ }
917
+ const doubled = wrapped_stream.getWritten();
918
+
919
+ try tty.writeAll(ansi.ANSI.screenDcsStart);
920
+ try tty.writeAll(doubled);
921
+ try tty.writeAll(ansi.ANSI.screenDcsEnd);
922
+ } else {
923
+ try tty.writeAll(osc52);
924
+ }
925
+ }
926
+ }
927
+
928
+ /// Check if we can write to the clipboard (TTY and OSC 52 supported)
929
+ fn canWriteClipboard(self: *Terminal) bool {
930
+ // In a real TTY environment, we'd check isTTY here
931
+ // For now, we just check if OSC 52 is supported
932
+ return self.caps.osc52;
933
+ }
934
+
935
+ /// Parse xtversion response string and extract terminal name and version
936
+ /// Examples: "kitty(0.40.1)", "ghostty 1.1.3", "tmux 3.5a"
937
+ fn parseXtversion(self: *Terminal, term_str: []const u8) void {
938
+ if (term_str.len == 0) return;
939
+
940
+ if (std.mem.indexOf(u8, term_str, "(")) |paren_pos| {
941
+ const name_len = @min(paren_pos, self.term_info.name.len);
942
+ @memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
943
+ self.term_info.name_len = name_len;
944
+
945
+ if (std.mem.indexOf(u8, term_str[paren_pos..], ")")) |close_offset| {
946
+ const ver_start = paren_pos + 1;
947
+ const ver_end = paren_pos + close_offset;
948
+ const ver_len = @min(ver_end - ver_start, self.term_info.version.len);
949
+ @memcpy(self.term_info.version[0..ver_len], term_str[ver_start .. ver_start + ver_len]);
950
+ self.term_info.version_len = ver_len;
951
+ }
952
+ } else {
953
+ if (std.mem.indexOf(u8, term_str, " ")) |space_pos| {
954
+ const name_len = @min(space_pos, self.term_info.name.len);
955
+ @memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
956
+ self.term_info.name_len = name_len;
957
+
958
+ const ver_start = space_pos + 1;
959
+ const ver_len = @min(term_str.len - ver_start, self.term_info.version.len);
960
+ @memcpy(self.term_info.version[0..ver_len], term_str[ver_start .. ver_start + ver_len]);
961
+ self.term_info.version_len = ver_len;
962
+ } else {
963
+ const name_len = @min(term_str.len, self.term_info.name.len);
964
+ @memcpy(self.term_info.name[0..name_len], term_str[0..name_len]);
965
+ self.term_info.name_len = name_len;
966
+ self.term_info.version_len = 0;
967
+ }
968
+ }
969
+
970
+ self.term_info.from_xtversion = true;
971
+ }
972
+
973
+ pub fn isXtversionTmux(self: *Terminal) bool {
974
+ return self.term_info.from_xtversion and std.mem.eql(u8, self.getTerminalName(), "tmux");
975
+ }
976
+
977
+ pub fn getTerminalInfo(self: *Terminal) TerminalInfo {
978
+ return self.term_info;
979
+ }
980
+
981
+ pub fn getTerminalName(self: *Terminal) []const u8 {
982
+ return self.term_info.name[0..self.term_info.name_len];
983
+ }
984
+
985
+ pub fn getTerminalVersion(self: *Terminal) []const u8 {
986
+ return self.term_info.version[0..self.term_info.version_len];
987
+ }