ratatui_ruby 0.3.1 → 0.4.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 (300) 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 +54 -13
  7. data/CHANGELOG.md +186 -1
  8. data/README.md +17 -15
  9. data/doc/application_architecture.md +116 -0
  10. data/doc/application_testing.md +12 -7
  11. data/doc/contributors/better_dx.md +543 -0
  12. data/doc/contributors/design/ruby_frontend.md +1 -1
  13. data/doc/contributors/developing_examples.md +203 -0
  14. data/doc/contributors/documentation_style.md +97 -0
  15. data/doc/contributors/dwim_dx.md +366 -0
  16. data/doc/contributors/example_analysis.md +82 -0
  17. data/doc/custom.css +14 -0
  18. data/doc/event_handling.md +119 -0
  19. data/doc/images/all_events.png +0 -0
  20. data/doc/images/analytics.png +0 -0
  21. data/doc/images/block_padding.png +0 -0
  22. data/doc/images/block_titles.png +0 -0
  23. data/doc/images/box_demo.png +0 -0
  24. data/doc/images/calendar_demo.png +0 -0
  25. data/doc/images/cell_demo.png +0 -0
  26. data/doc/images/chart_demo.png +0 -0
  27. data/doc/images/custom_widget.png +0 -0
  28. data/doc/images/flex_layout.png +0 -0
  29. data/doc/images/gauge_demo.png +0 -0
  30. data/doc/images/hit_test.png +0 -0
  31. data/doc/images/line_gauge_demo.png +0 -0
  32. data/doc/images/list_demo.png +0 -0
  33. data/doc/images/list_styles.png +0 -0
  34. data/doc/images/login_form.png +0 -0
  35. data/doc/images/map_demo.png +0 -0
  36. data/doc/images/mouse_events.png +0 -0
  37. data/doc/images/popup_demo.png +0 -0
  38. data/doc/images/quickstart_dsl.png +0 -0
  39. data/doc/images/quickstart_lifecycle.png +0 -0
  40. data/doc/images/ratatui_logo_demo.png +0 -0
  41. data/doc/images/readme_usage.png +0 -0
  42. data/doc/images/rich_text.png +0 -0
  43. data/doc/images/scroll_text.png +0 -0
  44. data/doc/images/scrollbar_demo.png +0 -0
  45. data/doc/images/sparkline_demo.png +0 -0
  46. data/doc/images/table_flex.png +0 -0
  47. data/doc/images/table_select.png +0 -0
  48. data/doc/images/widget_style_colors.png +0 -0
  49. data/doc/index.md +1 -0
  50. data/doc/interactive_design.md +121 -0
  51. data/doc/quickstart.md +147 -72
  52. data/examples/all_events/app.rb +169 -0
  53. data/examples/all_events/app.rbs +7 -0
  54. data/examples/all_events/test_app.rb +139 -0
  55. data/examples/analytics/app.rb +258 -0
  56. data/examples/analytics/app.rbs +7 -0
  57. data/examples/analytics/test_app.rb +132 -0
  58. data/examples/block_padding/app.rb +63 -0
  59. data/examples/block_padding/app.rbs +7 -0
  60. data/examples/block_padding/test_app.rb +31 -0
  61. data/examples/block_titles/app.rb +61 -0
  62. data/examples/block_titles/app.rbs +7 -0
  63. data/examples/block_titles/test_app.rb +34 -0
  64. data/examples/box_demo/app.rb +216 -0
  65. data/examples/box_demo/app.rbs +7 -0
  66. data/examples/box_demo/test_app.rb +88 -0
  67. data/examples/calendar_demo/app.rb +101 -0
  68. data/examples/calendar_demo/app.rbs +7 -0
  69. data/examples/calendar_demo/test_app.rb +108 -0
  70. data/examples/cell_demo/app.rb +108 -0
  71. data/examples/cell_demo/app.rbs +7 -0
  72. data/examples/cell_demo/test_app.rb +36 -0
  73. data/examples/chart_demo/app.rb +203 -0
  74. data/examples/chart_demo/app.rbs +7 -0
  75. data/examples/chart_demo/test_app.rb +102 -0
  76. data/examples/custom_widget/app.rb +51 -0
  77. data/examples/custom_widget/app.rbs +7 -0
  78. data/examples/custom_widget/test_app.rb +30 -0
  79. data/examples/flex_layout/app.rb +156 -0
  80. data/examples/flex_layout/app.rbs +7 -0
  81. data/examples/flex_layout/test_app.rb +65 -0
  82. data/examples/gauge_demo/app.rb +182 -0
  83. data/examples/gauge_demo/app.rbs +7 -0
  84. data/examples/gauge_demo/test_app.rb +120 -0
  85. data/examples/hit_test/app.rb +175 -0
  86. data/examples/hit_test/app.rbs +7 -0
  87. data/examples/hit_test/test_app.rb +102 -0
  88. data/examples/line_gauge_demo/app.rb +190 -0
  89. data/examples/line_gauge_demo/app.rbs +7 -0
  90. data/examples/line_gauge_demo/test_app.rb +129 -0
  91. data/examples/list_demo/app.rb +253 -0
  92. data/examples/list_demo/app.rbs +12 -0
  93. data/examples/list_demo/test_app.rb +237 -0
  94. data/examples/list_styles/app.rb +140 -0
  95. data/examples/list_styles/app.rbs +7 -0
  96. data/examples/list_styles/test_app.rb +157 -0
  97. data/examples/{login_form.rb → login_form/app.rb} +12 -16
  98. data/examples/login_form/app.rbs +7 -0
  99. data/examples/login_form/test_app.rb +51 -0
  100. data/examples/map_demo/app.rb +90 -0
  101. data/examples/map_demo/app.rbs +7 -0
  102. data/examples/map_demo/test_app.rb +149 -0
  103. data/examples/{mouse_events.rb → mouse_events/app.rb} +29 -27
  104. data/examples/mouse_events/app.rbs +7 -0
  105. data/examples/mouse_events/test_app.rb +53 -0
  106. data/examples/{popup_demo.rb → popup_demo/app.rb} +15 -17
  107. data/examples/popup_demo/app.rbs +7 -0
  108. data/examples/{test_popup_demo.rb → popup_demo/test_app.rb} +18 -26
  109. data/examples/quickstart_dsl/app.rb +36 -0
  110. data/examples/quickstart_dsl/app.rbs +7 -0
  111. data/examples/quickstart_dsl/test_app.rb +29 -0
  112. data/examples/quickstart_lifecycle/app.rb +39 -0
  113. data/examples/quickstart_lifecycle/app.rbs +7 -0
  114. data/examples/quickstart_lifecycle/test_app.rb +29 -0
  115. data/examples/ratatui_logo_demo/app.rb +79 -0
  116. data/examples/ratatui_logo_demo/app.rbs +7 -0
  117. data/examples/ratatui_logo_demo/test_app.rb +51 -0
  118. data/examples/ratatui_mascot_demo/app.rb +84 -0
  119. data/examples/ratatui_mascot_demo/app.rbs +7 -0
  120. data/examples/ratatui_mascot_demo/test_app.rb +47 -0
  121. data/examples/readme_usage/app.rb +29 -0
  122. data/examples/readme_usage/app.rbs +7 -0
  123. data/examples/readme_usage/test_app.rb +29 -0
  124. data/examples/rich_text/app.rb +141 -0
  125. data/examples/rich_text/app.rbs +7 -0
  126. data/examples/rich_text/test_app.rb +166 -0
  127. data/examples/scroll_text/app.rb +103 -0
  128. data/examples/scroll_text/app.rbs +7 -0
  129. data/examples/scroll_text/test_app.rb +110 -0
  130. data/examples/scrollbar_demo/app.rb +143 -0
  131. data/examples/scrollbar_demo/app.rbs +7 -0
  132. data/examples/scrollbar_demo/test_app.rb +77 -0
  133. data/examples/sparkline_demo/app.rb +240 -0
  134. data/examples/sparkline_demo/app.rbs +10 -0
  135. data/examples/sparkline_demo/test_app.rb +107 -0
  136. data/examples/table_flex/app.rb +65 -0
  137. data/examples/table_flex/app.rbs +7 -0
  138. data/examples/table_flex/test_app.rb +36 -0
  139. data/examples/table_select/app.rb +198 -0
  140. data/examples/table_select/app.rbs +7 -0
  141. data/examples/table_select/test_app.rb +180 -0
  142. data/examples/widget_style_colors/app.rb +104 -0
  143. data/examples/widget_style_colors/app.rbs +14 -0
  144. data/examples/widget_style_colors/test_app.rb +48 -0
  145. data/ext/ratatui_ruby/Cargo.lock +889 -115
  146. data/ext/ratatui_ruby/Cargo.toml +4 -3
  147. data/ext/ratatui_ruby/clippy.toml +7 -0
  148. data/ext/ratatui_ruby/extconf.rb +7 -0
  149. data/ext/ratatui_ruby/src/events.rs +218 -229
  150. data/ext/ratatui_ruby/src/lib.rs +38 -10
  151. data/ext/ratatui_ruby/src/rendering.rs +90 -10
  152. data/ext/ratatui_ruby/src/style.rs +281 -98
  153. data/ext/ratatui_ruby/src/terminal.rs +119 -25
  154. data/ext/ratatui_ruby/src/text.rs +171 -0
  155. data/ext/ratatui_ruby/src/widgets/barchart.rs +97 -24
  156. data/ext/ratatui_ruby/src/widgets/block.rs +31 -3
  157. data/ext/ratatui_ruby/src/widgets/calendar.rs +45 -44
  158. data/ext/ratatui_ruby/src/widgets/canvas.rs +46 -29
  159. data/ext/ratatui_ruby/src/widgets/chart.rs +69 -27
  160. data/ext/ratatui_ruby/src/widgets/clear.rs +3 -1
  161. data/ext/ratatui_ruby/src/widgets/gauge.rs +11 -4
  162. data/ext/ratatui_ruby/src/widgets/layout.rs +218 -15
  163. data/ext/ratatui_ruby/src/widgets/line_gauge.rs +92 -0
  164. data/ext/ratatui_ruby/src/widgets/list.rs +91 -11
  165. data/ext/ratatui_ruby/src/widgets/mod.rs +3 -0
  166. data/ext/ratatui_ruby/src/widgets/overlay.rs +3 -2
  167. data/ext/ratatui_ruby/src/widgets/paragraph.rs +35 -13
  168. data/ext/ratatui_ruby/src/widgets/ratatui_logo.rs +29 -0
  169. data/ext/ratatui_ruby/src/widgets/ratatui_mascot.rs +44 -0
  170. data/ext/ratatui_ruby/src/widgets/scrollbar.rs +59 -7
  171. data/ext/ratatui_ruby/src/widgets/sparkline.rs +70 -6
  172. data/ext/ratatui_ruby/src/widgets/table.rs +173 -64
  173. data/ext/ratatui_ruby/src/widgets/tabs.rs +105 -5
  174. data/lib/ratatui_ruby/cell.rb +166 -0
  175. data/lib/ratatui_ruby/event/focus_gained.rb +49 -0
  176. data/lib/ratatui_ruby/event/focus_lost.rb +50 -0
  177. data/lib/ratatui_ruby/event/key.rb +211 -0
  178. data/lib/ratatui_ruby/event/mouse.rb +124 -0
  179. data/lib/ratatui_ruby/event/paste.rb +71 -0
  180. data/lib/ratatui_ruby/event/resize.rb +80 -0
  181. data/lib/ratatui_ruby/event.rb +79 -0
  182. data/lib/ratatui_ruby/schema/bar_chart/bar.rb +45 -0
  183. data/lib/ratatui_ruby/schema/bar_chart/bar_group.rb +27 -0
  184. data/lib/ratatui_ruby/schema/bar_chart.rb +228 -19
  185. data/lib/ratatui_ruby/schema/block.rb +186 -14
  186. data/lib/ratatui_ruby/schema/calendar.rb +74 -17
  187. data/lib/ratatui_ruby/schema/canvas.rb +215 -48
  188. data/lib/ratatui_ruby/schema/center.rb +49 -11
  189. data/lib/ratatui_ruby/schema/chart.rb +151 -41
  190. data/lib/ratatui_ruby/schema/clear.rb +41 -72
  191. data/lib/ratatui_ruby/schema/constraint.rb +82 -22
  192. data/lib/ratatui_ruby/schema/cursor.rb +27 -9
  193. data/lib/ratatui_ruby/schema/draw.rb +53 -0
  194. data/lib/ratatui_ruby/schema/gauge.rb +59 -15
  195. data/lib/ratatui_ruby/schema/layout.rb +95 -13
  196. data/lib/ratatui_ruby/schema/line_gauge.rb +78 -0
  197. data/lib/ratatui_ruby/schema/list.rb +93 -19
  198. data/lib/ratatui_ruby/schema/overlay.rb +34 -8
  199. data/lib/ratatui_ruby/schema/paragraph.rb +87 -30
  200. data/lib/ratatui_ruby/schema/ratatui_logo.rb +25 -0
  201. data/lib/ratatui_ruby/schema/ratatui_mascot.rb +29 -0
  202. data/lib/ratatui_ruby/schema/rect.rb +64 -15
  203. data/lib/ratatui_ruby/schema/scrollbar.rb +132 -24
  204. data/lib/ratatui_ruby/schema/shape/label.rb +66 -0
  205. data/lib/ratatui_ruby/schema/sparkline.rb +122 -15
  206. data/lib/ratatui_ruby/schema/style.rb +49 -21
  207. data/lib/ratatui_ruby/schema/table.rb +119 -21
  208. data/lib/ratatui_ruby/schema/tabs.rb +75 -13
  209. data/lib/ratatui_ruby/schema/text.rb +90 -0
  210. data/lib/ratatui_ruby/session.rb +146 -0
  211. data/lib/ratatui_ruby/test_helper.rb +156 -13
  212. data/lib/ratatui_ruby/version.rb +1 -1
  213. data/lib/ratatui_ruby.rb +143 -23
  214. data/sig/ratatui_ruby/event.rbs +69 -0
  215. data/sig/ratatui_ruby/ratatui_ruby.rbs +2 -1
  216. data/sig/ratatui_ruby/schema/bar_chart/bar.rbs +16 -0
  217. data/sig/ratatui_ruby/schema/bar_chart/bar_group.rbs +13 -0
  218. data/sig/ratatui_ruby/schema/bar_chart.rbs +20 -2
  219. data/sig/ratatui_ruby/schema/block.rbs +5 -4
  220. data/sig/ratatui_ruby/schema/calendar.rbs +6 -2
  221. data/sig/ratatui_ruby/schema/canvas.rbs +52 -39
  222. data/sig/ratatui_ruby/schema/center.rbs +3 -3
  223. data/sig/ratatui_ruby/schema/chart.rbs +8 -5
  224. data/sig/ratatui_ruby/schema/constraint.rbs +8 -5
  225. data/sig/ratatui_ruby/schema/cursor.rbs +1 -1
  226. data/sig/ratatui_ruby/schema/draw.rbs +23 -0
  227. data/sig/ratatui_ruby/schema/gauge.rbs +4 -2
  228. data/sig/ratatui_ruby/schema/layout.rbs +11 -1
  229. data/sig/ratatui_ruby/schema/line_gauge.rbs +16 -0
  230. data/sig/ratatui_ruby/schema/list.rbs +5 -1
  231. data/sig/ratatui_ruby/schema/paragraph.rbs +4 -1
  232. data/{lib/ratatui_ruby/output.rb → sig/ratatui_ruby/schema/ratatui_logo.rbs} +3 -2
  233. data/sig/ratatui_ruby/{buffer.rbs → schema/ratatui_mascot.rbs} +4 -3
  234. data/sig/ratatui_ruby/schema/rect.rbs +2 -1
  235. data/sig/ratatui_ruby/schema/scrollbar.rbs +18 -2
  236. data/sig/ratatui_ruby/schema/sparkline.rbs +6 -2
  237. data/sig/ratatui_ruby/schema/table.rbs +8 -1
  238. data/sig/ratatui_ruby/schema/tabs.rbs +5 -1
  239. data/sig/ratatui_ruby/schema/text.rbs +22 -0
  240. data/tasks/resources/build.yml.erb +13 -11
  241. data/tasks/terminal_preview/app_screenshot.rb +35 -0
  242. data/tasks/terminal_preview/crash_report.rb +54 -0
  243. data/tasks/terminal_preview/example_app.rb +25 -0
  244. data/tasks/terminal_preview/launcher_script.rb +48 -0
  245. data/tasks/terminal_preview/preview_collection.rb +60 -0
  246. data/tasks/terminal_preview/preview_timing.rb +22 -0
  247. data/tasks/terminal_preview/safety_confirmation.rb +58 -0
  248. data/tasks/terminal_preview/saved_screenshot.rb +55 -0
  249. data/tasks/terminal_preview/system_appearance.rb +11 -0
  250. data/tasks/terminal_preview/terminal_window.rb +138 -0
  251. data/tasks/terminal_preview/window_id.rb +14 -0
  252. data/tasks/terminal_preview.rake +28 -0
  253. data/tasks/test.rake +1 -1
  254. metadata +174 -53
  255. data/doc/images/examples-analytics.rb.png +0 -0
  256. data/doc/images/examples-box_demo.rb.png +0 -0
  257. data/doc/images/examples-calendar_demo.rb.png +0 -0
  258. data/doc/images/examples-chart_demo.rb.png +0 -0
  259. data/doc/images/examples-custom_widget.rb.png +0 -0
  260. data/doc/images/examples-dashboard.rb.png +0 -0
  261. data/doc/images/examples-list_styles.rb.png +0 -0
  262. data/doc/images/examples-login_form.rb.png +0 -0
  263. data/doc/images/examples-map_demo.rb.png +0 -0
  264. data/doc/images/examples-mouse_events.rb.png +0 -0
  265. data/doc/images/examples-popup_demo.rb.gif +0 -0
  266. data/doc/images/examples-quickstart_lifecycle.rb.png +0 -0
  267. data/doc/images/examples-scroll_text.rb.png +0 -0
  268. data/doc/images/examples-scrollbar_demo.rb.png +0 -0
  269. data/doc/images/examples-stock_ticker.rb.png +0 -0
  270. data/doc/images/examples-system_monitor.rb.png +0 -0
  271. data/doc/images/examples-table_select.rb.png +0 -0
  272. data/examples/analytics.rb +0 -88
  273. data/examples/box_demo.rb +0 -71
  274. data/examples/calendar_demo.rb +0 -55
  275. data/examples/chart_demo.rb +0 -84
  276. data/examples/custom_widget.rb +0 -43
  277. data/examples/dashboard.rb +0 -72
  278. data/examples/list_styles.rb +0 -66
  279. data/examples/map_demo.rb +0 -58
  280. data/examples/quickstart_dsl.rb +0 -30
  281. data/examples/quickstart_lifecycle.rb +0 -40
  282. data/examples/readme_usage.rb +0 -21
  283. data/examples/scroll_text.rb +0 -74
  284. data/examples/scrollbar_demo.rb +0 -75
  285. data/examples/stock_ticker.rb +0 -93
  286. data/examples/system_monitor.rb +0 -94
  287. data/examples/table_select.rb +0 -70
  288. data/examples/test_analytics.rb +0 -65
  289. data/examples/test_box_demo.rb +0 -38
  290. data/examples/test_calendar_demo.rb +0 -66
  291. data/examples/test_dashboard.rb +0 -38
  292. data/examples/test_list_styles.rb +0 -61
  293. data/examples/test_login_form.rb +0 -63
  294. data/examples/test_map_demo.rb +0 -100
  295. data/examples/test_scroll_text.rb +0 -130
  296. data/examples/test_stock_ticker.rb +0 -39
  297. data/examples/test_system_monitor.rb +0 -40
  298. data/examples/test_table_select.rb +0 -37
  299. data/ext/ratatui_ruby/src/buffer.rs +0 -54
  300. data/lib/ratatui_ruby/dsl.rb +0 -64
