ratatui_ruby 0.3.1 → 0.5.0

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 (350) hide show
  1. checksums.yaml +4 -4
  2. data/.builds/ruby-3.2.yml +14 -12
  3. data/.builds/ruby-3.3.yml +14 -12
  4. data/.builds/ruby-3.4.yml +14 -12
  5. data/.builds/ruby-4.0.0.yml +14 -12
  6. data/AGENTS.md +89 -132
  7. data/CHANGELOG.md +223 -1
  8. data/README.md +23 -16
  9. data/REUSE.toml +20 -0
  10. data/doc/application_architecture.md +176 -0
  11. data/doc/application_testing.md +17 -10
  12. data/doc/contributors/design/ruby_frontend.md +11 -7
  13. data/doc/contributors/developing_examples.md +261 -0
  14. data/doc/contributors/documentation_style.md +104 -0
  15. data/doc/contributors/dwim_dx.md +366 -0
  16. data/doc/contributors/index.md +2 -0
  17. data/doc/custom.css +14 -0
  18. data/doc/event_handling.md +125 -0
  19. data/doc/images/app_all_events.png +0 -0
  20. data/doc/images/app_analytics.png +0 -0
  21. data/doc/images/app_color_picker.png +0 -0
  22. data/doc/images/app_custom_widget.png +0 -0
  23. data/doc/images/app_login_form.png +0 -0
  24. data/doc/images/app_map_demo.png +0 -0
  25. data/doc/images/app_mouse_events.png +0 -0
  26. data/doc/images/app_table_select.png +0 -0
  27. data/doc/images/verify_quickstart_dsl.png +0 -0
  28. data/doc/images/verify_quickstart_layout.png +0 -0
  29. data/doc/images/verify_quickstart_lifecycle.png +0 -0
  30. data/doc/images/verify_readme_usage.png +0 -0
  31. data/doc/images/widget_barchart_demo.png +0 -0
  32. data/doc/images/widget_block_padding.png +0 -0
  33. data/doc/images/widget_block_titles.png +0 -0
  34. data/doc/images/widget_box_demo.png +0 -0
  35. data/doc/images/widget_calendar_demo.png +0 -0
  36. data/doc/images/widget_cell_demo.png +0 -0
  37. data/doc/images/widget_chart_demo.png +0 -0
  38. data/doc/images/widget_gauge_demo.png +0 -0
  39. data/doc/images/widget_layout_split.png +0 -0
  40. data/doc/images/widget_line_gauge_demo.png +0 -0
  41. data/doc/images/widget_list_demo.png +0 -0
  42. data/doc/images/widget_list_styles.png +0 -0
  43. data/doc/images/widget_popup_demo.png +0 -0
  44. data/doc/images/widget_ratatui_logo_demo.png +0 -0
  45. data/doc/images/widget_ratatui_mascot_demo.png +0 -0
  46. data/doc/images/widget_rect.png +0 -0
  47. data/doc/images/widget_render.png +0 -0
  48. data/doc/images/widget_rich_text.png +0 -0
  49. data/doc/images/widget_scroll_text.png +0 -0
  50. data/doc/images/widget_scrollbar_demo.png +0 -0
  51. data/doc/images/widget_sparkline_demo.png +0 -0
  52. data/doc/images/widget_style_colors.png +0 -0
  53. data/doc/images/widget_table_flex.png +0 -0
  54. data/doc/images/widget_tabs_demo.png +0 -0
  55. data/doc/index.md +1 -0
  56. data/doc/interactive_design.md +116 -0
  57. data/doc/quickstart.md +186 -84
  58. data/examples/app_all_events/README.md +81 -0
  59. data/examples/app_all_events/app.rb +93 -0
  60. data/examples/app_all_events/model/event_color_cycle.rb +41 -0
  61. data/examples/app_all_events/model/event_entry.rb +75 -0
  62. data/examples/app_all_events/model/events.rb +180 -0
  63. data/examples/app_all_events/model/highlight.rb +57 -0
  64. data/examples/app_all_events/model/timestamp.rb +54 -0
  65. data/examples/app_all_events/test/snapshots/after_focus_lost.txt +24 -0
  66. data/examples/app_all_events/test/snapshots/after_focus_regained.txt +24 -0
  67. data/examples/app_all_events/test/snapshots/after_horizontal_resize.txt +24 -0
  68. data/examples/app_all_events/test/snapshots/after_key_a.txt +24 -0
  69. data/examples/app_all_events/test/snapshots/after_key_ctrl_x.txt +24 -0
  70. data/examples/app_all_events/test/snapshots/after_mouse_click.txt +24 -0
  71. data/examples/app_all_events/test/snapshots/after_mouse_drag.txt +24 -0
  72. data/examples/app_all_events/test/snapshots/after_multiple_events.txt +24 -0
  73. data/examples/app_all_events/test/snapshots/after_paste.txt +24 -0
  74. data/examples/app_all_events/test/snapshots/after_resize.txt +24 -0
  75. data/examples/app_all_events/test/snapshots/after_right_click.txt +24 -0
  76. data/examples/app_all_events/test/snapshots/after_vertical_resize.txt +24 -0
  77. data/examples/app_all_events/test/snapshots/initial_state.txt +24 -0
  78. data/examples/app_all_events/view/app_view.rb +78 -0
  79. data/examples/app_all_events/view/controls_view.rb +50 -0
  80. data/examples/app_all_events/view/counts_view.rb +55 -0
  81. data/examples/app_all_events/view/live_view.rb +69 -0
  82. data/examples/app_all_events/view/log_view.rb +60 -0
  83. data/{lib/ratatui_ruby/output.rb → examples/app_all_events/view.rb} +1 -1
  84. data/examples/app_all_events/view_state.rb +42 -0
  85. data/examples/app_color_picker/README.md +94 -0
  86. data/examples/app_color_picker/app.rb +112 -0
  87. data/examples/app_color_picker/clipboard.rb +84 -0
  88. data/examples/app_color_picker/color.rb +191 -0
  89. data/examples/app_color_picker/copy_dialog.rb +170 -0
  90. data/examples/app_color_picker/harmony.rb +56 -0
  91. data/examples/app_color_picker/input.rb +142 -0
  92. data/examples/app_color_picker/palette.rb +80 -0
  93. data/examples/app_color_picker/scene.rb +201 -0
  94. data/examples/app_login_form/app.rb +108 -0
  95. data/examples/app_map_demo/app.rb +93 -0
  96. data/examples/app_table_select/app.rb +201 -0
  97. data/examples/verify_quickstart_dsl/app.rb +45 -0
  98. data/examples/verify_quickstart_layout/app.rb +69 -0
  99. data/examples/verify_quickstart_lifecycle/app.rb +48 -0
  100. data/examples/verify_readme_usage/app.rb +34 -0
  101. data/examples/widget_barchart_demo/app.rb +238 -0
  102. data/examples/widget_block_padding/app.rb +67 -0
  103. data/examples/widget_block_titles/app.rb +69 -0
  104. data/examples/widget_box_demo/app.rb +250 -0
  105. data/examples/widget_calendar_demo/app.rb +109 -0
  106. data/examples/widget_cell_demo/app.rb +104 -0
  107. data/examples/widget_chart_demo/app.rb +213 -0
  108. data/examples/widget_gauge_demo/app.rb +212 -0
  109. data/examples/widget_layout_split/app.rb +246 -0
  110. data/examples/widget_line_gauge_demo/app.rb +217 -0
  111. data/examples/widget_list_demo/app.rb +382 -0
  112. data/examples/widget_list_styles/app.rb +141 -0
  113. data/examples/widget_popup_demo/app.rb +104 -0
  114. data/examples/widget_ratatui_logo_demo/app.rb +103 -0
  115. data/examples/widget_ratatui_mascot_demo/app.rb +93 -0
  116. data/examples/widget_rect/app.rb +205 -0
  117. data/examples/widget_render/app.rb +184 -0
  118. data/examples/widget_rich_text/app.rb +137 -0
  119. data/examples/widget_scroll_text/app.rb +108 -0
  120. data/examples/widget_scrollbar_demo/app.rb +153 -0
  121. data/examples/widget_sparkline_demo/app.rb +274 -0
  122. data/examples/widget_style_colors/app.rb +102 -0
  123. data/examples/widget_table_flex/app.rb +95 -0
  124. data/examples/widget_tabs_demo/app.rb +167 -0
  125. data/ext/ratatui_ruby/Cargo.lock +889 -115
  126. data/ext/ratatui_ruby/Cargo.toml +4 -3
  127. data/ext/ratatui_ruby/clippy.toml +7 -0
  128. data/ext/ratatui_ruby/extconf.rb +7 -0
  129. data/ext/ratatui_ruby/src/events.rs +293 -219
  130. data/ext/ratatui_ruby/src/frame.rs +115 -0
  131. data/ext/ratatui_ruby/src/lib.rs +105 -24
  132. data/ext/ratatui_ruby/src/rendering.rs +94 -10
  133. data/ext/ratatui_ruby/src/style.rs +357 -93
  134. data/ext/ratatui_ruby/src/terminal.rs +121 -31
  135. data/ext/ratatui_ruby/src/text.rs +178 -0
  136. data/ext/ratatui_ruby/src/widgets/barchart.rs +99 -24
  137. data/ext/ratatui_ruby/src/widgets/block.rs +32 -3
  138. data/ext/ratatui_ruby/src/widgets/calendar.rs +45 -44
  139. data/ext/ratatui_ruby/src/widgets/canvas.rs +44 -9
  140. data/ext/ratatui_ruby/src/widgets/chart.rs +79 -27
  141. data/ext/ratatui_ruby/src/widgets/clear.rs +3 -1
  142. data/ext/ratatui_ruby/src/widgets/gauge.rs +11 -4
  143. data/ext/ratatui_ruby/src/widgets/layout.rs +223 -15
  144. data/ext/ratatui_ruby/src/widgets/line_gauge.rs +92 -0
  145. data/ext/ratatui_ruby/src/widgets/list.rs +114 -11
  146. data/ext/ratatui_ruby/src/widgets/mod.rs +3 -0
  147. data/ext/ratatui_ruby/src/widgets/overlay.rs +4 -2
  148. data/ext/ratatui_ruby/src/widgets/paragraph.rs +35 -13
  149. data/ext/ratatui_ruby/src/widgets/ratatui_logo.rs +40 -0
  150. data/ext/ratatui_ruby/src/widgets/ratatui_mascot.rs +51 -0
  151. data/ext/ratatui_ruby/src/widgets/scrollbar.rs +61 -7
  152. data/ext/ratatui_ruby/src/widgets/sparkline.rs +73 -6
  153. data/ext/ratatui_ruby/src/widgets/table.rs +177 -64
  154. data/ext/ratatui_ruby/src/widgets/tabs.rs +105 -5
  155. data/lib/ratatui_ruby/cell.rb +166 -0
  156. data/lib/ratatui_ruby/event/focus_gained.rb +49 -0
  157. data/lib/ratatui_ruby/event/focus_lost.rb +50 -0
  158. data/lib/ratatui_ruby/event/key.rb +211 -0
  159. data/lib/ratatui_ruby/event/mouse.rb +124 -0
  160. data/lib/ratatui_ruby/event/none.rb +43 -0
  161. data/lib/ratatui_ruby/event/paste.rb +71 -0
  162. data/lib/ratatui_ruby/event/resize.rb +80 -0
  163. data/lib/ratatui_ruby/event.rb +131 -0
  164. data/lib/ratatui_ruby/frame.rb +87 -0
  165. data/lib/ratatui_ruby/schema/bar_chart/bar.rb +45 -0
  166. data/lib/ratatui_ruby/schema/bar_chart/bar_group.rb +23 -0
  167. data/lib/ratatui_ruby/schema/bar_chart.rb +226 -17
  168. data/lib/ratatui_ruby/schema/block.rb +178 -11
  169. data/lib/ratatui_ruby/schema/calendar.rb +70 -14
  170. data/lib/ratatui_ruby/schema/canvas.rb +213 -46
  171. data/lib/ratatui_ruby/schema/center.rb +46 -8
  172. data/lib/ratatui_ruby/schema/chart.rb +134 -32
  173. data/lib/ratatui_ruby/schema/clear.rb +22 -53
  174. data/lib/ratatui_ruby/schema/constraint.rb +72 -12
  175. data/lib/ratatui_ruby/schema/cursor.rb +23 -5
  176. data/lib/ratatui_ruby/schema/draw.rb +53 -0
  177. data/lib/ratatui_ruby/schema/gauge.rb +56 -12
  178. data/lib/ratatui_ruby/schema/layout.rb +91 -9
  179. data/lib/ratatui_ruby/schema/line_gauge.rb +78 -0
  180. data/lib/ratatui_ruby/schema/list.rb +92 -16
  181. data/lib/ratatui_ruby/schema/overlay.rb +29 -3
  182. data/lib/ratatui_ruby/schema/paragraph.rb +82 -25
  183. data/lib/ratatui_ruby/schema/ratatui_logo.rb +29 -0
  184. data/lib/ratatui_ruby/schema/ratatui_mascot.rb +34 -0
  185. data/lib/ratatui_ruby/schema/rect.rb +59 -10
  186. data/lib/ratatui_ruby/schema/scrollbar.rb +127 -19
  187. data/lib/ratatui_ruby/schema/shape/label.rb +66 -0
  188. data/lib/ratatui_ruby/schema/sparkline.rb +120 -12
  189. data/lib/ratatui_ruby/schema/style.rb +39 -11
  190. data/lib/ratatui_ruby/schema/table.rb +109 -18
  191. data/lib/ratatui_ruby/schema/tabs.rb +71 -10
  192. data/lib/ratatui_ruby/schema/text.rb +90 -0
  193. data/lib/ratatui_ruby/session/autodoc.rb +417 -0
  194. data/lib/ratatui_ruby/session.rb +163 -0
  195. data/lib/ratatui_ruby/test_helper.rb +322 -13
  196. data/lib/ratatui_ruby/version.rb +1 -1
  197. data/lib/ratatui_ruby.rb +184 -38
  198. data/sig/examples/app_all_events/app.rbs +11 -0
  199. data/sig/examples/app_all_events/model/event_entry.rbs +16 -0
  200. data/sig/examples/app_all_events/model/events.rbs +15 -0
  201. data/sig/examples/app_all_events/model/timestamp.rbs +11 -0
  202. data/sig/examples/app_all_events/view/app_view.rbs +8 -0
  203. data/sig/examples/app_all_events/view/controls_view.rbs +6 -0
  204. data/sig/examples/app_all_events/view/counts_view.rbs +6 -0
  205. data/sig/examples/app_all_events/view/live_view.rbs +6 -0
  206. data/sig/examples/app_all_events/view/log_view.rbs +6 -0
  207. data/sig/examples/app_all_events/view.rbs +8 -0
  208. data/sig/examples/app_all_events/view_state.rbs +15 -0
  209. data/sig/examples/app_color_picker/app.rbs +12 -0
  210. data/sig/examples/app_login_form/app.rbs +11 -0
  211. data/sig/examples/app_map_demo/app.rbs +11 -0
  212. data/sig/examples/app_table_select/app.rbs +11 -0
  213. data/sig/examples/verify_quickstart_dsl/app.rbs +11 -0
  214. data/sig/examples/verify_quickstart_lifecycle/app.rbs +11 -0
  215. data/sig/examples/verify_readme_usage/app.rbs +11 -0
  216. data/sig/examples/widget_block_padding/app.rbs +11 -0
  217. data/sig/examples/widget_block_titles/app.rbs +11 -0
  218. data/sig/examples/widget_box_demo/app.rbs +11 -0
  219. data/sig/examples/widget_calendar_demo/app.rbs +11 -0
  220. data/sig/examples/widget_cell_demo/app.rbs +11 -0
  221. data/sig/examples/widget_chart_demo/app.rbs +11 -0
  222. data/sig/examples/widget_gauge_demo/app.rbs +11 -0
  223. data/sig/examples/widget_layout_split/app.rbs +10 -0
  224. data/sig/examples/widget_line_gauge_demo/app.rbs +11 -0
  225. data/sig/examples/widget_list_demo/app.rbs +12 -0
  226. data/sig/examples/widget_list_styles/app.rbs +11 -0
  227. data/sig/examples/widget_popup_demo/app.rbs +11 -0
  228. data/sig/examples/widget_ratatui_logo_demo/app.rbs +11 -0
  229. data/sig/examples/widget_ratatui_mascot_demo/app.rbs +11 -0
  230. data/sig/examples/widget_rect/app.rbs +12 -0
  231. data/sig/examples/widget_render/app.rbs +10 -0
  232. data/sig/examples/widget_rich_text/app.rbs +11 -0
  233. data/sig/examples/widget_scroll_text/app.rbs +11 -0
  234. data/sig/examples/widget_scrollbar_demo/app.rbs +11 -0
  235. data/sig/examples/widget_sparkline_demo/app.rbs +10 -0
  236. data/sig/examples/widget_style_colors/app.rbs +14 -0
  237. data/sig/examples/widget_table_flex/app.rbs +11 -0
  238. data/sig/ratatui_ruby/event.rbs +69 -0
  239. data/sig/ratatui_ruby/frame.rbs +9 -0
  240. data/sig/ratatui_ruby/ratatui_ruby.rbs +5 -3
  241. data/sig/ratatui_ruby/schema/bar_chart/bar.rbs +16 -0
  242. data/sig/ratatui_ruby/schema/bar_chart/bar_group.rbs +13 -0
  243. data/sig/ratatui_ruby/schema/bar_chart.rbs +20 -2
  244. data/sig/ratatui_ruby/schema/block.rbs +5 -4
  245. data/sig/ratatui_ruby/schema/calendar.rbs +6 -2
  246. data/sig/ratatui_ruby/schema/canvas.rbs +52 -39
  247. data/sig/ratatui_ruby/schema/center.rbs +3 -3
  248. data/sig/ratatui_ruby/schema/chart.rbs +8 -5
  249. data/sig/ratatui_ruby/schema/constraint.rbs +8 -5
  250. data/sig/ratatui_ruby/schema/cursor.rbs +1 -1
  251. data/sig/ratatui_ruby/schema/draw.rbs +27 -0
  252. data/sig/ratatui_ruby/schema/gauge.rbs +4 -2
  253. data/sig/ratatui_ruby/schema/layout.rbs +11 -1
  254. data/sig/ratatui_ruby/schema/line_gauge.rbs +16 -0
  255. data/sig/ratatui_ruby/schema/list.rbs +5 -1
  256. data/sig/ratatui_ruby/schema/paragraph.rbs +4 -1
  257. data/sig/ratatui_ruby/schema/ratatui_logo.rbs +8 -0
  258. data/sig/ratatui_ruby/{buffer.rbs → schema/ratatui_mascot.rbs} +4 -3
  259. data/sig/ratatui_ruby/schema/rect.rbs +2 -1
  260. data/sig/ratatui_ruby/schema/scrollbar.rbs +18 -2
  261. data/sig/ratatui_ruby/schema/sparkline.rbs +6 -2
  262. data/sig/ratatui_ruby/schema/table.rbs +8 -1
  263. data/sig/ratatui_ruby/schema/tabs.rbs +5 -1
  264. data/sig/ratatui_ruby/schema/text.rbs +22 -0
  265. data/sig/ratatui_ruby/session.rbs +94 -0
  266. data/tasks/autodoc/inventory.rb +61 -0
  267. data/tasks/autodoc/member.rb +56 -0
  268. data/tasks/autodoc/name.rb +19 -0
  269. data/tasks/autodoc/notice.rb +26 -0
  270. data/tasks/autodoc/rbs.rb +38 -0
  271. data/tasks/autodoc/rdoc.rb +45 -0
  272. data/tasks/autodoc.rake +47 -0
  273. data/tasks/bump/history.rb +2 -2
  274. data/tasks/doc.rake +600 -6
  275. data/tasks/example_viewer.html.erb +172 -0
  276. data/tasks/lint.rake +8 -4
  277. data/tasks/resources/build.yml.erb +13 -11
  278. data/tasks/resources/index.html.erb +6 -0
  279. data/tasks/sourcehut.rake +4 -4
  280. data/tasks/terminal_preview/app_screenshot.rb +33 -0
  281. data/tasks/terminal_preview/crash_report.rb +52 -0
  282. data/tasks/terminal_preview/example_app.rb +25 -0
  283. data/tasks/terminal_preview/launcher_script.rb +46 -0
  284. data/tasks/terminal_preview/preview_collection.rb +58 -0
  285. data/tasks/terminal_preview/preview_timing.rb +22 -0
  286. data/tasks/terminal_preview/safety_confirmation.rb +56 -0
  287. data/tasks/terminal_preview/saved_screenshot.rb +53 -0
  288. data/tasks/terminal_preview/system_appearance.rb +11 -0
  289. data/tasks/terminal_preview/terminal_window.rb +136 -0
  290. data/tasks/terminal_preview/window_id.rb +14 -0
  291. data/tasks/terminal_preview.rake +28 -0
  292. data/tasks/test.rake +2 -2
  293. data/tasks/website/index_page.rb +3 -3
  294. data/tasks/website/version.rb +10 -10
  295. data/tasks/website/version_menu.rb +10 -12
  296. data/tasks/website/versioned_documentation.rb +49 -17
  297. data/tasks/website/website.rb +6 -8
  298. data/tasks/website.rake +4 -4
  299. metadata +206 -54
  300. data/LICENSES/BSD-2-Clause.txt +0 -9
  301. data/doc/images/examples-analytics.rb.png +0 -0
  302. data/doc/images/examples-box_demo.rb.png +0 -0
  303. data/doc/images/examples-calendar_demo.rb.png +0 -0
  304. data/doc/images/examples-chart_demo.rb.png +0 -0
  305. data/doc/images/examples-custom_widget.rb.png +0 -0
  306. data/doc/images/examples-dashboard.rb.png +0 -0
  307. data/doc/images/examples-list_styles.rb.png +0 -0
  308. data/doc/images/examples-login_form.rb.png +0 -0
  309. data/doc/images/examples-map_demo.rb.png +0 -0
  310. data/doc/images/examples-mouse_events.rb.png +0 -0
  311. data/doc/images/examples-popup_demo.rb.gif +0 -0
  312. data/doc/images/examples-quickstart_lifecycle.rb.png +0 -0
  313. data/doc/images/examples-scroll_text.rb.png +0 -0
  314. data/doc/images/examples-scrollbar_demo.rb.png +0 -0
  315. data/doc/images/examples-stock_ticker.rb.png +0 -0
  316. data/doc/images/examples-system_monitor.rb.png +0 -0
  317. data/doc/images/examples-table_select.rb.png +0 -0
  318. data/examples/analytics.rb +0 -88
  319. data/examples/box_demo.rb +0 -71
  320. data/examples/calendar_demo.rb +0 -55
  321. data/examples/chart_demo.rb +0 -84
  322. data/examples/custom_widget.rb +0 -43
  323. data/examples/dashboard.rb +0 -72
  324. data/examples/list_styles.rb +0 -66
  325. data/examples/login_form.rb +0 -115
  326. data/examples/map_demo.rb +0 -58
  327. data/examples/mouse_events.rb +0 -95
  328. data/examples/popup_demo.rb +0 -105
  329. data/examples/quickstart_dsl.rb +0 -30
  330. data/examples/quickstart_lifecycle.rb +0 -40
  331. data/examples/readme_usage.rb +0 -21
  332. data/examples/scroll_text.rb +0 -74
  333. data/examples/scrollbar_demo.rb +0 -75
  334. data/examples/stock_ticker.rb +0 -93
  335. data/examples/system_monitor.rb +0 -94
  336. data/examples/table_select.rb +0 -70
  337. data/examples/test_analytics.rb +0 -65
  338. data/examples/test_box_demo.rb +0 -38
  339. data/examples/test_calendar_demo.rb +0 -66
  340. data/examples/test_dashboard.rb +0 -38
  341. data/examples/test_list_styles.rb +0 -61
  342. data/examples/test_login_form.rb +0 -63
  343. data/examples/test_map_demo.rb +0 -100
  344. data/examples/test_popup_demo.rb +0 -62
  345. data/examples/test_scroll_text.rb +0 -130
  346. data/examples/test_stock_ticker.rb +0 -39
  347. data/examples/test_system_monitor.rb +0 -40
  348. data/examples/test_table_select.rb +0 -37
  349. data/ext/ratatui_ruby/src/buffer.rs +0 -54
  350. data/lib/ratatui_ruby/dsl.rb +0 -64
