@fairyhunter13/opentui-core 0.1.91 → 0.1.94

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