@@ -3,15 +3,16 @@
3
3
 
4
4
  [package]
5
5
  name = "ratatui_ruby"
6
- version = "0.3.1"
6
+ version = "0.4.0"
7
7
  edition = "2021"
8
8
 
9
9
  [lib]
10
- crate-type = ["cdylib"]
10
+ crate-type = ["cdylib", "staticlib"]
11
11
 
12
12
  [dependencies]
13
13
  magnus = "0.8.2"
14
- ratatui = { version = "0.29", features = ["widget-calendar"] }
14
+ ratatui = { version = "0.30", features = ["widget-calendar", "layout-cache", "unstable-rendered-line-info"] }
15
15
 
16
+ bumpalo = "3.16"
16
17
  lazy_static = "1.4"
17
18
  time = { version = "0.3", features = ["macros"] }
@@ -0,0 +1,7 @@
1
+ # SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
+ # SPDX-License-Identifier: AGPL-3.0-or-later
3
+
4
+ disallowed-methods = [
5
+ { path = "std::boxed::Box::leak", reason = "Use bumpalo arena allocation instead." },
6
+ { path = "std::vec::Vec::leak", reason = "Use bumpalo arena allocation instead." },
7
+ ]
@@ -9,4 +9,11 @@ require "rb_sys/mkmf"
9
9
  create_rust_makefile("ratatui_ruby/ratatui_ruby") do |r|
