@fairyhunter13/opentui-core 0.1.89 → 0.1.91

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 (571) hide show
  1. package/README.md +2 -2
  2. package/dev/keypress-debug-renderer.ts +148 -0
  3. package/dev/keypress-debug.ts +43 -0
  4. package/dev/print-env-vars.ts +32 -0
  5. package/dev/test-tmux-graphics-334.sh +68 -0
  6. package/dev/thai-debug-test.ts +68 -0
  7. package/docs/development.md +141 -0
  8. package/docs/env-vars.md +140 -0
  9. package/docs/getting-started.md +353 -0
  10. package/docs/renderables-vs-constructs.md +159 -0
  11. package/docs/tree-sitter.md +311 -0
  12. package/package.json +61 -52
  13. package/scripts/build.ts +400 -0
  14. package/scripts/publish.ts +60 -0
  15. package/src/3d/SpriteResourceManager.ts +286 -0
  16. package/src/3d/SpriteUtils.ts +71 -0
  17. package/src/3d/TextureUtils.ts +196 -0
  18. package/src/3d/ThreeRenderable.ts +197 -0
  19. package/src/3d/WGPURenderer.ts +294 -0
  20. package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
  21. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
  22. package/src/3d/animation/SpriteAnimator.ts +633 -0
  23. package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
  24. package/src/3d/canvas.ts +464 -0
  25. package/src/3d/index.ts +12 -0
  26. package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
  27. package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
  28. package/src/3d/physics/physics-interface.ts +31 -0
  29. package/src/3d/shaders/supersampling.wgsl +201 -0
  30. package/src/3d.ts +3 -0
  31. package/src/NativeSpanFeed.ts +300 -0
  32. package/src/Renderable.ts +1698 -0
  33. package/src/__snapshots__/buffer.test.ts.snap +28 -0
  34. package/src/animation/Timeline.test.ts +2709 -0
  35. package/src/animation/Timeline.ts +598 -0
  36. package/src/ansi.ts +18 -0
  37. package/src/benchmark/latest-all-bench-run.json +707 -0
  38. package/src/benchmark/latest-async-bench-run.json +336 -0
  39. package/src/benchmark/latest-default-bench-run.json +657 -0
  40. package/src/benchmark/latest-large-bench-run.json +707 -0
  41. package/src/benchmark/latest-quick-bench-run.json +207 -0
  42. package/src/benchmark/markdown-benchmark.ts +1804 -0
  43. package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
  44. package/src/benchmark/native-span-feed-benchmark.md +56 -0
  45. package/src/benchmark/native-span-feed-benchmark.ts +596 -0
  46. package/src/benchmark/native-span-feed-compare.ts +280 -0
  47. package/src/benchmark/renderer-benchmark.ts +754 -0
  48. package/src/benchmark/text-table-benchmark.ts +947 -0
  49. package/src/buffer.test.ts +291 -0
  50. package/src/buffer.ts +519 -0
  51. package/src/console.test.ts +612 -0
  52. package/src/console.ts +1255 -0
  53. package/src/edit-buffer.test.ts +1769 -0
  54. package/src/edit-buffer.ts +411 -0
  55. package/src/editor-view.test.ts +1032 -0
  56. package/src/editor-view.ts +284 -0
  57. package/src/examples/ascii-font-selection-demo.ts +245 -0
  58. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  59. package/src/examples/assets/concrete.png +0 -0
  60. package/src/examples/assets/crate.png +0 -0
  61. package/src/examples/assets/crate_emissive.png +0 -0
  62. package/src/examples/assets/forrest_background.png +0 -0
  63. package/src/examples/assets/hast-example.json +1018 -0
  64. package/src/examples/assets/heart.png +0 -0
  65. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  66. package/src/examples/assets/main_char_idle.png +0 -0
  67. package/src/examples/assets/main_char_jump_end.png +0 -0
  68. package/src/examples/assets/main_char_jump_landing.png +0 -0
  69. package/src/examples/assets/main_char_jump_start.png +0 -0
  70. package/src/examples/assets/main_char_run_loop.png +0 -0
  71. package/src/examples/assets/roughness_map.jpg +0 -0
  72. package/src/examples/build.ts +115 -0
  73. package/src/examples/code-demo.ts +584 -0
  74. package/src/examples/console-demo.ts +358 -0
  75. package/src/examples/core-plugin-slots-demo.ts +759 -0
  76. package/src/examples/diff-demo.ts +699 -0
  77. package/src/examples/draggable-three-demo.ts +259 -0
  78. package/src/examples/editor-demo.ts +322 -0
  79. package/src/examples/extmarks-demo.ts +204 -0
  80. package/src/examples/focus-restore-demo.ts +310 -0
  81. package/src/examples/fonts.ts +245 -0
  82. package/src/examples/fractal-shader-demo.ts +268 -0
  83. package/src/examples/framebuffer-demo.ts +674 -0
  84. package/src/examples/full-unicode-demo.ts +181 -0
  85. package/src/examples/golden-star-demo.ts +933 -0
  86. package/src/examples/grayscale-buffer-demo.ts +249 -0
  87. package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
  88. package/src/examples/index.ts +925 -0
  89. package/src/examples/input-demo.ts +377 -0
  90. package/src/examples/input-select-layout-demo.ts +425 -0
  91. package/src/examples/install.sh +143 -0
  92. package/src/examples/keypress-debug-demo.ts +452 -0
  93. package/src/examples/lib/HexList.ts +122 -0
  94. package/src/examples/lib/PaletteGrid.ts +125 -0
  95. package/src/examples/lib/standalone-keys.ts +25 -0
  96. package/src/examples/lib/tab-controller.ts +243 -0
  97. package/src/examples/lights-phong-demo.ts +290 -0
  98. package/src/examples/link-demo.ts +220 -0
  99. package/src/examples/live-state-demo.ts +480 -0
  100. package/src/examples/markdown-demo.ts +620 -0
  101. package/src/examples/mouse-interaction-demo.ts +428 -0
  102. package/src/examples/nested-zindex-demo.ts +357 -0
  103. package/src/examples/opacity-example.ts +235 -0
  104. package/src/examples/opentui-demo.ts +1057 -0
  105. package/src/examples/physx-planck-2d-demo.ts +507 -0
  106. package/src/examples/physx-rapier-2d-demo.ts +526 -0
  107. package/src/examples/relative-positioning-demo.ts +323 -0
  108. package/src/examples/scroll-example.ts +214 -0
  109. package/src/examples/scrollbox-mouse-test.ts +112 -0
  110. package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
  111. package/src/examples/select-demo.ts +237 -0
  112. package/src/examples/shader-cube-demo.ts +772 -0
  113. package/src/examples/simple-layout-example.ts +591 -0
  114. package/src/examples/slider-demo.ts +617 -0
  115. package/src/examples/split-mode-demo.ts +445 -0
  116. package/src/examples/sprite-animation-demo.ts +443 -0
  117. package/src/examples/sprite-particle-generator-demo.ts +486 -0
  118. package/src/examples/static-sprite-demo.ts +193 -0
  119. package/src/examples/sticky-scroll-example.ts +308 -0
  120. package/src/examples/styled-text-demo.ts +282 -0
  121. package/src/examples/tab-select-demo.ts +219 -0
  122. package/src/examples/terminal-title.ts +29 -0
  123. package/src/examples/terminal.ts +305 -0
  124. package/src/examples/text-node-demo.ts +416 -0
  125. package/src/examples/text-selection-demo.ts +377 -0
  126. package/src/examples/text-table-demo.ts +503 -0
  127. package/src/examples/text-truncation-demo.ts +481 -0
  128. package/src/examples/text-wrap.ts +757 -0
  129. package/src/examples/texture-loading-demo.ts +259 -0
  130. package/src/examples/timeline-example.ts +670 -0
  131. package/src/examples/transparency-demo.ts +241 -0
  132. package/src/examples/vnode-composition-demo.ts +404 -0
  133. package/src/index.ts +22 -0
  134. package/src/lib/KeyHandler.integration.test.ts +292 -0
  135. package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
  136. package/src/lib/KeyHandler.test.ts +662 -0
  137. package/src/lib/KeyHandler.ts +222 -0
  138. package/src/lib/RGBA.test.ts +984 -0
  139. package/src/lib/RGBA.ts +204 -0
  140. package/src/lib/ascii.font.ts +330 -0
  141. package/src/lib/border.test.ts +83 -0
  142. package/src/lib/border.ts +168 -0
  143. package/src/lib/bunfs.test.ts +27 -0
  144. package/src/lib/bunfs.ts +18 -0
  145. package/src/lib/clipboard.test.ts +41 -0
  146. package/src/lib/clipboard.ts +47 -0
  147. package/src/lib/clock.ts +31 -0
  148. package/src/lib/data-paths.test.ts +133 -0
  149. package/src/lib/data-paths.ts +109 -0
  150. package/src/lib/debounce.ts +106 -0
  151. package/src/lib/detect-links.test.ts +98 -0
  152. package/src/lib/detect-links.ts +56 -0
  153. package/src/lib/env.test.ts +228 -0
  154. package/src/lib/env.ts +209 -0
  155. package/src/lib/extmarks-history.ts +51 -0
  156. package/src/lib/extmarks-multiwidth.test.ts +322 -0
  157. package/src/lib/extmarks.test.ts +3457 -0
  158. package/src/lib/extmarks.ts +843 -0
  159. package/src/lib/fonts/block.json +405 -0
  160. package/src/lib/fonts/grid.json +265 -0
  161. package/src/lib/fonts/huge.json +741 -0
  162. package/src/lib/fonts/pallet.json +314 -0
  163. package/src/lib/fonts/shade.json +591 -0
  164. package/src/lib/fonts/slick.json +321 -0
  165. package/src/lib/fonts/tiny.json +69 -0
  166. package/src/lib/hast-styled-text.ts +59 -0
  167. package/src/lib/index.ts +21 -0
  168. package/src/lib/keymapping.test.ts +280 -0
  169. package/src/lib/keymapping.ts +87 -0
  170. package/src/lib/objects-in-viewport.test.ts +787 -0
  171. package/src/lib/objects-in-viewport.ts +153 -0
  172. package/src/lib/output.capture.ts +58 -0
  173. package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
  174. package/src/lib/parse.keypress-kitty.test.ts +663 -0
  175. package/src/lib/parse.keypress-kitty.ts +439 -0
  176. package/src/lib/parse.keypress.test.ts +1849 -0
  177. package/src/lib/parse.keypress.ts +397 -0
  178. package/src/lib/parse.mouse.test.ts +552 -0
  179. package/src/lib/parse.mouse.ts +232 -0
  180. package/src/lib/paste.ts +16 -0
  181. package/src/lib/queue.ts +65 -0
  182. package/src/lib/renderable.validations.test.ts +87 -0
  183. package/src/lib/renderable.validations.ts +83 -0
  184. package/src/lib/scroll-acceleration.ts +98 -0
  185. package/src/lib/selection.ts +240 -0
  186. package/src/lib/singleton.ts +28 -0
  187. package/src/lib/stdin-parser.test.ts +1676 -0
  188. package/src/lib/stdin-parser.ts +1248 -0
  189. package/src/lib/styled-text.ts +178 -0
  190. package/src/lib/terminal-capability-detection.test.ts +202 -0
  191. package/src/lib/terminal-capability-detection.ts +79 -0
  192. package/src/lib/terminal-palette.test.ts +878 -0
  193. package/src/lib/terminal-palette.ts +383 -0
  194. package/src/lib/tree-sitter/assets/README.md +118 -0
  195. package/src/lib/tree-sitter/assets/update.ts +331 -0
  196. package/src/lib/tree-sitter/assets.d.ts +9 -0
  197. package/src/lib/tree-sitter/cache.test.ts +270 -0
  198. package/src/lib/tree-sitter/client.test.ts +1061 -0
  199. package/src/lib/tree-sitter/client.ts +615 -0
  200. package/src/lib/tree-sitter/default-parsers.ts +80 -0
  201. package/src/lib/tree-sitter/download-utils.ts +148 -0
  202. package/src/lib/tree-sitter/index.ts +28 -0
  203. package/src/lib/tree-sitter/parser.worker.ts +1001 -0
  204. package/src/lib/tree-sitter/parsers-config.ts +75 -0
  205. package/src/lib/tree-sitter/resolve-ft.ts +62 -0
  206. package/src/lib/tree-sitter/types.ts +81 -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 +377 -0
  214. package/src/plugins/types.ts +46 -0
  215. package/src/post/filters.ts +888 -0
  216. package/src/renderables/ASCIIFont.ts +219 -0
  217. package/src/renderables/Box.test.ts +160 -0
  218. package/src/renderables/Box.ts +295 -0
  219. package/src/renderables/Code.test.ts +2062 -0
  220. package/src/renderables/Code.ts +357 -0
  221. package/src/renderables/Diff.regression.test.ts +226 -0
  222. package/src/renderables/Diff.test.ts +3027 -0
  223. package/src/renderables/Diff.ts +1209 -0
  224. package/src/renderables/EditBufferRenderable.ts +764 -0
  225. package/src/renderables/FrameBuffer.ts +47 -0
  226. package/src/renderables/Input.test.ts +1228 -0
  227. package/src/renderables/Input.ts +245 -0
  228. package/src/renderables/LineNumberRenderable.ts +675 -0
  229. package/src/renderables/Markdown.ts +1106 -0
  230. package/src/renderables/ScrollBar.ts +422 -0
  231. package/src/renderables/ScrollBox.ts +883 -0
  232. package/src/renderables/Select.test.ts +1010 -0
  233. package/src/renderables/Select.ts +523 -0
  234. package/src/renderables/Slider.test.ts +456 -0
  235. package/src/renderables/Slider.ts +347 -0
  236. package/src/renderables/TabSelect.test.ts +197 -0
  237. package/src/renderables/TabSelect.ts +455 -0
  238. package/src/renderables/Text.selection-buffer.test.ts +123 -0
  239. package/src/renderables/Text.test.ts +2660 -0
  240. package/src/renderables/Text.ts +147 -0
  241. package/src/renderables/TextBufferRenderable.ts +518 -0
  242. package/src/renderables/TextNode.test.ts +1058 -0
  243. package/src/renderables/TextNode.ts +325 -0
  244. package/src/renderables/TextTable.test.ts +1421 -0
  245. package/src/renderables/TextTable.ts +1344 -0
  246. package/src/renderables/Textarea.ts +732 -0
  247. package/src/renderables/TimeToFirstDraw.ts +89 -0
  248. package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
  249. package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
  250. package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
  251. package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
  252. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
  253. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
  254. package/src/renderables/__tests__/LineNumberRenderable.test.ts +1787 -0
  255. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
  256. package/src/renderables/__tests__/Markdown.test.ts +2287 -0
  257. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
  258. package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
  259. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
  260. package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
  261. package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
  262. package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
  263. package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
  264. package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
  265. package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
  266. package/src/renderables/__tests__/Textarea.rendering.test.ts +1864 -0
  267. package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
  268. package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
  269. package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
  270. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
  271. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
  272. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
  273. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
  274. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
  275. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
  276. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
  277. package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
  278. package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
  279. package/src/renderables/composition/README.md +8 -0
  280. package/src/renderables/composition/VRenderable.ts +32 -0
  281. package/src/renderables/composition/constructs.ts +127 -0
  282. package/src/renderables/composition/vnode.ts +289 -0
  283. package/src/renderables/index.ts +22 -0
  284. package/src/renderables/markdown-parser.ts +66 -0
  285. package/src/renderer.ts +2363 -0
  286. package/src/runtime-plugin-support.ts +39 -0
  287. package/src/runtime-plugin.ts +144 -0
  288. package/src/syntax-style.test.ts +841 -0
  289. package/src/syntax-style.ts +264 -0
  290. package/src/testing/README.md +210 -0
  291. package/src/testing/capture-spans.test.ts +194 -0
  292. package/src/testing/integration.test.ts +276 -0
  293. package/src/testing/manual-clock.ts +106 -0
  294. package/src/testing/mock-keys.test.ts +1356 -0
  295. package/src/testing/mock-keys.ts +449 -0
  296. package/src/testing/mock-mouse.test.ts +218 -0
  297. package/src/testing/mock-mouse.ts +247 -0
  298. package/src/testing/mock-tree-sitter-client.ts +73 -0
  299. package/src/testing/spy.ts +13 -0
  300. package/src/testing/test-recorder.test.ts +415 -0
  301. package/src/testing/test-recorder.ts +145 -0
  302. package/src/testing/test-renderer.ts +116 -0
  303. package/src/testing.ts +7 -0
  304. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
  305. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
  306. package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
  307. package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
  308. package/src/tests/allocator-stats.test.ts +38 -0
  309. package/src/tests/destroy-during-render.test.ts +200 -0
  310. package/src/tests/hover-cursor.test.ts +98 -0
  311. package/src/tests/native-span-feed-async.test.ts +173 -0
  312. package/src/tests/native-span-feed-close.test.ts +120 -0
  313. package/src/tests/native-span-feed-coverage.test.ts +227 -0
  314. package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
  315. package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
  316. package/src/tests/opacity.test.ts +123 -0
  317. package/src/tests/renderable.snapshot.test.ts +524 -0
  318. package/src/tests/renderable.test.ts +1281 -0
  319. package/src/tests/renderer.console-startup.test.ts +65 -0
  320. package/src/tests/renderer.control.test.ts +364 -0
  321. package/src/tests/renderer.core-slot-binding.test.ts +952 -0
  322. package/src/tests/renderer.cursor.test.ts +26 -0
  323. package/src/tests/renderer.destroy-during-render.test.ts +110 -0
  324. package/src/tests/renderer.focus-restore.test.ts +228 -0
  325. package/src/tests/renderer.focus.test.ts +251 -0
  326. package/src/tests/renderer.idle.test.ts +219 -0
  327. package/src/tests/renderer.input.test.ts +2145 -0
  328. package/src/tests/renderer.kitty-flags.test.ts +195 -0
  329. package/src/tests/renderer.mouse.test.ts +1269 -0
  330. package/src/tests/renderer.palette.test.ts +629 -0
  331. package/src/tests/renderer.selection.test.ts +49 -0
  332. package/src/tests/renderer.slot-registry.test.ts +649 -0
  333. package/src/tests/renderer.useMouse.test.ts +50 -0
  334. package/src/tests/runtime-plugin-support.fixture.ts +11 -0
  335. package/src/tests/runtime-plugin-support.test.ts +28 -0
  336. package/src/tests/runtime-plugin.fixture.ts +40 -0
  337. package/src/tests/runtime-plugin.test.ts +190 -0
  338. package/src/tests/scrollbox-culling-bug.test.ts +114 -0
  339. package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
  340. package/src/tests/scrollbox-hitgrid.test.ts +909 -0
  341. package/src/tests/scrollbox.test.ts +1530 -0
  342. package/src/tests/wrap-resize-perf.test.ts +229 -0
  343. package/src/tests/yoga-setters.test.ts +921 -0
  344. package/src/text-buffer-view.test.ts +705 -0
  345. package/src/text-buffer-view.ts +189 -0
  346. package/src/text-buffer.test.ts +347 -0
  347. package/src/text-buffer.ts +250 -0
  348. package/src/types.ts +152 -0
  349. package/src/utils.ts +88 -0
  350. package/src/zig/ansi.zig +268 -0
  351. package/src/zig/bench/README.md +50 -0
  352. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
  353. package/src/zig/bench/edit-buffer_bench.zig +476 -0
  354. package/src/zig/bench/native-span-feed_bench.zig +100 -0
  355. package/src/zig/bench/rope-markers_bench.zig +713 -0
  356. package/src/zig/bench/rope_bench.zig +514 -0
  357. package/src/zig/bench/styled-text_bench.zig +470 -0
  358. package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
  359. package/src/zig/bench/text-buffer-view_bench.zig +459 -0
  360. package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
  361. package/src/zig/bench/utf8_bench.zig +799 -0
  362. package/src/zig/bench-utils.zig +431 -0
  363. package/src/zig/bench.zig +217 -0
  364. package/src/zig/buffer.zig +2223 -0
  365. package/src/zig/build.zig +289 -0
  366. package/src/zig/build.zig.zon +16 -0
  367. package/src/zig/edit-buffer.zig +825 -0
  368. package/src/zig/editor-view.zig +802 -0
  369. package/src/zig/event-bus.zig +13 -0
  370. package/src/zig/event-emitter.zig +65 -0
  371. package/src/zig/file-logger.zig +92 -0
  372. package/src/zig/grapheme.zig +599 -0
  373. package/src/zig/lib.zig +1834 -0
  374. package/src/zig/link.zig +333 -0
  375. package/src/zig/logger.zig +43 -0
  376. package/src/zig/mem-registry.zig +125 -0
  377. package/src/zig/native-span-feed-bench-lib.zig +7 -0
  378. package/src/zig/native-span-feed.zig +708 -0
  379. package/src/zig/renderer.zig +1386 -0
  380. package/src/zig/rope.zig +1220 -0
  381. package/src/zig/syntax-style.zig +161 -0
  382. package/src/zig/terminal.zig +975 -0
  383. package/src/zig/test.zig +70 -0
  384. package/src/zig/tests/README.md +18 -0
  385. package/src/zig/tests/buffer_test.zig +2526 -0
  386. package/src/zig/tests/edit-buffer-history_test.zig +271 -0
  387. package/src/zig/tests/edit-buffer_test.zig +1689 -0
  388. package/src/zig/tests/editor-view_test.zig +3299 -0
  389. package/src/zig/tests/event-emitter_test.zig +249 -0
  390. package/src/zig/tests/grapheme_test.zig +1304 -0
  391. package/src/zig/tests/link_test.zig +190 -0
  392. package/src/zig/tests/mem-registry_test.zig +473 -0
  393. package/src/zig/tests/memory_leak_regression_test.zig +159 -0
  394. package/src/zig/tests/native-span-feed_test.zig +1264 -0
  395. package/src/zig/tests/renderer_test.zig +1010 -0
  396. package/src/zig/tests/rope-nested_test.zig +712 -0
  397. package/src/zig/tests/rope_fuzz_test.zig +238 -0
  398. package/src/zig/tests/rope_test.zig +2362 -0
  399. package/src/zig/tests/segment-merge.test.zig +148 -0
  400. package/src/zig/tests/syntax-style_test.zig +557 -0
  401. package/src/zig/tests/terminal_test.zig +719 -0
  402. package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
  403. package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
  404. package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
  405. package/src/zig/tests/text-buffer-segment_test.zig +320 -0
  406. package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
  407. package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
  408. package/src/zig/tests/text-buffer-view_test.zig +3649 -0
  409. package/src/zig/tests/text-buffer_test.zig +2191 -0
  410. package/src/zig/tests/unicode-width-map.zon +3909 -0
  411. package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
  412. package/src/zig/tests/utf8_test.zig +4057 -0
  413. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
  414. package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
  415. package/src/zig/tests/word-wrap-editing_test.zig +498 -0
  416. package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
  417. package/src/zig/text-buffer-iterators.zig +499 -0
  418. package/src/zig/text-buffer-segment.zig +404 -0
  419. package/src/zig/text-buffer-view.zig +1371 -0
  420. package/src/zig/text-buffer.zig +1180 -0
  421. package/src/zig/utf8.zig +1948 -0
  422. package/src/zig/utils.zig +9 -0
  423. package/src/zig-structs.ts +261 -0
  424. package/src/zig.ts +3843 -0
  425. package/tsconfig.build.json +22 -0
  426. package/tsconfig.json +28 -0
  427. package/3d/SpriteResourceManager.d.ts +0 -74
  428. package/3d/SpriteUtils.d.ts +0 -13
  429. package/3d/TextureUtils.d.ts +0 -24
  430. package/3d/ThreeRenderable.d.ts +0 -40
  431. package/3d/WGPURenderer.d.ts +0 -61
  432. package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
  433. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
  434. package/3d/animation/SpriteAnimator.d.ts +0 -124
  435. package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
  436. package/3d/canvas.d.ts +0 -44
  437. package/3d/index.d.ts +0 -12
  438. package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
  439. package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
  440. package/3d/physics/physics-interface.d.ts +0 -27
  441. package/3d.d.ts +0 -2
  442. package/3d.js +0 -34042
  443. package/3d.js.map +0 -155
  444. package/LICENSE +0 -21
  445. package/NativeSpanFeed.d.ts +0 -41
  446. package/Renderable.d.ts +0 -334
  447. package/animation/Timeline.d.ts +0 -126
  448. package/ansi.d.ts +0 -13
  449. package/buffer.d.ts +0 -107
  450. package/console.d.ts +0 -143
  451. package/edit-buffer.d.ts +0 -98
  452. package/editor-view.d.ts +0 -73
  453. package/index-e4hzc2j2.js +0 -113
  454. package/index-e4hzc2j2.js.map +0 -10
  455. package/index-nkrr8a4c.js +0 -18415
  456. package/index-nkrr8a4c.js.map +0 -64
  457. package/index-nyw5p3ep.js +0 -12619
  458. package/index-nyw5p3ep.js.map +0 -43
  459. package/index.d.ts +0 -21
  460. package/index.js +0 -430
  461. package/index.js.map +0 -9
  462. package/lib/KeyHandler.d.ts +0 -61
  463. package/lib/RGBA.d.ts +0 -25
  464. package/lib/ascii.font.d.ts +0 -508
  465. package/lib/border.d.ts +0 -49
  466. package/lib/bunfs.d.ts +0 -7
  467. package/lib/clipboard.d.ts +0 -17
  468. package/lib/clock.d.ts +0 -15
  469. package/lib/data-paths.d.ts +0 -26
  470. package/lib/debounce.d.ts +0 -42
  471. package/lib/detect-links.d.ts +0 -6
  472. package/lib/env.d.ts +0 -42
  473. package/lib/extmarks-history.d.ts +0 -17
  474. package/lib/extmarks.d.ts +0 -89
  475. package/lib/hast-styled-text.d.ts +0 -17
  476. package/lib/index.d.ts +0 -21
  477. package/lib/keymapping.d.ts +0 -25
  478. package/lib/objects-in-viewport.d.ts +0 -24
  479. package/lib/output.capture.d.ts +0 -24
  480. package/lib/parse.keypress-kitty.d.ts +0 -2
  481. package/lib/parse.keypress.d.ts +0 -26
  482. package/lib/parse.mouse.d.ts +0 -30
  483. package/lib/paste.d.ts +0 -7
  484. package/lib/queue.d.ts +0 -15
  485. package/lib/renderable.validations.d.ts +0 -12
  486. package/lib/scroll-acceleration.d.ts +0 -43
  487. package/lib/selection.d.ts +0 -63
  488. package/lib/singleton.d.ts +0 -7
  489. package/lib/stdin-parser.d.ts +0 -76
  490. package/lib/styled-text.d.ts +0 -63
  491. package/lib/terminal-capability-detection.d.ts +0 -30
  492. package/lib/terminal-palette.d.ts +0 -50
  493. package/lib/tree-sitter/assets/update.d.ts +0 -11
  494. package/lib/tree-sitter/client.d.ts +0 -47
  495. package/lib/tree-sitter/default-parsers.d.ts +0 -2
  496. package/lib/tree-sitter/download-utils.d.ts +0 -21
  497. package/lib/tree-sitter/index.d.ts +0 -8
  498. package/lib/tree-sitter/parser.worker.d.ts +0 -1
  499. package/lib/tree-sitter/parsers-config.d.ts +0 -38
  500. package/lib/tree-sitter/resolve-ft.d.ts +0 -2
  501. package/lib/tree-sitter/types.d.ts +0 -81
  502. package/lib/tree-sitter-styled-text.d.ts +0 -14
  503. package/lib/validate-dir-name.d.ts +0 -1
  504. package/lib/yoga.options.d.ts +0 -32
  505. package/parser.worker.js +0 -869
  506. package/parser.worker.js.map +0 -12
  507. package/plugins/core-slot.d.ts +0 -72
  508. package/plugins/registry.d.ts +0 -38
  509. package/plugins/types.d.ts +0 -34
  510. package/post/filters.d.ts +0 -105
  511. package/renderables/ASCIIFont.d.ts +0 -52
  512. package/renderables/Box.d.ts +0 -72
  513. package/renderables/Code.d.ts +0 -78
  514. package/renderables/Diff.d.ts +0 -142
  515. package/renderables/EditBufferRenderable.d.ts +0 -162
  516. package/renderables/FrameBuffer.d.ts +0 -16
  517. package/renderables/Input.d.ts +0 -67
  518. package/renderables/LineNumberRenderable.d.ts +0 -74
  519. package/renderables/Markdown.d.ts +0 -173
  520. package/renderables/ScrollBar.d.ts +0 -77
  521. package/renderables/ScrollBox.d.ts +0 -124
  522. package/renderables/Select.d.ts +0 -115
  523. package/renderables/Slider.d.ts +0 -44
  524. package/renderables/TabSelect.d.ts +0 -96
  525. package/renderables/Text.d.ts +0 -36
  526. package/renderables/TextBufferRenderable.d.ts +0 -105
  527. package/renderables/TextNode.d.ts +0 -91
  528. package/renderables/TextTable.d.ts +0 -140
  529. package/renderables/Textarea.d.ts +0 -114
  530. package/renderables/TimeToFirstDraw.d.ts +0 -24
  531. package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
  532. package/renderables/composition/VRenderable.d.ts +0 -16
  533. package/renderables/composition/constructs.d.ts +0 -35
  534. package/renderables/composition/vnode.d.ts +0 -46
  535. package/renderables/index.d.ts +0 -22
  536. package/renderables/markdown-parser.d.ts +0 -10
  537. package/renderer.d.ts +0 -388
  538. package/runtime-plugin-support.d.ts +0 -3
  539. package/runtime-plugin-support.js +0 -29
  540. package/runtime-plugin-support.js.map +0 -10
  541. package/runtime-plugin.d.ts +0 -11
  542. package/runtime-plugin.js +0 -16
  543. package/runtime-plugin.js.map +0 -9
  544. package/syntax-style.d.ts +0 -54
  545. package/testing/manual-clock.d.ts +0 -16
  546. package/testing/mock-keys.d.ts +0 -81
  547. package/testing/mock-mouse.d.ts +0 -38
  548. package/testing/mock-tree-sitter-client.d.ts +0 -23
  549. package/testing/spy.d.ts +0 -7
  550. package/testing/test-recorder.d.ts +0 -61
  551. package/testing/test-renderer.d.ts +0 -23
  552. package/testing.d.ts +0 -6
  553. package/testing.js +0 -675
  554. package/testing.js.map +0 -15
  555. package/text-buffer-view.d.ts +0 -42
  556. package/text-buffer.d.ts +0 -67
  557. package/types.d.ts +0 -131
  558. package/utils.d.ts +0 -14
  559. package/zig-structs.d.ts +0 -155
  560. package/zig.d.ts +0 -351
  561. /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
  562. /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
  563. /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
  564. /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
  565. /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
  566. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
  567. /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  568. /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
  569. /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
  570. /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
  571. /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
