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