10
10
  # Optional: Force release profile if needed
11
11
  # r.profile = ENV.fetch("RB_SYS_CARGO_PROFILE", :release).to_sym
12
+
13
+ # Force static linking on musl to avoid "cdylib" issues
14
+ if RbConfig::CONFIG["target_os"].include?("linux-musl") || RbConfig::CONFIG["host_os"].include?("linux-musl")
15
+ r.extra_rustc_args = ["--crate-type", "staticlib"]
16
+ else
17
+ r.extra_rustc_args = ["--crate-type", "cdylib"]
18
+ end
12
19
  end
@@ -4,136 +4,152 @@
4
4
  use magnus::{Error, IntoValue, TryConvert, Value};
5
5
  use std::sync::Mutex;
6
6
 
7
- lazy_static::lazy_static! {
8
- static ref EVENT_QUEUE: Mutex<Vec<ratatui::crossterm::event::Event>> = Mutex::new(Vec::new());
9
- }
7
+ static EVENT_QUEUE: Mutex<Vec<ratatui::crossterm::event::Event>> = Mutex::new(Vec::new());
10
8
 
9
+ #[allow(clippy::needless_pass_by_value)]
11
10
  pub fn inject_test_event(event_type: String, data: magnus::RHash) -> Result<(), Error> {
12
11
  let ruby = magnus::Ruby::get().unwrap();
13
12
  let event = match event_type.as_str() {
14
- "key" => {
15
- let code_val: Value = data.get(ruby.to_symbol("code")).ok_or_else(|| {
16
- Error::new(
17
- ruby.exception_arg_error(),
18
- "Missing 'code' in key event",
19
- )
20
- })?;
21
- let code_str: String = String::try_convert(code_val)?;
22
- let code = match code_str.as_str() {
23
- "up" => ratatui::crossterm::event::KeyCode::Up,
24
- "down" => ratatui::crossterm::event::KeyCode::Down,
25
- "left" => ratatui::crossterm::event::KeyCode::Left,
26
- "right" => ratatui::crossterm::event::KeyCode::Right,
27
- "enter" => ratatui::crossterm::event::KeyCode::Enter,
28
- "esc" => ratatui::crossterm::event::KeyCode::Esc,
29
- "backspace" => ratatui::crossterm::event::KeyCode::Backspace,
30
- "tab" => ratatui::crossterm::event::KeyCode::Tab,
31
- c if c.len() == 1 => ratatui::crossterm::event::KeyCode::Char(c.chars().next().unwrap()),
32
- _ => ratatui::crossterm::event::KeyCode::Null,
33
- };
13
+ "key" => parse_key_event(data, &ruby)?,
14
+ "mouse" => parse_mouse_event(data, &ruby)?,
15
+ "resize" => parse_resize_event(data, &ruby)?,
16
+ "paste" => parse_paste_event(data, &ruby)?,
17
+ "focus_gained" => ratatui::crossterm::event::Event::FocusGained,
18
+ "focus_lost" => ratatui::crossterm::event::Event::FocusLost,
19
+ _ => {
20
+ return Err(Error::new(
21
+ ruby.exception_arg_error(),
22
+ format!("Unknown event type: {event_type}"),
23
+ ))
24
+ }
25
+ };
34
26
 
35
- let mut modifiers = ratatui::crossterm::event::KeyModifiers::empty();
36
- if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
37
- let mods: Vec<String> = Vec::try_convert(mods_val)?;
38
- for m in mods {
39
- match m.as_str() {
40
- "ctrl" => modifiers |= ratatui::crossterm::event::KeyModifiers::CONTROL,
41
- "alt" => modifiers |= ratatui::crossterm::event::KeyModifiers::ALT,
42
- "shift" => modifiers |= ratatui::crossterm::event::KeyModifiers::SHIFT,
43
- _ => {}
44
- }
45
- }
46
- }
27
+ EVENT_QUEUE.lock().unwrap().push(event);
28
+ Ok(())
29
+ }
30
+
31
+ fn parse_key_event(data: magnus::RHash, ruby: &magnus::Ruby) -> Result<ratatui::crossterm::event::Event, Error> {
32
+ let code_val: Value = data.get(ruby.to_symbol("code")).ok_or_else(|| {
33
+ Error::new(ruby.exception_arg_error(), "Missing 'code' in key event")
34
+ })?;
35
+ let code_str: String = String::try_convert(code_val)?;
36
+ let code = match code_str.as_str() {
37
+ "up" => ratatui::crossterm::event::KeyCode::Up,
38
+ "down" => ratatui::crossterm::event::KeyCode::Down,
39
+ "left" => ratatui::crossterm::event::KeyCode::Left,
40
+ "right" => ratatui::crossterm::event::KeyCode::Right,
41
+ "enter" => ratatui::crossterm::event::KeyCode::Enter,
42
+ "esc" => ratatui::crossterm::event::KeyCode::Esc,
43
+ "backspace" => ratatui::crossterm::event::KeyCode::Backspace,
44
+ "tab" => ratatui::crossterm::event::KeyCode::Tab,
45
+ c if c.len() == 1 => ratatui::crossterm::event::KeyCode::Char(c.chars().next().unwrap()),
46
+ _ => ratatui::crossterm::event::KeyCode::Null,
47
+ };
47
48
 
48
- ratatui::crossterm::event::Event::Key(ratatui::crossterm::event::KeyEvent::new(code, modifiers))
49
+ let mut modifiers = ratatui::crossterm::event::KeyModifiers::empty();
50
+ if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
51
+ let mods: Vec<String> = Vec::try_convert(mods_val)?;
52
+ for m in mods {
53
+ match m.as_str() {
54
+ "ctrl" => modifiers |= ratatui::crossterm::event::KeyModifiers::CONTROL,
55
+ "alt" => modifiers |= ratatui::crossterm::event::KeyModifiers::ALT,
56
+ "shift" => modifiers |= ratatui::crossterm::event::KeyModifiers::SHIFT,
57
+ _ => {}
58
+ }
49
59
  }
50
- "mouse" => {
51
- let kind_val: Value = data.get(ruby.to_symbol("kind")).ok_or_else(|| {
52
- Error::new(
53
- ruby.exception_arg_error(),
54
- "Missing 'kind' in mouse event",
55
- )
56
- })?;
57
- let kind_str: String = String::try_convert(kind_val)?;
60
+ }
58
61
 
59
- let button = if let Some(btn_val) = data.get(ruby.to_symbol("button")) {
60
- let button_str: String = String::try_convert(btn_val)?;
61
- match button_str.as_str() {
62
- "right" => ratatui::crossterm::event::MouseButton::Right,
63
- "middle" => ratatui::crossterm::event::MouseButton::Middle,
64
- _ => ratatui::crossterm::event::MouseButton::Left,
65
- }
66
- } else {
67
- ratatui::crossterm::event::MouseButton::Left
68
- };
62
+ Ok(ratatui::crossterm::event::Event::Key(ratatui::crossterm::event::KeyEvent::new(code, modifiers)))
63
+ }
69
64
 