package/src/console.ts ADDED
@@ -0,0 +1,1255 @@
1
+ import { EventEmitter } from "events"
2
+ import { Console } from "node:console"
3
+ import fs from "node:fs"
4
+ import path from "node:path"
5
+ import util from "node:util"
6
+ import type { CliRenderer, ColorInput, MouseEvent } from "./index.js"
7
+ import { OptimizedBuffer } from "./buffer.js"
8
+ import { type Clock, SystemClock } from "./lib/clock.js"
9
+ import { Capture, CapturedWritableStream } from "./lib/output.capture.js"
10
+ import { parseColor, RGBA } from "./lib/RGBA.js"
11
+ import { singleton } from "./lib/singleton.js"
12
+ import { env, registerEnvVar } from "./lib/env.js"
13
+ import type { KeyEvent } from "./lib/KeyHandler.js"
14
+ import {
15
+ type KeyBinding as BaseKeyBinding,
16
+ mergeKeyBindings,
17
+ getKeyBindingKey,
18
+ buildKeyBindingsMap,
19
+ type KeyAliasMap,
20
+ defaultKeyAliases,
21
+ mergeKeyAliases,
22
+ keyBindingToString,
23
+ } from "./lib/keymapping.js"
24
+
25
+ interface CallerInfo {
26
+ functionName: string
27
+ fullPath: string
28
+ fileName: string
29
+ lineNumber: number
30
+ columnNumber: number
31
+ }
32
+
33
+ function getCallerInfo(): CallerInfo | null {
34
+ const err = new Error()
35
+ const stackLines = err.stack?.split("\n").slice(5) || []
36
+ if (!stackLines.length) return null
37
+
38
+ const callerLine = stackLines[0].trim()
39
+
40
+ const regex = /at\s+(?:([\w$.<>]+)\s+\()?((?:\/|[A-Za-z]:\\)[^:]+):(\d+):(\d+)\)?/
41
+ const match = callerLine.match(regex)
42
+
43
+ if (!match) return null
44
+
45
+ // Extract details from the match.
46
+ const functionName = match[1] || "<anonymous>"
47
+ const fullPath = match[2]
48
+ const fileName = fullPath.split(/[\\/]/).pop() || "<unknown>"
49
+ const lineNumber = parseInt(match[3], 10) || 0
50
+ const columnNumber = parseInt(match[4], 10) || 0
51
+
52
+ return { functionName, fullPath, fileName, lineNumber, columnNumber }
53
+ }
54
+
55
+ enum LogLevel {
56
+ LOG = "LOG",
57
+ INFO = "INFO",
58
+ WARN = "WARN",
59
+ ERROR = "ERROR",
60
+ DEBUG = "DEBUG",
61
+ }
62
+
63
+ export const capture = singleton("ConsoleCapture", () => new Capture())
64
+
65
+ registerEnvVar({
66
+ name: "OTUI_USE_CONSOLE",
67
+ description: "Whether to use the console. Will not capture console output if set to false.",
68
+ type: "boolean",
69
+ default: true,
70
+ })
71
+
72
+ registerEnvVar({
73
+ name: "SHOW_CONSOLE",
74
+ description: "Show the console at startup if set to true.",
75
+ type: "boolean",
76
+ default: false,
77
+ })
78
+
79
+ class TerminalConsoleCache extends EventEmitter {
80
+ private _cachedLogs: [Date, LogLevel, any[], CallerInfo | null][] = []
81
+ private readonly MAX_CACHE_SIZE = 1000
82
+ private _collectCallerInfo: boolean = false
83
+ private _cachingEnabled: boolean = true
84
+ private _originalConsole: typeof console | null = null
85
+
86
+ get cachedLogs(): [Date, LogLevel, any[], CallerInfo | null][] {
87
+ return this._cachedLogs
88
+ }
89
+
90
+ constructor() {
91
+ super()
92
+
93
+ // Note: Console activation will be handled by the renderer when needed
94
+ // Don't activate on import to avoid hiding console.log globally
95
+ }
96
+
97
+ public activate(): void {
98
+ if (!this._originalConsole) {
99
+ this._originalConsole = global.console
100
+ }
101
+ this.setupConsoleCapture()
102
+ this.overrideConsoleMethods()
103
+ }
104
+
105
+ private setupConsoleCapture(): void {
106
+ if (!env.OTUI_USE_CONSOLE) return
107
+
108
+ const mockStdout = new CapturedWritableStream("stdout", capture)
109
+ const mockStderr = new CapturedWritableStream("stderr", capture)
110
+
111
+ // TODO: The Console constructor doesn't return a full Console interface implementation,
112
+ // it only provides a subset of methods (log, info, warn, error, debug, etc.).
113
+ // TypeScript's Console interface requires all methods (assert, clear, count, etc.).
114
+ // Using 'as any' as a workaround since we override the methods we use immediately after.
115
+ global.console = new Console({
116
+ stdout: mockStdout,
117
+ stderr: mockStderr,
118
+ colorMode: true,
119
+ inspectOptions: {
120
+ compact: false,
121
+ breakLength: 80,
122
+ depth: 2,
123
+ },
124
+ }) as any
125
+ }
126
+
127
+ private overrideConsoleMethods(): void {
128
+ console.log = (...args: any[]) => {
129
+ this.appendToConsole(LogLevel.LOG, ...args)
130
+ }
131
+
132
+ console.info = (...args: any[]) => {
133
+ this.appendToConsole(LogLevel.INFO, ...args)
134
+ }
135
+
136
+ console.warn = (...args: any[]) => {
137
+ this.appendToConsole(LogLevel.WARN, ...args)
138
+ }
139
+
140
+ console.error = (...args: any[]) => {
141
+ this.appendToConsole(LogLevel.ERROR, ...args)
142
+ }
143
+
144
+ console.debug = (...args: any[]) => {
145
+ this.appendToConsole(LogLevel.DEBUG, ...args)
146
+ }
147
+ }
148
+
149
+ public setCollectCallerInfo(enabled: boolean): void {
150
+ this._collectCallerInfo = enabled
151
+ }
152
+
153
+ public clearConsole(): void {
154
+ this._cachedLogs = []
155
+ }
156
+
157
+ public setCachingEnabled(enabled: boolean): void {
158
+ this._cachingEnabled = enabled
159
+ }
160
+
161
+ public deactivate(): void {
162
+ this.restoreOriginalConsole()
163
+ }
164
+
165
+ private restoreOriginalConsole(): void {
166
+ if (this._originalConsole) {
167
+ global.console = this._originalConsole
168
+ }
169
+
170
+ this.setupConsoleCapture()
171
+ }
172
+
173
+ public addLogEntry(level: LogLevel, ...args: any[]) {
174
+ const callerInfo = this._collectCallerInfo ? getCallerInfo() : null
175
+ const logEntry: [Date, LogLevel, any[], CallerInfo | null] = [new Date(), level, args, callerInfo]
176
+
177
+ if (this._cachingEnabled) {
178
+ if (this._cachedLogs.length >= this.MAX_CACHE_SIZE) {
179
+ this._cachedLogs.shift()
180
+ }
181
+ this._cachedLogs.push(logEntry)
182
+ }
183
+
184
+ return logEntry
185
+ }
186
+
187
+ private appendToConsole(level: LogLevel, ...args: any[]): void {
188
+ if (this._cachedLogs.length >= this.MAX_CACHE_SIZE) {
189
+ this._cachedLogs.shift()
190
+ }
191
+ const entry = this.addLogEntry(level, ...args)
192
+ this.emit("entry", entry)
193
+ }
194
+
195
+ public destroy(): void {
196
+ this.deactivate()
197
+ }
198
+ }
199
+
200
+ const terminalConsoleCache = singleton("TerminalConsoleCache", () => {
201
+ const terminalConsoleCache = new TerminalConsoleCache()
202
+ process.on("exit", () => {
203
+ terminalConsoleCache.destroy()
204
+ })
205
+ return terminalConsoleCache
206
+ })
207
+
208
+ export enum ConsolePosition {
209
+ TOP = "top",
210
+ BOTTOM = "bottom",
211
+ LEFT = "left",
212
+ RIGHT = "right",
213
+ }
214
+
215
+ interface ConsoleSelection {
216
+ startLine: number
217
+ startCol: number
218
+ endLine: number
219
+ endCol: number
220
+ }
221
+
222
+ export type ConsoleAction =
223
+ | "scroll-up"
224
+ | "scroll-down"
225
+ | "scroll-to-top"
226
+ | "scroll-to-bottom"
227
+ | "position-previous"
228
+ | "position-next"
229
+ | "size-increase"
230
+ | "size-decrease"
231
+ | "save-logs"
232
+ | "copy-selection"
233
+
234
+ export type ConsoleKeyBinding = BaseKeyBinding<ConsoleAction>
235
+
236
+ const defaultConsoleKeybindings: ConsoleKeyBinding[] = [
237
+ { name: "up", action: "scroll-up" },
238
+ { name: "down", action: "scroll-down" },
239
+ { name: "up", shift: true, action: "scroll-to-top" },
240
+ { name: "down", shift: true, action: "scroll-to-bottom" },
241
+ { name: "p", ctrl: true, action: "position-previous" },
242
+ { name: "o", ctrl: true, action: "position-next" },
243
+ { name: "+", action: "size-increase" },
244
+ { name: "=", shift: true, action: "size-increase" },
245
+ { name: "-", action: "size-decrease" },
246
+ { name: "s", ctrl: true, action: "save-logs" },
247
+ { name: "c", ctrl: true, shift: true, action: "copy-selection" },
248
+ ]
249
+
250
+ export interface ConsoleOptions {
251
+ position?: ConsolePosition
252
+ sizePercent?: number
253
+ zIndex?: number
254
+ colorInfo?: ColorInput
255
+ colorWarn?: ColorInput
256
+ colorError?: ColorInput
257
+ colorDebug?: ColorInput
258
+ colorDefault?: ColorInput
259
+ backgroundColor?: ColorInput
260
+ startInDebugMode?: boolean
261
+ title?: string
262
+ titleBarColor?: ColorInput
263
+ titleBarTextColor?: ColorInput
264
+ cursorColor?: ColorInput
265
+ maxStoredLogs?: number
266
+ maxDisplayLines?: number
267
+ onCopySelection?: (text: string) => void
268
+ keyBindings?: ConsoleKeyBinding[]
269
+ keyAliasMap?: KeyAliasMap
270
+ selectionColor?: ColorInput
271
+ copyButtonColor?: ColorInput
272
+ clock?: Clock
273
+ }
274
+
275
+ const DEFAULT_CONSOLE_OPTIONS: Required<
276
+ Omit<ConsoleOptions, "onCopySelection" | "keyBindings" | "keyAliasMap" | "clock">
277
+ > & {
278
+ onCopySelection?: (text: string) => void
279
+ keyBindings?: ConsoleKeyBinding[]
280
+ keyAliasMap?: KeyAliasMap
281
+ } = {
282
+ position: ConsolePosition.BOTTOM,
283
+ sizePercent: 30,
284
+ zIndex: Infinity,
285
+ colorInfo: "#00FFFF", // Cyan
286
+ colorWarn: "#FFFF00", // Yellow
287
+ colorError: "#FF0000", // Red
288
+ colorDebug: "#808080", // Gray
289
+ colorDefault: "#FFFFFF", // White
290
+ backgroundColor: RGBA.fromValues(0.1, 0.1, 0.1, 0.7),
291
+ startInDebugMode: false,
292
+ title: "Console",
293
+ titleBarColor: RGBA.fromValues(0.05, 0.05, 0.05, 0.7),
294
+ titleBarTextColor: "#FFFFFF",
295
+ cursorColor: "#00A0FF",
296
+ maxStoredLogs: 2000,
297
+ maxDisplayLines: 3000,
298
+ onCopySelection: undefined,
299
+ keyBindings: undefined,
300
+ keyAliasMap: undefined,
301
+ selectionColor: RGBA.fromValues(0.3, 0.5, 0.8, 0.5),
302
+ copyButtonColor: "#00A0FF",
303
+ }
304
+
305
+ const INDENT_WIDTH = 2
306
+
307
+ interface DisplayLine {
308
+ text: string
309
+ level: LogLevel
310
+ indent: boolean
311
+ }
312
+
313
+ export class TerminalConsole extends EventEmitter {
314
+ private isVisible: boolean = false
315
+ private isFocused: boolean = false
316
+ private renderer: CliRenderer
317
+ private keyHandler: (event: KeyEvent) => void
318
+ private options: Required<Omit<ConsoleOptions, "onCopySelection" | "keyBindings" | "keyAliasMap" | "clock">> & {
319
+ onCopySelection?: (text: string) => void
320
+ keyBindings?: ConsoleKeyBinding[]
321
+ keyAliasMap?: KeyAliasMap
322
+ }
323
+ private _debugModeEnabled: boolean = false
324
+
325
+ private frameBuffer: OptimizedBuffer | null = null
326
+ private consoleX: number = 0
327
+ private consoleY: number = 0
328
+ private consoleWidth: number = 0
329
+ private consoleHeight: number = 0
330
+ private scrollTopIndex: number = 0
331
+ private isScrolledToBottom: boolean = true
332
+ private currentLineIndex: number = 0
333
+ private _displayLines: DisplayLine[] = []
334
+ private _allLogEntries: [Date, LogLevel, any[], CallerInfo | null][] = []
335
+ private _needsFrameBufferUpdate: boolean = false
336
+ private _entryListener: (logEntry: [Date, LogLevel, any[], CallerInfo | null]) => void
337
+
338
+ private _selectionStart: { line: number; col: number } | null = null
339
+ private _selectionEnd: { line: number; col: number } | null = null
340
+ private _isDragging: boolean = false
341
+ private _copyButtonBounds: { x: number; y: number; width: number; height: number } = {
342
+ x: 0,
343
+ y: 0,
344
+ width: 0,
345
+ height: 0,
346
+ }
347
+ private _autoScrollInterval: ReturnType<Clock["setInterval"]> | null = null
348
+ private readonly clock: Clock
349
+
350
+ private _keyBindingsMap: Map<string, ConsoleAction>
351
+ private _keyAliasMap: KeyAliasMap
352
+ private _keyBindings: ConsoleKeyBinding[]
353
+ private _mergedKeyBindings: ConsoleKeyBinding[]
354
+ private _actionHandlers: Map<ConsoleAction, () => boolean>
355
+
356
+ private markNeedsRerender(): void {
357
+ this._needsFrameBufferUpdate = true
358
+ this.renderer.requestRender()
359
+ }
360
+
361
+ private getCopyButtonLabel(): string {
362
+ const copyBindings = this._mergedKeyBindings.filter((b) => b.action === "copy-selection")
363
+ const copyBinding = copyBindings[copyBindings.length - 1]
364
+ if (copyBinding) {
365
+ const shortcut = keyBindingToString(copyBinding)
366
+ return `[Copy (${shortcut})]`
367
+ }
368
+ return "[Copy]"
369
+ }
370
+
371
+ private _rgbaInfo: RGBA
372
+ private _rgbaWarn: RGBA
373
+ private _rgbaError: RGBA
374
+ private _rgbaDebug: RGBA
375
+ private _rgbaDefault: RGBA
376
+ private backgroundColor: RGBA
377
+ private _rgbaTitleBar: RGBA
378
+ private _rgbaTitleBarText: RGBA
379
+ private _title: string
380
+ private _rgbaCursor: RGBA
381
+ private _rgbaSelection: RGBA
382
+ private _rgbaCopyButton: RGBA
383
+
384
+ private _positions: ConsolePosition[] = [
385
+ ConsolePosition.TOP,
386
+ ConsolePosition.RIGHT,
387
+ ConsolePosition.BOTTOM,
388
+ ConsolePosition.LEFT,
389
+ ]
390
+
391
+ constructor(renderer: CliRenderer, options: ConsoleOptions = {}) {
392
+ super()
393
+ this.renderer = renderer
394
+ this.clock = options.clock ?? new SystemClock()
395
+ this.options = { ...DEFAULT_CONSOLE_OPTIONS, ...options }
396
+ this.keyHandler = this.handleKeyPress.bind(this)
397
+ this._debugModeEnabled = this.options.startInDebugMode
398
+ terminalConsoleCache.setCollectCallerInfo(this._debugModeEnabled)
399
+
400
+ this._rgbaInfo = parseColor(this.options.colorInfo)
401
+ this._rgbaWarn = parseColor(this.options.colorWarn)
402
+ this._rgbaError = parseColor(this.options.colorError)
403
+ this._rgbaDebug = parseColor(this.options.colorDebug)
404
+ this._rgbaDefault = parseColor(this.options.colorDefault)
405
+ this.backgroundColor = parseColor(this.options.backgroundColor)
406
+ this._rgbaTitleBar = parseColor(this.options.titleBarColor)
407
+ this._rgbaTitleBarText = parseColor(this.options.titleBarTextColor || this.options.colorDefault)
408
+ this._title = this.options.title
409
+ this._rgbaCursor = parseColor(this.options.cursorColor)
410
+ this._rgbaSelection = parseColor(this.options.selectionColor)
411
+ this._rgbaCopyButton = parseColor(this.options.copyButtonColor)
412
+
413
+ this._keyAliasMap = mergeKeyAliases(defaultKeyAliases, options.keyAliasMap || {})
414
+ this._keyBindings = options.keyBindings || []
415
+ this._mergedKeyBindings = mergeKeyBindings(defaultConsoleKeybindings, this._keyBindings)
416
+ this._keyBindingsMap = buildKeyBindingsMap(this._mergedKeyBindings, this._keyAliasMap)
417
+ this._actionHandlers = this.buildActionHandlers()
418
+
419
+ this._updateConsoleDimensions()
420
+ this._scrollToBottom(true)
421
+
422
+ this._entryListener = (logEntry: [Date, LogLevel, any[], CallerInfo | null]) => {
423
+ this._handleNewLog(logEntry)
424
+ }
425
+ terminalConsoleCache.on("entry", this._entryListener)
426
+
427
+ if (env.SHOW_CONSOLE) {
428
+ this.show()
429
+ }
430
+ }
431
+
432
+ private buildActionHandlers(): Map<ConsoleAction, () => boolean> {
433
+ return new Map([
434
+ ["scroll-up", () => this.scrollUp()],
435
+ ["scroll-down", () => this.scrollDown()],
436
+ ["scroll-to-top", () => this.scrollToTop()],
437
+ ["scroll-to-bottom", () => this.scrollToBottomAction()],
438
+ ["position-previous", () => this.positionPrevious()],
439
+ ["position-next", () => this.positionNext()],
440
+ ["size-increase", () => this.sizeIncrease()],
441
+ ["size-decrease", () => this.sizeDecrease()],
442
+ ["save-logs", () => this.saveLogsAction()],
443
+ ["copy-selection", () => this.triggerCopyAction()],
444
+ ])
445
+ }
446
+
447
+ public activate(): void {
448
+ terminalConsoleCache.activate()
449
+ }
450
+
451
+ public deactivate(): void {
452
+ terminalConsoleCache.deactivate()
453
+ }
454
+
455
+ // Handles a single new log entry *while the console is visible*
456
+ private _handleNewLog(logEntry: [Date, LogLevel, any[], CallerInfo | null]): void {
457
+ if (!this.isVisible) return
458
+
459
+ this._allLogEntries.push(logEntry)
460
+
461
+ if (this._allLogEntries.length > this.options.maxStoredLogs) {
462
+ this._allLogEntries.splice(0, this._allLogEntries.length - this.options.maxStoredLogs)
463
+ }
464
+
465
+ const newDisplayLines = this._processLogEntry(logEntry)
466
+ this._displayLines.push(...newDisplayLines)
467
+
468
+ if (this._displayLines.length > this.options.maxDisplayLines) {
469
+ this._displayLines.splice(0, this._displayLines.length - this.options.maxDisplayLines)
470
+ const linesRemoved = this._displayLines.length - this.options.maxDisplayLines
471
+ this.scrollTopIndex = Math.max(0, this.scrollTopIndex - linesRemoved)
472
+ }
473
+
474
+ if (this.isScrolledToBottom) {
475
+ this._scrollToBottom()
476
+ }
477
+ this.markNeedsRerender()
478
+ }
479
+
480
+ private _updateConsoleDimensions(termWidth?: number, termHeight?: number): void {
481
+ const width = termWidth ?? this.renderer.width
482
+ const height = termHeight ?? this.renderer.height
483
+ const sizePercent = this.options.sizePercent / 100
484
+
485
+ switch (this.options.position) {
486
+ case ConsolePosition.TOP:
487
+ this.consoleX = 0
488
+ this.consoleY = 0
489
+ this.consoleWidth = width
490
+ this.consoleHeight = Math.max(1, Math.floor(height * sizePercent))
491
+ break
492
+ case ConsolePosition.BOTTOM:
493
+ this.consoleHeight = Math.max(1, Math.floor(height * sizePercent))
494
+ this.consoleWidth = width
495
+ this.consoleX = 0
496
+ this.consoleY = height - this.consoleHeight
497
+ break
498
+ case ConsolePosition.LEFT:
499
+ this.consoleWidth = Math.max(1, Math.floor(width * sizePercent))
500
+ this.consoleHeight = height
501
+ this.consoleX = 0
502
+ this.consoleY = 0
503
+ break
504
+ case ConsolePosition.RIGHT:
505
+ this.consoleWidth = Math.max(1, Math.floor(width * sizePercent))
506
+ this.consoleHeight = height
507
+ this.consoleY = 0
508
+ this.consoleX = width - this.consoleWidth
509
+ break
510
+ }
511
+ this.currentLineIndex = Math.max(0, Math.min(this.currentLineIndex, this.consoleHeight - 1))
512
+ }
513
+
514
+ private handleKeyPress(event: KeyEvent): void {
515
+ if (event.name === "escape") {
516
+ this.blur()
517
+ return
518
+ }
519
+
520
+ const bindingKey = getKeyBindingKey({
521
+ name: event.name,
522
+ ctrl: event.ctrl,
523
+ shift: event.shift,
524
+ meta: event.meta,
525
+ super: event.super,
526
+ action: "scroll-up" as ConsoleAction,
527
+ })
528
+
529
+ const action = this._keyBindingsMap.get(bindingKey)
530
+
531
+ if (action) {
532
+ const handler = this._actionHandlers.get(action)
533
+ if (handler) {
534
+ handler()
535
+ return
536
+ }
537
+ }
538
+ }
539
+
540
+ private scrollUp(): boolean {
541
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
542
+
543
+ if (this.currentLineIndex > 0) {
544
+ this.currentLineIndex--
545
+ this.markNeedsRerender()
546
+ } else if (this.scrollTopIndex > 0) {
547
+ this.scrollTopIndex--
548
+ this.isScrolledToBottom = false
549
+ this.markNeedsRerender()
550
+ }
551
+ return true
552
+ }
553
+
554
+ private scrollDown(): boolean {
555
+ const displayLineCount = this._displayLines.length
556
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
557
+ const maxScrollTop = Math.max(0, displayLineCount - logAreaHeight)
558
+ const canCursorMoveDown =
559
+ this.currentLineIndex < logAreaHeight - 1 && this.scrollTopIndex + this.currentLineIndex < displayLineCount - 1
560
+
561
+ if (canCursorMoveDown) {
562
+ this.currentLineIndex++
563
+ this.markNeedsRerender()
564
+ } else if (this.scrollTopIndex < maxScrollTop) {
565
+ this.scrollTopIndex++
566
+ this.isScrolledToBottom = this.scrollTopIndex === maxScrollTop
567
+ this.markNeedsRerender()
568
+ }
569
+ return true
570
+ }
571
+
572
+ private scrollToTop(): boolean {
573
+ if (this.scrollTopIndex > 0 || this.currentLineIndex > 0) {
574
+ this.scrollTopIndex = 0
575
+ this.currentLineIndex = 0
576
+ this.isScrolledToBottom = this._displayLines.length <= Math.max(1, this.consoleHeight - 1)
577
+ this.markNeedsRerender()
578
+ }
579
+ return true
580
+ }
581
+
582
+ private scrollToBottomAction(): boolean {
583
+ const logAreaHeightForScroll = Math.max(1, this.consoleHeight - 1)
584
+ const maxScrollPossible = Math.max(0, this._displayLines.length - logAreaHeightForScroll)
585
+ if (this.scrollTopIndex < maxScrollPossible || !this.isScrolledToBottom) {
586
+ this._scrollToBottom(true)
587
+ this.markNeedsRerender()
588
+ }
589
+ return true
590
+ }
591
+
592
+ private positionPrevious(): boolean {
593
+ const currentPositionIndex = this._positions.indexOf(this.options.position)
594
+ const prevIndex = (currentPositionIndex - 1 + this._positions.length) % this._positions.length
595
+ this.options.position = this._positions[prevIndex]
596
+ this.resize(this.renderer.width, this.renderer.height)
597
+ return true
598
+ }
599
+
600
+ private positionNext(): boolean {
601
+ const currentPositionIndex = this._positions.indexOf(this.options.position)
602
+ const nextIndex = (currentPositionIndex + 1) % this._positions.length
603
+ this.options.position = this._positions[nextIndex]
604
+ this.resize(this.renderer.width, this.renderer.height)
605
+ return true
606
+ }
607
+
608
+ private sizeIncrease(): boolean {
609
+ this.options.sizePercent = Math.min(100, this.options.sizePercent + 5)
610
+ this.resize(this.renderer.width, this.renderer.height)
611
+ return true
612
+ }
613
+
614
+ private sizeDecrease(): boolean {
615
+ this.options.sizePercent = Math.max(10, this.options.sizePercent - 5)
616
+ this.resize(this.renderer.width, this.renderer.height)
617
+ return true
618
+ }
619
+
620
+ private saveLogsAction(): boolean {
621
+ this.saveLogsToFile()
622
+ return true
623
+ }
624
+
625
+ private triggerCopyAction(): boolean {
626
+ this.triggerCopy()
627
+ return true
628
+ }
629
+
630
+ private attachStdin(): void {
631
+ if (this.isFocused) return
632
+ this.renderer.keyInput.on("keypress", this.keyHandler)
633
+ this.isFocused = true
634
+ }
635
+
636
+ private detachStdin(): void {
637
+ if (!this.isFocused) return
638
+ this.renderer.keyInput.off("keypress", this.keyHandler)
639
+ this.isFocused = false
640
+ }
641
+
642
+ private formatTimestamp(date: Date): string {
643
+ return new Intl.DateTimeFormat("en-US", {
644
+ hour: "2-digit",
645
+ minute: "2-digit",
646
+ second: "2-digit",
647
+ hour12: false,
648
+ }).format(date)
649
+ }
650
+
651
+ private formatArguments(args: any[]): string {
652
+ return args
653
+ .map((arg) => {
654
+ if (arg instanceof Error) {
655
+ const errorProps = arg
656
+ return `Error: ${errorProps.message}\n` + (errorProps.stack ? `${errorProps.stack}\n` : "")
657
+ }
658
+ if (typeof arg === "object" && arg !== null) {
659
+ try {
660
+ return util.inspect(arg, { depth: 2 })
661
+ } catch (e) {
662
+ return String(arg)
663
+ }
664
+ }
665
+ try {
666
+ return util.inspect(arg, { depth: 2 })
667
+ } catch (e) {
668
+ return String(arg)
669
+ }
670
+ })
671
+ .join(" ")
672
+ }
673
+
674
+ public resize(width: number, height: number): void {
675
+ this._updateConsoleDimensions(width, height)
676
+
677
+ if (this.frameBuffer) {
678
+ this.frameBuffer.resize(this.consoleWidth, this.consoleHeight)
679
+
680
+ const displayLineCount = this._displayLines.length
681
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
682
+ const maxScrollTop = Math.max(0, displayLineCount - logAreaHeight)
683
+ this.scrollTopIndex = Math.min(this.scrollTopIndex, maxScrollTop)
684
+ this.isScrolledToBottom = this.scrollTopIndex === maxScrollTop
685
+ const visibleLineCount = Math.min(logAreaHeight, displayLineCount - this.scrollTopIndex)
686
+ this.currentLineIndex = Math.max(0, Math.min(this.currentLineIndex, visibleLineCount - 1))
687
+
688
+ if (this.isVisible) {
689
+ this.markNeedsRerender()
690
+ }
691
+ }
692
+ }
693
+
694
+ public clear(): void {
695
+ terminalConsoleCache.clearConsole()
696
+ this._allLogEntries = []
697
+ this._displayLines = []
698
+ this.markNeedsRerender()
699
+ }
700
+
701
+ public toggle(): void {
702
+ if (this.isVisible) {
703
+ if (this.isFocused) {
704
+ this.hide()
705
+ } else {
706
+ this.focus()
707
+ }
708
+ } else {
709
+ this.show()
710
+ }
711
+ if (!this.renderer.isRunning) {
712
+ this.renderer.requestRender()
713
+ }
714
+ }
715
+
716
+ public focus(): void {
717
+ this.attachStdin()
718
+ this._scrollToBottom(true)
719
+ this.markNeedsRerender()
720
+ }
721
+
722
+ public blur(): void {
723
+ this.detachStdin()
724
+ this.markNeedsRerender()
725
+ }
726
+
727
+ public show(): void {
728
+ if (!this.isVisible) {
729
+ this.isVisible = true
730
+ this._processCachedLogs()
731
+ terminalConsoleCache.setCachingEnabled(false)
732
+
733
+ if (!this.frameBuffer) {
734
+ this.frameBuffer = OptimizedBuffer.create(this.consoleWidth, this.consoleHeight, this.renderer.widthMethod, {
735
+ respectAlpha: this.backgroundColor.a < 1,
736
+ id: "console framebuffer",
737
+ })
738
+ }
739
+ const logCount = terminalConsoleCache.cachedLogs.length
740
+ const visibleLogLines = Math.min(this.consoleHeight, logCount)
741
+ this.currentLineIndex = Math.max(0, visibleLogLines - 1)
742
+ this.scrollTopIndex = 0
743
+ this._scrollToBottom(true)
744
+
745
+ this.focus()
746
+ this.markNeedsRerender()
747
+ }
748
+ }
749
+
750
+ public hide(): void {
751
+ if (this.isVisible) {
752
+ this.isVisible = false
753
+ this.blur()
754
+ terminalConsoleCache.setCachingEnabled(true)
755
+ }
756
+ }
757
+
758
+ public destroy(): void {
759
+ this.stopAutoScroll()
760
+ this.hide()
761
+ this.deactivate()
762
+ terminalConsoleCache.off("entry", this._entryListener)
763
+ }
764
+
765
+ public getCachedLogs(): string {
766
+ return terminalConsoleCache.cachedLogs
767
+ .map((logEntry) => logEntry[0].toISOString() + " " + logEntry.slice(1).join(" "))
768
+ .join("\n")
769
+ }
770
+
771
+ private updateFrameBuffer(): void {
772
+ if (!this.frameBuffer) return
773
+
774
+ this.frameBuffer.clear(this.backgroundColor)
775
+
776
+ const displayLines = this._displayLines
777
+ const displayLineCount = displayLines.length
778
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
779
+
780
+ // --- Draw Title Bar ---
781
+ this.frameBuffer.fillRect(0, 0, this.consoleWidth, 1, this._rgbaTitleBar)
782
+ const dynamicTitle = `${this._title}${this.isFocused ? " (Focused)" : ""}`
783
+ const titleX = Math.max(0, Math.floor((this.consoleWidth - dynamicTitle.length) / 2))
784
+ this.frameBuffer.drawText(dynamicTitle, titleX, 0, this._rgbaTitleBarText, this._rgbaTitleBar)
785
+
786
+ // --- Draw [Copy] Button ---
787
+ const copyLabel = this.getCopyButtonLabel()
788
+ const copyButtonX = this.consoleWidth - copyLabel.length - 1
789
+ if (copyButtonX >= 0) {
790
+ const copyButtonEnabled = this.hasSelection()
791
+ const disabledColor = RGBA.fromInts(100, 100, 100, 255)
792
+ const copyColor = copyButtonEnabled ? this._rgbaCopyButton : disabledColor
793
+ this.frameBuffer.drawText(copyLabel, copyButtonX, 0, copyColor, this._rgbaTitleBar)
794
+ this._copyButtonBounds = { x: copyButtonX, y: 0, width: copyLabel.length, height: 1 }
795
+ } else {
796
+ this._copyButtonBounds = { x: -1, y: -1, width: 0, height: 0 }
797
+ }
798
+
799
+ const startIndex = this.scrollTopIndex
800
+ const endIndex = Math.min(startIndex + logAreaHeight, displayLineCount)
801
+ const visibleDisplayLines = displayLines.slice(startIndex, endIndex)
802
+
803
+ let lineY = 1
804
+ for (let i = 0; i < visibleDisplayLines.length; i++) {
805
+ if (lineY >= this.consoleHeight) break
806
+
807
+ const displayLine = visibleDisplayLines[i]
808
+ const absoluteLineIndex = startIndex + i
809
+
810
+ let levelColor = this._rgbaDefault
811
+ switch (displayLine.level) {
812
+ case LogLevel.INFO:
813
+ levelColor = this._rgbaInfo
814
+ break
815
+ case LogLevel.WARN:
816
+ levelColor = this._rgbaWarn
817
+ break
818
+ case LogLevel.ERROR:
819
+ levelColor = this._rgbaError
820
+ break
821
+ case LogLevel.DEBUG:
822
+ levelColor = this._rgbaDebug
823
+ break
824
+ }
825
+
826
+ const linePrefix = displayLine.indent ? " ".repeat(INDENT_WIDTH) : ""
827
+ const textToDraw = displayLine.text
828
+ const textAvailableWidth = this.consoleWidth - 1 - (displayLine.indent ? INDENT_WIDTH : 0)
829
+ const showCursor = this.isFocused && lineY - 1 === this.currentLineIndex
830
+
831
+ if (showCursor) {
832
+ this.frameBuffer.drawText(">", 0, lineY, this._rgbaCursor, this.backgroundColor)
833
+ } else {
834
+ this.frameBuffer.drawText(" ", 0, lineY, this._rgbaDefault, this.backgroundColor)
835
+ }
836
+
837
+ const fullText = `${linePrefix}${textToDraw.substring(0, textAvailableWidth)}`
838
+ const selectionRange = this.getLineSelectionRange(absoluteLineIndex)
839
+
840
+ if (selectionRange) {
841
+ const adjustedStart = Math.max(0, selectionRange.start)
842
+ const adjustedEnd = Math.min(fullText.length, selectionRange.end)
843
+
844
+ if (adjustedStart > 0) {
845
+ this.frameBuffer.drawText(fullText.substring(0, adjustedStart), 1, lineY, levelColor)
846
+ }
847
+
848
+ if (adjustedStart < adjustedEnd) {
849
+ this.frameBuffer.fillRect(1 + adjustedStart, lineY, adjustedEnd - adjustedStart, 1, this._rgbaSelection)
850
+ this.frameBuffer.drawText(
851
+ fullText.substring(adjustedStart, adjustedEnd),
852
+ 1 + adjustedStart,
853
+ lineY,
854
+ levelColor,
855
+ this._rgbaSelection,
856
+ )
857
+ }
858
+
859
+ if (adjustedEnd < fullText.length) {
860
+ this.frameBuffer.drawText(fullText.substring(adjustedEnd), 1 + adjustedEnd, lineY, levelColor)
861
+ }
862
+ } else {
863
+ this.frameBuffer.drawText(fullText, 1, lineY, levelColor)
864
+ }
865
+
866
+ lineY++
867
+ }
868
+ }
869
+
870
+ public renderToBuffer(buffer: OptimizedBuffer): void {
871
+ if (!this.isVisible || !this.frameBuffer) return
872
+
873
+ if (this._needsFrameBufferUpdate) {
874
+ this.updateFrameBuffer()
875
+ this._needsFrameBufferUpdate = false
876
+ }
877
+
878
+ buffer.drawFrameBuffer(this.consoleX, this.consoleY, this.frameBuffer)
879
+ }
880
+
881
+ public setDebugMode(enabled: boolean): void {
882
+ this._debugModeEnabled = enabled
883
+ terminalConsoleCache.setCollectCallerInfo(enabled)
884
+ if (this.isVisible) {
885
+ this.markNeedsRerender()
886
+ }
887
+ }
888
+
889
+ public toggleDebugMode(): void {
890
+ this.setDebugMode(!this._debugModeEnabled)
891
+ }
892
+
893
+ public set keyBindings(bindings: ConsoleKeyBinding[]) {
894
+ this._keyBindings = bindings
895
+ this._mergedKeyBindings = mergeKeyBindings(defaultConsoleKeybindings, bindings)
896
+ this._keyBindingsMap = buildKeyBindingsMap(this._mergedKeyBindings, this._keyAliasMap)
897
+ this.markNeedsRerender()
898
+ }
899
+
900
+ public set keyAliasMap(aliases: KeyAliasMap) {
901
+ this._keyAliasMap = mergeKeyAliases(defaultKeyAliases, aliases)
902
+ this._mergedKeyBindings = mergeKeyBindings(defaultConsoleKeybindings, this._keyBindings)
903
+ this._keyBindingsMap = buildKeyBindingsMap(this._mergedKeyBindings, this._keyAliasMap)
904
+ this.markNeedsRerender()
905
+ }
906
+
907
+ public set onCopySelection(callback: ((text: string) => void) | undefined) {
908
+ this.options.onCopySelection = callback
909
+ }
910
+
911
+ public get onCopySelection(): ((text: string) => void) | undefined {
912
+ return this.options.onCopySelection
913
+ }
914
+
915
+ private _scrollToBottom(forceCursorToLastLine: boolean = false): void {
916
+ const displayLineCount = this._displayLines.length
917
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
918
+ const maxScrollTop = Math.max(0, displayLineCount - logAreaHeight)
919
+ this.scrollTopIndex = maxScrollTop
920
+ this.isScrolledToBottom = true
921
+
922
+ const visibleLineCount = Math.min(logAreaHeight, displayLineCount - this.scrollTopIndex)
923
+ if (forceCursorToLastLine || this.currentLineIndex >= visibleLineCount) {
924
+ this.currentLineIndex = Math.max(0, visibleLineCount - 1)
925
+ }
926
+ }
927
+
928
+ private _processLogEntry(logEntry: [Date, LogLevel, any[], CallerInfo | null]): DisplayLine[] {
929
+ const [date, level, args, callerInfo] = logEntry
930
+ const displayLines: DisplayLine[] = []
931
+
932
+ const timestamp = this.formatTimestamp(date)
933
+ const callerSource = callerInfo ? `${callerInfo.fileName}:${callerInfo.lineNumber}` : "unknown"
934
+ const prefix = `[${timestamp}] [${level}]` + (this._debugModeEnabled ? ` [${callerSource}]` : "") + " "
935
+
936
+ const formattedArgs = this.formatArguments(args)
937
+ const initialLines = formattedArgs.split("\n")
938
+
939
+ for (let i = 0; i < initialLines.length; i++) {
940
+ const lineText = initialLines[i]
941
+ const isFirstLineOfEntry = i === 0
942
+ const availableWidth = this.consoleWidth - 1 - (isFirstLineOfEntry ? 0 : INDENT_WIDTH)
943
+ const linePrefix = isFirstLineOfEntry ? prefix : " ".repeat(INDENT_WIDTH)
944
+ const textToWrap = isFirstLineOfEntry ? linePrefix + lineText : lineText
945
+
946
+ let currentPos = 0
947
+ while (currentPos < textToWrap.length || (isFirstLineOfEntry && currentPos === 0 && textToWrap.length === 0)) {
948
+ const segment = textToWrap.substring(currentPos, currentPos + availableWidth)
949
+ const isFirstSegmentOfLine = currentPos === 0
950
+
951
+ displayLines.push({
952
+ text: isFirstSegmentOfLine && !isFirstLineOfEntry ? linePrefix + segment : segment,
953
+ level: level,
954
+ indent: !isFirstLineOfEntry || !isFirstSegmentOfLine,
955
+ })
956
+
957
+ currentPos += availableWidth
958
+ if (isFirstLineOfEntry && currentPos === 0 && textToWrap.length === 0) break
959
+ }
960
+ }
961
+
962
+ return displayLines
963
+ }
964
+
965
+ private _processCachedLogs(): void {
966
+ const logsToProcess = [...terminalConsoleCache.cachedLogs]
967
+ terminalConsoleCache.clearConsole()
968
+
969
+ this._allLogEntries.push(...logsToProcess)
970
+
971
+ if (this._allLogEntries.length > this.options.maxStoredLogs) {
972
+ this._allLogEntries.splice(0, this._allLogEntries.length - this.options.maxStoredLogs)
973
+ }
974
+
975
+ for (const logEntry of logsToProcess) {
976
+ const processed = this._processLogEntry(logEntry)
977
+ this._displayLines.push(...processed)
978
+ }
979
+
980
+ if (this._displayLines.length > this.options.maxDisplayLines) {
981
+ this._displayLines.splice(0, this._displayLines.length - this.options.maxDisplayLines)
982
+ }
983
+ }
984
+
985
+ private hasSelection(): boolean {
986
+ if (this._selectionStart === null || this._selectionEnd === null) return false
987
+
988
+ return this._selectionStart.line !== this._selectionEnd.line || this._selectionStart.col !== this._selectionEnd.col
989
+ }
990
+
991
+ private normalizeSelection(): ConsoleSelection | null {
992
+ if (!this._selectionStart || !this._selectionEnd) return null
993
+
994
+ const start = this._selectionStart
995
+ const end = this._selectionEnd
996
+
997
+ const startBeforeEnd = start.line < end.line || (start.line === end.line && start.col <= end.col)
998
+
999
+ if (startBeforeEnd) {
1000
+ return {
1001
+ startLine: start.line,
1002
+ startCol: start.col,
1003
+ endLine: end.line,
1004
+ endCol: end.col,
1005
+ }
1006
+ } else {
1007
+ return {
1008
+ startLine: end.line,
1009
+ startCol: end.col,
1010
+ endLine: start.line,
1011
+ endCol: start.col,
1012
+ }
1013
+ }
1014
+ }
1015
+
1016
+ private getSelectedText(): string {
1017
+ const selection = this.normalizeSelection()
1018
+ if (!selection) return ""
1019
+
1020
+ const lines: string[] = []
1021
+ for (let i = selection.startLine; i <= selection.endLine; i++) {
1022
+ if (i < 0 || i >= this._displayLines.length) continue
1023
+ const line = this._displayLines[i]
1024
+ const linePrefix = line.indent ? " ".repeat(INDENT_WIDTH) : ""
1025
+ const textAvailableWidth = this.consoleWidth - 1 - (line.indent ? INDENT_WIDTH : 0)
1026
+ const fullText = linePrefix + line.text.substring(0, textAvailableWidth)
1027
+ let text = fullText
1028
+
1029
+ if (i === selection.startLine && i === selection.endLine) {
1030
+ text = fullText.substring(selection.startCol, selection.endCol)
1031
+ } else if (i === selection.startLine) {
1032
+ text = fullText.substring(selection.startCol)
1033
+ } else if (i === selection.endLine) {
1034
+ text = fullText.substring(0, selection.endCol)
1035
+ }
1036
+
1037
+ lines.push(text)
1038
+ }
1039
+
1040
+ return lines.join("\n")
1041
+ }
1042
+
1043
+ private clearSelection(): void {
1044
+ this._selectionStart = null
1045
+ this._selectionEnd = null
1046
+ this._isDragging = false
1047
+ this.stopAutoScroll()
1048
+ }
1049
+
1050
+ private stopAutoScroll(): void {
1051
+ if (this._autoScrollInterval !== null) {
1052
+ this.clock.clearInterval(this._autoScrollInterval)
1053
+ this._autoScrollInterval = null
1054
+ }
1055
+ }
1056
+
1057
+ private startAutoScroll(direction: "up" | "down"): void {
1058
+ this.stopAutoScroll()
1059
+ this._autoScrollInterval = this.clock.setInterval(() => {
1060
+ if (direction === "up") {
1061
+ if (this.scrollTopIndex > 0) {
1062
+ this.scrollTopIndex--
1063
+ this.isScrolledToBottom = false
1064
+ if (this._selectionEnd) {
1065
+ this._selectionEnd = {
1066
+ line: this.scrollTopIndex,
1067
+ col: this._selectionEnd.col,
1068
+ }
1069
+ }
1070
+ this.markNeedsRerender()
1071
+ } else {
1072
+ this.stopAutoScroll()
1073
+ }
1074
+ } else {
1075
+ const displayLineCount = this._displayLines.length
1076
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
1077
+ const maxScrollTop = Math.max(0, displayLineCount - logAreaHeight)
1078
+ if (this.scrollTopIndex < maxScrollTop) {
1079
+ this.scrollTopIndex++
1080
+ this.isScrolledToBottom = this.scrollTopIndex === maxScrollTop
1081
+ if (this._selectionEnd) {
1082
+ const maxLine = this.scrollTopIndex + logAreaHeight - 1
1083
+ this._selectionEnd = {
1084
+ line: Math.min(maxLine, displayLineCount - 1),
1085
+ col: this._selectionEnd.col,
1086
+ }
1087
+ }
1088
+ this.markNeedsRerender()
1089
+ } else {
1090
+ this.stopAutoScroll()
1091
+ }
1092
+ }
1093
+ }, 50)
1094
+ }
1095
+
1096
+ private triggerCopy(): void {
1097
+ if (!this.hasSelection()) return
1098
+ const text = this.getSelectedText()
1099
+ if (text && this.options.onCopySelection) {
1100
+ try {
1101
+ this.options.onCopySelection(text)
1102
+ } catch {}
1103
+ this.clearSelection()
1104
+ this.markNeedsRerender()
1105
+ }
1106
+ }
1107
+
1108
+ private getLineSelectionRange(lineIndex: number): { start: number; end: number } | null {
1109
+ const selection = this.normalizeSelection()
1110
+ if (!selection) return null
1111
+
1112
+ if (lineIndex < selection.startLine || lineIndex > selection.endLine) {
1113
+ return null
1114
+ }
1115
+
1116
+ const line = this._displayLines[lineIndex]
1117
+ if (!line) return null
1118
+
1119
+ const linePrefix = line.indent ? " ".repeat(INDENT_WIDTH) : ""
1120
+ const textAvailableWidth = this.consoleWidth - 1 - (line.indent ? INDENT_WIDTH : 0)
1121
+ const fullTextLength = linePrefix.length + Math.min(line.text.length, textAvailableWidth)
1122
+
1123
+ let start = 0
1124
+ let end = fullTextLength
1125
+
1126
+ if (lineIndex === selection.startLine) {
1127
+ start = Math.max(0, selection.startCol)
1128
+ }
1129
+ if (lineIndex === selection.endLine) {
1130
+ end = Math.min(fullTextLength, selection.endCol)
1131
+ }
1132
+
1133
+ if (start >= end) return null
1134
+ return { start, end }
1135
+ }
1136
+
1137
+ public handleMouse(event: MouseEvent): boolean {
1138
+ if (!this.isVisible) return false
1139
+
1140
+ const localX = event.x - this.consoleX
1141
+ const localY = event.y - this.consoleY
1142
+
1143
+ if (localX < 0 || localX >= this.consoleWidth || localY < 0 || localY >= this.consoleHeight) {
1144
+ return false
1145
+ }
1146
+
1147
+ if (event.type === "scroll" && event.scroll) {
1148
+ if (event.scroll.direction === "up") {
1149
+ this.scrollUp()
1150
+ } else if (event.scroll.direction === "down") {
1151
+ this.scrollDown()
1152
+ }
1153
+ return true
1154
+ }
1155
+
1156
+ if (localY === 0) {
1157
+ if (
1158
+ event.type === "down" &&
1159
+ event.button === 0 &&
1160
+ localX >= this._copyButtonBounds.x &&
1161
+ localX < this._copyButtonBounds.x + this._copyButtonBounds.width
1162
+ ) {
1163
+ this.triggerCopy()
1164
+ return true
1165
+ }
1166
+ return true
1167
+ }
1168
+
1169
+ const lineIndex = this.scrollTopIndex + (localY - 1)
1170
+ const colIndex = Math.max(0, localX - 1)
1171
+
1172
+ if (event.type === "down" && event.button === 0) {
1173
+ this.clearSelection()
1174
+ this._selectionStart = { line: lineIndex, col: colIndex }
1175
+ this._selectionEnd = { line: lineIndex, col: colIndex }
1176
+ this._isDragging = true
1177
+ this.markNeedsRerender()
1178
+ return true
1179
+ }
1180
+
1181
+ if (event.type === "drag" && this._isDragging) {
1182
+ this._selectionEnd = { line: lineIndex, col: colIndex }
1183
+
1184
+ // Check if drag is at the edge and trigger auto-scroll
1185
+ const logAreaHeight = Math.max(1, this.consoleHeight - 1)
1186
+ const relativeY = localY - 1 // Subtract 1 for title bar
1187
+
1188
+ if (relativeY <= 0) {
1189
+ // Dragging at top edge
1190
+ this.startAutoScroll("up")
1191
+ } else if (relativeY >= logAreaHeight - 1) {
1192
+ // Dragging at bottom edge
1193
+ this.startAutoScroll("down")
1194
+ } else {
1195
+ // Not at edge, stop auto-scrolling
1196
+ this.stopAutoScroll()
1197
+ }
1198
+
1199
+ this.markNeedsRerender()
1200
+ return true
1201
+ }
1202
+
1203
+ if (event.type === "up") {
1204
+ if (this._isDragging) {
1205
+ this._selectionEnd = { line: lineIndex, col: colIndex }
1206
+ this._isDragging = false
1207
+ this.stopAutoScroll()
1208
+ this.markNeedsRerender()
1209
+ }
1210
+ return true
1211
+ }
1212
+
1213
+ return true
1214
+ }
1215
+
1216
+ public get visible(): boolean {
1217
+ return this.isVisible
1218
+ }
1219
+
1220
+ public get bounds(): { x: number; y: number; width: number; height: number } {
1221
+ return {
1222
+ x: this.consoleX,
1223
+ y: this.consoleY,
1224
+ width: this.consoleWidth,
1225
+ height: this.consoleHeight,
1226
+ }
1227
+ }
1228
+
1229
+ private saveLogsToFile(): void {
1230
+ try {
1231
+ const timestamp = Date.now()
1232
+ const filename = `_console_${timestamp}.log`
1233
+ const filepath = path.join(process.cwd(), filename)
1234
+
1235
+ const allLogEntries = [...this._allLogEntries, ...terminalConsoleCache.cachedLogs]
1236
+
1237
+ const logLines: string[] = []
1238
+
1239
+ for (const [date, level, args, callerInfo] of allLogEntries) {
1240
+ const timestampStr = this.formatTimestamp(date)
1241
+ const callerSource = callerInfo ? `${callerInfo.fileName}:${callerInfo.lineNumber}` : "unknown"
1242
+ const prefix = `[${timestampStr}] [${level}]` + (this._debugModeEnabled ? ` [${callerSource}]` : "") + " "
1243
+ const formattedArgs = this.formatArguments(args)
1244
+ logLines.push(prefix + formattedArgs)
1245
+ }
1246
+
1247
+ const content = logLines.join("\n")
1248
+ fs.writeFileSync(filepath, content, "utf8")
1249
+
1250
+ console.info(`Console logs saved to: ${filename}`)
1251
+ } catch (error) {
1252
+ console.error(`Failed to save console logs:`, error)
1253
+ }
1254
+ }
1255
+ }