ratatui_ruby 1.4.0-x86_64-linux

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 (292) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE +15 -0
  3. data/LICENSES/AGPL-3.0-or-later.txt +661 -0
  4. data/LICENSES/CC-BY-SA-4.0.txt +427 -0
  5. data/LICENSES/CC0-1.0.txt +121 -0
  6. data/LICENSES/LGPL-3.0-or-later.txt +304 -0
  7. data/LICENSES/MIT-0.txt +16 -0
  8. data/LICENSES/MIT.txt +21 -0
  9. data/REUSE.toml +42 -0
  10. data/exe/.gitkeep +0 -0
  11. data/ext/ratatui_ruby/.cargo/config.toml +13 -0
  12. data/ext/ratatui_ruby/.gitignore +4 -0
  13. data/ext/ratatui_ruby/Cargo.lock +1737 -0
  14. data/ext/ratatui_ruby/Cargo.toml +24 -0
  15. data/ext/ratatui_ruby/clippy.toml +7 -0
  16. data/ext/ratatui_ruby/extconf.rb +21 -0
  17. data/ext/ratatui_ruby/src/color.rs +82 -0
  18. data/ext/ratatui_ruby/src/errors.rs +28 -0
  19. data/ext/ratatui_ruby/src/events.rs +700 -0
  20. data/ext/ratatui_ruby/src/frame.rs +241 -0
  21. data/ext/ratatui_ruby/src/lib.rs +343 -0
  22. data/ext/ratatui_ruby/src/lib_header.rs +11 -0
  23. data/ext/ratatui_ruby/src/rendering.rs +158 -0
  24. data/ext/ratatui_ruby/src/string_width.rs +101 -0
  25. data/ext/ratatui_ruby/src/style.rs +469 -0
  26. data/ext/ratatui_ruby/src/terminal/capabilities.rs +46 -0
  27. data/ext/ratatui_ruby/src/terminal/init.rs +233 -0
  28. data/ext/ratatui_ruby/src/terminal/mod.rs +42 -0
  29. data/ext/ratatui_ruby/src/terminal/mutations.rs +158 -0
  30. data/ext/ratatui_ruby/src/terminal/queries.rs +231 -0
  31. data/ext/ratatui_ruby/src/terminal/query.rs +400 -0
  32. data/ext/ratatui_ruby/src/terminal/storage.rs +109 -0
  33. data/ext/ratatui_ruby/src/terminal/wrapper.rs +16 -0
  34. data/ext/ratatui_ruby/src/text.rs +225 -0
  35. data/ext/ratatui_ruby/src/widgets/barchart.rs +169 -0
  36. data/ext/ratatui_ruby/src/widgets/block.rs +41 -0
  37. data/ext/ratatui_ruby/src/widgets/calendar.rs +84 -0
  38. data/ext/ratatui_ruby/src/widgets/canvas.rs +183 -0
  39. data/ext/ratatui_ruby/src/widgets/center.rs +79 -0
  40. data/ext/ratatui_ruby/src/widgets/chart.rs +222 -0
  41. data/ext/ratatui_ruby/src/widgets/clear.rs +39 -0
  42. data/ext/ratatui_ruby/src/widgets/cursor.rs +32 -0
  43. data/ext/ratatui_ruby/src/widgets/gauge.rs +65 -0
  44. data/ext/ratatui_ruby/src/widgets/layout.rs +379 -0
  45. data/ext/ratatui_ruby/src/widgets/line_gauge.rs +100 -0
  46. data/ext/ratatui_ruby/src/widgets/list.rs +378 -0
  47. data/ext/ratatui_ruby/src/widgets/list_state.rs +173 -0
  48. data/ext/ratatui_ruby/src/widgets/mod.rs +26 -0
  49. data/ext/ratatui_ruby/src/widgets/overlay.rs +24 -0
  50. data/ext/ratatui_ruby/src/widgets/paragraph.rs +87 -0
  51. data/ext/ratatui_ruby/src/widgets/ratatui_logo.rs +40 -0
  52. data/ext/ratatui_ruby/src/widgets/ratatui_mascot.rs +55 -0
  53. data/ext/ratatui_ruby/src/widgets/scrollbar.rs +214 -0
  54. data/ext/ratatui_ruby/src/widgets/scrollbar_state.rs +169 -0
  55. data/ext/ratatui_ruby/src/widgets/sparkline.rs +127 -0
  56. data/ext/ratatui_ruby/src/widgets/table.rs +415 -0
  57. data/ext/ratatui_ruby/src/widgets/table_state.rs +203 -0
  58. data/ext/ratatui_ruby/src/widgets/tabs.rs +194 -0
  59. data/lib/ratatui_ruby/backend/window_size.rb +50 -0
  60. data/lib/ratatui_ruby/backend.rb +59 -0
  61. data/lib/ratatui_ruby/buffer/cell.rb +212 -0
  62. data/lib/ratatui_ruby/buffer.rb +149 -0
  63. data/lib/ratatui_ruby/cell.rb +208 -0
  64. data/lib/ratatui_ruby/debug.rb +215 -0
  65. data/lib/ratatui_ruby/draw.rb +63 -0
  66. data/lib/ratatui_ruby/event/focus_gained.rb +125 -0
  67. data/lib/ratatui_ruby/event/focus_lost.rb +127 -0
  68. data/lib/ratatui_ruby/event/key/character.rb +53 -0
  69. data/lib/ratatui_ruby/event/key/dwim.rb +301 -0
  70. data/lib/ratatui_ruby/event/key/media.rb +46 -0
  71. data/lib/ratatui_ruby/event/key/modifier.rb +107 -0
  72. data/lib/ratatui_ruby/event/key/navigation.rb +72 -0
  73. data/lib/ratatui_ruby/event/key/system.rb +47 -0
  74. data/lib/ratatui_ruby/event/key.rb +479 -0
  75. data/lib/ratatui_ruby/event/mouse.rb +291 -0
  76. data/lib/ratatui_ruby/event/none.rb +53 -0
  77. data/lib/ratatui_ruby/event/paste.rb +130 -0
  78. data/lib/ratatui_ruby/event/resize.rb +221 -0
  79. data/lib/ratatui_ruby/event/sync.rb +52 -0
  80. data/lib/ratatui_ruby/event.rb +163 -0
  81. data/lib/ratatui_ruby/frame.rb +257 -0
  82. data/lib/ratatui_ruby/labs/a11y.rb +182 -0
  83. data/lib/ratatui_ruby/labs/frame_a11y_capture.rb +50 -0
  84. data/lib/ratatui_ruby/labs.rb +47 -0
  85. data/lib/ratatui_ruby/layout/alignment.rb +91 -0
  86. data/lib/ratatui_ruby/layout/constraint.rb +337 -0
  87. data/lib/ratatui_ruby/layout/layout.rb +258 -0
  88. data/lib/ratatui_ruby/layout/position.rb +81 -0
  89. data/lib/ratatui_ruby/layout/rect.rb +733 -0
  90. data/lib/ratatui_ruby/layout/size.rb +62 -0
  91. data/lib/ratatui_ruby/layout.rb +29 -0
  92. data/lib/ratatui_ruby/list_state.rb +201 -0
  93. data/lib/ratatui_ruby/output_guard.rb +171 -0
  94. data/lib/ratatui_ruby/ratatui_ruby.so +0 -0
  95. data/lib/ratatui_ruby/scrollbar_state.rb +122 -0
  96. data/lib/ratatui_ruby/style/color.rb +149 -0
  97. data/lib/ratatui_ruby/style/style.rb +147 -0
  98. data/lib/ratatui_ruby/style.rb +19 -0
  99. data/lib/ratatui_ruby/symbols.rb +435 -0
  100. data/lib/ratatui_ruby/synthetic_events.rb +106 -0
  101. data/lib/ratatui_ruby/table_state.rb +251 -0
  102. data/lib/ratatui_ruby/terminal/capabilities.rb +316 -0
  103. data/lib/ratatui_ruby/terminal/viewport.rb +80 -0
  104. data/lib/ratatui_ruby/terminal.rb +66 -0
  105. data/lib/ratatui_ruby/terminal_lifecycle.rb +303 -0
  106. data/lib/ratatui_ruby/terminal_lifecycle.rb.bak +197 -0
  107. data/lib/ratatui_ruby/test_helper/event_injection.rb +241 -0
  108. data/lib/ratatui_ruby/test_helper/global_state.rb +111 -0
  109. data/lib/ratatui_ruby/test_helper/snapshot.rb +568 -0
  110. data/lib/ratatui_ruby/test_helper/snapshots/axis_labels_alignment.ansi +24 -0
  111. data/lib/ratatui_ruby/test_helper/snapshots/axis_labels_alignment.txt +24 -0
  112. data/lib/ratatui_ruby/test_helper/snapshots/barchart_styled_label.ansi +5 -0
  113. data/lib/ratatui_ruby/test_helper/snapshots/barchart_styled_label.txt +5 -0
  114. data/lib/ratatui_ruby/test_helper/snapshots/chart_rendering.ansi +24 -0
  115. data/lib/ratatui_ruby/test_helper/snapshots/chart_rendering.txt +24 -0
  116. data/lib/ratatui_ruby/test_helper/snapshots/half_block_marker.ansi +12 -0
  117. data/lib/ratatui_ruby/test_helper/snapshots/half_block_marker.txt +12 -0
  118. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_bottom.ansi +12 -0
  119. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_bottom.txt +12 -0
  120. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_left.ansi +12 -0
  121. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_left.txt +12 -0
  122. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_right.ansi +12 -0
  123. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_right.txt +12 -0
  124. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_top.ansi +12 -0
  125. data/lib/ratatui_ruby/test_helper/snapshots/legend_position_top.txt +12 -0
  126. data/lib/ratatui_ruby/test_helper/snapshots/my_snapshot.txt +1 -0
  127. data/lib/ratatui_ruby/test_helper/snapshots/styled_axis_title.ansi +10 -0
  128. data/lib/ratatui_ruby/test_helper/snapshots/styled_axis_title.txt +10 -0
  129. data/lib/ratatui_ruby/test_helper/snapshots/styled_dataset_name.ansi +10 -0
  130. data/lib/ratatui_ruby/test_helper/snapshots/styled_dataset_name.txt +10 -0
  131. data/lib/ratatui_ruby/test_helper/style_assertions.rb +449 -0
  132. data/lib/ratatui_ruby/test_helper/subprocess_timeout.rb +35 -0
  133. data/lib/ratatui_ruby/test_helper/terminal.rb +187 -0
  134. data/lib/ratatui_ruby/test_helper/test_doubles.rb +86 -0
  135. data/lib/ratatui_ruby/test_helper.rb +115 -0
  136. data/lib/ratatui_ruby/text/line.rb +245 -0
  137. data/lib/ratatui_ruby/text/span.rb +158 -0
  138. data/lib/ratatui_ruby/text.rb +99 -0
  139. data/lib/ratatui_ruby/tui/buffer_factories.rb +22 -0
  140. data/lib/ratatui_ruby/tui/canvas_factories.rb +149 -0
  141. data/lib/ratatui_ruby/tui/core.rb +67 -0
  142. data/lib/ratatui_ruby/tui/layout_factories.rb +153 -0
  143. data/lib/ratatui_ruby/tui/state_factories.rb +77 -0
  144. data/lib/ratatui_ruby/tui/style_factories.rb +22 -0
  145. data/lib/ratatui_ruby/tui/text_factories.rb +86 -0
  146. data/lib/ratatui_ruby/tui/widget_factories.rb +272 -0
  147. data/lib/ratatui_ruby/tui.rb +106 -0
  148. data/lib/ratatui_ruby/version.rb +12 -0
  149. data/lib/ratatui_ruby/widgets/bar_chart/bar.rb +51 -0
  150. data/lib/ratatui_ruby/widgets/bar_chart/bar_group.rb +29 -0
  151. data/lib/ratatui_ruby/widgets/bar_chart.rb +308 -0
  152. data/lib/ratatui_ruby/widgets/block.rb +266 -0
  153. data/lib/ratatui_ruby/widgets/calendar.rb +88 -0
  154. data/lib/ratatui_ruby/widgets/canvas.rb +297 -0
  155. data/lib/ratatui_ruby/widgets/cell.rb +59 -0
  156. data/lib/ratatui_ruby/widgets/center.rb +71 -0
  157. data/lib/ratatui_ruby/widgets/chart.rb +172 -0
  158. data/lib/ratatui_ruby/widgets/clear.rb +66 -0
  159. data/lib/ratatui_ruby/widgets/coerceable_widget.rb +77 -0
  160. data/lib/ratatui_ruby/widgets/cursor.rb +54 -0
  161. data/lib/ratatui_ruby/widgets/gauge.rb +146 -0
  162. data/lib/ratatui_ruby/widgets/line_gauge.rb +158 -0
  163. data/lib/ratatui_ruby/widgets/list.rb +252 -0
  164. data/lib/ratatui_ruby/widgets/list_item.rb +55 -0
  165. data/lib/ratatui_ruby/widgets/overlay.rb +55 -0
  166. data/lib/ratatui_ruby/widgets/paragraph.rb +113 -0
  167. data/lib/ratatui_ruby/widgets/ratatui_logo.rb +35 -0
  168. data/lib/ratatui_ruby/widgets/ratatui_mascot.rb +40 -0
  169. data/lib/ratatui_ruby/widgets/row.rb +123 -0
  170. data/lib/ratatui_ruby/widgets/scrollbar.rb +147 -0
  171. data/lib/ratatui_ruby/widgets/shape/label.rb +80 -0
  172. data/lib/ratatui_ruby/widgets/sparkline.rb +153 -0
  173. data/lib/ratatui_ruby/widgets/table.rb +213 -0
  174. data/lib/ratatui_ruby/widgets/tabs.rb +91 -0
  175. data/lib/ratatui_ruby/widgets.rb +43 -0
  176. data/lib/ratatui_ruby.rb +555 -0
  177. data/sig/examples/app_all_events/app.rbs +11 -0
  178. data/sig/examples/app_all_events/model/app_model.rbs +23 -0
  179. data/sig/examples/app_all_events/model/event_entry.rbs +23 -0
  180. data/sig/examples/app_all_events/model/timestamp.rbs +11 -0
  181. data/sig/examples/app_all_events/view/app_view.rbs +8 -0
  182. data/sig/examples/app_all_events/view/controls_view.rbs +6 -0
  183. data/sig/examples/app_all_events/view/counts_view.rbs +6 -0
  184. data/sig/examples/app_all_events/view/live_view.rbs +6 -0
  185. data/sig/examples/app_all_events/view/log_view.rbs +6 -0
  186. data/sig/examples/app_all_events/view.rbs +14 -0
  187. data/sig/examples/app_cli_rich_moments/app.rbs +12 -0
  188. data/sig/examples/app_color_picker/app.rbs +17 -0
  189. data/sig/examples/app_external_editor/app.rbs +12 -0
  190. data/sig/examples/app_login_form/app.rbs +11 -0
  191. data/sig/examples/app_stateful_interaction/app.rbs +39 -0
  192. data/sig/examples/verify_quickstart_dsl/app.rbs +17 -0
  193. data/sig/examples/verify_quickstart_lifecycle/app.rbs +17 -0
  194. data/sig/examples/verify_readme_usage/app.rbs +17 -0
  195. data/sig/examples/widget_block_demo/app.rbs +38 -0
  196. data/sig/examples/widget_box_demo/app.rbs +17 -0
  197. data/sig/examples/widget_calendar_demo/app.rbs +17 -0
  198. data/sig/examples/widget_cell_demo/app.rbs +17 -0
  199. data/sig/examples/widget_chart_demo/app.rbs +17 -0
  200. data/sig/examples/widget_gauge_demo/app.rbs +17 -0
  201. data/sig/examples/widget_layout_split/app.rbs +16 -0
  202. data/sig/examples/widget_line_gauge_demo/app.rbs +17 -0
  203. data/sig/examples/widget_list_demo/app.rbs +17 -0
  204. data/sig/examples/widget_map_demo/app.rbs +17 -0
  205. data/sig/examples/widget_popup_demo/app.rbs +17 -0
  206. data/sig/examples/widget_ratatui_logo_demo/app.rbs +17 -0
  207. data/sig/examples/widget_ratatui_mascot_demo/app.rbs +17 -0
  208. data/sig/examples/widget_rect/app.rbs +18 -0
  209. data/sig/examples/widget_render/app.rbs +16 -0
  210. data/sig/examples/widget_rich_text/app.rbs +17 -0
  211. data/sig/examples/widget_scroll_text/app.rbs +17 -0
  212. data/sig/examples/widget_scrollbar_demo/app.rbs +17 -0
  213. data/sig/examples/widget_sparkline_demo/app.rbs +16 -0
  214. data/sig/examples/widget_style_colors/app.rbs +20 -0
  215. data/sig/examples/widget_table_demo/app.rbs +17 -0
  216. data/sig/examples/widget_text_width/app.rbs +16 -0
  217. data/sig/generated/event_key_predicates.rbs +1348 -0
  218. data/sig/manifest.yaml +5 -0
  219. data/sig/patches/data.rbs +26 -0
  220. data/sig/patches/debugger__.rbs +8 -0
  221. data/sig/ratatui_ruby/backend/window_size.rbs +17 -0
  222. data/sig/ratatui_ruby/backend.rbs +12 -0
  223. data/sig/ratatui_ruby/buffer/cell.rbs +46 -0
  224. data/sig/ratatui_ruby/buffer.rbs +18 -0
  225. data/sig/ratatui_ruby/cell.rbs +44 -0
  226. data/sig/ratatui_ruby/clear.rbs +18 -0
  227. data/sig/ratatui_ruby/constraint.rbs +26 -0
  228. data/sig/ratatui_ruby/debug.rbs +45 -0
  229. data/sig/ratatui_ruby/draw.rbs +30 -0
  230. data/sig/ratatui_ruby/event.rbs +249 -0
  231. data/sig/ratatui_ruby/frame.rbs +23 -0
  232. data/sig/ratatui_ruby/interfaces.rbs +25 -0
  233. data/sig/ratatui_ruby/labs.rbs +90 -0
  234. data/sig/ratatui_ruby/layout/alignment.rbs +26 -0
  235. data/sig/ratatui_ruby/layout/constraint.rbs +39 -0
  236. data/sig/ratatui_ruby/layout/layout.rbs +45 -0
  237. data/sig/ratatui_ruby/layout/position.rbs +18 -0
  238. data/sig/ratatui_ruby/layout/rect.rbs +64 -0
  239. data/sig/ratatui_ruby/layout/size.rbs +18 -0
  240. data/sig/ratatui_ruby/list_state.rbs +23 -0
  241. data/sig/ratatui_ruby/output_guard.rbs +23 -0
  242. data/sig/ratatui_ruby/ratatui_ruby.rbs +113 -0
  243. data/sig/ratatui_ruby/rect.rbs +17 -0
  244. data/sig/ratatui_ruby/scrollbar_state.rbs +24 -0
  245. data/sig/ratatui_ruby/session.rbs +93 -0
  246. data/sig/ratatui_ruby/style/color.rbs +22 -0
  247. data/sig/ratatui_ruby/style/style.rbs +29 -0
  248. data/sig/ratatui_ruby/symbols.rbs +141 -0
  249. data/sig/ratatui_ruby/synthetic_events.rbs +24 -0
  250. data/sig/ratatui_ruby/table_state.rbs +27 -0
  251. data/sig/ratatui_ruby/terminal/capabilities.rbs +38 -0
  252. data/sig/ratatui_ruby/terminal/viewport.rbs +33 -0
  253. data/sig/ratatui_ruby/terminal_lifecycle.rbs +39 -0
  254. data/sig/ratatui_ruby/test_helper/event_injection.rbs +22 -0
  255. data/sig/ratatui_ruby/test_helper/snapshot.rbs +37 -0
  256. data/sig/ratatui_ruby/test_helper/style_assertions.rbs +77 -0
  257. data/sig/ratatui_ruby/test_helper/terminal.rbs +20 -0
  258. data/sig/ratatui_ruby/test_helper/test_doubles.rbs +32 -0
  259. data/sig/ratatui_ruby/test_helper.rbs +18 -0
  260. data/sig/ratatui_ruby/text/line.rbs +27 -0
  261. data/sig/ratatui_ruby/text/span.rbs +23 -0
  262. data/sig/ratatui_ruby/text.rbs +12 -0
  263. data/sig/ratatui_ruby/tui/buffer_factories.rbs +16 -0
  264. data/sig/ratatui_ruby/tui/canvas_factories.rbs +38 -0
  265. data/sig/ratatui_ruby/tui/core.rbs +23 -0
  266. data/sig/ratatui_ruby/tui/layout_factories.rbs +39 -0
  267. data/sig/ratatui_ruby/tui/state_factories.rbs +23 -0
  268. data/sig/ratatui_ruby/tui/style_factories.rbs +18 -0
  269. data/sig/ratatui_ruby/tui/text_factories.rbs +23 -0
  270. data/sig/ratatui_ruby/tui/widget_factories.rbs +138 -0
  271. data/sig/ratatui_ruby/tui.rbs +25 -0
  272. data/sig/ratatui_ruby/version.rbs +12 -0
  273. data/sig/ratatui_ruby/widgets/bar_chart.rbs +95 -0
  274. data/sig/ratatui_ruby/widgets/block.rbs +51 -0
  275. data/sig/ratatui_ruby/widgets/calendar.rbs +45 -0
  276. data/sig/ratatui_ruby/widgets/canvas.rbs +95 -0
  277. data/sig/ratatui_ruby/widgets/chart.rbs +91 -0
  278. data/sig/ratatui_ruby/widgets/coerceable_widget.rbs +26 -0
  279. data/sig/ratatui_ruby/widgets/gauge.rbs +44 -0
  280. data/sig/ratatui_ruby/widgets/line_gauge.rbs +48 -0
  281. data/sig/ratatui_ruby/widgets/list.rbs +63 -0
  282. data/sig/ratatui_ruby/widgets/misc.rbs +158 -0
  283. data/sig/ratatui_ruby/widgets/paragraph.rbs +45 -0
  284. data/sig/ratatui_ruby/widgets/row.rbs +43 -0
  285. data/sig/ratatui_ruby/widgets/scrollbar.rbs +53 -0
  286. data/sig/ratatui_ruby/widgets/shape/label.rbs +37 -0
  287. data/sig/ratatui_ruby/widgets/sparkline.rbs +45 -0
  288. data/sig/ratatui_ruby/widgets/table.rbs +78 -0
  289. data/sig/ratatui_ruby/widgets/tabs.rbs +44 -0
  290. data/sig/ratatui_ruby/widgets.rbs +16 -0
  291. data/vendor/goodcop/base.yml +1047 -0
  292. metadata +729 -0