70
- let x_val: Value = data.get(ruby.to_symbol("x")).ok_or_else(|| {
71
- Error::new(ruby.exception_arg_error(), "Missing 'x' in mouse event")
72
- })?;
73
- let x: u16 = u16::try_convert(x_val)?;
65
+ fn parse_mouse_event(data: magnus::RHash, ruby: &magnus::Ruby) -> Result<ratatui::crossterm::event::Event, Error> {
66
+ let kind_val: Value = data.get(ruby.to_symbol("kind")).ok_or_else(|| {
67
+ Error::new(ruby.exception_arg_error(), "Missing 'kind' in mouse event")
68
+ })?;
69
+ let kind_str: String = String::try_convert(kind_val)?;
74
70
 
75
- let y_val: Value = data.get(ruby.to_symbol("y")).ok_or_else(|| {
76
- Error::new(ruby.exception_arg_error(), "Missing 'y' in mouse event")
77
- })?;
78
- let y: u16 = u16::try_convert(y_val)?;
71
+ let button = if let Some(btn_val) = data.get(ruby.to_symbol("button")) {
72
+ let button_str: String = String::try_convert(btn_val)?;
73
+ match button_str.as_str() {
74
+ "right" => ratatui::crossterm::event::MouseButton::Right,
75
+ "middle" => ratatui::crossterm::event::MouseButton::Middle,
76
+ _ => ratatui::crossterm::event::MouseButton::Left,
77
+ }
78
+ } else {
79
+ ratatui::crossterm::event::MouseButton::Left
80
+ };
79
81
 
80
- let kind = match kind_str.as_str() {
81
- "down" => ratatui::crossterm::event::MouseEventKind::Down(button),
82
- "up" => ratatui::crossterm::event::MouseEventKind::Up(button),
83
- "drag" => ratatui::crossterm::event::MouseEventKind::Drag(button),
84
- "moved" => ratatui::crossterm::event::MouseEventKind::Moved,
85
- "scroll_down" => ratatui::crossterm::event::MouseEventKind::ScrollDown,
86
- "scroll_up" => ratatui::crossterm::event::MouseEventKind::ScrollUp,
87
- "scroll_left" => ratatui::crossterm::event::MouseEventKind::ScrollLeft,
88
- "scroll_right" => ratatui::crossterm::event::MouseEventKind::ScrollRight,
89
- _ => {
90
- return Err(Error::new(
91
- ruby.exception_arg_error(),
92
- format!("Unknown mouse kind: {}", kind_str),
93
- ))
94
- }
95
- };
82
+ let x_val: Value = data.get(ruby.to_symbol("x")).ok_or_else(|| {
83
+ Error::new(ruby.exception_arg_error(), "Missing 'x' in mouse event")
84
+ })?;
85
+ let x: u16 = u16::try_convert(x_val)?;
96
86
 
97
- let mut modifiers = ratatui::crossterm::event::KeyModifiers::empty();
98
- if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
99
- let mods: Vec<String> = Vec::try_convert(mods_val)?;
100
- for m in mods {
101
- match m.as_str() {
102
- "ctrl" => modifiers |= ratatui::crossterm::event::KeyModifiers::CONTROL,
103
- "alt" => modifiers |= ratatui::crossterm::event::KeyModifiers::ALT,
104
- "shift" => modifiers |= ratatui::crossterm::event::KeyModifiers::SHIFT,
105
- _ => {}
106
- }
107
- }
108
- }
87
+ let y_val: Value = data.get(ruby.to_symbol("y")).ok_or_else(|| {
88
+ Error::new(ruby.exception_arg_error(), "Missing 'y' in mouse event")
89
+ })?;
90
+ let y: u16 = u16::try_convert(y_val)?;
109
91
 