@@ -1,6 +1,7 @@
1
1
  // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
2
  // SPDX-License-Identifier: AGPL-3.0-or-later
3
3
 
4
+ use magnus::value::ReprValue;
4
5
  use magnus::Error;
5
6
  use ratatui::{
6
7
  backend::{CrosstermBackend, TestBackend},
@@ -11,14 +12,12 @@ use std::sync::Mutex;
11
12
 
12
13
  pub enum TerminalWrapper {
13
14
  Crossterm(Terminal<CrosstermBackend<io::Stdout>>),
14
- Test(Terminal<TestBackend>), // We don't need Mutex inside the enum variant because the global is a Mutex
15
+ Test(Terminal<TestBackend>),
15
16
  }
16
17
 
17
- lazy_static::lazy_static! {
18
- pub static ref TERMINAL: Mutex<Option<TerminalWrapper>> = Mutex::new(None);
19
- }
18
+ pub static TERMINAL: Mutex<Option<TerminalWrapper>> = Mutex::new(None);
20
19
 
21
- pub fn init_terminal() -> Result<(), Error> {
20
+ pub fn init_terminal(focus_events: bool, bracketed_paste: bool) -> Result<(), Error> {
22
21
  let ruby = magnus::Ruby::get().unwrap();
23
22
  let mut term_lock = TERMINAL.lock().unwrap();
24
23
  if term_lock.is_none() {
@@ -31,6 +30,16 @@ pub fn init_terminal() -> Result<(), Error> {
31
30
  ratatui::crossterm::event::EnableMouseCapture
32
31
  )
33
32
  .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
33
+
34
+ if focus_events {
35
+ ratatui::crossterm::execute!(stdout, ratatui::crossterm::event::EnableFocusChange)
36
+ .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
37
+ }
38
+ if bracketed_paste {
39
+ ratatui::crossterm::execute!(stdout, ratatui::crossterm::event::EnableBracketedPaste)
40
+ .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
41
+ }
42
+
34
43
  let backend = CrosstermBackend::new(stdout);
35
44
  let terminal = Terminal::new(backend)
36
45
  .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
@@ -49,30 +58,30 @@ pub fn init_test_terminal(width: u16, height: u16) -> Result<(), Error> {
49
58
  Ok(())
50
59
  }
51
60
 
52
- pub fn restore_terminal() -> Result<(), Error> {
61
+ pub fn restore_terminal() {
53
62
  let mut term_lock = TERMINAL.lock().unwrap();
54
- if let Some(TerminalWrapper::Crossterm(mut terminal)) = term_lock.take() {
55
- let _ = ratatui::crossterm::terminal::disable_raw_mode();
56
- let _ = ratatui::crossterm::execute!(
57
- terminal.backend_mut(),
58
- ratatui::crossterm::terminal::LeaveAlternateScreen,
59
- ratatui::crossterm::event::DisableMouseCapture
60
- );
63
+ if let Some(wrapper) = term_lock.take() {
64
+ match wrapper {
65
+ TerminalWrapper::Crossterm(mut t) => {
66
+ let _ = ratatui::crossterm::terminal::disable_raw_mode();
67
+ let _ = ratatui::crossterm::execute!(
68
+ t.backend_mut(),
69
+ ratatui::crossterm::terminal::LeaveAlternateScreen,
70
+ ratatui::crossterm::event::DisableMouseCapture,
71
+ ratatui::crossterm::event::DisableFocusChange,
72
+ ratatui::crossterm::event::DisableBracketedPaste
73
+ );
74
+ }
75
+ TerminalWrapper::Test(_) => {}
76
+ }
61
77
  }
62
- Ok(())
63
78
  }
64
79
 
65
80
  pub fn get_buffer_content() -> Result<String, Error> {
66
81
  let ruby = magnus::Ruby::get().unwrap();
67
82
  let term_lock = TERMINAL.lock().unwrap();
68
83
  if let Some(TerminalWrapper::Test(terminal)) = term_lock.as_ref() {
69
- // We need to access the buffer.
70
- // Since we are mocking, we can just print the buffer to a string.
71
84
  let buffer = terminal.backend().buffer();
72
- // Simple representation: each cell's symbol.
73
- // For a more complex representation we could return an array of strings.
74
- // Let's just return the full string representation for now which is useful for debugging/asserting.
75
- // Actually, let's reconstruct it line by line.
76
85
  let area = buffer.area;
77
86
  let mut result = String::new();
78
87
  for y in 0..area.height {
@@ -112,23 +121,104 @@ pub fn resize_terminal(width: u16, height: u16) -> Result<(), Error> {
112
121
  let mut term_lock = TERMINAL.lock().unwrap();
113
122
  if let Some(wrapper) = term_lock.as_mut() {
114
123
  match wrapper {
115
- TerminalWrapper::Crossterm(_) => {
116
- // Resize happens automatically for Crossterm via signals usually,
117
- // but we can't easily force it here without OS interaction.
118
- // Ignoring for now as it's less critical for unit testing the logic.
119
- }
124
+ TerminalWrapper::Crossterm(_) => {}
120
125
  TerminalWrapper::Test(terminal) => {
121
126
  terminal.backend_mut().resize(width, height);
122
- // Also resize the terminal wrapper itself if needed, but TestBackend resize handles the buffer.
123
- // We might need to call terminal.resize() too if Ratatui caches the size.
124
127
  if let Err(e) = terminal.resize(ratatui::layout::Rect::new(0, 0, width, height)) {
125
- return Err(Error::new(
126
- ruby.exception_runtime_error(),
127
- e.to_string(),
128
- ));
128
+ return Err(Error::new(ruby.exception_runtime_error(), e.to_string()));
129
129
  }
130
130
  }
131
131
  }
132
132
  }
133
133
  Ok(())
134
134
  }
135
+
136
+ use magnus::Value;
137
+
138
+ pub fn get_cell_at(x: u16, y: u16) -> Result<magnus::RHash, Error> {
139
+ let ruby = magnus::Ruby::get().unwrap();
140
+ let term_lock = TERMINAL.lock().unwrap();
141
+ if let Some(TerminalWrapper::Test(terminal)) = term_lock.as_ref() {
142
+ let buffer = terminal.backend().buffer();
143
+ if let Some(cell) = buffer.cell((x, y)) {
144
+ let hash = ruby.hash_new();
145
+ hash.aset("char", cell.symbol())?;
146
+ hash.aset("fg", color_to_value(cell.fg))?;
147
+ hash.aset("bg", color_to_value(cell.bg))?;
148
+ hash.aset("modifiers", modifiers_to_value(cell.modifier))?;
149
+ Ok(hash)
150
+ } else {
151
+ Err(Error::new(
152
+ ruby.exception_runtime_error(),
153
+ format!("Coordinates ({x}, {y}) out of bounds"),
154
+ ))
155
+ }
156
+ } else {
157
+ Err(Error::new(
158
+ ruby.exception_runtime_error(),
159
+ "Terminal is not initialized as TestBackend",
160
+ ))
161
+ }
162
+ }
163
+
164
+ fn color_to_value(color: ratatui::style::Color) -> Value {
165
+ let ruby = magnus::Ruby::get().unwrap();
166
+ match color {
167
+ ratatui::style::Color::Reset => ruby.qnil().as_value(),
168
+ ratatui::style::Color::Black => ruby.to_symbol("black").as_value(),
169
+ ratatui::style::Color::Red => ruby.to_symbol("red").as_value(),
170
+ ratatui::style::Color::Green => ruby.to_symbol("green").as_value(),
171
+ ratatui::style::Color::Yellow => ruby.to_symbol("yellow").as_value(),
172
+ ratatui::style::Color::Blue => ruby.to_symbol("blue").as_value(),
173
+ ratatui::style::Color::Magenta => ruby.to_symbol("magenta").as_value(),
174
+ ratatui::style::Color::Cyan => ruby.to_symbol("cyan").as_value(),
175
+ ratatui::style::Color::Gray => ruby.to_symbol("gray").as_value(),
176
+ ratatui::style::Color::DarkGray => ruby.to_symbol("dark_gray").as_value(),
177
+ ratatui::style::Color::LightRed => ruby.to_symbol("light_red").as_value(),
178
+ ratatui::style::Color::LightGreen => ruby.to_symbol("light_green").as_value(),
179
+ ratatui::style::Color::LightYellow => ruby.to_symbol("light_yellow").as_value(),
180
+ ratatui::style::Color::LightBlue => ruby.to_symbol("light_blue").as_value(),
181
+ ratatui::style::Color::LightMagenta => ruby.to_symbol("light_magenta").as_value(),
182
+ ratatui::style::Color::LightCyan => ruby.to_symbol("light_cyan").as_value(),
183
+ ratatui::style::Color::White => ruby.to_symbol("white").as_value(),
184
+ ratatui::style::Color::Rgb(r, g, b) => ruby
185
+ .str_new(&(format!("#{r:02x}{g:02x}{b:02x}")))
186
+ .as_value(),
187
+ ratatui::style::Color::Indexed(i) => ruby.to_symbol(format!("indexed_{i}")).as_value(),
188
+ }
189
+ }
190
+
191
+ fn modifiers_to_value(modifier: ratatui::style::Modifier) -> Value {
192
+ let ruby = magnus::Ruby::get().unwrap();
193
+ let ary = ruby.ary_new();
194
+
195
+ if modifier.contains(ratatui::style::Modifier::BOLD) {
196
+ let _ = ary.push(ruby.str_new("bold"));
197
+ }
198
+ if modifier.contains(ratatui::style::Modifier::ITALIC) {
199
+ let _ = ary.push(ruby.str_new("italic"));
200
+ }
201
+ if modifier.contains(ratatui::style::Modifier::DIM) {
202
+ let _ = ary.push(ruby.str_new("dim"));
203
+ }
204
+ if modifier.contains(ratatui::style::Modifier::UNDERLINED) {
205
+ let _ = ary.push(ruby.str_new("underlined"));
206
+ }
207
+ if modifier.contains(ratatui::style::Modifier::REVERSED) {
208
+ let _ = ary.push(ruby.str_new("reversed"));
209
+ }
210
+ if modifier.contains(ratatui::style::Modifier::HIDDEN) {
211
+ let _ = ary.push(ruby.str_new("hidden"));
212
+ }
213
+ if modifier.contains(ratatui::style::Modifier::CROSSED_OUT) {
214
+ let _ = ary.push(ruby.str_new("crossed_out"));
215
+ }
216
+ if modifier.contains(ratatui::style::Modifier::SLOW_BLINK) {
217
+ let _ = ary.push(ruby.str_new("slow_blink"));
218
+ }
219
+ if modifier.contains(ratatui::style::Modifier::RAPID_BLINK) {
220
+ let _ = ary.push(ruby.str_new("rapid_blink"));
221
+ }
222
+
223
+ ary.as_value()
224
+ }
@@ -0,0 +1,178 @@
1
+ // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
+ // SPDX-License-Identifier: AGPL-3.0-or-later
3
+
4
+ use crate::style::parse_style;
5
+ use magnus::{prelude::*, Error, Value};
6
+ use ratatui::text::{Line, Span};
7
+
8
+ /// Parses a Ruby value into a ratatui Text structure.
9
+ ///
10
+ /// Supports:
11
+ /// - String: Plain text without styling
12
+ /// - `Text::Span`: A single styled fragment
13
+ /// - `Text::Line`: A line composed of multiple spans
14
+ /// - Array: Array of `Text::Lines` or Strings
15
+ pub fn parse_text(value: Value) -> Result<Vec<Line<'static>>, Error> {
16
+ let ruby = magnus::Ruby::get().unwrap();
17
+
18
+ if value.is_nil() {
19
+ return Ok(vec![Line::from("")]);
20
+ }
21
+
22
+ // Check if it's a String
23
+ if let Ok(s) = String::try_convert(value) {
24
+ // Split on newlines and create a Line for each.
25
+ // We need to own the strings, so we convert each line string to a String
26
+ let lines: Vec<Line> = s
27
+ .split('\n')
28
+ .map(|line| Line::from(line.to_string()))
29
+ .collect();
30
+ return if lines.is_empty() {
31
+ Ok(vec![Line::from("")])
32
+ } else {
33
+ Ok(lines)
34
+ };
35
+ }
36
+
37
+ // Check if it's an Array
38
+ if let Some(arr) = magnus::RArray::from_value(value) {
39
+ let mut lines = Vec::new();
40
+ for i in 0..arr.len() {
41
+ let ruby = magnus::Ruby::get().unwrap();
42
+ let index = isize::try_from(i)
43
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
44
+ let elem: Value = arr.entry(index)?;
45
+
46
+ // Try to convert to String
47
+ if let Ok(s) = String::try_convert(elem) {
48
+ lines.push(Line::from(s));
49
+ } else if let Ok(line) = parse_line(elem) {
50
+ lines.push(line);
51
+ } else if let Ok(span) = parse_span(elem) {
52
+ lines.push(Line::from(vec![span]));
53
+ }
54
+ }
55
+ return if lines.is_empty() {
56
+ Ok(vec![Line::from("")])
57
+ } else {
58
+ Ok(lines)
59
+ };
60
+ }
61
+
62
+ // Try to parse as Line
63
+ if let Ok(line) = parse_line(value) {
64
+ return Ok(vec![line]);
65
+ }
66
+
67
+ // Try to parse as Span
68
+ if let Ok(span) = parse_span(value) {
69
+ return Ok(vec![Line::from(vec![span])]);
70
+ }
71
+
72
+ // Fallback: try to convert to string
73
+ match String::try_convert(value) {
74
+ Ok(s) => {
75
+ let lines: Vec<Line> = s
76
+ .split('\n')
77
+ .map(|line| Line::from(line.to_string()))
78
+ .collect();
79
+ if lines.is_empty() {
80
+ Ok(vec![Line::from("")])
81
+ } else {
82
+ Ok(lines)
83
+ }
84
+ }
85
+ Err(_) => Err(Error::new(
86
+ ruby.exception_type_error(),
87
+ "expected String, Text::Span, Text::Line, or Array of Text::Lines/Spans",
88
+ )),
89
+ }
90
+ }
91
+
92
+ /// Parses a Ruby Span object into a ratatui Span.
93
+ fn parse_span(value: Value) -> Result<Span<'static>, Error> {
94
+ let ruby = magnus::Ruby::get().unwrap();
95
+
96
+ // Get class name
97
+ let class_obj: Value = value.funcall("class", ())?;
98
+ let class_name: String = class_obj.funcall("name", ())?;
99
+
100
+ if !class_name.contains("Span") {
101
+ return Err(Error::new(
102
+ ruby.exception_type_error(),
103
+ "expected a Text::Span object",
104
+ ));
105
+ }
106
+
107
+ // Extract content and style from the Ruby Span
108
+ let content: Value = value.funcall("content", ())?;
109
+ let style_val: Value = value.funcall("style", ())?;
110
+
111
+ let content_str: String = content.funcall("to_s", ())?;
112
+ let style = parse_style(style_val)?;
113
+
114
+ Ok(Span::styled(content_str, style))
115
+ }
116
+
117
+ /// Parses a Ruby `Text::Line` object into a ratatui Line.
118
+ pub fn parse_line(value: Value) -> Result<Line<'static>, Error> {
119
+ let ruby = magnus::Ruby::get().unwrap();
120
+
121
+ // Get class name
122
+ let class_obj: Value = value.funcall("class", ())?;
123
+ let class_name: String = class_obj.funcall("name", ())?;
124
+
125
+ if !class_name.contains("Line") {
126
+ return Err(Error::new(
127
+ ruby.exception_type_error(),
128
+ "expected a Text::Line object",
129
+ ));
130
+ }
131
+
132
+ // Extract spans from the Ruby Line
133
+ let spans_val: Value = value.funcall("spans", ())?;
134
+
135
+ if spans_val.is_nil() {
136
+ return Ok(Line::from(""));
137
+ }
138
+
139
+ let spans_array = magnus::RArray::from_value(spans_val).ok_or_else(|| {
140
+ Error::new(
141
+ ruby.exception_type_error(),
142
+ "expected array of Spans in Text::Line.spans",
143
+ )
144
+ })?;
145
+
146
+ let mut spans = Vec::new();
147
+ for i in 0..spans_array.len() {
148
+ let ruby = magnus::Ruby::get().unwrap();
149
+ let index = isize::try_from(i)
150
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
151
+ let span_elem: Value = spans_array.entry(index)?;
152
+
153
+ // If it's a string, convert to span without style
154
+ if let Ok(s) = String::try_convert(span_elem) {
155
+ spans.push(Span::raw(s));
156
+ } else {
157
+ // Try to parse as Span object
158
+ if let Ok(span) = parse_span(span_elem) {
159
+ spans.push(span);
160
+ } else if let Ok(s) = String::try_convert(span_elem) {
161
+ // If it fails, try converting to string
162
+ spans.push(Span::raw(s));
163
+ }
164
+ }
165
+ }
166
+
167
+ if spans.is_empty() {
168
+ Ok(Line::from(""))
169
+ } else {
170
+ Ok(Line::from(spans))
171
+ }
172
+ }
173
+
174
+ #[cfg(test)]
175
+ mod tests {
176
+ #[test]
177
+ fn test_parse_plain_string() {}
178
+ }
@@ -1,40 +1,103 @@
1
1
  // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
2
  // SPDX-License-Identifier: AGPL-3.0-or-later
3
3
 
4
- use crate::style::{parse_block, parse_style};
5
- use magnus::{prelude::*, Error, Value};
6
- use ratatui::{layout::Rect, widgets::BarChart, Frame};
4
+ use crate::style::{parse_bar_set, parse_block, parse_style};
5
+ use bumpalo::Bump;
6
+ use magnus::{prelude::*, Error, RArray, Symbol, Value};
7
+ use ratatui::{
8
+ layout::{Direction, Rect},
9
+ text::Line,
10
+ widgets::{Bar, BarChart, BarGroup},
11
+ Frame,
12
+ };
7
13
 
8
14
  pub fn render(frame: &mut Frame, area: Rect, node: Value) -> Result<(), Error> {
9
- let data_val: magnus::RHash = node.funcall("data", ())?;
15
+ let bump = Bump::new();
16
+ let data_val: Value = node.funcall("data", ())?;
10
17
  let bar_width: u16 = node.funcall("bar_width", ())?;
11
18
  let bar_gap: u16 = node.funcall("bar_gap", ())?;
19
+ let group_gap: u16 = node.funcall("group_gap", ())?;
12
20
  let max_val: Value = node.funcall("max", ())?;
13
21
  let style_val: Value = node.funcall("style", ())?;
14
22
  let block_val: Value = node.funcall("block", ())?;
23
+ let direction_sym: Symbol = node.funcall("direction", ())?;
24
+ let label_style_val: Value = node.funcall("label_style", ())?;
25
+ let value_style_val: Value = node.funcall("value_style", ())?;
26
+ let bar_set_val: Value = node.funcall("bar_set", ())?;
15
27
 
16
- let keys: magnus::RArray = data_val.funcall("keys", ())?;
17
- let mut labels = Vec::new();
18
- let mut data_vec = Vec::new();
19
-
20
- for i in 0..keys.len() {
21
- let key: Value = keys.entry(i as isize)?;
22
- let val: u64 = data_val.funcall("[]", (key,))?;
23
- let label: String = key.funcall("to_s", ())?;
24
- labels.push(label);
25
- data_vec.push(val);
26
- }
27
-
28
- let chart_data: Vec<(&str, u64)> = labels
29
- .iter()
30
- .zip(data_vec.iter())
31
- .map(|(l, v)| (l.as_str(), *v))
32
- .collect();
28
+ let direction = if direction_sym.to_string() == "horizontal" {
29
+ Direction::Horizontal
30
+ } else {
31
+ Direction::Vertical
32
+ };
33
33
 
34
34
  let mut bar_chart = BarChart::default()
35
- .data(&chart_data)
36
35
  .bar_width(bar_width)
37
- .bar_gap(bar_gap);
36
+ .bar_gap(bar_gap)
37
+ .group_gap(group_gap)
38
+ .direction(direction);
39
+
40
+ // Data parsing
41
+ let ruby = magnus::Ruby::get().unwrap();
42
+ if let Some(array) = RArray::from_value(data_val) {
43
+ for i in 0..array.len() {
44
+ let index = isize::try_from(i)
45
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
46
+ let group_obj: Value = array.entry(index)?;
47
+
48
+ let label_val: Value = group_obj.funcall("label", ())?;
49
+ let label_str: String = if label_val.is_nil() {
50
+ String::new()
51
+ } else {
52
+ label_val.funcall("to_s", ())?
53
+ };
54
+ let label_ref = bump.alloc_str(&label_str) as &str;
55
+
56
+ let bars_array: RArray = group_obj.funcall("bars", ())?;
57
+ let mut bars: Vec<Bar> = Vec::new();
58
+
59
+ for j in 0..bars_array.len() {
60
+ let bar_idx = isize::try_from(j)
61
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
62
+ let bar_obj: Value = bars_array.entry(bar_idx)?;
63
+
64
+ let value: u64 = bar_obj.funcall("value", ())?;
65
+ let mut bar = Bar::default().value(value);
66
+
67
+ let label_val: Value = bar_obj.funcall("label", ())?;
68
+ if !label_val.is_nil() {
69
+ let s: String = label_val.funcall("to_s", ())?;
70
+ let s_ref = bump.alloc_str(&s) as &str;
71
+ bar = bar.label(Line::from(s_ref));
72
+ }
73
+
74
+ let text_val: Value = bar_obj.funcall("text_value", ())?;
75
+ if !text_val.is_nil() {
76
+ let s: String = text_val.funcall("to_s", ())?;
77
+ let s_ref = bump.alloc_str(&s) as &str;
78
+ bar = bar.text_value(s_ref);
79
+ }
80
+
81
+ let style_val: Value = bar_obj.funcall("style", ())?;
82
+ if !style_val.is_nil() {
83
+ bar = bar.style(parse_style(style_val)?);
84
+ }
85
+
86
+ let val_style_val: Value = bar_obj.funcall("value_style", ())?;
87
+ if !val_style_val.is_nil() {
88
+ bar = bar.value_style(parse_style(val_style_val)?);
89
+ }
90
+
91
+ bars.push(bar);
92
+ }
93
+
94
+ let mut group = BarGroup::new(bars);
95
+ if !label_ref.is_empty() {
96
+ group = group.label(Line::from(label_ref));
97
+ }
98
+ bar_chart = bar_chart.data(group);
99
+ }
100
+ }
38
101
 
39
102
  if !max_val.is_nil() {
40
103
  let max: u64 = u64::try_convert(max_val)?;
@@ -46,7 +109,19 @@ pub fn render(frame: &mut Frame, area: Rect, node: Value) -> Result<(), Error> {
46
109
  }
47
110
 
48
111
  if !block_val.is_nil() {
49
- bar_chart = bar_chart.block(parse_block(block_val)?);
112
+ bar_chart = bar_chart.block(parse_block(block_val, &bump)?);
113
+ }
114
+
115
+ if !label_style_val.is_nil() {
116
+ bar_chart = bar_chart.label_style(parse_style(label_style_val)?);
117
+ }
118
+
119
+ if !value_style_val.is_nil() {
120
+ bar_chart = bar_chart.value_style(parse_style(value_style_val)?);
121
+ }
122
+
123
+ if !bar_set_val.is_nil() {
124
+ bar_chart = bar_chart.bar_set(parse_bar_set(bar_set_val, &bump)?);
50
125
  }
51
126
 
52
127
  frame.render_widget(bar_chart, area);
@@ -1,12 +1,41 @@
1
1
  // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
2
  // SPDX-License-Identifier: AGPL-3.0-or-later
3
3
 
4
+ use crate::rendering::render_node;
4
5
  use crate::style::parse_block;
5
- use magnus::{Error, Value};
6
+ use bumpalo::Bump;
7
+ use magnus::{prelude::*, Error, Value};
6
8
  use ratatui::{layout::Rect, widgets::Widget, Frame};
7
9
 
8
10
  pub fn render(frame: &mut Frame, area: Rect, node: Value) -> Result<(), Error> {
9
- let block = parse_block(node)?;
10
- block.render(area, frame.buffer_mut());
11
+ let bump = Bump::new();
12
+ let block = parse_block(node, &bump)?;
13
+ let block_clone = block.clone();
14
+
15
+ // Render the block itself (borders, styling)
16
+ block_clone.render(area, frame.buffer_mut());
17
+
18
+ // Get children and render them within the block's inner area
19
+ let children_val: Value = node.funcall("children", ())?;
20
+ let children_array = magnus::RArray::from_value(children_val);
21
+
22
+ if let Some(arr) = children_array {
23
+ if !arr.is_empty() {
24
+ // Calculate the inner area of the block (excluding borders and padding)
25
+ let inner = block.inner(area);
26
+
27
+ // Render each child in the block's inner area
28
+ for i in 0..arr.len() {
29
+ let ruby = magnus::Ruby::get().unwrap();
30
+ let index = isize::try_from(i)
31
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
32
+ let child: Value = arr.entry(index)?;
33
+ if let Err(e) = render_node(frame, inner, child) {
34
+ eprintln!("Error rendering block child {i}: {e:?}");
35
+ }
36
+ }
37
+ }
38
+ }
39
+
11
40
  Ok(())
12
41
  }