@fairyhunter13/opentui-core 0.1.114 → 0.1.115

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-j4m38kjn.js +411 -0
  28. package/index-j4m38kjn.js.map +10 -0
  29. package/index-tse8gzh0.js +20614 -0
  30. package/index-tse8gzh0.js.map +67 -0
  31. package/index-vv2jcd4r.js +12299 -0
  32. package/index-vv2jcd4r.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 +50 -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
@@ -1,1866 +0,0 @@
1
- import { describe, expect, it, beforeEach, afterEach } from "bun:test"
2
- import { createTestRenderer, type TestRenderer, type MockInput } from "../../testing/test-renderer.js"
3
- import { createTextareaRenderable } from "./renderable-test-utils.js"
4
- import { RGBA } from "../../lib/RGBA.js"
5
- import { SyntaxStyle } from "../../syntax-style.js"
6
- import { OptimizedBuffer } from "../../buffer.js"
7
- import { fg, t } from "../../lib/index.js"
8
- import { BoxRenderable } from "../Box.js"
9
- import { TextareaRenderable } from "../Textarea.js"
10
- import { TextRenderable } from "../Text.js"
11
-
12
- let currentRenderer: TestRenderer
13
- let renderOnce: () => Promise<void>
14
- let currentMockInput: MockInput
15
- let captureFrame: () => string
16
- let resize: (width: number, height: number) => void
17
-
18
- describe("Textarea - Rendering Tests", () => {
19
- beforeEach(async () => {
20
- ;({
21
- renderer: currentRenderer,
22
- renderOnce,
23
- captureCharFrame: captureFrame,
24
- mockInput: currentMockInput,
25
- resize,
26
- } = await createTestRenderer({
27
- width: 80,
28
- height: 24,
29
- }))
30
- })
31
-
32
- afterEach(() => {
33
- currentRenderer.destroy()
34
- })
35
-
36
- describe("Wrapping", () => {
37
- it("should move cursor down through all wrapped visual lines at column 0", async () => {
38
- // Create a long line that will wrap into multiple visual lines
39
- const longText =
40
- "This is a very long line that will definitely wrap into multiple visual lines when the viewport is small"
41
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
42
- initialValue: longText,
43
- width: 20, // Small viewport to force wrapping
44
- height: 10,
45
- wrapMode: "word",
46
- })
47
-
48
- editor.focus()
49
-
50
- // Set cursor at the beginning (0, 0) - logical position
51
- editor.editBuffer.setCursor(0, 0)
52
- await renderOnce()
53
-
54
- // Get initial visual cursor position - should be at visual 0, 0
55
- let visualCursor = editor.editorView.getVisualCursor()
56
- expect(visualCursor.visualRow).toBe(0)
57
- expect(visualCursor.visualCol).toBe(0)
58
-
59
- // Verify we have multiple wrapped lines (should be 7 for this text)
60
- const vlineCount = editor.editorView.getVirtualLineCount()
61
- expect(vlineCount).toBeGreaterThan(1)
62
-
63
- // Move down through each wrapped line - cursor should stay at column 0
64
- for (let i = 1; i < vlineCount; i++) {
65
- currentMockInput.pressArrow("down")
66
- await renderOnce()
67
-
68
- visualCursor = editor.editorView.getVisualCursor()
69
-
70
- // Cursor should have moved down to the next visual line
71
- expect(visualCursor.visualRow).toBe(i)
72
-
73
- // Cursor should be at column 0 (beginning of each wrapped line)
74
- expect(visualCursor.visualCol).toBe(0)
75
- }
76
-
77
- // After moving through all wrapped lines, we should be at the last wrapped line
78
- expect(visualCursor.visualRow).toBe(vlineCount - 1)
79
- expect(visualCursor.visualCol).toBe(0)
80
- })
81
-
82
- it("should move cursor up through all wrapped visual lines at column 0", async () => {
83
- // Create a long line that will wrap into multiple visual lines
84
- const longText =
85
- "This is a very long line that will definitely wrap into multiple visual lines when the viewport is small"
86
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
87
- initialValue: longText,
88
- width: 20, // Small viewport to force wrapping
89
- height: 10,
90
- wrapMode: "word",
91
- })
92
-
93
- editor.focus()
94
-
95
- // Verify we have multiple wrapped lines
96
- const vlineCount = editor.editorView.getVirtualLineCount()
97
- expect(vlineCount).toBeGreaterThan(1)
98
-
99
- // Start at the END of the line (which will be on the last wrapped visual line)
100
- const eol = editor.editBuffer.getEOL()
101
- editor.editBuffer.setCursor(eol.row, eol.col)
102
- await renderOnce()
103
-
104
- // Move to the beginning of the last wrapped line (column 0 of last visual line)
105
- let visualCursor = editor.editorView.getVisualCursor()
106
- const lastVisualRow = visualCursor.visualRow
107
-
108
- // Set cursor to column 0 of the last wrapped visual line by finding its logical column
109
- // Last visual line starts at a specific logical column - we need to find it
110
- const lastVlineStartCol = editor.logicalCursor.col - visualCursor.visualCol
111
- editor.editBuffer.setCursor(0, lastVlineStartCol)
112
- await renderOnce()
113
-
114
- visualCursor = editor.editorView.getVisualCursor()
115
- expect(visualCursor.visualRow).toBe(lastVisualRow)
116
- expect(visualCursor.visualCol).toBe(0)
117
-
118
- // Now move UP through each wrapped line - cursor should stay at column 0
119
- for (let i = lastVisualRow - 1; i >= 0; i--) {
120
- currentMockInput.pressArrow("up")
121
- await renderOnce()
122
-
123
- visualCursor = editor.editorView.getVisualCursor()
124
-
125
- // Cursor should have moved up to the previous visual line
126
- expect(visualCursor.visualRow).toBe(i)
127
-
128
- // Cursor should be at column 0 (beginning of each wrapped line)
129
- expect(visualCursor.visualCol).toBe(0)
130
- }
131
-
132
- // After moving through all wrapped lines, we should be at the first wrapped line
133
- expect(visualCursor.visualRow).toBe(0)
134
- expect(visualCursor.visualCol).toBe(0)
135
- })
136
-
137
- it("should handle wrap mode property", async () => {
138
- const longText = "A".repeat(100)
139
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
140
- initialValue: longText,
141
- width: 20,
142
- height: 10,
143
- wrapMode: "word",
144
- })
145
-
146
- expect(editor.wrapMode).toBe("word")
147
- const wrappedCount = editor.editorView.getVirtualLineCount()
148
- expect(wrappedCount).toBeGreaterThan(1)
149
-
150
- editor.wrapMode = "none"
151
- expect(editor.wrapMode).toBe("none")
152
- const unwrappedCount = editor.editorView.getVirtualLineCount()
153
- expect(unwrappedCount).toBe(1)
154
- })
155
-
156
- it("should handle wrapMode changes", async () => {
157
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
158
- initialValue: "Hello wonderful world",
159
- width: 12,
160
- height: 10,
161
- wrapMode: "char",
162
- })
163
-
164
- expect(editor.wrapMode).toBe("char")
165
-
166
- editor.wrapMode = "word"
167
- expect(editor.wrapMode).toBe("word")
168
- })
169
-
170
- it("should render with tab indicator correctly", async () => {
171
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
172
- initialValue: "Line 1\tTabbed\nLine 2\t\tDouble tab",
173
- tabIndicator: "→",
174
- tabIndicatorColor: RGBA.fromValues(0.5, 0.5, 0.5, 1),
175
- width: 40,
176
- height: 10,
177
- })
178
-
179
- await renderOnce()
180
- const frame = captureFrame()
181
- expect(frame).toMatchSnapshot()
182
- })
183
- })
184
-
185
- describe("Height and Width Measurement", () => {
186
- it("should grow height for multiline text without wrapping", async () => {
187
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
188
- initialValue: "Line 1\nLine 2\nLine 3\nLine 4\nLine 5",
189
- wrapMode: "none",
190
- width: 40,
191
- })
192
-
193
- await renderOnce()
194
-
195
- expect(editor.height).toBe(5)
196
- expect(editor.width).toBeGreaterThanOrEqual(6)
197
- })
198
-
199
- it("should grow height for wrapped text when wrapping enabled", async () => {
200
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
201
- initialValue: "This is a very long line that will definitely wrap to multiple lines",
202
- wrapMode: "word",
203
- width: 15,
204
- })
205
-
206
- await renderOnce()
207
-
208
- expect(editor.height).toBeGreaterThan(1)
209
- expect(editor.width).toBeLessThanOrEqual(15)
210
- })
211
-
212
- it("should measure full width when wrapping is disabled and not constrained by parent", async () => {
213
- const longLine = "This is a very long line that would wrap but wrapping is disabled"
214
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
215
- initialValue: longLine,
216
- wrapMode: "none",
217
- position: "absolute",
218
- })
219
-
220
- await renderOnce()
221
-
222
- expect(editor.height).toBe(1)
223
- expect(editor.width).toBe(longLine.length)
224
- })
225
-
226
- it("should shrink height when deleting lines via value setter", async () => {
227
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
228
- initialValue: "Line 1\nLine 2\nLine 3\nLine 4\nLine 5",
229
- width: 40,
230
- wrapMode: "none",
231
- })
232
-
233
- editor.focus()
234
- await renderOnce()
235
- expect(editor.height).toBe(5)
236
-
237
- // Remove lines by setting new value
238
- editor.setText("Line 1\nLine 2")
239
- await renderOnce()
240
-
241
- expect(editor.height).toBe(2)
242
- expect(editor.plainText).toBe("Line 1\nLine 2")
243
- })
244
-
245
- it("should update height when content changes from single to multiline", async () => {
246
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
247
- initialValue: "Single line",
248
- wrapMode: "none",
249
- })
250
-
251
- await renderOnce()
252
- expect(editor.height).toBe(1)
253
-
254
- editor.setText("Line 1\nLine 2\nLine 3")
255
- await renderOnce()
256
-
257
- expect(editor.height).toBe(3)
258
- })
259
-
260
- it("should grow height when pressing Enter to add newlines", async () => {
261
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
262
- initialValue: "Single line",
263
- width: 40,
264
- wrapMode: "none",
265
- })
266
-
267
- // Add a second textarea below to verify layout reflow
268
- const { textarea: belowEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
269
- initialValue: "Below",
270
- width: 40,
271
- })
272
-
273
- await renderOnce()
274
- expect(editor.height).toBe(1)
275
- const initialHeight = editor.height
276
- const initialBelowY = belowEditor.y
277
-
278
- editor.focus()
279
- editor.gotoLine(9999) // Move to end
280
-
281
- // Press Enter 3 times to add 3 newlines
282
- currentMockInput.pressEnter()
283
- expect(editor.plainText).toBe("Single line\n")
284
- await renderOnce() // Wait for layout recalculation
285
-
286
- currentMockInput.pressEnter()
287
- expect(editor.plainText).toBe("Single line\n\n")
288
- await renderOnce() // Wait for layout recalculation
289
-
290
- currentMockInput.pressEnter()
291
- expect(editor.plainText).toBe("Single line\n\n\n")
292
- await renderOnce() // Wait for layout recalculation
293
-
294
- // The editor should have grown
295
- expect(editor.height).toBeGreaterThan(initialHeight)
296
- expect(editor.height).toBe(4) // 1 original line + 3 new lines
297
- expect(editor.plainText).toBe("Single line\n\n\n")
298
-
299
- // The element below should have moved down
300
- expect(belowEditor.y).toBeGreaterThan(initialBelowY)
301
- expect(belowEditor.y).toBe(4) // After the 4-line editor
302
- })
303
- })
304
-
305
- describe("Unicode Support", () => {
306
- it("should handle emoji insertion", async () => {
307
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
308
- initialValue: "Hello",
309
- width: 40,
310
- height: 10,
311
- })
312
-
313
- editor.focus()
314
- editor.gotoLine(9999) // Move to end
315
- editor.insertText(" 🌟")
316
-
317
- expect(editor.plainText).toBe("Hello 🌟")
318
- })
319
-
320
- it("should handle CJK characters", async () => {
321
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
322
- initialValue: "Hello",
323
- width: 40,
324
- height: 10,
325
- })
326
-
327
- editor.focus()
328
- editor.gotoLine(9999) // Move to end
329
- editor.insertText(" 世界")
330
-
331
- expect(editor.plainText).toBe("Hello 世界")
332
- })
333
-
334
- it("should handle emoji cursor movement", async () => {
335
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
336
- initialValue: "A🌟B",
337
- width: 40,
338
- height: 10,
339
- })
340
-
341
- editor.focus()
342
- expect(editor.logicalCursor.col).toBe(0)
343
-
344
- currentMockInput.pressArrow("right") // Move past A
345
- expect(editor.logicalCursor.col).toBe(1)
346
-
347
- currentMockInput.pressArrow("right") // Move past emoji (2 cells)
348
- expect(editor.logicalCursor.col).toBe(3)
349
-
350
- currentMockInput.pressArrow("right") // Move past B
351
- expect(editor.logicalCursor.col).toBe(4)
352
- })
353
- })
354
-
355
- describe("Content Property", () => {
356
- it("should update content programmatically", async () => {
357
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
358
- initialValue: "Initial",
359
- width: 40,
360
- height: 10,
361
- })
362
-
363
- editor.setText("Updated")
364
- expect(editor.plainText).toBe("Updated")
365
- expect(editor.plainText).toBe("Updated")
366
- })
367
-
368
- it("should reset cursor when content changes", async () => {
369
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
370
- initialValue: "Hello World",
371
- width: 40,
372
- height: 10,
373
- })
374
-
375
- editor.gotoLine(9999) // Move to end
376
- expect(editor.logicalCursor.col).toBe(11)
377
-
378
- editor.setText("New")
379
- // Cursor should reset to start
380
- expect(editor.logicalCursor.row).toBe(0)
381
- expect(editor.logicalCursor.col).toBe(0)
382
- })
383
-
384
- it("should clear text with clear() method", async () => {
385
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
386
- initialValue: "Hello World",
387
- width: 40,
388
- height: 10,
389
- })
390
-
391
- expect(editor.plainText).toBe("Hello World")
392
-
393
- editor.clear()
394
- expect(editor.plainText).toBe("")
395
- })
396
-
397
- it("should clear highlights with clear() method", async () => {
398
- const style = SyntaxStyle.create()
399
- const styleId = style.registerStyle("highlight", {
400
- fg: RGBA.fromValues(1, 0, 0, 1),
401
- })
402
-
403
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
404
- initialValue: "Hello World",
405
- width: 40,
406
- height: 10,
407
- syntaxStyle: style,
408
- })
409
-
410
- editor.addHighlightByCharRange({
411
- start: 0,
412
- end: 5,
413
- styleId: styleId,
414
- priority: 0,
415
- })
416
-
417
- const highlightsBefore = editor.getLineHighlights(0)
418
- expect(highlightsBefore.length).toBeGreaterThan(0)
419
-
420
- editor.clear()
421
-
422
- expect(editor.plainText).toBe("")
423
- const highlightsAfter = editor.getLineHighlights(0)
424
- expect(highlightsAfter.length).toBe(0)
425
- })
426
-
427
- it("should clear both text and highlights together", async () => {
428
- const style = SyntaxStyle.create()
429
- const styleId = style.registerStyle("highlight", {
430
- fg: RGBA.fromValues(1, 0, 0, 1),
431
- })
432
-
433
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
434
- initialValue: "Line 1\nLine 2\nLine 3",
435
- width: 40,
436
- height: 10,
437
- syntaxStyle: style,
438
- })
439
-
440
- editor.addHighlight(0, { start: 0, end: 6, styleId: styleId, priority: 0 })
441
- editor.addHighlight(1, { start: 0, end: 6, styleId: styleId, priority: 0 })
442
-
443
- expect(editor.plainText).toBe("Line 1\nLine 2\nLine 3")
444
- expect(editor.getLineHighlights(0).length).toBe(1)
445
- expect(editor.getLineHighlights(1).length).toBe(1)
446
-
447
- editor.clear()
448
-
449
- expect(editor.plainText).toBe("")
450
- expect(editor.getLineHighlights(0).length).toBe(0)
451
- expect(editor.getLineHighlights(1).length).toBe(0)
452
- })
453
-
454
- it("should allow typing after clear()", async () => {
455
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
456
- initialValue: "Hello World",
457
- width: 40,
458
- height: 10,
459
- })
460
-
461
- editor.focus()
462
- expect(editor.plainText).toBe("Hello World")
463
-
464
- currentMockInput.pressKey("!")
465
- expect(editor.plainText).toBe("!Hello World")
466
-
467
- editor.clear()
468
- expect(editor.plainText).toBe("")
469
-
470
- currentMockInput.pressKey("N")
471
- currentMockInput.pressKey("e")
472
- currentMockInput.pressKey("w")
473
- expect(editor.plainText).toBe("New")
474
-
475
- currentMockInput.pressKey(" ")
476
- currentMockInput.pressKey("T")
477
- currentMockInput.pressKey("e")
478
- currentMockInput.pressKey("x")
479
- currentMockInput.pressKey("t")
480
- expect(editor.plainText).toBe("New Text")
481
- })
482
- })
483
-
484
- describe("Rendering After Edits", () => {
485
- it("should render correctly after insert text", async () => {
486
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
487
- initialValue: "Test",
488
- width: 40,
489
- height: 10,
490
- })
491
-
492
- editor.focus()
493
- editor.insertText("x")
494
-
495
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
496
- buffer.drawEditorView(editor.editorView, 0, 0)
497
-
498
- expect(editor.plainText).toBe("xTest")
499
- expect(editor.logicalCursor.col).toBe(1)
500
-
501
- buffer.destroy()
502
- })
503
-
504
- it("should render correctly after rapid edits", async () => {
505
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
506
- initialValue: "",
507
- width: 40,
508
- height: 10,
509
- })
510
-
511
- editor.focus()
512
-
513
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
514
-
515
- for (let i = 0; i < 5; i++) {
516
- editor.insertText("a")
517
- buffer.drawEditorView(editor.editorView, 0, 0)
518
- }
519
-
520
- expect(editor.plainText).toBe("aaaaa")
521
- expect(editor.logicalCursor.col).toBe(5)
522
-
523
- buffer.destroy()
524
- })
525
-
526
- it("should render correctly after newline", async () => {
527
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
528
- initialValue: "Hello",
529
- width: 40,
530
- height: 10,
531
- })
532
-
533
- editor.focus()
534
- editor.gotoLine(9999)
535
-
536
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
537
-
538
- editor.newLine()
539
- buffer.drawEditorView(editor.editorView, 0, 0)
540
-
541
- expect(editor.plainText).toBe("Hello\n")
542
- expect(editor.logicalCursor.row).toBe(1)
543
- expect(editor.logicalCursor.col).toBe(0)
544
-
545
- buffer.destroy()
546
- })
547
-
548
- it("should render correctly after backspace", async () => {
549
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
550
- initialValue: "Hello",
551
- width: 40,
552
- height: 10,
553
- })
554
-
555
- editor.focus()
556
- editor.gotoLine(9999)
557
-
558
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
559
-
560
- editor.deleteCharBackward()
561
- buffer.drawEditorView(editor.editorView, 0, 0)
562
-
563
- expect(editor.plainText).toBe("Hell")
564
- expect(editor.logicalCursor.col).toBe(4)
565
-
566
- buffer.destroy()
567
- })
568
-
569
- it("should render correctly with draw-edit-draw pattern", async () => {
570
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
571
- initialValue: "Test",
572
- width: 40,
573
- height: 10,
574
- })
575
-
576
- editor.focus()
577
-
578
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
579
-
580
- buffer.drawEditorView(editor.editorView, 0, 0)
581
- editor.insertText("x")
582
- buffer.drawEditorView(editor.editorView, 0, 0)
583
-
584
- expect(editor.plainText).toBe("xTest")
585
- expect(editor.logicalCursor.col).toBe(1)
586
-
587
- buffer.destroy()
588
- })
589
-
590
- it("should render correctly after multiple text buffer modifications", async () => {
591
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
592
- initialValue: "Line1\nLine2\nLine3",
593
- width: 40,
594
- height: 10,
595
- })
596
-
597
- editor.focus()
598
-
599
- const buffer = OptimizedBuffer.create(80, 24, "wcwidth")
600
-
601
- buffer.drawEditorView(editor.editorView, 0, 0)
602
-
603
- editor.insertText("X")
604
- buffer.drawEditorView(editor.editorView, 0, 0)
605
- expect(editor.plainText).toBe("XLine1\nLine2\nLine3")
606
-
607
- editor.newLine()
608
- buffer.drawEditorView(editor.editorView, 0, 0)
609
- expect(editor.plainText).toBe("X\nLine1\nLine2\nLine3")
610
-
611
- editor.deleteCharBackward()
612
- buffer.drawEditorView(editor.editorView, 0, 0)
613
- expect(editor.plainText).toBe("XLine1\nLine2\nLine3")
614
-
615
- buffer.destroy()
616
- })
617
- })
618
-
619
- describe("Viewport Scrolling", () => {
620
- it("should scroll viewport down when cursor moves below visible area", async () => {
621
- // Create editor with small viewport
622
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
623
- initialValue: "Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9",
624
- width: 40,
625
- height: 5, // Only 5 lines visible
626
- })
627
-
628
- editor.focus()
629
-
630
- // Initial viewport should show lines 0-4
631
- let viewport = editor.editorView.getViewport()
632
- expect(viewport.offsetY).toBe(0)
633
- expect(viewport.height).toBe(5)
634
-
635
- // Move cursor to line 7 (beyond viewport)
636
- editor.gotoLine(7)
637
-
638
- // Viewport should have scrolled to keep cursor visible
639
- viewport = editor.editorView.getViewport()
640
- // With scroll margin of 0.2 (20% = 1 line), viewport should scroll to show line 7
641
- // Expected: offsetY should be at least 3 (to show lines 3-7)
642
- expect(viewport.offsetY).toBeGreaterThanOrEqual(3)
643
- })
644
-
645
- it("should scroll viewport up when cursor moves above visible area", async () => {
646
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
647
- initialValue: "Line 0\nLine 1\nLine 2\nLine 3\nLine 4\nLine 5\nLine 6\nLine 7\nLine 8\nLine 9",
648
- width: 40,
649
- height: 5,
650
- })
651
-
652
- editor.focus()
653
-
654
- // Start at line 8
655
- editor.gotoLine(8)
656
-
657
- let viewport = editor.editorView.getViewport()
658
- // Viewport should have automatically scrolled to show line 8
659
- expect(viewport.offsetY).toBeGreaterThan(0)
660
-
661
- // Now move to line 1 (above viewport)
662
- editor.gotoLine(1)
663
-
664
- viewport = editor.editorView.getViewport()
665
- // Viewport should have scrolled up to show line 1
666
- expect(viewport.offsetY).toBeLessThanOrEqual(1)
667
- })
668
-
669
- it("should scroll viewport when using arrow keys to move beyond visible area", async () => {
670
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
671
- initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
672
- width: 40,
673
- height: 5,
674
- })
675
-
676
- editor.focus()
677
-
678
- let viewport = editor.editorView.getViewport()
679
- expect(viewport.offsetY).toBe(0)
680
-
681
- // Press down arrow 6 times to move beyond initial viewport
682
- for (let i = 0; i < 6; i++) {
683
- currentMockInput.pressArrow("down")
684
- }
685
-
686
- viewport = editor.editorView.getViewport()
687
- // Should have scrolled
688
- expect(viewport.offsetY).toBeGreaterThan(0)
689
- })
690
-
691
- it("should maintain scroll margin when moving cursor", async () => {
692
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
693
- initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
694
- width: 40,
695
- height: 10,
696
- scrollMargin: 0.2, // 20% = 2 lines margin
697
- })
698
-
699
- editor.focus()
700
-
701
- // Move to line 8 (near bottom of initial viewport)
702
- editor.gotoLine(8)
703
-
704
- let viewport = editor.editorView.getViewport()
705
-
706
- // With 2-line margin, cursor at line 8 should trigger scroll
707
- // so that line 8 is at most at position 8 in viewport
708
- expect(viewport.offsetY).toBeGreaterThanOrEqual(0)
709
- })
710
-
711
- it("should handle viewport scrolling with text wrapping", async () => {
712
- const longLine = "word ".repeat(50) // Creates line that will wrap
713
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
714
- initialValue: Array.from({ length: 10 }, (_, i) => (i === 5 ? longLine : `Line ${i}`)).join("\n"),
715
- width: 20,
716
- height: 5,
717
- wrapMode: "word",
718
- })
719
-
720
- editor.focus()
721
-
722
- // Move to the long line
723
- editor.gotoLine(5)
724
-
725
- const vlineCount = editor.editorView.getTotalVirtualLineCount()
726
- expect(vlineCount).toBeGreaterThan(10) // Should be more due to wrapping
727
-
728
- // Move to end of long line
729
- const cursor = editor.logicalCursor
730
- editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
731
-
732
- let viewport = editor.editorView.getViewport()
733
-
734
- // Viewport should have scrolled to show cursor
735
- // This is complex with wrapping - we need virtual line scrolling
736
- })
737
-
738
- it("should verify viewport follows cursor to line 10", async () => {
739
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
740
- initialValue: Array.from({ length: 20 }, (_, i) => `Line ${i}`).join("\n"),
741
- width: 40,
742
- height: 8,
743
- })
744
-
745
- editor.focus()
746
-
747
- // Move to line 10
748
- editor.gotoLine(10)
749
-
750
- const viewport = editor.editorView.getViewport()
751
-
752
- // Viewport should have scrolled to show line 10
753
- // With height=8 and scroll margin, line 10 should be visible
754
- expect(viewport.offsetY).toBeGreaterThan(0)
755
- expect(viewport.offsetY).toBeLessThanOrEqual(10)
756
-
757
- // Line 10 should be within the viewport range
758
- const viewportEnd = viewport.offsetY + viewport.height
759
- expect(10).toBeGreaterThanOrEqual(viewport.offsetY)
760
- expect(10).toBeLessThan(viewportEnd)
761
- })
762
-
763
- it("should track viewport offset as cursor moves through document", async () => {
764
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
765
- initialValue: Array.from({ length: 15 }, (_, i) => `Line ${i}`).join("\n"),
766
- width: 30,
767
- height: 5,
768
- })
769
-
770
- editor.focus()
771
-
772
- const viewportOffsets: number[] = []
773
-
774
- // Track viewport offset at different cursor positions
775
- for (const line of [0, 2, 4, 6, 8, 10, 12]) {
776
- editor.gotoLine(line)
777
- const viewport = editor.editorView.getViewport()
778
- viewportOffsets.push(viewport.offsetY)
779
- }
780
-
781
- // Viewport should generally increase as cursor moves down
782
- // (with possible plateaus when cursor is already visible)
783
- const lastOffset = viewportOffsets[viewportOffsets.length - 1]
784
- const firstOffset = viewportOffsets[0]
785
- expect(lastOffset).toBeGreaterThan(firstOffset)
786
-
787
- // At line 0, viewport should be at 0
788
- expect(viewportOffsets[0]).toBe(0)
789
-
790
- // At line 12, viewport should have scrolled
791
- expect(viewportOffsets[viewportOffsets.length - 1]).toBeGreaterThan(5)
792
- })
793
-
794
- it("should scroll viewport when cursor moves with Page Up/Page Down", async () => {
795
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
796
- initialValue: Array.from({ length: 30 }, (_, i) => `Line ${i}`).join("\n"),
797
- width: 40,
798
- height: 10,
799
- })
800
-
801
- editor.focus()
802
-
803
- let viewport = editor.editorView.getViewport()
804
- expect(viewport.offsetY).toBe(0)
805
-
806
- // Move down 15 lines (more than viewport height)
807
- for (let i = 0; i < 15; i++) {
808
- editor.moveCursorDown()
809
- }
810
-
811
- viewport = editor.editorView.getViewport()
812
-
813
- // Should have scrolled
814
- expect(viewport.offsetY).toBeGreaterThan(0)
815
- expect(editor.logicalCursor.row).toBe(15)
816
- })
817
-
818
- it("should scroll viewport down when pressing Enter repeatedly", async () => {
819
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
820
- initialValue: "Start",
821
- width: 40,
822
- height: 5,
823
- })
824
-
825
- editor.focus()
826
- editor.gotoLine(9999) // Move to end
827
-
828
- let viewport = editor.editorView.getViewport()
829
- expect(viewport.offsetY).toBe(0)
830
- expect(editor.logicalCursor.row).toBe(0)
831
-
832
- // Press Enter 8 times to create 8 new lines
833
- for (let i = 0; i < 8; i++) {
834
- currentMockInput.pressEnter()
835
- }
836
-
837
- // After 8 Enters, we should have 9 lines total (0-8)
838
- expect(editor.logicalCursor.row).toBe(8)
839
-
840
- // Viewport should have scrolled to keep cursor visible
841
- viewport = editor.editorView.getViewport()
842
- expect(viewport.offsetY).toBeGreaterThan(0)
843
-
844
- // Cursor should be visible in viewport
845
- const cursorLine = editor.logicalCursor.row
846
- expect(cursorLine).toBeGreaterThanOrEqual(viewport.offsetY)
847
- expect(cursorLine).toBeLessThan(viewport.offsetY + viewport.height)
848
- })
849
-
850
- it("should scroll viewport up when pressing Backspace to delete characters and move up", async () => {
851
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
852
- initialValue: Array.from({ length: 15 }, (_, i) => `Line ${i}`).join("\n"),
853
- width: 40,
854
- height: 5,
855
- })
856
-
857
- editor.focus()
858
-
859
- // Start at line 10, move to end so we have characters to delete
860
- editor.gotoLine(10)
861
- let cursor = editor.logicalCursor
862
- editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
863
-
864
- let viewport = editor.editorView.getViewport()
865
- expect(viewport.offsetY).toBeGreaterThan(0)
866
- const initialOffset = viewport.offsetY
867
-
868
- // Delete all text and move cursor up to line 0
869
- // Press Ctrl+A to go to start, then move to line 2, then backspace repeatedly
870
- editor.gotoLine(0) // Move to start
871
- editor.gotoLine(2)
872
- cursor = editor.logicalCursor
873
- editor.editBuffer.setCursorToLineCol(cursor.row, 9999) // Move to end of line
874
-
875
- // Now we're at line 2, and viewport should have scrolled up
876
- viewport = editor.editorView.getViewport()
877
-
878
- // Viewport should have scrolled up from initial position
879
- expect(viewport.offsetY).toBeLessThan(initialOffset)
880
- expect(editor.logicalCursor.row).toBe(2)
881
- })
882
-
883
- it("should scroll viewport when typing at end creates wrapped lines beyond viewport", async () => {
884
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
885
- initialValue: "Start",
886
- width: 20,
887
- height: 5,
888
- wrapMode: "word",
889
- })
890
-
891
- editor.focus()
892
- editor.gotoLine(9999) // Move to end
893
-
894
- let viewport = editor.editorView.getViewport()
895
- expect(viewport.offsetY).toBe(0)
896
-
897
- // Type enough to create multiple wrapped lines
898
- const longText = " word".repeat(50)
899
- for (const char of longText) {
900
- currentMockInput.pressKey(char)
901
- }
902
-
903
- viewport = editor.editorView.getViewport()
904
- const vlineCount = editor.editorView.getTotalVirtualLineCount()
905
-
906
- // Should have created multiple virtual lines
907
- expect(vlineCount).toBeGreaterThan(5)
908
-
909
- // Viewport should have scrolled to keep cursor visible
910
- // (This test may fail if virtual line scrolling isn't implemented yet)
911
- expect(viewport.offsetY).toBeGreaterThanOrEqual(0)
912
- })
913
-
914
- it("should scroll viewport when using Enter to add lines, then Backspace to remove them", async () => {
915
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
916
- initialValue: "Line 0\nLine 1\nLine 2",
917
- width: 40,
918
- height: 5,
919
- })
920
-
921
- editor.focus()
922
- editor.gotoLine(9999) // Move to end
923
-
924
- let viewport = editor.editorView.getViewport()
925
- const initialOffset = viewport.offsetY
926
-
927
- // Add 6 new lines
928
- for (let i = 0; i < 6; i++) {
929
- currentMockInput.pressEnter()
930
- currentMockInput.pressKey("X")
931
- }
932
-
933
- // Should have scrolled down
934
- viewport = editor.editorView.getViewport()
935
- expect(viewport.offsetY).toBeGreaterThan(initialOffset)
936
- const maxOffset = viewport.offsetY
937
-
938
- // Now delete those lines by backspacing
939
- for (let i = 0; i < 12; i++) {
940
- // 12 backspaces to delete 6 "X\n" pairs
941
- currentMockInput.pressBackspace()
942
- }
943
-
944
- // Should have scrolled back up
945
- viewport = editor.editorView.getViewport()
946
- expect(viewport.offsetY).toBeLessThan(maxOffset)
947
- })
948
-
949
- it("should show last line at bottom of viewport with no gap", async () => {
950
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
951
- initialValue: Array.from({ length: 10 }, (_, i) => `Line ${i}`).join("\n"),
952
- width: 40,
953
- height: 5,
954
- })
955
-
956
- editor.focus()
957
-
958
- // Move to last line (line 9)
959
- editor.gotoLine(9)
960
-
961
- let viewport = editor.editorView.getViewport()
962
-
963
- // With 10 lines (0-9) and viewport height 5, max offset is 10 - 5 = 5
964
- // Viewport should be at offset 5, showing lines 5-9 with line 9 at the bottom
965
- expect(viewport.offsetY).toBe(5)
966
-
967
- // Verify cursor line is visible
968
- expect(9).toBeGreaterThanOrEqual(viewport.offsetY)
969
- expect(9).toBeLessThan(viewport.offsetY + viewport.height)
970
-
971
- // No gap - last visible line should be the last line of content
972
- const lastVisibleLine = viewport.offsetY + viewport.height - 1
973
- expect(lastVisibleLine).toBe(9)
974
- })
975
-
976
- it("should not scroll past end when document is smaller than viewport", async () => {
977
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
978
- initialValue: "Line 0\nLine 1\nLine 2",
979
- width: 40,
980
- height: 10, // Viewport bigger than content
981
- })
982
-
983
- editor.focus()
984
-
985
- // Move to last line
986
- editor.gotoLine(2)
987
-
988
- let viewport = editor.editorView.getViewport()
989
-
990
- // Should NOT scroll at all - content fits in viewport
991
- expect(viewport.offsetY).toBe(0)
992
- })
993
- })
994
-
995
- describe("Placeholder Support", () => {
996
- it("should display placeholder when empty", async () => {
997
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
998
- initialValue: "",
999
- width: 40,
1000
- height: 10,
1001
- placeholder: "Enter text here...",
1002
- })
1003
-
1004
- // plainText should return empty (placeholder is display-only)
1005
- expect(editor.plainText).toBe("")
1006
- expect(editor.placeholder).toBe("Enter text here...")
1007
- })
1008
-
1009
- it("should hide placeholder when text is inserted", async () => {
1010
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1011
- initialValue: "",
1012
- width: 40,
1013
- height: 10,
1014
- placeholder: "Type something...",
1015
- })
1016
-
1017
- editor.focus()
1018
- expect(editor.plainText).toBe("")
1019
-
1020
- currentMockInput.pressKey("H")
1021
- currentMockInput.pressKey("i")
1022
-
1023
- expect(editor.plainText).toBe("Hi")
1024
- })
1025
-
1026
- it("should reactivate placeholder when all text is deleted", async () => {
1027
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1028
- initialValue: "Test",
1029
- width: 40,
1030
- height: 10,
1031
- placeholder: "Empty buffer...",
1032
- })
1033
-
1034
- editor.focus()
1035
- expect(editor.plainText).toBe("Test")
1036
-
1037
- // Move to end, then delete all text
1038
- editor.gotoLine(9999)
1039
- for (let i = 0; i < 4; i++) {
1040
- currentMockInput.pressBackspace()
1041
- }
1042
-
1043
- expect(editor.plainText).toBe("")
1044
- })
1045
-
1046
- it("should update placeholder text dynamically", async () => {
1047
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1048
- initialValue: "",
1049
- width: 40,
1050
- height: 10,
1051
- placeholder: "First placeholder",
1052
- })
1053
-
1054
- expect(editor.placeholder).toBe("First placeholder")
1055
- expect(editor.plainText).toBe("")
1056
-
1057
- editor.placeholder = "Second placeholder"
1058
- expect(editor.placeholder).toBe("Second placeholder")
1059
- expect(editor.plainText).toBe("")
1060
- })
1061
-
1062
- it("should update placeholder with styled text dynamically", async () => {
1063
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1064
- initialValue: "",
1065
- width: 40,
1066
- height: 10,
1067
- placeholder: "Colored placeholder",
1068
- })
1069
-
1070
- expect(editor.plainText).toBe("")
1071
-
1072
- // Update placeholder with styled text
1073
- editor.placeholder = t`${fg("#FF0000")("Red placeholder")}`
1074
- expect(editor.plainText).toBe("")
1075
- })
1076
-
1077
- it("should work with value property setter", async () => {
1078
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1079
- initialValue: "",
1080
- width: 40,
1081
- height: 10,
1082
- placeholder: "Empty state",
1083
- })
1084
-
1085
- expect(editor.plainText).toBe("")
1086
-
1087
- editor.setText("New content")
1088
- expect(editor.plainText).toBe("New content")
1089
-
1090
- editor.setText("")
1091
- expect(editor.plainText).toBe("")
1092
- })
1093
-
1094
- it("should handle placeholder with focus changes", async () => {
1095
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1096
- initialValue: "",
1097
- width: 40,
1098
- height: 10,
1099
- placeholder: "Click to edit",
1100
- })
1101
-
1102
- // Placeholder should show regardless of focus
1103
- expect(editor.plainText).toBe("")
1104
-
1105
- editor.focus()
1106
- expect(editor.plainText).toBe("")
1107
-
1108
- editor.blur()
1109
- expect(editor.plainText).toBe("")
1110
- })
1111
-
1112
- it("should handle typing after placeholder is shown", async () => {
1113
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1114
- initialValue: "",
1115
- width: 40,
1116
- height: 10,
1117
- placeholder: "Start typing...",
1118
- })
1119
-
1120
- editor.focus()
1121
- expect(editor.plainText).toBe("")
1122
-
1123
- currentMockInput.pressKey("H")
1124
- currentMockInput.pressKey("e")
1125
- currentMockInput.pressKey("l")
1126
- currentMockInput.pressKey("l")
1127
- currentMockInput.pressKey("o")
1128
-
1129
- expect(editor.plainText).toBe("Hello")
1130
- })
1131
-
1132
- it("should show placeholder after deleting all typed text", async () => {
1133
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1134
- initialValue: "",
1135
- width: 40,
1136
- height: 10,
1137
- placeholder: "Type here",
1138
- })
1139
-
1140
- editor.focus()
1141
-
1142
- // Type "Test"
1143
- currentMockInput.pressKey("T")
1144
- currentMockInput.pressKey("e")
1145
- currentMockInput.pressKey("s")
1146
- currentMockInput.pressKey("t")
1147
- expect(editor.plainText).toBe("Test")
1148
-
1149
- // Backspace all
1150
- for (let i = 0; i < 4; i++) {
1151
- currentMockInput.pressBackspace()
1152
- }
1153
-
1154
- expect(editor.plainText).toBe("")
1155
- })
1156
-
1157
- it("should handle placeholder with newlines", async () => {
1158
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1159
- initialValue: "",
1160
- width: 40,
1161
- height: 10,
1162
- placeholder: "Line 1\nLine 2",
1163
- })
1164
-
1165
- expect(editor.plainText).toBe("")
1166
-
1167
- editor.insertText("Content")
1168
- expect(editor.plainText).toBe("Content")
1169
- })
1170
-
1171
- it("should handle null placeholder (no placeholder)", async () => {
1172
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1173
- initialValue: "",
1174
- width: 40,
1175
- height: 10,
1176
- placeholder: null,
1177
- })
1178
-
1179
- expect(editor.placeholder).toBe(null)
1180
- expect(editor.plainText).toBe("")
1181
-
1182
- editor.insertText("Content")
1183
- expect(editor.plainText).toBe("Content")
1184
- })
1185
-
1186
- it("should clear placeholder when set to null", async () => {
1187
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1188
- initialValue: "",
1189
- width: 40,
1190
- height: 10,
1191
- placeholder: "Initial placeholder",
1192
- })
1193
-
1194
- expect(editor.placeholder).toBe("Initial placeholder")
1195
- expect(editor.plainText).toBe("")
1196
-
1197
- editor.placeholder = null
1198
- expect(editor.placeholder).toBe(null)
1199
- expect(editor.plainText).toBe("")
1200
- })
1201
-
1202
- it("should reset placeholder when set to undefined", async () => {
1203
- const { textarea: editor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1204
- initialValue: "",
1205
- width: 40,
1206
- height: 10,
1207
- placeholder: "Initial placeholder",
1208
- })
1209
-
1210
- expect(editor.placeholder).toBe("Initial placeholder")
1211
-
1212
- expect(() => {
1213
- editor.placeholder = undefined
1214
- }).not.toThrow()
1215
-
1216
- expect(editor.placeholder).toBe(null)
1217
- expect(editor.plainText).toBe("")
1218
- })
1219
- })
1220
-
1221
- describe("Textarea Content Snapshots", () => {
1222
- it("should render basic text content correctly", async () => {
1223
- await createTextareaRenderable(currentRenderer, renderOnce, {
1224
- initialValue: "Hello World",
1225
- left: 5,
1226
- top: 3,
1227
- width: 20,
1228
- height: 5,
1229
- })
1230
-
1231
- const frame = captureFrame()
1232
- expect(frame).toMatchSnapshot()
1233
- })
1234
-
1235
- it("should render multiline text content correctly", async () => {
1236
- await createTextareaRenderable(currentRenderer, renderOnce, {
1237
- initialValue: "Line 1: Hello\nLine 2: World\nLine 3: Testing\nLine 4: Multiline",
1238
- left: 1,
1239
- top: 1,
1240
- width: 30,
1241
- height: 10,
1242
- })
1243
-
1244
- const frame = captureFrame()
1245
- expect(frame).toMatchSnapshot()
1246
- })
1247
-
1248
- it("should render text with character wrapping correctly", async () => {
1249
- await createTextareaRenderable(currentRenderer, renderOnce, {
1250
- initialValue: "This is a very long text that should wrap to multiple lines when wrap is enabled",
1251
- wrapMode: "char",
1252
- width: 15,
1253
- left: 0,
1254
- top: 0,
1255
- })
1256
-
1257
- const frame = captureFrame()
1258
- expect(frame).toMatchSnapshot()
1259
- })
1260
-
1261
- it("should render text with word wrapping and punctuation", async () => {
1262
- await createTextareaRenderable(currentRenderer, renderOnce, {
1263
- initialValue: "Hello,World.Test-Example/Path with various punctuation marks!",
1264
- wrapMode: "word",
1265
- width: 12,
1266
- left: 0,
1267
- top: 0,
1268
- })
1269
-
1270
- const frame = captureFrame()
1271
- expect(frame).toMatchSnapshot()
1272
- })
1273
-
1274
- it("should render placeholder when creating textarea with placeholder directly", async () => {
1275
- await createTextareaRenderable(currentRenderer, renderOnce, {
1276
- placeholder: "Enter text here...",
1277
- left: 1,
1278
- top: 1,
1279
- width: 30,
1280
- height: 5,
1281
- })
1282
-
1283
- const frame = captureFrame()
1284
- expect(frame).toMatchSnapshot()
1285
- })
1286
-
1287
- it("should render placeholder when set programmatically after creation", async () => {
1288
- const { textarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1289
- left: 1,
1290
- top: 1,
1291
- width: 30,
1292
- height: 5,
1293
- })
1294
-
1295
- textarea.placeholder = "Type something..."
1296
- await renderOnce()
1297
-
1298
- const frame = captureFrame()
1299
- expect(frame).toMatchSnapshot()
1300
- })
1301
-
1302
- it("should resize correctly when typing return as first input with placeholder", async () => {
1303
- resize(40, 10)
1304
-
1305
- const container = new BoxRenderable(currentRenderer, {
1306
- border: true,
1307
- left: 1,
1308
- top: 1,
1309
- })
1310
- currentRenderer.root.add(container)
1311
-
1312
- const textarea = new TextareaRenderable(currentRenderer, {
1313
- placeholder: "Enter your message...",
1314
- width: 30,
1315
- minHeight: 1,
1316
- maxHeight: 3,
1317
- })
1318
- container.add(textarea)
1319
-
1320
- textarea.focus()
1321
- await renderOnce()
1322
-
1323
- const frameBeforeEnter = captureFrame()
1324
- expect(textarea.height).toBe(1)
1325
-
1326
- currentMockInput.pressEnter()
1327
- await renderOnce()
1328
- await renderOnce()
1329
-
1330
- const frameAfterEnter = captureFrame()
1331
- expect(frameAfterEnter).toMatchSnapshot()
1332
- expect(textarea.height).toBe(2)
1333
- expect(textarea.plainText).toBe("\n")
1334
- })
1335
- })
1336
-
1337
- describe("Layout Reflow on Size Change", () => {
1338
- it("should reflow subsequent elements when textarea grows and shrinks", async () => {
1339
- const { textarea: firstEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1340
- initialValue: "Short",
1341
- width: 20,
1342
- wrapMode: "word",
1343
- })
1344
-
1345
- const { textarea: secondEditor } = await createTextareaRenderable(currentRenderer, renderOnce, {
1346
- initialValue: "I am below the first textarea",
1347
- width: 30,
1348
- })
1349
-
1350
- await renderOnce()
1351
-
1352
- // Initially, first editor is 1 line high
1353
- expect(firstEditor.height).toBe(1)
1354
- const initialSecondY = secondEditor.y
1355
- expect(initialSecondY).toBe(1) // Right after first editor
1356
-
1357
- // Expand first editor with wrapped content
1358
- firstEditor.setText("This is a very long line that will wrap to multiple lines and push the second textarea down")
1359
- await renderOnce()
1360
-
1361
- // First editor should now be taller
1362
- expect(firstEditor.height).toBeGreaterThan(1)
1363
- // Second editor should have moved down
1364
- expect(secondEditor.y).toBeGreaterThan(initialSecondY)
1365
- const expandedSecondY = secondEditor.y
1366
-
1367
- // Shrink first editor back
1368
- firstEditor.setText("Short again")
1369
- await renderOnce()
1370
-
1371
- // First editor should be 1 line again
1372
- expect(firstEditor.height).toBe(1)
1373
- // Second editor should have moved back up
1374
- expect(secondEditor.y).toBeLessThan(expandedSecondY)
1375
- expect(secondEditor.y).toBe(initialSecondY)
1376
- })
1377
- })
1378
-
1379
- describe("Width/Height Setter Layout Tests", () => {
1380
- it("should not shrink box when width is set via setter", async () => {
1381
- resize(40, 10)
1382
-
1383
- const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
1384
- currentRenderer.root.add(container)
1385
-
1386
- const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
1387
- container.add(row)
1388
-
1389
- const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00" })
1390
- row.add(indicator)
1391
-
1392
- const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
1393
- indicator.add(indicatorText)
1394
-
1395
- const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
1396
- row.add(content)
1397
-
1398
- const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
1399
- content.add(contentText)
1400
-
1401
- await renderOnce()
1402
-
1403
- const initialIndicatorWidth = indicator.width
1404
-
1405
- indicator.width = 5
1406
- await renderOnce()
1407
-
1408
- const frame = captureFrame()
1409
- expect(frame).toMatchSnapshot()
1410
-
1411
- expect(indicator.width).toBe(5)
1412
- expect(content.width).toBeGreaterThan(0)
1413
- expect(content.width).toBeLessThan(30)
1414
- })
1415
-
1416
- it("should not shrink box when height is set via setter in column layout with textarea", async () => {
1417
- resize(30, 15)
1418
-
1419
- const outerBox = new BoxRenderable(currentRenderer, { border: true, width: 25, height: 10 })
1420
- currentRenderer.root.add(outerBox)
1421
-
1422
- const column = new BoxRenderable(currentRenderer, { flexDirection: "column", height: "100%" })
1423
- outerBox.add(column)
1424
-
1425
- const header = new BoxRenderable(currentRenderer, { backgroundColor: "#f00" })
1426
- column.add(header)
1427
-
1428
- const headerText = new TextRenderable(currentRenderer, { content: "Header" })
1429
- header.add(headerText)
1430
-
1431
- const mainContent = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
1432
- column.add(mainContent)
1433
-
1434
- const { textarea: mainTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1435
- initialValue: "Line1\nLine2\nLine3\nLine4\nLine5\nLine6\nLine7\nLine8",
1436
- })
1437
- mainContent.add(mainTextarea)
1438
-
1439
- const footer = new BoxRenderable(currentRenderer, { height: 2, backgroundColor: "#00f" })
1440
- column.add(footer)
1441
-
1442
- const footerText = new TextRenderable(currentRenderer, { content: "Footer" })
1443
- footer.add(footerText)
1444
-
1445
- await renderOnce()
1446
-
1447
- header.height = 3
1448
- await renderOnce()
1449
-
1450
- const frame = captureFrame()
1451
- expect(frame).toMatchSnapshot()
1452
-
1453
- expect(header.height).toBe(3)
1454
- expect(mainContent.height).toBeGreaterThan(0)
1455
- expect(footer.height).toBe(2)
1456
- })
1457
-
1458
- it("should not shrink box when minWidth is set via setter", async () => {
1459
- resize(40, 10)
1460
-
1461
- const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
1462
- currentRenderer.root.add(container)
1463
-
1464
- const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
1465
- container.add(row)
1466
-
1467
- const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
1468
- row.add(indicator)
1469
-
1470
- const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
1471
- indicator.add(indicatorText)
1472
-
1473
- const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
1474
- row.add(content)
1475
-
1476
- const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
1477
- content.add(contentText)
1478
-
1479
- await renderOnce()
1480
-
1481
- indicator.minWidth = 5
1482
- await renderOnce()
1483
-
1484
- const frame = captureFrame()
1485
- expect(frame).toMatchSnapshot()
1486
- expect(indicator.width).toBeGreaterThanOrEqual(5)
1487
- expect(content.width).toBeGreaterThan(0)
1488
- })
1489
-
1490
- it("should not shrink box when minHeight is set via setter in column layout with textarea", async () => {
1491
- resize(30, 15)
1492
-
1493
- const outerBox = new BoxRenderable(currentRenderer, { border: true, width: 25, height: 10 })
1494
- currentRenderer.root.add(outerBox)
1495
-
1496
- const column = new BoxRenderable(currentRenderer, { flexDirection: "column", height: "100%" })
1497
- outerBox.add(column)
1498
-
1499
- const header = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
1500
- column.add(header)
1501
-
1502
- const headerText = new TextRenderable(currentRenderer, { content: "Header" })
1503
- header.add(headerText)
1504
-
1505
- const mainContent = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
1506
- column.add(mainContent)
1507
-
1508
- const { textarea: mainTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1509
- initialValue: "Line1\nLine2\nLine3\nLine4\nLine5\nLine6\nLine7\nLine8",
1510
- })
1511
- mainContent.add(mainTextarea)
1512
-
1513
- const footer = new BoxRenderable(currentRenderer, { height: 2, backgroundColor: "#00f" })
1514
- column.add(footer)
1515
-
1516
- const footerText = new TextRenderable(currentRenderer, { content: "Footer" })
1517
- footer.add(footerText)
1518
-
1519
- await renderOnce()
1520
-
1521
- header.minHeight = 3
1522
- await renderOnce()
1523
-
1524
- const frame = captureFrame()
1525
- expect(frame).toMatchSnapshot()
1526
-
1527
- expect(header.height).toBeGreaterThanOrEqual(3)
1528
- expect(mainContent.height).toBeGreaterThan(0)
1529
- expect(footer.height).toBe(2)
1530
- })
1531
-
1532
- it("should not shrink box when width is set from undefined via setter", async () => {
1533
- resize(40, 10)
1534
-
1535
- const container = new BoxRenderable(currentRenderer, { border: true, width: 30 })
1536
- currentRenderer.root.add(container)
1537
-
1538
- const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
1539
- container.add(row)
1540
-
1541
- const indicator = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
1542
- row.add(indicator)
1543
-
1544
- const indicatorText = new TextRenderable(currentRenderer, { content: ">" })
1545
- indicator.add(indicatorText)
1546
-
1547
- const content = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexGrow: 1 })
1548
- row.add(content)
1549
-
1550
- const contentText = new TextRenderable(currentRenderer, { content: "Content that takes up space" })
1551
- content.add(contentText)
1552
-
1553
- await renderOnce()
1554
-
1555
- indicator.width = 5
1556
- await renderOnce()
1557
-
1558
- const frame = captureFrame()
1559
- expect(frame).toMatchSnapshot()
1560
-
1561
- expect(indicator.width).toBe(5)
1562
- expect(content.width).toBeGreaterThan(0)
1563
- })
1564
-
1565
- it("should verify dimensions are actually respected under extreme pressure", async () => {
1566
- resize(30, 10)
1567
-
1568
- const container = new BoxRenderable(currentRenderer, { border: true, width: 20 })
1569
- currentRenderer.root.add(container)
1570
-
1571
- const row = new BoxRenderable(currentRenderer, { flexDirection: "row", width: "100%" })
1572
- container.add(row)
1573
-
1574
- const box1 = new BoxRenderable(currentRenderer, { backgroundColor: "#f00", flexShrink: 1 })
1575
- row.add(box1)
1576
- const text1 = new TextRenderable(currentRenderer, { content: "AAA" })
1577
- box1.add(text1)
1578
-
1579
- const box2 = new BoxRenderable(currentRenderer, { backgroundColor: "#0f0", flexShrink: 1 })
1580
- row.add(box2)
1581
- const text2 = new TextRenderable(currentRenderer, { content: "BBB" })
1582
- box2.add(text2)
1583
-
1584
- const box3 = new BoxRenderable(currentRenderer, { backgroundColor: "#00f", flexGrow: 1 })
1585
- row.add(box3)
1586
- const text3 = new TextRenderable(currentRenderer, { content: "CCC" })
1587
- box3.add(text3)
1588
-
1589
- await renderOnce()
1590
-
1591
- box1.width = 7
1592
- box2.minWidth = 5
1593
- await renderOnce()
1594
-
1595
- expect(box1.width).toBe(7)
1596
- expect(box2.width).toBeGreaterThanOrEqual(5)
1597
- expect(box3.width).toBeGreaterThan(0)
1598
-
1599
- const total = box1.width + box2.width + box3.width
1600
- expect(total).toBeLessThanOrEqual(18)
1601
- })
1602
- })
1603
-
1604
- describe("Absolute Positioned Box with Textarea", () => {
1605
- it("should render textarea in absolute positioned box with padding and borders correctly", async () => {
1606
- resize(80, 20)
1607
-
1608
- const notificationBox = new BoxRenderable(currentRenderer, {
1609
- position: "absolute",
1610
- justifyContent: "center",
1611
- alignItems: "flex-start",
1612
- top: 2,
1613
- right: 2,
1614
- maxWidth: Math.min(60, 80 - 6),
1615
- paddingLeft: 2,
1616
- paddingRight: 2,
1617
- paddingTop: 1,
1618
- paddingBottom: 1,
1619
- backgroundColor: "#1e293b",
1620
- borderColor: "#3b82f6",
1621
- border: ["left", "right"],
1622
- })
1623
-
1624
- currentRenderer.root.add(notificationBox)
1625
-
1626
- const outerWrapperBox = new BoxRenderable(currentRenderer, {
1627
- flexDirection: "row",
1628
- paddingBottom: 1,
1629
- paddingTop: 1,
1630
- paddingLeft: 2,
1631
- paddingRight: 2,
1632
- gap: 2,
1633
- })
1634
- notificationBox.add(outerWrapperBox)
1635
-
1636
- const innerContentBox = new BoxRenderable(currentRenderer, {
1637
- flexGrow: 1,
1638
- gap: 1,
1639
- })
1640
- outerWrapperBox.add(innerContentBox)
1641
-
1642
- const titleText = new TextRenderable(currentRenderer, {
1643
- content: "Important Notification",
1644
- attributes: 1,
1645
- marginBottom: 1,
1646
- fg: "#f8fafc",
1647
- })
1648
- innerContentBox.add(titleText)
1649
-
1650
- const { textarea: messageTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1651
- initialValue:
1652
- "This is a longer message that should wrap properly within the absolutely positioned box with appropriate width constraints and padding applied.",
1653
- textColor: "#e2e8f0",
1654
- wrapMode: "word",
1655
- width: "100%",
1656
- })
1657
- innerContentBox.add(messageTextarea)
1658
-
1659
- await renderOnce()
1660
-
1661
- const frame = captureFrame()
1662
- expect(frame).toMatchSnapshot()
1663
-
1664
- expect(notificationBox.x).toBeGreaterThan(0)
1665
- expect(notificationBox.y).toBe(2)
1666
- expect(notificationBox.width).toBeGreaterThan(25)
1667
-
1668
- expect(outerWrapperBox.width).toBeGreaterThan(15)
1669
- expect(innerContentBox.width).toBeGreaterThan(15)
1670
-
1671
- expect(titleText.width).toBeGreaterThan(15)
1672
- expect(titleText.plainText).toBe("Important Notification")
1673
- expect(titleText.height).toBe(1)
1674
-
1675
- expect(messageTextarea.width).toBeGreaterThan(15)
1676
- expect(messageTextarea.height).toBeGreaterThanOrEqual(1)
1677
- expect(messageTextarea.plainText).toBe(
1678
- "This is a longer message that should wrap properly within the absolutely positioned box with appropriate width constraints and padding applied.",
1679
- )
1680
- })
1681
-
1682
- it("should render textarea fully visible in absolute positioned box at various positions", async () => {
1683
- resize(100, 25)
1684
-
1685
- const topRightBox = new BoxRenderable(currentRenderer, {
1686
- position: "absolute",
1687
- top: 1,
1688
- right: 1,
1689
- maxWidth: 40,
1690
- paddingLeft: 1,
1691
- paddingRight: 1,
1692
- paddingTop: 0,
1693
- paddingBottom: 0,
1694
- backgroundColor: "#fef2f2",
1695
- borderColor: "#ef4444",
1696
- border: true,
1697
- })
1698
- currentRenderer.root.add(topRightBox)
1699
-
1700
- const { textarea: topRightTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1701
- initialValue: "Error: File not found in the specified directory path",
1702
- textColor: "#991b1b",
1703
- wrapMode: "word",
1704
- width: "100%",
1705
- })
1706
- topRightBox.add(topRightTextarea)
1707
-
1708
- const bottomLeftBox = new BoxRenderable(currentRenderer, {
1709
- position: "absolute",
1710
- bottom: 1,
1711
- left: 1,
1712
- maxWidth: 35,
1713
- paddingLeft: 1,
1714
- paddingRight: 1,
1715
- backgroundColor: "#f0fdf4",
1716
- borderColor: "#22c55e",
1717
- border: ["top", "bottom"],
1718
- })
1719
- currentRenderer.root.add(bottomLeftBox)
1720
-
1721
- const { textarea: bottomLeftTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1722
- initialValue: "Success: Operation completed successfully!",
1723
- textColor: "#166534",
1724
- wrapMode: "word",
1725
- width: "100%",
1726
- })
1727
- bottomLeftBox.add(bottomLeftTextarea)
1728
-
1729
- await renderOnce()
1730
-
1731
- const frame = captureFrame()
1732
- expect(frame).toMatchSnapshot()
1733
-
1734
- expect(topRightBox.y).toBe(1)
1735
- expect(topRightBox.x).toBeGreaterThan(50)
1736
- expect(topRightBox.width).toBeGreaterThan(30)
1737
- expect(topRightBox.width).toBeLessThanOrEqual(40)
1738
-
1739
- expect(topRightTextarea.plainText).toBe("Error: File not found in the specified directory path")
1740
- expect(topRightTextarea.width).toBeGreaterThan(25)
1741
- expect(topRightTextarea.width).toBeLessThanOrEqual(38)
1742
- expect(topRightTextarea.height).toBeGreaterThan(1)
1743
-
1744
- expect(bottomLeftBox.x).toBe(1)
1745
- expect(bottomLeftBox.y).toBeGreaterThan(15)
1746
- expect(bottomLeftBox.width).toBeGreaterThan(25)
1747
- expect(bottomLeftBox.width).toBeLessThanOrEqual(35)
1748
-
1749
- expect(bottomLeftTextarea.plainText).toBe("Success: Operation completed successfully!")
1750
- expect(bottomLeftTextarea.width).toBeGreaterThan(25)
1751
- expect(bottomLeftTextarea.width).toBeLessThanOrEqual(33)
1752
- expect(bottomLeftTextarea.height).toBeGreaterThan(1)
1753
- })
1754
-
1755
- it("should handle width:100% textarea in absolute positioned box with constrained maxWidth", async () => {
1756
- resize(70, 15)
1757
-
1758
- const constrainedBox = new BoxRenderable(currentRenderer, {
1759
- position: "absolute",
1760
- top: 5,
1761
- left: 10,
1762
- maxWidth: 50,
1763
- paddingLeft: 3,
1764
- paddingRight: 3,
1765
- paddingTop: 2,
1766
- paddingBottom: 2,
1767
- backgroundColor: "#1e1e2e",
1768
- })
1769
- currentRenderer.root.add(constrainedBox)
1770
-
1771
- const { textarea: longTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1772
- initialValue:
1773
- "This is an extremely long piece of text that needs to wrap multiple times within the constrained width of the absolutely positioned container box with significant padding on all sides.",
1774
- textColor: "#cdd6f4",
1775
- wrapMode: "word",
1776
- width: "100%",
1777
- })
1778
- constrainedBox.add(longTextarea)
1779
-
1780
- await renderOnce()
1781
-
1782
- const frame = captureFrame()
1783
- expect(frame).toMatchSnapshot()
1784
-
1785
- expect(constrainedBox.width).toBeLessThanOrEqual(50)
1786
- expect(constrainedBox.width).toBeGreaterThan(40)
1787
- expect(constrainedBox.x).toBe(10)
1788
- expect(constrainedBox.y).toBe(5)
1789
-
1790
- expect(longTextarea.width).toBeGreaterThan(35)
1791
- expect(longTextarea.width).toBeLessThanOrEqual(44)
1792
- expect(longTextarea.height).toBeGreaterThanOrEqual(5)
1793
- expect(longTextarea.plainText).toBe(
1794
- "This is an extremely long piece of text that needs to wrap multiple times within the constrained width of the absolutely positioned container box with significant padding on all sides.",
1795
- )
1796
- })
1797
-
1798
- it("should render multiple textarea elements in absolute positioned box with proper spacing", async () => {
1799
- resize(90, 20)
1800
-
1801
- const infoBox = new BoxRenderable(currentRenderer, {
1802
- position: "absolute",
1803
- justifyContent: "flex-start",
1804
- alignItems: "flex-start",
1805
- top: 3,
1806
- right: 5,
1807
- maxWidth: 45,
1808
- paddingLeft: 2,
1809
- paddingRight: 2,
1810
- paddingTop: 1,
1811
- paddingBottom: 1,
1812
- backgroundColor: "#eff6ff",
1813
- borderColor: "#3b82f6",
1814
- border: true,
1815
- })
1816
- currentRenderer.root.add(infoBox)
1817
-
1818
- const headerText = new TextRenderable(currentRenderer, {
1819
- content: "System Update",
1820
- attributes: 1,
1821
- fg: "#1e40af",
1822
- })
1823
- infoBox.add(headerText)
1824
-
1825
- const { textarea: bodyTextarea } = await createTextareaRenderable(currentRenderer, renderOnce, {
1826
- initialValue: "A new version is available with bug fixes and performance improvements.",
1827
- textColor: "#1e3a8a",
1828
- wrapMode: "word",
1829
- width: "100%",
1830
- marginTop: 1,
1831
- })
1832
- infoBox.add(bodyTextarea)
1833
-
1834
- const footerText = new TextRenderable(currentRenderer, {
1835
- content: "Click to install",
1836
- fg: "#60a5fa",
1837
- marginTop: 1,
1838
- })
1839
- infoBox.add(footerText)
1840
-
1841
- await renderOnce()
1842
-
1843
- const frame = captureFrame()
1844
- expect(frame).toMatchSnapshot()
1845
-
1846
- expect(headerText.plainText).toBe("System Update")
1847
- expect(bodyTextarea.plainText).toBe("A new version is available with bug fixes and performance improvements.")
1848
- expect(footerText.plainText).toBe("Click to install")
1849
-
1850
- expect(infoBox.width).toBeGreaterThan(35)
1851
- expect(infoBox.width).toBeLessThanOrEqual(45)
1852
-
1853
- expect(headerText.width).toBeGreaterThan(10)
1854
- expect(headerText.height).toBe(1)
1855
-
1856
- expect(bodyTextarea.width).toBeGreaterThan(30)
1857
- expect(bodyTextarea.height).toBeGreaterThanOrEqual(2)
1858
-
1859
- expect(footerText.width).toBeGreaterThan(10)
1860
- expect(footerText.height).toBe(1)
1861
-
1862
- expect(bodyTextarea.y).toBeGreaterThan(headerText.y)
1863
- expect(footerText.y).toBeGreaterThan(bodyTextarea.y)
1864
- })
1865
- })
1866
- })