110
- ratatui::crossterm::event::Event::Mouse(ratatui::crossterm::event::MouseEvent {
111
- kind,
112
- column: x,
113
- row: y,
114
- modifiers,
115
- })
116
- }
117
- _ => {
118
- return Err(Error::new(
119
- ruby.exception_arg_error(),
120
- format!("Unknown event type: {}", event_type),
121
- ))
122
- }
92
+ let kind = match kind_str.as_str() {
93
+ "down" => ratatui::crossterm::event::MouseEventKind::Down(button),
94
+ "up" => ratatui::crossterm::event::MouseEventKind::Up(button),
95
+ "drag" => ratatui::crossterm::event::MouseEventKind::Drag(button),
96
+ "moved" => ratatui::crossterm::event::MouseEventKind::Moved,
97
+ "scroll_down" => ratatui::crossterm::event::MouseEventKind::ScrollDown,
98
+ "scroll_up" => ratatui::crossterm::event::MouseEventKind::ScrollUp,
99
+ "scroll_left" => ratatui::crossterm::event::MouseEventKind::ScrollLeft,
100
+ "scroll_right" => ratatui::crossterm::event::MouseEventKind::ScrollRight,
101
+ _ => return Err(Error::new(ruby.exception_arg_error(), format!("Unknown mouse kind: {kind_str}"))),
123
102
  };
124
103
 
125
- EVENT_QUEUE.lock().unwrap().push(event);
126
- Ok(())
104
+ let mut modifiers = ratatui::crossterm::event::KeyModifiers::empty();
105
+ if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
106
+ let mods: Vec<String> = Vec::try_convert(mods_val)?;
107
+ for m in mods {
108
+ match m.as_str() {
109
+ "ctrl" => modifiers |= ratatui::crossterm::event::KeyModifiers::CONTROL,
110
+ "alt" => modifiers |= ratatui::crossterm::event::KeyModifiers::ALT,
111
+ "shift" => modifiers |= ratatui::crossterm::event::KeyModifiers::SHIFT,
112
+ _ => {}
113
+ }
114
+ }
115
+ }
116
+
117
+ Ok(ratatui::crossterm::event::Event::Mouse(ratatui::crossterm::event::MouseEvent { kind, column: x, row: y, modifiers }))
127
118
  }
128
119
 