@@ -0,0 +1,469 @@
1
+ // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
+ //
3
+ // SPDX-License-Identifier: AGPL-3.0-or-later
4
+
5
+ use crate::errors::type_error_with_context;
6
+ use bumpalo::Bump;
7
+ use magnus::{prelude::*, Error, Symbol, Value};
8
+ use ratatui::{
9
+ layout::Alignment,
10
+ style::{Color, Modifier, Style},
11
+ symbols,
12
+ text::Line,
13
+ widgets::{Block, BorderType, Borders, Padding},
14
+ };
15
+
16
+ use crate::text::parse_line;
17
+
18
+ pub fn parse_color(color_str: &str) -> Option<Color> {
19
+ // Try standard ratatui parsing first (named colors, indexed, etc.)
20
+ if let Ok(color) = color_str.parse::<Color>() {
21
+ return Some(color);
22
+ }
23
+
24
+ // Try hex parsing manually: #RRGGBB
25
+ let hex_str = color_str.trim();
26
+ if hex_str.starts_with('#') && hex_str.len() == 7 {
27
+ if let Ok(hex_val) = u32::from_str_radix(&hex_str[1..], 16) {
28
+ let r = ((hex_val >> 16) & 0xFF) as u8;
29
+ let g = ((hex_val >> 8) & 0xFF) as u8;
30
+ let b = (hex_val & 0xFF) as u8;
31
+ return Some(Color::Rgb(r, g, b));
32
+ }
33
+ }
34
+
35
+ None
36
+ }
37
+
38
+ pub fn parse_color_value(val: Value) -> Result<Option<Color>, Error> {
39
+ if val.is_nil() {
40
+ return Ok(None);
41
+ }
42
+ let s: String = val.funcall("to_s", ())?;
43
+ Ok(parse_color(&s))
44
+ }
45
+
46
+ pub fn parse_modifier_str(s: &str) -> Option<Modifier> {
47
+ match s {
48
+ "bold" => Some(Modifier::BOLD),
49
+ "italic" => Some(Modifier::ITALIC),
50
+ "dim" => Some(Modifier::DIM),
51
+ "reversed" => Some(Modifier::REVERSED),
52
+ "underlined" => Some(Modifier::UNDERLINED),
53
+ "slow_blink" => Some(Modifier::SLOW_BLINK),
54
+ "rapid_blink" => Some(Modifier::RAPID_BLINK),
55
+ "crossed_out" => Some(Modifier::CROSSED_OUT),
56
+ "hidden" => Some(Modifier::HIDDEN),
57
+ _ => None,
58
+ }
59
+ }
60
+
61
+ pub fn parse_style(style_val: Value) -> Result<Style, Error> {
62
+ let ruby = magnus::Ruby::get().unwrap();
63
+ if style_val.is_nil() {
64
+ return Ok(Style::default());
65
+ }
66
+
67
+ let mut style = Style::default();
68
+
69
+ let (fg, bg, underline_color, modifiers_val, remove_modifiers_val) =
70
+ if let Some(hash) = magnus::RHash::from_value(style_val) {
71
+ (
72
+ hash.lookup(ruby.to_symbol("fg"))
73
+ .unwrap_or_else(|_| ruby.qnil().as_value()),
74
+ hash.lookup(ruby.to_symbol("bg"))
75
+ .unwrap_or_else(|_| ruby.qnil().as_value()),
76
+ hash.lookup(ruby.to_symbol("underline_color"))
77
+ .unwrap_or_else(|_| ruby.qnil().as_value()),
78
+ hash.lookup(ruby.to_symbol("modifiers"))
79
+ .unwrap_or_else(|_| ruby.qnil().as_value()),
80
+ hash.lookup(ruby.to_symbol("remove_modifiers"))
81
+ .unwrap_or_else(|_| ruby.qnil().as_value()),
82
+ )
83
+ } else {
84
+ (
85
+ style_val.funcall("fg", ())?,
86
+ style_val.funcall("bg", ())?,
87
+ style_val.funcall("underline_color", ())?,
88
+ style_val.funcall("modifiers", ())?,
89
+ style_val.funcall("remove_modifiers", ())?,
90
+ )
91
+ };
92
+
93
+ if !fg.is_nil() {
94
+ if let Ok(fg_str) = fg.funcall::<_, _, String>("to_s", ()) {
95
+ if let Some(color) = parse_color(&fg_str) {
96
+ style = style.fg(color);
97
+ }
98
+ }
99
+ }
100
+
101
+ if !bg.is_nil() {
102
+ if let Ok(bg_str) = bg.funcall::<_, _, String>("to_s", ()) {
103
+ if let Some(color) = parse_color(&bg_str) {
104
+ style = style.bg(color);
105
+ }
106
+ }
107
+ }
108
+
109
+ if !underline_color.is_nil() {
110
+ if let Ok(uc_str) = underline_color.funcall::<_, _, String>("to_s", ()) {
111
+ if let Some(color) = parse_color(&uc_str) {
112
+ style = style.underline_color(color);
113
+ }
114
+ }
115
+ }
116
+
117
+ if !modifiers_val.is_nil() {
118
+ if let Some(modifiers_array) = magnus::RArray::from_value(modifiers_val) {
119
+ for i in 0..modifiers_array.len() {
120
+ let index = isize::try_from(i)
121
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
122
+ if let Ok(sym) = modifiers_array.entry::<Symbol>(index) {
123
+ if let Some(m) = parse_modifier_str(&sym.to_string()) {
124
+ style = style.add_modifier(m);
125
+ }
126
+ }
127
+ }
128
+ }
129
+ }
130
+
131
+ if !remove_modifiers_val.is_nil() {
132
+ if let Some(remove_modifiers_array) = magnus::RArray::from_value(remove_modifiers_val) {
133
+ for i in 0..remove_modifiers_array.len() {
134
+ let index = isize::try_from(i)
135
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
136
+ if let Ok(sym) = remove_modifiers_array.entry::<Symbol>(index) {
137
+ if let Some(m) = parse_modifier_str(&sym.to_string()) {
138
+ style = style.remove_modifier(m);
139
+ }
140
+ }
141
+ }
142
+ }
143
+ }
144
+
145
+ Ok(style)
146
+ }
147
+
148
+ pub fn parse_border_set<'a>(
149
+ set_val: Value,
150
+ bump: &'a Bump,
151
+ ) -> Result<symbols::border::Set<'a>, Error> {
152
+ let ruby = magnus::Ruby::get().unwrap();
153
+ let hash = magnus::RHash::from_value(set_val)
154
+ .ok_or_else(|| type_error_with_context(&ruby, "expected hash for border_set", set_val))?;
155
+
156
+ let get_char = |key: &str| -> Result<Option<&'a str>, Error> {
157
+ let mut val: Value = hash
158
+ .lookup(ruby.to_symbol(key))
159
+ .unwrap_or_else(|_| ruby.qnil().as_value());
160
+ if val.is_nil() {
161
+ val = hash
162
+ .lookup(ruby.str_new(key))
163
+ .unwrap_or_else(|_| ruby.qnil().as_value());
164
+ }
165
+ if val.is_nil() {
166
+ Ok(None)
167
+ } else {
168
+ let s: String = val.funcall("to_s", ())?;
169
+ Ok(Some(bump.alloc_str(&s)))
170
+ }
171
+ };
172
+
173
+ let mut set = symbols::border::Set::default();
174
+ if let Some(s) = get_char("top_left")? {
175
+ set.top_left = s;
176
+ }
177
+ if let Some(s) = get_char("top_right")? {
178
+ set.top_right = s;
179
+ }
180
+ if let Some(s) = get_char("bottom_left")? {
181
+ set.bottom_left = s;
182
+ }
183
+ if let Some(s) = get_char("bottom_right")? {
184
+ set.bottom_right = s;
185
+ }
186
+ if let Some(s) = get_char("vertical_left")? {
187
+ set.vertical_left = s;
188
+ }
189
+ if let Some(s) = get_char("vertical_right")? {
190
+ set.vertical_right = s;
191
+ }
192
+ if let Some(s) = get_char("horizontal_top")? {
193
+ set.horizontal_top = s;
194
+ }
195
+ if let Some(s) = get_char("horizontal_bottom")? {
196
+ set.horizontal_bottom = s;
197
+ }
198
+
199
+ Ok(set)
200
+ }
201
+
202
+ pub fn parse_bar_set<'a>(set_val: Value, bump: &'a Bump) -> Result<symbols::bar::Set<'a>, Error> {
203
+ let ruby = magnus::Ruby::get().unwrap();
204
+
205
+ // Check if set_val is a symbol shortcut
206
+ if let Ok(sym) = magnus::Symbol::from_value(set_val)
207
+ .ok_or::<Error>(Error::new(ruby.exception_type_error(), ""))
208
+ {
209
+ let sym_str = sym.name().unwrap_or_default();
210
+ return match sym_str.as_ref() {
211
+ "nine_levels" => Ok(symbols::bar::NINE_LEVELS),
212
+ "three_levels" => Ok(symbols::bar::THREE_LEVELS),
213
+ _ => Err(Error::new(
214
+ ruby.exception_arg_error(),
215
+ format!("unknown bar_set symbol: :{sym_str}"),
216
+ )),
217
+ };
218
+ }
219
+
220
+ let hash = magnus::RHash::from_value(set_val).ok_or_else(|| {
221
+ type_error_with_context(&ruby, "expected symbol or hash for bar_set", set_val)
222
+ })?;
223
+
224
+ let get_char = |key: &str| -> Result<Option<&'a str>, Error> {
225
+ let mut val: Value = hash
226
+ .lookup(ruby.to_symbol(key))
227
+ .unwrap_or_else(|_| ruby.qnil().as_value());
228
+ if val.is_nil() {
229
+ val = hash
230
+ .lookup(ruby.str_new(key))
231
+ .unwrap_or_else(|_| ruby.qnil().as_value());
232
+ }
233
+ if val.is_nil() {
234
+ Ok(None)
235
+ } else {
236
+ let s: String = val.funcall("to_s", ())?;
237
+ Ok(Some(bump.alloc_str(&s)))
238
+ }
239
+ };
240
+
241
+ let mut set = symbols::bar::Set::default();
242
+ if let Some(s) = get_char("empty")? {
243
+ set.empty = s;
244
+ }
245
+ if let Some(s) = get_char("one_eighth")? {
246
+ set.one_eighth = s;
247
+ }
248
+ if let Some(s) = get_char("one_quarter")? {
249
+ set.one_quarter = s;
250
+ }
251
+ if let Some(s) = get_char("three_eighths")? {
252
+ set.three_eighths = s;
253
+ }
254
+ if let Some(s) = get_char("half")? {
255
+ set.half = s;
256
+ }
257
+ if let Some(s) = get_char("five_eighths")? {
258
+ set.five_eighths = s;
259
+ }
260
+ if let Some(s) = get_char("three_quarters")? {
261
+ set.three_quarters = s;
262
+ }
263
+ if let Some(s) = get_char("seven_eighths")? {
264
+ set.seven_eighths = s;
265
+ }
266
+ if let Some(s) = get_char("full")? {
267
+ set.full = s;
268
+ }
269
+
270
+ Ok(set)
271
+ }
272
+
273
+ pub fn parse_block(block_val: Value, bump: &Bump) -> Result<Block<'_>, Error> {
274
+ if block_val.is_nil() {
275
+ return Ok(Block::default());
276
+ }
277
+
278
+ let mut block = Block::default();
279
+ if let Ok(v) = block_val.funcall::<&str, _, Value>("style", ()) {
280
+ if !v.is_nil() {
281
+ block = block.style(parse_style(v)?);
282
+ }
283
+ }
284
+
285
+ if let Ok(v) = block_val.funcall::<&str, _, Value>("title_style", ()) {
286
+ if !v.is_nil() {
287
+ block = block.title_style(parse_style(v)?);
288
+ }
289
+ }
290
+
291
+ if let Ok(title) = block_val.funcall::<&str, _, Value>("title", ()) {
292
+ if !title.is_nil() {
293
+ let s: String = title.funcall("to_s", ())?;
294
+ block = block.title(Line::from(s));
295
+ }
296
+ }
297
+
298
+ if let Ok(v) = block_val.funcall::<&str, _, Value>("title_alignment", ()) {
299
+ if let Some(align_sym) = Symbol::from_value(v) {
300
+ match align_sym.to_string().as_str() {
301
+ "center" => block = block.title_alignment(Alignment::Center),
302
+ "right" => block = block.title_alignment(Alignment::Right),
303
+ _ => block = block.title_alignment(Alignment::Left),
304
+ }
305
+ }
306
+ }
307
+
308
+ block = parse_titles(block_val, block)?;
309
+ block = parse_borders(block_val, block, bump)?;
310
+ block = parse_padding(block_val, block);
311
+
312
+ Ok(block)
313
+ }
314
+
315
+ fn parse_titles(block_val: Value, mut block: Block<'_>) -> Result<Block<'_>, Error> {
316
+ if let Ok(titles_val) = block_val.funcall::<&str, _, Value>("titles", ()) {
317
+ if titles_val.is_nil() {
318
+ return Ok(block);
319
+ }
320
+ if let Some(titles_array) = magnus::RArray::from_value(titles_val) {
321
+ for i in 0..titles_array.len() {
322
+ let ruby = magnus::Ruby::get().unwrap();
323
+ let index = isize::try_from(i)
324
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
325
+ let title_item: Value = titles_array.entry(index)?;
326
+ let mut alignment = Alignment::Left;
327
+ let mut is_bottom = false;
328
+ let mut style = Style::default();
329
+ let mut line: Option<Line<'static>> = None;
330
+
331
+ if let Some(hash) = magnus::RHash::from_value(title_item) {
332
+ if let Ok(v) = hash.lookup::<_, Value>(ruby.to_symbol("content")) {
333
+ if !v.is_nil() {
334
+ // First, try to parse as a Line object (preserves styling)
335
+ if let Ok(parsed_line) = parse_line(v) {
336
+ line = Some(parsed_line);
337
+ } else {
338
+ // Fallback to string
339
+ let content: String = v.funcall("to_s", ())?;
340
+ line = Some(Line::from(content));
341
+ }
342
+ }
343
+ }
344
+ if let Ok(v) = hash.lookup::<_, Value>(ruby.to_symbol("alignment")) {
345
+ if let Some(s) = Symbol::from_value(v) {
346
+ match s.to_string().as_str() {
347
+ "center" => alignment = Alignment::Center,
348
+ "right" => alignment = Alignment::Right,
349
+ _ => {}
350
+ }
351
+ }
352
+ }
353
+ if let Ok(v) = hash.lookup::<_, Value>(ruby.to_symbol("position")) {
354
+ if let Some(s) = Symbol::from_value(v) {
355
+ if s.to_string().as_str() == "bottom" {
356
+ is_bottom = true;
357
+ }
358
+ }
359
+ }
360
+ if let Ok(v) = hash.lookup::<_, Value>(ruby.to_symbol("style")) {
361
+ if !v.is_nil() {
362
+ style = parse_style(v)?;
363
+ }
364
+ }
365
+ } else {
366
+ let content: String = title_item.funcall("to_s", ())?;
367
+ line = Some(Line::from(content));
368
+ }
369
+
370
+ if let Some(mut l) = line {
371
+ l = l.alignment(alignment);
372
+ // Only apply style if the line doesn't already have styled spans
373
+ if style != Style::default() {
374
+ l = l.style(style);
375
+ }
376
+ block = if is_bottom {
377
+ block.title_bottom(l)
378
+ } else {
379
+ block.title_top(l)
380
+ };
381
+ }
382
+ }
383
+ }
384
+ }
385
+ Ok(block)
386
+ }
387
+
388
+ fn parse_borders<'a>(
389
+ block_val: Value,
390
+ mut block: Block<'a>,
391
+ bump: &'a Bump,
392
+ ) -> Result<Block<'a>, Error> {
393
+ if let Ok(borders_val) = block_val.funcall::<&str, _, Value>("borders", ()) {
394
+ if !borders_val.is_nil() {
395
+ let mut ratatui_borders = Borders::NONE;
396
+ if let Some(sym) = Symbol::from_value(borders_val) {
397
+ match sym.to_string().as_str() {
398
+ "all" => ratatui_borders = Borders::ALL,
399
+ "top" => ratatui_borders = Borders::TOP,
400
+ "bottom" => ratatui_borders = Borders::BOTTOM,
401
+ "left" => ratatui_borders = Borders::LEFT,
402
+ "right" => ratatui_borders = Borders::RIGHT,
403
+ _ => {}
404
+ }
405
+ } else if let Some(arr) = magnus::RArray::from_value(borders_val) {
406
+ for i in 0..arr.len() {
407
+ let ruby = magnus::Ruby::get().unwrap();
408
+ let index = isize::try_from(i)
409
+ .map_err(|e| Error::new(ruby.exception_range_error(), e.to_string()))?;
410
+ let sym: Symbol = arr.entry(index)?;
411
+ match sym.to_string().as_str() {
412
+ "all" => ratatui_borders |= Borders::ALL,
413
+ "top" => ratatui_borders |= Borders::TOP,
414
+ "bottom" => ratatui_borders |= Borders::BOTTOM,
415
+ "left" => ratatui_borders |= Borders::LEFT,
416
+ "right" => ratatui_borders |= Borders::RIGHT,
417
+ _ => {}
418
+ }
419
+ }
420
+ }
421
+ block = block.borders(ratatui_borders);
422
+ }
423
+ }
424
+
425
+ if let Ok(v) = block_val.funcall::<&str, _, Value>("border_style", ()) {
426
+ if !v.is_nil() {
427
+ block = block.border_style(parse_style(v)?);
428
+ }
429
+ }
430
+
431
+ if let Ok(v) = block_val.funcall::<&str, _, Value>("border_set", ()) {
432
+ if !v.is_nil() {
433
+ block = block.border_set(parse_border_set(v, bump)?);
434
+ } else if let Ok(v) = block_val.funcall::<&str, _, Value>("border_type", ()) {
435
+ if let Some(sym) = Symbol::from_value(v) {
436
+ match sym.to_string().as_str() {
437
+ "rounded" => block = block.border_type(BorderType::Rounded),
438
+ "double" => block = block.border_type(BorderType::Double),
439
+ "thick" => block = block.border_type(BorderType::Thick),
440
+ "quadrant_inside" => block = block.border_type(BorderType::QuadrantInside),
441
+ "quadrant_outside" => block = block.border_type(BorderType::QuadrantOutside),
442
+ _ => block = block.border_type(BorderType::Plain),
443
+ }
444
+ }
445
+ }
446
+ }
447
+ Ok(block)
448
+ }
449
+
450
+ fn parse_padding(block_val: Value, block: Block<'_>) -> Block<'_> {
451
+ if let Ok(padding_val) = block_val.funcall::<&str, _, Value>("padding", ()) {
452
+ if padding_val.is_nil() {
453
+ return block;
454
+ }
455
+ if let Ok(p) = u16::try_convert(padding_val) {
456
+ return block.padding(Padding::uniform(p));
457
+ }
458
+ if let Some(arr) = magnus::RArray::from_value(padding_val) {
459
+ if arr.len() == 4 {
460
+ let left: u16 = arr.entry(0).unwrap_or(0);
461
+ let right: u16 = arr.entry(1).unwrap_or(0);
462
+ let top: u16 = arr.entry(2).unwrap_or(0);
463
+ let bottom: u16 = arr.entry(3).unwrap_or(0);
464
+ return block.padding(Padding::new(left, right, top, bottom));
465
+ }
466
+ }
467
+ }
468
+ block
469
+ }
@@ -0,0 +1,46 @@
1
+ // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
+ // SPDX-License-Identifier: AGPL-3.0-or-later
3
+
4
+ //! Terminal capability detection functions.
5
+
6
+ use magnus::{Error, Ruby};
7
+
8
+ /// Returns color support level (8, 256, or `u16::MAX` for truecolor)
9
+ ///
10
+ /// Wraps `crossterm::style::available_color_count()` which checks COLORTERM and TERM env vars.
11
+ pub fn available_color_count() -> u16 {
12
+ ratatui::crossterm::style::available_color_count()
13
+ }
14
+
15
+ /// Query if terminal supports Kitty keyboard protocol
16
+ ///
17
+ /// Note: This requires raw mode and may return errors in some environments.
18
+ pub fn supports_keyboard_enhancement() -> Result<bool, Error> {
19
+ ratatui::crossterm::terminal::supports_keyboard_enhancement().map_err(|e| {
20
+ Error::new(
21
+ Ruby::get().unwrap().exception_runtime_error(),
22
+ e.to_string(),
23
+ )
24
+ })
25
+ }
26
+
27
+ /// Query terminal window size in characters and pixels
28
+ ///
29
+ /// Wraps `crossterm::terminal::window_size()`. Returns
30
+ /// Some((columns, rows, `pixel_width`, `pixel_height`)) or None if query fails.
31
+ /// Note: Pixel dimensions may be 0 on some systems (marked as "unused" by Unix drivers,
32
+ /// not implemented on Windows).
33
+ pub fn terminal_window_size() -> Option<(u16, u16, u16, u16)> {
34
+ match ratatui::crossterm::terminal::window_size() {
35
+ Ok(size) => Some((size.columns, size.rows, size.width, size.height)),
36
+ Err(_) => None,
37
+ }
38
+ }
39
+
40
+ /// Globally override `NO_COLOR` detection
41
+ ///
42
+ /// Wraps `crossterm::style::force_color_output()`. When enabled, color output will
43
+ /// be forced even if `NO_COLOR` is set. Useful for `--color=always` flags.
44
+ pub fn force_color_output(enable: bool) {
45
+ ratatui::crossterm::style::force_color_output(enable);
46
+ }