@fairyhunter13/opentui-core 0.1.114 → 0.1.116

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/3d/SpriteResourceManager.d.ts +74 -0
  2. package/3d/SpriteUtils.d.ts +13 -0
  3. package/3d/TextureUtils.d.ts +24 -0
  4. package/3d/ThreeRenderable.d.ts +40 -0
  5. package/3d/WGPURenderer.d.ts +61 -0
  6. package/3d/animation/ExplodingSpriteEffect.d.ts +71 -0
  7. package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +76 -0
  8. package/3d/animation/SpriteAnimator.d.ts +124 -0
  9. package/3d/animation/SpriteParticleGenerator.d.ts +62 -0
  10. package/3d/canvas.d.ts +44 -0
  11. package/3d/index.d.ts +12 -0
  12. package/3d/physics/PlanckPhysicsAdapter.d.ts +19 -0
  13. package/3d/physics/RapierPhysicsAdapter.d.ts +19 -0
  14. package/3d/physics/physics-interface.d.ts +27 -0
  15. package/3d.d.ts +2 -0
  16. package/3d.js +34041 -0
  17. package/3d.js.map +155 -0
  18. package/LICENSE +21 -0
  19. package/NativeSpanFeed.d.ts +41 -0
  20. package/Renderable.d.ts +334 -0
  21. package/animation/Timeline.d.ts +126 -0
  22. package/ansi.d.ts +13 -0
  23. package/buffer.d.ts +111 -0
  24. package/console.d.ts +144 -0
  25. package/edit-buffer.d.ts +98 -0
  26. package/editor-view.d.ts +73 -0
  27. package/index-dcj62y8t.js +20614 -0
  28. package/index-dcj62y8t.js.map +67 -0
  29. package/index-jyrhjc34.js +411 -0
  30. package/index-jyrhjc34.js.map +10 -0
  31. package/index-wc7ae60z.js +12299 -0
  32. package/index-wc7ae60z.js.map +42 -0
  33. package/index.d.ts +23 -0
  34. package/index.js +478 -0
  35. package/index.js.map +9 -0
  36. package/lib/KeyHandler.d.ts +61 -0
  37. package/lib/RGBA.d.ts +25 -0
  38. package/lib/ascii.font.d.ts +508 -0
  39. package/lib/border.d.ts +51 -0
  40. package/lib/bunfs.d.ts +7 -0
  41. package/lib/clipboard.d.ts +17 -0
  42. package/lib/clock.d.ts +15 -0
  43. package/lib/data-paths.d.ts +26 -0
  44. package/lib/debounce.d.ts +42 -0
  45. package/lib/detect-links.d.ts +6 -0
  46. package/lib/env.d.ts +42 -0
  47. package/lib/extmarks-history.d.ts +17 -0
  48. package/lib/extmarks.d.ts +89 -0
  49. package/lib/hast-styled-text.d.ts +17 -0
  50. package/lib/index.d.ts +21 -0
  51. package/lib/keymapping.d.ts +25 -0
  52. package/lib/objects-in-viewport.d.ts +24 -0
  53. package/lib/output.capture.d.ts +24 -0
  54. package/lib/parse.keypress-kitty.d.ts +2 -0
  55. package/lib/parse.keypress.d.ts +26 -0
  56. package/lib/parse.mouse.d.ts +30 -0
  57. package/lib/paste.d.ts +7 -0
  58. package/lib/queue.d.ts +15 -0
  59. package/lib/renderable.validations.d.ts +12 -0
  60. package/lib/scroll-acceleration.d.ts +43 -0
  61. package/lib/selection.d.ts +63 -0
  62. package/lib/singleton.d.ts +7 -0
  63. package/lib/stdin-parser.d.ts +87 -0
  64. package/lib/styled-text.d.ts +63 -0
  65. package/lib/terminal-capability-detection.d.ts +30 -0
  66. package/lib/terminal-palette.d.ts +50 -0
  67. package/lib/tree-sitter/assets/update.d.ts +11 -0
  68. package/lib/tree-sitter/client.d.ts +47 -0
  69. package/lib/tree-sitter/default-parsers.d.ts +2 -0
  70. package/lib/tree-sitter/download-utils.d.ts +21 -0
  71. package/lib/tree-sitter/index.d.ts +8 -0
  72. package/lib/tree-sitter/parser.worker.d.ts +1 -0
  73. package/lib/tree-sitter/parsers-config.d.ts +53 -0
  74. package/lib/tree-sitter/resolve-ft.d.ts +5 -0
  75. package/lib/tree-sitter/types.d.ts +82 -0
  76. package/lib/tree-sitter-styled-text.d.ts +14 -0
  77. package/lib/validate-dir-name.d.ts +1 -0
  78. package/lib/yoga.options.d.ts +32 -0
  79. package/package.json +53 -62
  80. package/parser.worker.js +899 -0
  81. package/parser.worker.js.map +12 -0
  82. package/plugins/core-slot.d.ts +72 -0
  83. package/plugins/registry.d.ts +42 -0
  84. package/plugins/types.d.ts +34 -0
  85. package/post/effects.d.ts +147 -0
  86. package/post/filters.d.ts +65 -0
  87. package/post/matrices.d.ts +20 -0
  88. package/renderables/ASCIIFont.d.ts +52 -0
  89. package/renderables/Box.d.ts +81 -0
  90. package/renderables/Code.d.ts +78 -0
  91. package/renderables/Diff.d.ts +142 -0
  92. package/renderables/EditBufferRenderable.d.ts +237 -0
  93. package/renderables/FrameBuffer.d.ts +16 -0
  94. package/renderables/Input.d.ts +67 -0
  95. package/renderables/LineNumberRenderable.d.ts +78 -0
  96. package/renderables/Markdown.d.ts +185 -0
  97. package/renderables/ScrollBar.d.ts +77 -0
  98. package/renderables/ScrollBox.d.ts +124 -0
  99. package/renderables/Select.d.ts +115 -0
  100. package/renderables/Slider.d.ts +47 -0
  101. package/renderables/TabSelect.d.ts +96 -0
  102. package/renderables/Text.d.ts +36 -0
  103. package/renderables/TextBufferRenderable.d.ts +105 -0
  104. package/renderables/TextNode.d.ts +91 -0
  105. package/renderables/TextTable.d.ts +140 -0
  106. package/renderables/Textarea.d.ts +63 -0
  107. package/renderables/TimeToFirstDraw.d.ts +24 -0
  108. package/renderables/__tests__/renderable-test-utils.d.ts +12 -0
  109. package/renderables/composition/VRenderable.d.ts +16 -0
  110. package/renderables/composition/constructs.d.ts +35 -0
  111. package/renderables/composition/vnode.d.ts +46 -0
  112. package/renderables/index.d.ts +23 -0
  113. package/renderables/markdown-parser.d.ts +10 -0
  114. package/renderer.d.ts +419 -0
  115. package/runtime-plugin-support.d.ts +3 -0
  116. package/runtime-plugin-support.js +29 -0
  117. package/runtime-plugin-support.js.map +10 -0
  118. package/runtime-plugin.d.ts +16 -0
  119. package/runtime-plugin.js +16 -0
  120. package/runtime-plugin.js.map +9 -0
  121. package/syntax-style.d.ts +54 -0
  122. package/testing/manual-clock.d.ts +17 -0
  123. package/testing/mock-keys.d.ts +81 -0
  124. package/testing/mock-mouse.d.ts +38 -0
  125. package/testing/mock-tree-sitter-client.d.ts +23 -0
  126. package/testing/spy.d.ts +7 -0
  127. package/testing/test-recorder.d.ts +61 -0
  128. package/testing/test-renderer.d.ts +23 -0
  129. package/testing.d.ts +6 -0
  130. package/testing.js +697 -0
  131. package/testing.js.map +15 -0
  132. package/text-buffer-view.d.ts +42 -0
  133. package/text-buffer.d.ts +67 -0
  134. package/types.d.ts +139 -0
  135. package/utils.d.ts +14 -0
  136. package/zig-structs.d.ts +155 -0
  137. package/zig.d.ts +353 -0
  138. package/dev/keypress-debug-renderer.ts +0 -148
  139. package/dev/keypress-debug.ts +0 -43
  140. package/dev/print-env-vars.ts +0 -32
  141. package/dev/test-tmux-graphics-334.sh +0 -68
  142. package/dev/thai-debug-test.ts +0 -68
  143. package/docs/development.md +0 -144
  144. package/scripts/build.ts +0 -400
  145. package/scripts/publish.ts +0 -60
  146. package/src/3d/SpriteResourceManager.ts +0 -286
  147. package/src/3d/SpriteUtils.ts +0 -70
  148. package/src/3d/TextureUtils.ts +0 -196
  149. package/src/3d/ThreeRenderable.ts +0 -197
  150. package/src/3d/WGPURenderer.ts +0 -294
  151. package/src/3d/animation/ExplodingSpriteEffect.ts +0 -513
  152. package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +0 -429
  153. package/src/3d/animation/SpriteAnimator.ts +0 -633
  154. package/src/3d/animation/SpriteParticleGenerator.ts +0 -435
  155. package/src/3d/canvas.ts +0 -464
  156. package/src/3d/index.ts +0 -12
  157. package/src/3d/physics/PlanckPhysicsAdapter.ts +0 -72
  158. package/src/3d/physics/RapierPhysicsAdapter.ts +0 -66
  159. package/src/3d/physics/physics-interface.ts +0 -31
  160. package/src/3d/shaders/supersampling.wgsl +0 -201
  161. package/src/3d.ts +0 -3
  162. package/src/NativeSpanFeed.ts +0 -300
  163. package/src/Renderable.ts +0 -1704
  164. package/src/__snapshots__/buffer.test.ts.snap +0 -28
  165. package/src/animation/Timeline.test.ts +0 -2709
  166. package/src/animation/Timeline.ts +0 -598
  167. package/src/ansi.ts +0 -18
  168. package/src/benchmark/attenuation-benchmark.ts +0 -81
  169. package/src/benchmark/colormatrix-benchmark.ts +0 -128
  170. package/src/benchmark/gain-benchmark.ts +0 -80
  171. package/src/benchmark/latest-all-bench-run.json +0 -707
  172. package/src/benchmark/latest-async-bench-run.json +0 -336
  173. package/src/benchmark/latest-default-bench-run.json +0 -657
  174. package/src/benchmark/latest-large-bench-run.json +0 -707
  175. package/src/benchmark/latest-quick-bench-run.json +0 -207
  176. package/src/benchmark/markdown-benchmark.ts +0 -1796
  177. package/src/benchmark/native-span-feed-async-benchmark.ts +0 -355
  178. package/src/benchmark/native-span-feed-benchmark.md +0 -56
  179. package/src/benchmark/native-span-feed-benchmark.ts +0 -596
  180. package/src/benchmark/native-span-feed-compare.ts +0 -280
  181. package/src/benchmark/renderer-benchmark.ts +0 -754
  182. package/src/benchmark/text-table-benchmark.ts +0 -948
  183. package/src/buffer.test.ts +0 -291
  184. package/src/buffer.ts +0 -554
  185. package/src/console.test.ts +0 -612
  186. package/src/console.ts +0 -1254
  187. package/src/edit-buffer.test.ts +0 -1769
  188. package/src/edit-buffer.ts +0 -411
  189. package/src/editor-view.test.ts +0 -1032
  190. package/src/editor-view.ts +0 -284
  191. package/src/examples/ascii-font-selection-demo.ts +0 -245
  192. package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
  193. package/src/examples/assets/concrete.png +0 -0
  194. package/src/examples/assets/crate.png +0 -0
  195. package/src/examples/assets/crate_emissive.png +0 -0
  196. package/src/examples/assets/forrest_background.png +0 -0
  197. package/src/examples/assets/hast-example.json +0 -1018
  198. package/src/examples/assets/heart.png +0 -0
  199. package/src/examples/assets/main_char_heavy_attack.png +0 -0
  200. package/src/examples/assets/main_char_idle.png +0 -0
  201. package/src/examples/assets/main_char_jump_end.png +0 -0
  202. package/src/examples/assets/main_char_jump_landing.png +0 -0
  203. package/src/examples/assets/main_char_jump_start.png +0 -0
  204. package/src/examples/assets/main_char_run_loop.png +0 -0
  205. package/src/examples/assets/roughness_map.jpg +0 -0
  206. package/src/examples/build.ts +0 -115
  207. package/src/examples/code-demo.ts +0 -924
  208. package/src/examples/console-demo.ts +0 -358
  209. package/src/examples/core-plugin-slots-demo.ts +0 -759
  210. package/src/examples/diff-demo.ts +0 -701
  211. package/src/examples/draggable-three-demo.ts +0 -259
  212. package/src/examples/editor-demo.ts +0 -322
  213. package/src/examples/extmarks-demo.ts +0 -196
  214. package/src/examples/focus-restore-demo.ts +0 -310
  215. package/src/examples/fonts.ts +0 -245
  216. package/src/examples/fractal-shader-demo.ts +0 -268
  217. package/src/examples/framebuffer-demo.ts +0 -674
  218. package/src/examples/full-unicode-demo.ts +0 -241
  219. package/src/examples/golden-star-demo.ts +0 -933
  220. package/src/examples/grayscale-buffer-demo.ts +0 -249
  221. package/src/examples/hast-syntax-highlighting-demo.ts +0 -129
  222. package/src/examples/index.ts +0 -926
  223. package/src/examples/input-demo.ts +0 -377
  224. package/src/examples/input-select-layout-demo.ts +0 -425
  225. package/src/examples/install.sh +0 -143
  226. package/src/examples/keypress-debug-demo.ts +0 -452
  227. package/src/examples/lib/HexList.ts +0 -122
  228. package/src/examples/lib/PaletteGrid.ts +0 -125
  229. package/src/examples/lib/standalone-keys.ts +0 -25
  230. package/src/examples/lib/tab-controller.ts +0 -243
  231. package/src/examples/lights-phong-demo.ts +0 -290
  232. package/src/examples/link-demo.ts +0 -220
  233. package/src/examples/live-state-demo.ts +0 -480
  234. package/src/examples/markdown-demo.ts +0 -725
  235. package/src/examples/mouse-interaction-demo.ts +0 -428
  236. package/src/examples/nested-zindex-demo.ts +0 -357
  237. package/src/examples/opacity-example.ts +0 -235
  238. package/src/examples/opentui-demo.ts +0 -1057
  239. package/src/examples/physx-planck-2d-demo.ts +0 -623
  240. package/src/examples/physx-rapier-2d-demo.ts +0 -655
  241. package/src/examples/relative-positioning-demo.ts +0 -323
  242. package/src/examples/scroll-example.ts +0 -214
  243. package/src/examples/scrollbox-mouse-test.ts +0 -112
  244. package/src/examples/scrollbox-overlay-hit-test.ts +0 -206
  245. package/src/examples/select-demo.ts +0 -237
  246. package/src/examples/shader-cube-demo.ts +0 -1015
  247. package/src/examples/simple-layout-example.ts +0 -591
  248. package/src/examples/slider-demo.ts +0 -617
  249. package/src/examples/split-mode-demo.ts +0 -453
  250. package/src/examples/sprite-animation-demo.ts +0 -443
  251. package/src/examples/sprite-particle-generator-demo.ts +0 -486
  252. package/src/examples/static-sprite-demo.ts +0 -193
  253. package/src/examples/sticky-scroll-example.ts +0 -308
  254. package/src/examples/styled-text-demo.ts +0 -282
  255. package/src/examples/tab-select-demo.ts +0 -219
  256. package/src/examples/terminal-title.ts +0 -29
  257. package/src/examples/terminal.ts +0 -305
  258. package/src/examples/text-node-demo.ts +0 -416
  259. package/src/examples/text-selection-demo.ts +0 -377
  260. package/src/examples/text-table-demo.ts +0 -503
  261. package/src/examples/text-truncation-demo.ts +0 -481
  262. package/src/examples/text-wrap.ts +0 -757
  263. package/src/examples/texture-loading-demo.ts +0 -259
  264. package/src/examples/timeline-example.ts +0 -670
  265. package/src/examples/transparency-demo.ts +0 -400
  266. package/src/examples/vnode-composition-demo.ts +0 -404
  267. package/src/examples/wide-grapheme-overlay-demo.ts +0 -280
  268. package/src/index.ts +0 -24
  269. package/src/lib/KeyHandler.integration.test.ts +0 -292
  270. package/src/lib/KeyHandler.stopPropagation.test.ts +0 -289
  271. package/src/lib/KeyHandler.test.ts +0 -662
  272. package/src/lib/KeyHandler.ts +0 -222
  273. package/src/lib/RGBA.test.ts +0 -984
  274. package/src/lib/RGBA.ts +0 -204
  275. package/src/lib/ascii.font.ts +0 -330
  276. package/src/lib/border.test.ts +0 -83
  277. package/src/lib/border.ts +0 -170
  278. package/src/lib/bunfs.test.ts +0 -27
  279. package/src/lib/bunfs.ts +0 -18
  280. package/src/lib/clipboard.test.ts +0 -41
  281. package/src/lib/clipboard.ts +0 -47
  282. package/src/lib/clock.ts +0 -35
  283. package/src/lib/data-paths.test.ts +0 -133
  284. package/src/lib/data-paths.ts +0 -109
  285. package/src/lib/debounce.ts +0 -106
  286. package/src/lib/detect-links.test.ts +0 -98
  287. package/src/lib/detect-links.ts +0 -56
  288. package/src/lib/env.test.ts +0 -228
  289. package/src/lib/env.ts +0 -209
  290. package/src/lib/extmarks-history.ts +0 -51
  291. package/src/lib/extmarks-multiwidth.test.ts +0 -322
  292. package/src/lib/extmarks.test.ts +0 -3457
  293. package/src/lib/extmarks.ts +0 -843
  294. package/src/lib/fonts/block.json +0 -405
  295. package/src/lib/fonts/grid.json +0 -265
  296. package/src/lib/fonts/huge.json +0 -741
  297. package/src/lib/fonts/pallet.json +0 -314
  298. package/src/lib/fonts/shade.json +0 -591
  299. package/src/lib/fonts/slick.json +0 -321
  300. package/src/lib/fonts/tiny.json +0 -69
  301. package/src/lib/hast-styled-text.ts +0 -59
  302. package/src/lib/index.ts +0 -21
  303. package/src/lib/keymapping.test.ts +0 -317
  304. package/src/lib/keymapping.ts +0 -115
  305. package/src/lib/objects-in-viewport.test.ts +0 -787
  306. package/src/lib/objects-in-viewport.ts +0 -153
  307. package/src/lib/output.capture.ts +0 -58
  308. package/src/lib/parse.keypress-kitty.protocol.test.ts +0 -340
  309. package/src/lib/parse.keypress-kitty.test.ts +0 -663
  310. package/src/lib/parse.keypress-kitty.ts +0 -439
  311. package/src/lib/parse.keypress.test.ts +0 -1849
  312. package/src/lib/parse.keypress.ts +0 -397
  313. package/src/lib/parse.mouse.test.ts +0 -552
  314. package/src/lib/parse.mouse.ts +0 -232
  315. package/src/lib/paste.ts +0 -16
  316. package/src/lib/queue.ts +0 -65
  317. package/src/lib/renderable.validations.test.ts +0 -87
  318. package/src/lib/renderable.validations.ts +0 -83
  319. package/src/lib/scroll-acceleration.ts +0 -98
  320. package/src/lib/selection.ts +0 -240
  321. package/src/lib/singleton.ts +0 -28
  322. package/src/lib/stdin-parser.test.ts +0 -2290
  323. package/src/lib/stdin-parser.ts +0 -1810
  324. package/src/lib/styled-text.ts +0 -178
  325. package/src/lib/terminal-capability-detection.test.ts +0 -202
  326. package/src/lib/terminal-capability-detection.ts +0 -79
  327. package/src/lib/terminal-palette.test.ts +0 -878
  328. package/src/lib/terminal-palette.ts +0 -383
  329. package/src/lib/tree-sitter/assets/README.md +0 -118
  330. package/src/lib/tree-sitter/assets/update.ts +0 -334
  331. package/src/lib/tree-sitter/assets.d.ts +0 -9
  332. package/src/lib/tree-sitter/cache.test.ts +0 -273
  333. package/src/lib/tree-sitter/client.test.ts +0 -1165
  334. package/src/lib/tree-sitter/client.ts +0 -607
  335. package/src/lib/tree-sitter/default-parsers.ts +0 -86
  336. package/src/lib/tree-sitter/download-utils.ts +0 -148
  337. package/src/lib/tree-sitter/index.ts +0 -28
  338. package/src/lib/tree-sitter/parser.worker.ts +0 -1042
  339. package/src/lib/tree-sitter/parsers-config.ts +0 -81
  340. package/src/lib/tree-sitter/resolve-ft.test.ts +0 -55
  341. package/src/lib/tree-sitter/resolve-ft.ts +0 -189
  342. package/src/lib/tree-sitter/types.ts +0 -82
  343. package/src/lib/tree-sitter-styled-text.test.ts +0 -1253
  344. package/src/lib/tree-sitter-styled-text.ts +0 -306
  345. package/src/lib/validate-dir-name.ts +0 -55
  346. package/src/lib/yoga.options.test.ts +0 -628
  347. package/src/lib/yoga.options.ts +0 -346
  348. package/src/plugins/core-slot.ts +0 -579
  349. package/src/plugins/registry.ts +0 -402
  350. package/src/plugins/types.ts +0 -46
  351. package/src/post/effects.ts +0 -930
  352. package/src/post/filters.ts +0 -489
  353. package/src/post/matrices.ts +0 -288
  354. package/src/renderables/ASCIIFont.ts +0 -219
  355. package/src/renderables/Box.test.ts +0 -205
  356. package/src/renderables/Box.ts +0 -326
  357. package/src/renderables/Code.test.ts +0 -2062
  358. package/src/renderables/Code.ts +0 -357
  359. package/src/renderables/Diff.regression.test.ts +0 -226
  360. package/src/renderables/Diff.test.ts +0 -3101
  361. package/src/renderables/Diff.ts +0 -1211
  362. package/src/renderables/EditBufferRenderable.test.ts +0 -288
  363. package/src/renderables/EditBufferRenderable.ts +0 -1166
  364. package/src/renderables/FrameBuffer.ts +0 -47
  365. package/src/renderables/Input.test.ts +0 -1228
  366. package/src/renderables/Input.ts +0 -247
  367. package/src/renderables/LineNumberRenderable.ts +0 -724
  368. package/src/renderables/Markdown.ts +0 -1393
  369. package/src/renderables/ScrollBar.ts +0 -422
  370. package/src/renderables/ScrollBox.ts +0 -883
  371. package/src/renderables/Select.test.ts +0 -1033
  372. package/src/renderables/Select.ts +0 -524
  373. package/src/renderables/Slider.test.ts +0 -456
  374. package/src/renderables/Slider.ts +0 -342
  375. package/src/renderables/TabSelect.test.ts +0 -197
  376. package/src/renderables/TabSelect.ts +0 -455
  377. package/src/renderables/Text.selection-buffer.test.ts +0 -123
  378. package/src/renderables/Text.test.ts +0 -2660
  379. package/src/renderables/Text.ts +0 -147
  380. package/src/renderables/TextBufferRenderable.ts +0 -518
  381. package/src/renderables/TextNode.test.ts +0 -1058
  382. package/src/renderables/TextNode.ts +0 -325
  383. package/src/renderables/TextTable.test.ts +0 -1421
  384. package/src/renderables/TextTable.ts +0 -1344
  385. package/src/renderables/Textarea.ts +0 -430
  386. package/src/renderables/TimeToFirstDraw.ts +0 -89
  387. package/src/renderables/__snapshots__/Code.test.ts.snap +0 -13
  388. package/src/renderables/__snapshots__/Diff.test.ts.snap +0 -785
  389. package/src/renderables/__snapshots__/Text.test.ts.snap +0 -421
  390. package/src/renderables/__snapshots__/TextTable.test.ts.snap +0 -215
  391. package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +0 -144
  392. package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +0 -816
  393. package/src/renderables/__tests__/LineNumberRenderable.test.ts +0 -1865
  394. package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +0 -85
  395. package/src/renderables/__tests__/Markdown.code-colors.test.ts +0 -242
  396. package/src/renderables/__tests__/Markdown.test.ts +0 -2518
  397. package/src/renderables/__tests__/MultiRenderable.selection.test.ts +0 -87
  398. package/src/renderables/__tests__/Textarea.buffer.test.ts +0 -682
  399. package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +0 -675
  400. package/src/renderables/__tests__/Textarea.editing.test.ts +0 -2041
  401. package/src/renderables/__tests__/Textarea.error-handling.test.ts +0 -35
  402. package/src/renderables/__tests__/Textarea.events.test.ts +0 -738
  403. package/src/renderables/__tests__/Textarea.highlights.test.ts +0 -590
  404. package/src/renderables/__tests__/Textarea.keybinding.test.ts +0 -3149
  405. package/src/renderables/__tests__/Textarea.paste.test.ts +0 -357
  406. package/src/renderables/__tests__/Textarea.rendering.test.ts +0 -1866
  407. package/src/renderables/__tests__/Textarea.scroll.test.ts +0 -733
  408. package/src/renderables/__tests__/Textarea.selection.test.ts +0 -1590
  409. package/src/renderables/__tests__/Textarea.stress.test.ts +0 -670
  410. package/src/renderables/__tests__/Textarea.undo-redo.test.ts +0 -383
  411. package/src/renderables/__tests__/Textarea.visual-lines.test.ts +0 -310
  412. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +0 -221
  413. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +0 -89
  414. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +0 -457
  415. package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +0 -158
  416. package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +0 -387
  417. package/src/renderables/__tests__/markdown-parser.test.ts +0 -217
  418. package/src/renderables/__tests__/renderable-test-utils.ts +0 -60
  419. package/src/renderables/composition/README.md +0 -8
  420. package/src/renderables/composition/VRenderable.ts +0 -32
  421. package/src/renderables/composition/constructs.ts +0 -127
  422. package/src/renderables/composition/vnode.ts +0 -289
  423. package/src/renderables/index.ts +0 -23
  424. package/src/renderables/markdown-parser.ts +0 -66
  425. package/src/renderer.ts +0 -2681
  426. package/src/runtime-plugin-support.ts +0 -39
  427. package/src/runtime-plugin.ts +0 -615
  428. package/src/syntax-style.test.ts +0 -841
  429. package/src/syntax-style.ts +0 -257
  430. package/src/testing/README.md +0 -210
  431. package/src/testing/capture-spans.test.ts +0 -194
  432. package/src/testing/integration.test.ts +0 -276
  433. package/src/testing/manual-clock.ts +0 -117
  434. package/src/testing/mock-keys.test.ts +0 -1378
  435. package/src/testing/mock-keys.ts +0 -457
  436. package/src/testing/mock-mouse.test.ts +0 -218
  437. package/src/testing/mock-mouse.ts +0 -247
  438. package/src/testing/mock-tree-sitter-client.ts +0 -73
  439. package/src/testing/spy.ts +0 -13
  440. package/src/testing/test-recorder.test.ts +0 -415
  441. package/src/testing/test-recorder.ts +0 -145
  442. package/src/testing/test-renderer.ts +0 -132
  443. package/src/testing.ts +0 -7
  444. package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +0 -481
  445. package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +0 -19
  446. package/src/tests/__snapshots__/scrollbox.test.ts.snap +0 -29
  447. package/src/tests/absolute-positioning.snapshot.test.ts +0 -638
  448. package/src/tests/allocator-stats.test.ts +0 -38
  449. package/src/tests/destroy-during-render.test.ts +0 -200
  450. package/src/tests/destroy-on-exit.fixture.ts +0 -36
  451. package/src/tests/destroy-on-exit.test.ts +0 -41
  452. package/src/tests/hover-cursor.test.ts +0 -98
  453. package/src/tests/native-span-feed-async.test.ts +0 -173
  454. package/src/tests/native-span-feed-close.test.ts +0 -120
  455. package/src/tests/native-span-feed-coverage.test.ts +0 -227
  456. package/src/tests/native-span-feed-edge-cases.test.ts +0 -352
  457. package/src/tests/native-span-feed-use-after-free.test.ts +0 -45
  458. package/src/tests/opacity.test.ts +0 -123
  459. package/src/tests/renderable.snapshot.test.ts +0 -524
  460. package/src/tests/renderable.test.ts +0 -1281
  461. package/src/tests/renderer.clock.test.ts +0 -158
  462. package/src/tests/renderer.console-startup.test.ts +0 -185
  463. package/src/tests/renderer.control.test.ts +0 -425
  464. package/src/tests/renderer.core-slot-binding.test.ts +0 -952
  465. package/src/tests/renderer.cursor.test.ts +0 -26
  466. package/src/tests/renderer.destroy-during-render.test.ts +0 -147
  467. package/src/tests/renderer.focus-restore.test.ts +0 -257
  468. package/src/tests/renderer.focus.test.ts +0 -294
  469. package/src/tests/renderer.idle.test.ts +0 -219
  470. package/src/tests/renderer.input.test.ts +0 -2237
  471. package/src/tests/renderer.kitty-flags.test.ts +0 -195
  472. package/src/tests/renderer.mouse.test.ts +0 -1274
  473. package/src/tests/renderer.palette.test.ts +0 -629
  474. package/src/tests/renderer.selection.test.ts +0 -49
  475. package/src/tests/renderer.slot-registry.test.ts +0 -684
  476. package/src/tests/renderer.useMouse.test.ts +0 -47
  477. package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +0 -76
  478. package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +0 -43
  479. package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +0 -67
  480. package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +0 -72
  481. package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +0 -44
  482. package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +0 -85
  483. package/src/tests/runtime-plugin-path-alias.fixture.ts +0 -43
  484. package/src/tests/runtime-plugin-resolve-roots.fixture.ts +0 -65
  485. package/src/tests/runtime-plugin-support.fixture.ts +0 -11
  486. package/src/tests/runtime-plugin-support.test.ts +0 -19
  487. package/src/tests/runtime-plugin-windows-file-url.fixture.ts +0 -30
  488. package/src/tests/runtime-plugin.fixture.ts +0 -40
  489. package/src/tests/runtime-plugin.test.ts +0 -354
  490. package/src/tests/scrollbox-culling-bug.test.ts +0 -114
  491. package/src/tests/scrollbox-hitgrid-resize.test.ts +0 -136
  492. package/src/tests/scrollbox-hitgrid.test.ts +0 -909
  493. package/src/tests/scrollbox.test.ts +0 -1530
  494. package/src/tests/wrap-resize-perf.test.ts +0 -276
  495. package/src/tests/yoga-setters.test.ts +0 -921
  496. package/src/text-buffer-view.test.ts +0 -705
  497. package/src/text-buffer-view.ts +0 -189
  498. package/src/text-buffer.test.ts +0 -347
  499. package/src/text-buffer.ts +0 -250
  500. package/src/types.ts +0 -161
  501. package/src/utils.ts +0 -88
  502. package/src/zig/ansi.zig +0 -268
  503. package/src/zig/bench/README.md +0 -50
  504. package/src/zig/bench/buffer-draw-text-buffer_bench.zig +0 -887
  505. package/src/zig/bench/edit-buffer_bench.zig +0 -476
  506. package/src/zig/bench/native-span-feed_bench.zig +0 -100
  507. package/src/zig/bench/rope-markers_bench.zig +0 -713
  508. package/src/zig/bench/rope_bench.zig +0 -514
  509. package/src/zig/bench/styled-text_bench.zig +0 -470
  510. package/src/zig/bench/text-buffer-coords_bench.zig +0 -362
  511. package/src/zig/bench/text-buffer-view_bench.zig +0 -459
  512. package/src/zig/bench/text-chunk-graphemes_bench.zig +0 -273
  513. package/src/zig/bench/utf8_bench.zig +0 -799
  514. package/src/zig/bench-utils.zig +0 -431
  515. package/src/zig/bench.zig +0 -217
  516. package/src/zig/buffer-methods.zig +0 -211
  517. package/src/zig/buffer.zig +0 -2281
  518. package/src/zig/build.zig +0 -289
  519. package/src/zig/build.zig.zon +0 -16
  520. package/src/zig/edit-buffer.zig +0 -825
  521. package/src/zig/editor-view.zig +0 -802
  522. package/src/zig/event-bus.zig +0 -13
  523. package/src/zig/event-emitter.zig +0 -65
  524. package/src/zig/file-logger.zig +0 -92
  525. package/src/zig/grapheme.zig +0 -599
  526. package/src/zig/lib.zig +0 -1854
  527. package/src/zig/link.zig +0 -333
  528. package/src/zig/logger.zig +0 -43
  529. package/src/zig/mem-registry.zig +0 -125
  530. package/src/zig/native-span-feed-bench-lib.zig +0 -7
  531. package/src/zig/native-span-feed.zig +0 -708
  532. package/src/zig/renderer.zig +0 -1393
  533. package/src/zig/rope.zig +0 -1220
  534. package/src/zig/syntax-style.zig +0 -161
  535. package/src/zig/terminal.zig +0 -987
  536. package/src/zig/test.zig +0 -72
  537. package/src/zig/tests/README.md +0 -18
  538. package/src/zig/tests/buffer-methods_test.zig +0 -1109
  539. package/src/zig/tests/buffer_test.zig +0 -2557
  540. package/src/zig/tests/edit-buffer-history_test.zig +0 -271
  541. package/src/zig/tests/edit-buffer_test.zig +0 -1689
  542. package/src/zig/tests/editor-view_test.zig +0 -3299
  543. package/src/zig/tests/event-emitter_test.zig +0 -249
  544. package/src/zig/tests/grapheme_test.zig +0 -1304
  545. package/src/zig/tests/link_test.zig +0 -190
  546. package/src/zig/tests/mem-registry_test.zig +0 -473
  547. package/src/zig/tests/memory_leak_regression_test.zig +0 -159
  548. package/src/zig/tests/native-span-feed_test.zig +0 -1264
  549. package/src/zig/tests/renderer_test.zig +0 -1017
  550. package/src/zig/tests/rope-nested_test.zig +0 -712
  551. package/src/zig/tests/rope_fuzz_test.zig +0 -238
  552. package/src/zig/tests/rope_test.zig +0 -2362
  553. package/src/zig/tests/segment-merge.test.zig +0 -148
  554. package/src/zig/tests/syntax-style_test.zig +0 -557
  555. package/src/zig/tests/terminal_test.zig +0 -754
  556. package/src/zig/tests/text-buffer-drawing_test.zig +0 -3237
  557. package/src/zig/tests/text-buffer-highlights_test.zig +0 -666
  558. package/src/zig/tests/text-buffer-iterators_test.zig +0 -776
  559. package/src/zig/tests/text-buffer-segment_test.zig +0 -320
  560. package/src/zig/tests/text-buffer-selection_test.zig +0 -1035
  561. package/src/zig/tests/text-buffer-selection_viewport_test.zig +0 -358
  562. package/src/zig/tests/text-buffer-view_test.zig +0 -3649
  563. package/src/zig/tests/text-buffer_test.zig +0 -2191
  564. package/src/zig/tests/unicode-width-map.zon +0 -3909
  565. package/src/zig/tests/utf8_no_zwj_test.zig +0 -260
  566. package/src/zig/tests/utf8_test.zig +0 -4057
  567. package/src/zig/tests/utf8_wcwidth_cursor_test.zig +0 -267
  568. package/src/zig/tests/utf8_wcwidth_test.zig +0 -357
  569. package/src/zig/tests/word-wrap-editing_test.zig +0 -498
  570. package/src/zig/tests/wrap-cache-perf_test.zig +0 -113
  571. package/src/zig/text-buffer-iterators.zig +0 -499
  572. package/src/zig/text-buffer-segment.zig +0 -404
  573. package/src/zig/text-buffer-view.zig +0 -1371
  574. package/src/zig/text-buffer.zig +0 -1180
  575. package/src/zig/utf8.zig +0 -1948
  576. package/src/zig/utils.zig +0 -9
  577. package/src/zig-structs.ts +0 -261
  578. package/src/zig.ts +0 -3884
  579. package/tsconfig.build.json +0 -24
  580. package/tsconfig.json +0 -27
  581. /package/{src/lib/tree-sitter/assets → assets}/javascript/highlights.scm +0 -0
  582. /package/{src/lib/tree-sitter/assets → assets}/javascript/tree-sitter-javascript.wasm +0 -0
  583. /package/{src/lib/tree-sitter/assets → assets}/markdown/highlights.scm +0 -0
  584. /package/{src/lib/tree-sitter/assets → assets}/markdown/injections.scm +0 -0
  585. /package/{src/lib/tree-sitter/assets → assets}/markdown/tree-sitter-markdown.wasm +0 -0
  586. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/highlights.scm +0 -0
  587. /package/{src/lib/tree-sitter/assets → assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
  588. /package/{src/lib/tree-sitter/assets → assets}/typescript/highlights.scm +0 -0
  589. /package/{src/lib/tree-sitter/assets → assets}/typescript/tree-sitter-typescript.wasm +0 -0
  590. /package/{src/lib/tree-sitter/assets → assets}/zig/highlights.scm +0 -0
  591. /package/{src/lib/tree-sitter/assets → assets}/zig/tree-sitter-zig.wasm +0 -0
package/src/console.ts DELETED
@@ -1,1254 +0,0 @@
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
- }