129
- pub fn poll_event() -> Result<Value, Error> {
130
- let ruby = magnus::Ruby::get().unwrap();
120
+ fn parse_resize_event(data: magnus::RHash, ruby: &magnus::Ruby) -> Result<ratatui::crossterm::event::Event, Error> {
121
+ let width_val: Value = data.get(ruby.to_symbol("width")).ok_or_else(|| {
122
+ Error::new(ruby.exception_arg_error(), "Missing 'width' in resize event")
123
+ })?;
124
+ let width: u16 = u16::try_convert(width_val)?;
125
+
126
+ let height_val: Value = data.get(ruby.to_symbol("height")).ok_or_else(|| {
127
+ Error::new(ruby.exception_arg_error(), "Missing 'height' in resize event")
128
+ })?;
129
+ let height: u16 = u16::try_convert(height_val)?;
130
+
131
+ Ok(ratatui::crossterm::event::Event::Resize(width, height))
132
+ }
133
+
134
+ fn parse_paste_event(data: magnus::RHash, ruby: &magnus::Ruby) -> Result<ratatui::crossterm::event::Event, Error> {
135
+ let content_val: Value = data.get(ruby.to_symbol("content")).ok_or_else(|| {
136
+ Error::new(ruby.exception_arg_error(), "Missing 'content' in paste event")
137
+ })?;
138
+ let content: String = String::try_convert(content_val)?;
139
+ Ok(ratatui::crossterm::event::Event::Paste(content))
140
+ }
141
+
142
+ pub fn clear_events() {
143
+ EVENT_QUEUE.lock().unwrap().clear();
144
+ }
145
+
146
+ pub fn poll_event(ruby: &magnus::Ruby) -> Result<Value, Error> {
131
147
  let event = {
132
148
  let mut queue = EVENT_QUEUE.lock().unwrap();
133
- if !queue.is_empty() {
134
- Some(queue.remove(0))
135
- } else {
149
+ if queue.is_empty() {
136
150
  None
151
+ } else {
152
+ Some(queue.remove(0))
137
153
  }
138
154
  };
139
155
 
@@ -141,17 +157,13 @@ pub fn poll_event() -> Result<Value, Error> {
141
157
  return handle_event(e);
142
158
  }
143
159
 
144
- // Check if we are in test mode. If so, don't poll crossterm.
145
160
  let is_test_mode = {
146
161
  let term_lock = crate::terminal::TERMINAL.lock().unwrap();
147
- matches!(
148
- term_lock.as_ref(),
149
- Some(crate::terminal::TerminalWrapper::Test(_))
150
- )
162
+ matches!(term_lock.as_ref(), Some(crate::terminal::TerminalWrapper::Test(_)))
151
163
  };
152
164
 
153
165
  if is_test_mode {
154
- return Ok(ruby.qnil().into_value_with(&magnus::Ruby::get().unwrap()));
166
+ return Ok(ruby.qnil().into_value_with(ruby));
155
167
  }
156
168
 
157
169
  if ratatui::crossterm::event::poll(std::time::Duration::from_millis(16))
@@ -161,127 +173,104 @@ pub fn poll_event() -> Result<Value, Error> {
161
173
  .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
162
174
  handle_event(event)
163
175
  } else {
164
- Ok(ruby.qnil().into_value_with(&magnus::Ruby::get().unwrap()))
176
+ Ok(ruby.qnil().into_value_with(ruby))
165
177
  }
166
178
  }
167
179
 
168
180
  fn handle_event(event: ratatui::crossterm::event::Event) -> Result<Value, Error> {
169
- let ruby = magnus::Ruby::get().unwrap();
170
181
  match event {
171
- ratatui::crossterm::event::Event::Key(key) => {
172
- if key.kind == ratatui::crossterm::event::KeyEventKind::Press {
173
- let ruby = magnus::Ruby::get().unwrap();
174
- let hash = ruby.hash_new();
175
- hash.aset(ruby.to_symbol("type"), ruby.to_symbol("key"))?;
176
-
177
- let code = match key.code {
178
- ratatui::crossterm::event::KeyCode::Char(c) => c.to_string(),
179
- ratatui::crossterm::event::KeyCode::Up => "up".to_string(),
180
- ratatui::crossterm::event::KeyCode::Down => "down".to_string(),
181
- ratatui::crossterm::event::KeyCode::Left => "left".to_string(),
182
- ratatui::crossterm::event::KeyCode::Right => "right".to_string(),
183
- ratatui::crossterm::event::KeyCode::Enter => "enter".to_string(),
184
- ratatui::crossterm::event::KeyCode::Esc => "esc".to_string(),
185
- ratatui::crossterm::event::KeyCode::Backspace => "backspace".to_string(),
186
- ratatui::crossterm::event::KeyCode::Tab => "tab".to_string(),
187
- _ => "unknown".to_string(),
188
- };
189
- hash.aset(ruby.to_symbol("code"), code)?;
190
-
191
- let mut modifiers = Vec::new();
192
- if key
193
- .modifiers
194
- .contains(ratatui::crossterm::event::KeyModifiers::CONTROL)
195
- {
196
- modifiers.push("ctrl");
197
- }
198
- if key.modifiers.contains(ratatui::crossterm::event::KeyModifiers::ALT) {
199
- modifiers.push("alt");
200
- }
201
- if key
202
- .modifiers
203
- .contains(ratatui::crossterm::event::KeyModifiers::SHIFT)
204
- {
205
- modifiers.push("shift");
206
- }
207
- if !modifiers.is_empty() {
208
- hash.aset(ruby.to_symbol("modifiers"), modifiers)?;
209
- }
210
-
211
- return Ok(hash.into_value_with(&ruby));
212
- }
213
- }
214
- ratatui::crossterm::event::Event::Mouse(event) => {
215
- let ruby = magnus::Ruby::get().unwrap();
216
- let hash = ruby.hash_new();
217
- hash.aset(ruby.to_symbol("type"), ruby.to_symbol("mouse"))?;
218
-
219
- let (kind, button) = match event.kind {
220
- ratatui::crossterm::event::MouseEventKind::Down(btn) => ("down", btn),
221
- ratatui::crossterm::event::MouseEventKind::Up(btn) => ("up", btn),
222
- ratatui::crossterm::event::MouseEventKind::Drag(btn) => ("drag", btn),
223
- ratatui::crossterm::event::MouseEventKind::Moved => {
224
- ("moved", ratatui::crossterm::event::MouseButton::Left)
225
- } // button is ignored for moved
226
- ratatui::crossterm::event::MouseEventKind::ScrollDown => {
227
- ("scroll_down", ratatui::crossterm::event::MouseButton::Left)
228
- } // button is ignored for scroll
229
- ratatui::crossterm::event::MouseEventKind::ScrollUp => {
230
- ("scroll_up", ratatui::crossterm::event::MouseButton::Left)
231
- } // button is ignored for scroll
232
- ratatui::crossterm::event::MouseEventKind::ScrollLeft => {
233
- ("scroll_left", ratatui::crossterm::event::MouseButton::Left)
234
- } // button is ignored for scroll
235
- ratatui::crossterm::event::MouseEventKind::ScrollRight => {
236
- ("scroll_right", ratatui::crossterm::event::MouseButton::Left)
237
- } // button is ignored for scroll
238
- };
182
+ ratatui::crossterm::event::Event::Key(key) => handle_key_event(key),
183
+ ratatui::crossterm::event::Event::Mouse(event) => handle_mouse_event(event),
184
+ ratatui::crossterm::event::Event::Resize(w, h) => handle_resize_event(w, h),
185
+ ratatui::crossterm::event::Event::Paste(content) => handle_paste_event(content),
186
+ ratatui::crossterm::event::Event::FocusGained => handle_focus_event("focus_gained"),
187
+ ratatui::crossterm::event::Event::FocusLost => handle_focus_event("focus_lost"),
188
+ }
189
+ }
239
190
 
240
- hash.aset(ruby.to_symbol("kind"), ruby.to_symbol(kind))?;
191
+ fn handle_key_event(key: ratatui::crossterm::event::KeyEvent) -> Result<Value, Error> {
192
+ let ruby = magnus::Ruby::get().unwrap();
193
+ if key.kind != ratatui::crossterm::event::KeyEventKind::Press {
194
+ return Ok(ruby.qnil().into_value_with(&ruby));
195
+ }
196
+ let hash = ruby.hash_new();
197
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("key"))?;
198
+ let code = match key.code {
199
+ ratatui::crossterm::event::KeyCode::Char(c) => c.to_string(),
200
+ ratatui::crossterm::event::KeyCode::Up => "up".to_string(),
201
+ ratatui::crossterm::event::KeyCode::Down => "down".to_string(),
202
+ ratatui::crossterm::event::KeyCode::Left => "left".to_string(),
203
+ ratatui::crossterm::event::KeyCode::Right => "right".to_string(),
204
+ ratatui::crossterm::event::KeyCode::Enter => "enter".to_string(),
205
+ ratatui::crossterm::event::KeyCode::Esc => "esc".to_string(),
206
+ ratatui::crossterm::event::KeyCode::Backspace => "backspace".to_string(),
207
+ ratatui::crossterm::event::KeyCode::Tab => "tab".to_string(),
208
+ _ => "unknown".to_string(),
209
+ };
210
+ hash.aset(ruby.to_symbol("code"), code)?;
211
+ let mut modifiers = Vec::new();
212
+ if key.modifiers.contains(ratatui::crossterm::event::KeyModifiers::CONTROL) { modifiers.push("ctrl"); }
213
+ if key.modifiers.contains(ratatui::crossterm::event::KeyModifiers::ALT) { modifiers.push("alt"); }
214
+ if key.modifiers.contains(ratatui::crossterm::event::KeyModifiers::SHIFT) { modifiers.push("shift"); }
215
+ if !modifiers.is_empty() { hash.aset(ruby.to_symbol("modifiers"), modifiers)?; }
216
+ Ok(hash.into_value_with(&ruby))
217
+ }
241
218
 
242
- if matches!(
243
- event.kind,
244
- ratatui::crossterm::event::MouseEventKind::Down(_)
245
- | ratatui::crossterm::event::MouseEventKind::Up(_)
246
- | ratatui::crossterm::event::MouseEventKind::Drag(_)
247
- ) {
248
- let btn_sym = match button {
249
- ratatui::crossterm::event::MouseButton::Left => "left",
250
- ratatui::crossterm::event::MouseButton::Right => "right",
251
- ratatui::crossterm::event::MouseButton::Middle => "middle",
252
- };
253
- hash.aset(ruby.to_symbol("button"), ruby.to_symbol(btn_sym))?;
254
- } else {
255
- hash.aset(ruby.to_symbol("button"), ruby.to_symbol("none"))?;
256
- }
219
+ fn handle_mouse_event(event: ratatui::crossterm::event::MouseEvent) -> Result<Value, Error> {
220
+ let ruby = magnus::Ruby::get().unwrap();
221
+ let hash = ruby.hash_new();
222
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("mouse"))?;
223
+ let (kind, button) = match event.kind {
224
+ ratatui::crossterm::event::MouseEventKind::Down(btn) => ("down", btn),
225
+ ratatui::crossterm::event::MouseEventKind::Up(btn) => ("up", btn),
226
+ ratatui::crossterm::event::MouseEventKind::Drag(btn) => ("drag", btn),
227
+ ratatui::crossterm::event::MouseEventKind::Moved => ("moved", ratatui::crossterm::event::MouseButton::Left),
228
+ ratatui::crossterm::event::MouseEventKind::ScrollDown => ("scroll_down", ratatui::crossterm::event::MouseButton::Left),
229
+ ratatui::crossterm::event::MouseEventKind::ScrollUp => ("scroll_up", ratatui::crossterm::event::MouseButton::Left),
230
+ ratatui::crossterm::event::MouseEventKind::ScrollLeft => ("scroll_left", ratatui::crossterm::event::MouseButton::Left),
231
+ ratatui::crossterm::event::MouseEventKind::ScrollRight => ("scroll_right", ratatui::crossterm::event::MouseButton::Left),
232
+ };
233
+ hash.aset(ruby.to_symbol("kind"), ruby.to_symbol(kind))?;
234
+ if matches!(event.kind, ratatui::crossterm::event::MouseEventKind::Down(_) | ratatui::crossterm::event::MouseEventKind::Up(_) | ratatui::crossterm::event::MouseEventKind::Drag(_)) {
235
+ let btn_sym = match button {
236
+ ratatui::crossterm::event::MouseButton::Left => "left",
237
+ ratatui::crossterm::event::MouseButton::Right => "right",
238
+ ratatui::crossterm::event::MouseButton::Middle => "middle",
239
+ };
240
+ hash.aset(ruby.to_symbol("button"), ruby.to_symbol(btn_sym))?;
241
+ } else {
242
+ hash.aset(ruby.to_symbol("button"), ruby.to_symbol("none"))?;
243
+ }
244
+ hash.aset(ruby.to_symbol("x"), event.column)?;
245
+ hash.aset(ruby.to_symbol("y"), event.row)?;
246
+ let mut modifiers = Vec::new();
247
+ if event.modifiers.contains(ratatui::crossterm::event::KeyModifiers::CONTROL) { modifiers.push("ctrl"); }
248
+ if event.modifiers.contains(ratatui::crossterm::event::KeyModifiers::ALT) { modifiers.push("alt"); }
249
+ if event.modifiers.contains(ratatui::crossterm::event::KeyModifiers::SHIFT) { modifiers.push("shift"); }
250
+ hash.aset(ruby.to_symbol("modifiers"), modifiers)?;
251
+ Ok(hash.into_value_with(&ruby))
252
+ }
257
253
 
258
- hash.aset(ruby.to_symbol("x"), event.column)?;
259
- hash.aset(ruby.to_symbol("y"), event.row)?;
254
+ fn handle_resize_event(w: u16, h: u16) -> Result<Value, Error> {
255
+ let ruby = magnus::Ruby::get().unwrap();
256
+ let hash = ruby.hash_new();
257
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("resize"))?;
258
+ hash.aset(ruby.to_symbol("width"), w)?;
259
+ hash.aset(ruby.to_symbol("height"), h)?;
260
+ Ok(hash.into_value_with(&ruby))
261
+ }
260
262
 
261
- let mut modifiers = Vec::new();
262
- if event
263
- .modifiers
264
- .contains(ratatui::crossterm::event::KeyModifiers::CONTROL)
265
- {
266
- modifiers.push("ctrl");
267
- }
268
- if event
269
- .modifiers
270
- .contains(ratatui::crossterm::event::KeyModifiers::ALT)
271
- {
272
- modifiers.push("alt");
273
- }
274
- if event
275
- .modifiers
276
- .contains(ratatui::crossterm::event::KeyModifiers::SHIFT)
277
- {
278
- modifiers.push("shift");
279
- }
280
- hash.aset(ruby.to_symbol("modifiers"), modifiers)?;
263
+ fn handle_paste_event(content: String) -> Result<Value, Error> {
264
+ let ruby = magnus::Ruby::get().unwrap();
265
+ let hash = ruby.hash_new();
266
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("paste"))?;
267
+ hash.aset(ruby.to_symbol("content"), content)?;
268
+ Ok(hash.into_value_with(&ruby))
269
+ }
281
270
 
282
- return Ok(hash.into_value_with(&ruby));
283
- }
284
- _ => {}
285
- }
286
- Ok(ruby.qnil().into_value_with(&magnus::Ruby::get().unwrap()))
271
+ fn handle_focus_event(event_type: &str) -> Result<Value, Error> {
272
+ let ruby = magnus::Ruby::get().unwrap();
273
+ let hash = ruby.hash_new();
274
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol(event_type))?;
275
+ Ok(hash.into_value_with(&ruby))
287
276
  }
@@ -1,25 +1,35 @@
1
1
  // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
2
  // SPDX-License-Identifier: AGPL-3.0-or-later
3
3
 
4
- mod buffer;
4
+ // Require SAFETY comments on all unsafe blocks
5
+ #![warn(clippy::undocumented_unsafe_blocks)]
6
+ // Enable pedantic lints for stricter code quality
7
+ #![warn(clippy::pedantic)]
8
+ // Allow certain pedantic lints that are too noisy for FFI code
9
+ #![allow(clippy::missing_errors_doc)]
10
+ #![allow(clippy::missing_panics_doc)]
11
+ #![allow(clippy::module_name_repetitions)]
12
+
5
13
  mod events;
6
14
  mod rendering;
7
15
  mod style;
8
16
  mod terminal;
17
+ mod text;
9
18
  mod widgets;
10
19
 
11
- use magnus::{function, Class, Error, Module, Value};
20
+ use magnus::{function, Error, Module, Object, Value};
12
21
  use terminal::{init_terminal, restore_terminal, TERMINAL};
13
22
 
14
23
  fn draw(tree: Value) -> Result<(), Error> {
15
24
  let ruby = magnus::Ruby::get().unwrap();
16
25
  let mut term_lock = TERMINAL.lock().unwrap();
26
+ let mut render_error = None;
17
27
  if let Some(wrapper) = term_lock.as_mut() {
18
28
  match wrapper {
19
29
  terminal::TerminalWrapper::Crossterm(term) => {
20
30
  term.draw(|f| {
21
31
  if let Err(e) = rendering::render_node(f, f.area(), tree) {
22
- eprintln!("Render error: {:?}", e);
32
+ render_error = Some(e);
23
33
  }
24
34
  })
25
35
  .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
@@ -27,7 +37,7 @@ fn draw(tree: Value) -> Result<(), Error> {
27
37
  terminal::TerminalWrapper::Test(term) => {
28
38
  term.draw(|f| {
29
39
  if let Err(e) = rendering::render_node(f, f.area(), tree) {
30
- eprintln!("Render error: {:?}", e);
40
+ render_error = Some(e);
31
41
  }
32
42
  })
33
43
  .map_err(|e| Error::new(ruby.exception_runtime_error(), e.to_string()))?;
@@ -36,6 +46,11 @@ fn draw(tree: Value) -> Result<(), Error> {
36
46
  } else {
37
47
  eprintln!("Terminal is None!");
38
48
  }
49
+
50
+ if let Some(e) = render_error {
51
+ return Err(e);
52
+ }
53
+
39
54
  Ok(())
40
55
  }
41
56
 
@@ -44,16 +59,14 @@ fn init() -> Result<(), Error> {
44
59
  let ruby = magnus::Ruby::get().unwrap();
45
60
  let m = ruby.define_module("RatatuiRuby")?;
46
61
 
47
- let buffer_class = m.define_class("Buffer", ruby.class_object())?;
48
- buffer_class.undef_default_alloc_func();
49
- buffer_class.define_method("set_string", magnus::method!(buffer::BufferWrapper::set_string, 4))?;
50
- buffer_class.define_method("area", magnus::method!(buffer::BufferWrapper::area, 0))?;
51
62
 
52
- m.define_module_function("init_terminal", function!(init_terminal, 0))?;
63
+
64
+ m.define_module_function("_init_terminal", function!(init_terminal, 2))?;
53
65
  m.define_module_function("restore_terminal", function!(restore_terminal, 0))?;
54
66
  m.define_module_function("draw", function!(draw, 1))?;
55
- m.define_module_function("poll_event", function!(events::poll_event, 0))?;
67
+ m.define_module_function("_poll_event", function!(events::poll_event, 0))?;
56
68
  m.define_module_function("inject_test_event", function!(events::inject_test_event, 2))?;
69
+ m.define_module_function("clear_events", function!(events::clear_events, 0))?;
57
70
 
58
71
  // Test backend helpers
59
72
  m.define_module_function(
@@ -68,8 +81,23 @@ fn init() -> Result<(), Error> {
68
81
  "get_cursor_position",
69
82
  function!(terminal::get_cursor_position, 0),
70
83
  )?;
84
+ m.define_module_function(
85
+ "_get_cell_at",
86
+ function!(terminal::get_cell_at, 2),
87
+ )?;
71
88
  m.define_module_function("resize_terminal", function!(terminal::resize_terminal, 2))?;
72
89
 
90
+ // Register Layout.split on the Layout class
91
+ let layout_class = m.const_get::<_, magnus::RClass>("Layout")?;
92
+ layout_class.define_singleton_method("_split", function!(widgets::layout::split_layout, 4))?;
93
+
94
+ // Paragraph metrics
95
+ m.define_module_function("_paragraph_line_count", function!(widgets::paragraph::line_count, 2))?;
96
+ m.define_module_function("_paragraph_line_width", function!(widgets::paragraph::line_width, 1))?;
97
+
98
+ // Tabs metrics
99
+ m.define_module_function("_tabs_width", function!(widgets::tabs::width, 1))?;
100
+
73
101
  Ok(())
74
102
  }
75
103