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,700 @@
1
+ // SPDX-FileCopyrightText: 2025 Kerrick Long <me@kerricklong.com>
2
+ // SPDX-License-Identifier: AGPL-3.0-or-later
3
+
4
+ use magnus::{Error, IntoValue, TryConvert, Value};
5
+ use rb_sys::rb_thread_call_without_gvl;
6
+ use std::cell::RefCell;
7
+ use std::os::raw::c_void;
8
+
9
+ /// Wrapper enum for test events - includes crossterm events and our Sync event.
10
+ #[derive(Debug, Clone)]
11
+ enum TestEvent {
12
+ Crossterm(ratatui::crossterm::event::Event),
13
+ Sync,
14
+ }
15
+
16
+ thread_local! {
17
+ static EVENT_QUEUE: RefCell<Vec<TestEvent>> = const { RefCell::new(Vec::new()) };
18
+ }
19
+
20
+ use ratatui::crossterm::event::{KeyCode, KeyModifiers, MediaKeyCode, ModifierKeyCode};
21
+
22
+ /// Single source of truth for base key code mappings.
23
+ const BASE_KEY_MAPPINGS: &[(&str, KeyCode)] = &[
24
+ // Arrow keys
25
+ ("up", KeyCode::Up),
26
+ ("down", KeyCode::Down),
27
+ ("left", KeyCode::Left),
28
+ ("right", KeyCode::Right),
29
+ // Common keys
30
+ ("enter", KeyCode::Enter),
31
+ ("esc", KeyCode::Esc),
32
+ ("backspace", KeyCode::Backspace),
33
+ ("tab", KeyCode::Tab),
34
+ ("back_tab", KeyCode::BackTab),
35
+ ("null", KeyCode::Null),
36
+ // Navigation keys
37
+ ("home", KeyCode::Home),
38
+ ("end", KeyCode::End),
39
+ ("page_up", KeyCode::PageUp),
40
+ ("page_down", KeyCode::PageDown),
41
+ ("insert", KeyCode::Insert),
42
+ ("delete", KeyCode::Delete),
43
+ // Lock keys
44
+ ("caps_lock", KeyCode::CapsLock),
45
+ ("scroll_lock", KeyCode::ScrollLock),
46
+ ("num_lock", KeyCode::NumLock),
47
+ // System keys
48
+ ("print_screen", KeyCode::PrintScreen),
49
+ ("pause", KeyCode::Pause),
50
+ ("menu", KeyCode::Menu),
51
+ ("keypad_begin", KeyCode::KeypadBegin),
52
+ ];
53
+
54
+ /// Single source of truth for media key mappings.
55
+ const MEDIA_KEY_MAPPINGS: &[(&str, MediaKeyCode)] = &[
56
+ ("media_play", MediaKeyCode::Play),
57
+ ("media_pause", MediaKeyCode::Pause),
58
+ ("media_play_pause", MediaKeyCode::PlayPause),
59
+ ("media_reverse", MediaKeyCode::Reverse),
60
+ ("media_stop", MediaKeyCode::Stop),
61
+ ("media_fast_forward", MediaKeyCode::FastForward),
62
+ ("media_rewind", MediaKeyCode::Rewind),
63
+ ("media_track_next", MediaKeyCode::TrackNext),
64
+ ("media_track_previous", MediaKeyCode::TrackPrevious),
65
+ ("media_record", MediaKeyCode::Record),
66
+ ("media_lower_volume", MediaKeyCode::LowerVolume),
67
+ ("media_raise_volume", MediaKeyCode::RaiseVolume),
68
+ ("media_mute_volume", MediaKeyCode::MuteVolume),
69
+ ];
70
+
71
+ /// Single source of truth for modifier key mappings.
72
+ const MODIFIER_KEY_MAPPINGS: &[(&str, ModifierKeyCode)] = &[
73
+ ("left_shift", ModifierKeyCode::LeftShift),
74
+ ("left_control", ModifierKeyCode::LeftControl),
75
+ ("left_alt", ModifierKeyCode::LeftAlt),
76
+ ("left_super", ModifierKeyCode::LeftSuper),
77
+ ("left_hyper", ModifierKeyCode::LeftHyper),
78
+ ("left_meta", ModifierKeyCode::LeftMeta),
79
+ ("right_shift", ModifierKeyCode::RightShift),
80
+ ("right_control", ModifierKeyCode::RightControl),
81
+ ("right_alt", ModifierKeyCode::RightAlt),
82
+ ("right_super", ModifierKeyCode::RightSuper),
83
+ ("right_hyper", ModifierKeyCode::RightHyper),
84
+ ("right_meta", ModifierKeyCode::RightMeta),
85
+ ("iso_level3_shift", ModifierKeyCode::IsoLevel3Shift),
86
+ ("iso_level5_shift", ModifierKeyCode::IsoLevel5Shift),
87
+ ];
88
+
89
+ /// Single source of truth for keyboard modifier flag mappings.
90
+ const KEYBOARD_MODIFIER_MAPPINGS: &[(&str, KeyModifiers)] = &[
91
+ ("ctrl", KeyModifiers::CONTROL),
92
+ ("alt", KeyModifiers::ALT),
93
+ ("shift", KeyModifiers::SHIFT),
94
+ ];
95
+
96
+ /// Returns all supported key codes for RBS generation.
97
+ pub fn all_key_codes() -> magnus::RHash {
98
+ let ruby = magnus::Ruby::get().unwrap();
99
+ let hash = ruby.hash_new();
100
+
101
+ let base: Vec<&str> = BASE_KEY_MAPPINGS.iter().map(|(s, _)| *s).collect();
102
+ let media: Vec<&str> = MEDIA_KEY_MAPPINGS.iter().map(|(s, _)| *s).collect();
103
+ let modifier_keys: Vec<&str> = MODIFIER_KEY_MAPPINGS.iter().map(|(s, _)| *s).collect();
104
+ let keyboard_modifiers: Vec<&str> =
105
+ KEYBOARD_MODIFIER_MAPPINGS.iter().map(|(s, _)| *s).collect();
106
+
107
+ let _ = hash.aset(ruby.to_symbol("base_keys"), base);
108
+ let _ = hash.aset(ruby.to_symbol("media_keys"), media);
109
+ let _ = hash.aset(ruby.to_symbol("modifier_keys"), modifier_keys);
110
+ let _ = hash.aset(ruby.to_symbol("keyboard_modifiers"), keyboard_modifiers);
111
+
112
+ hash
113
+ }
114
+
115
+ #[allow(clippy::needless_pass_by_value)]
116
+ pub fn inject_test_event(event_type: String, data: magnus::RHash) -> Result<(), Error> {
117
+ let ruby = magnus::Ruby::get().unwrap();
118
+ let event = match event_type.as_str() {
119
+ "key" => TestEvent::Crossterm(parse_key_event(data, &ruby)?),
120
+ "mouse" => TestEvent::Crossterm(parse_mouse_event(data, &ruby)?),
121
+ "resize" => TestEvent::Crossterm(parse_resize_event(data, &ruby)?),
122
+ "paste" => TestEvent::Crossterm(parse_paste_event(data, &ruby)?),
123
+ "focus_gained" => TestEvent::Crossterm(ratatui::crossterm::event::Event::FocusGained),
124
+ "focus_lost" => TestEvent::Crossterm(ratatui::crossterm::event::Event::FocusLost),
125
+ "sync" => TestEvent::Sync,
126
+ _ => {
127
+ return Err(Error::new(
128
+ ruby.exception_arg_error(),
129
+ format!("Unknown event type: {event_type}"),
130
+ ))
131
+ }
132
+ };
133
+
134
+ EVENT_QUEUE.with(|q| q.borrow_mut().push(event));
135
+ Ok(())
136
+ }
137
+
138
+ fn parse_base_key(s: &str) -> Option<KeyCode> {
139
+ BASE_KEY_MAPPINGS
140
+ .iter()
141
+ .find(|(key, _)| *key == s)
142
+ .map(|(_, code)| *code)
143
+ }
144
+
145
+ fn parse_media_key(s: &str) -> Option<MediaKeyCode> {
146
+ MEDIA_KEY_MAPPINGS
147
+ .iter()
148
+ .find(|(key, _)| *key == s)
149
+ .map(|(_, code)| *code)
150
+ }
151
+
152
+ fn parse_modifier_key(s: &str) -> Option<ModifierKeyCode> {
153
+ MODIFIER_KEY_MAPPINGS
154
+ .iter()
155
+ .find(|(key, _)| *key == s)
156
+ .map(|(_, code)| *code)
157
+ }
158
+
159
+ fn parse_keyboard_modifier(s: &str) -> Option<KeyModifiers> {
160
+ KEYBOARD_MODIFIER_MAPPINGS
161
+ .iter()
162
+ .find(|(key, _)| *key == s)
163
+ .map(|(_, mods)| *mods)
164
+ }
165
+
166
+ fn parse_key_event(
167
+ data: magnus::RHash,
168
+ ruby: &magnus::Ruby,
169
+ ) -> Result<ratatui::crossterm::event::Event, Error> {
170
+ let code_val: Value = data
171
+ .get(ruby.to_symbol("code"))
172
+ .ok_or_else(|| Error::new(ruby.exception_arg_error(), "Missing 'code' in key event"))?;
173
+ let code_str: String = String::try_convert(code_val)?;
174
+
175
+ let code = if let Some(kc) = parse_base_key(&code_str) {
176
+ kc
177
+ } else if let Some(m) = parse_media_key(&code_str) {
178
+ KeyCode::Media(m)
179
+ } else if let Some(m) = parse_modifier_key(&code_str) {
180
+ KeyCode::Modifier(m)
181
+ } else if let Some(num_str) = code_str.strip_prefix('f') {
182
+ if let Ok(n) = num_str.parse::<u8>() {
183
+ KeyCode::F(n)
184
+ } else {
185
+ KeyCode::Char(code_str.chars().next().unwrap_or('\0'))
186
+ }
187
+ } else if code_str.len() == 1 {
188
+ KeyCode::Char(code_str.chars().next().unwrap())
189
+ } else {
190
+ KeyCode::Null
191
+ };
192
+
193
+ let mut modifiers = KeyModifiers::empty();
194
+ if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
195
+ let mods: Vec<String> = Vec::try_convert(mods_val)?;
196
+ for m in mods {
197
+ if let Some(mod_flag) = parse_keyboard_modifier(&m) {
198
+ modifiers |= mod_flag;
199
+ }
200
+ }
201
+ }
202
+
203
+ Ok(ratatui::crossterm::event::Event::Key(
204
+ ratatui::crossterm::event::KeyEvent::new(code, modifiers),
205
+ ))
206
+ }
207
+
208
+ fn parse_mouse_event(
209
+ data: magnus::RHash,
210
+ ruby: &magnus::Ruby,
211
+ ) -> Result<ratatui::crossterm::event::Event, Error> {
212
+ let kind_val: Value = data
213
+ .get(ruby.to_symbol("kind"))
214
+ .ok_or_else(|| Error::new(ruby.exception_arg_error(), "Missing 'kind' in mouse event"))?;
215
+ let kind_str: String = String::try_convert(kind_val)?;
216
+
217
+ let button = if let Some(btn_val) = data.get(ruby.to_symbol("button")) {
218
+ let button_str: String = String::try_convert(btn_val)?;
219
+ match button_str.as_str() {
220
+ "right" => ratatui::crossterm::event::MouseButton::Right,
221
+ "middle" => ratatui::crossterm::event::MouseButton::Middle,
222
+ _ => ratatui::crossterm::event::MouseButton::Left,
223
+ }
224
+ } else {
225
+ ratatui::crossterm::event::MouseButton::Left
226
+ };
227
+
228
+ let x_val: Value = data
229
+ .get(ruby.to_symbol("x"))
230
+ .ok_or_else(|| Error::new(ruby.exception_arg_error(), "Missing 'x' in mouse event"))?;
231
+ let x: u16 = u16::try_convert(x_val)?;
232
+
233
+ let y_val: Value = data
234
+ .get(ruby.to_symbol("y"))
235
+ .ok_or_else(|| Error::new(ruby.exception_arg_error(), "Missing 'y' in mouse event"))?;
236
+ let y: u16 = u16::try_convert(y_val)?;
237
+
238
+ let kind = match kind_str.as_str() {
239
+ "down" => ratatui::crossterm::event::MouseEventKind::Down(button),
240
+ "up" => ratatui::crossterm::event::MouseEventKind::Up(button),
241
+ "drag" => ratatui::crossterm::event::MouseEventKind::Drag(button),
242
+ "moved" => ratatui::crossterm::event::MouseEventKind::Moved,
243
+ "scroll_down" => ratatui::crossterm::event::MouseEventKind::ScrollDown,
244
+ "scroll_up" => ratatui::crossterm::event::MouseEventKind::ScrollUp,
245
+ "scroll_left" => ratatui::crossterm::event::MouseEventKind::ScrollLeft,
246
+ "scroll_right" => ratatui::crossterm::event::MouseEventKind::ScrollRight,
247
+ _ => {
248
+ return Err(Error::new(
249
+ ruby.exception_arg_error(),
250
+ format!("Unknown mouse kind: {kind_str}"),
251
+ ))
252
+ }
253
+ };
254
+
255
+ let mut modifiers = ratatui::crossterm::event::KeyModifiers::empty();
256
+ if let Some(mods_val) = data.get(ruby.to_symbol("modifiers")) {
257
+ let mods: Vec<String> = Vec::try_convert(mods_val)?;
258
+ for m in mods {
259
+ match m.as_str() {
260
+ "ctrl" => modifiers |= ratatui::crossterm::event::KeyModifiers::CONTROL,
261
+ "alt" => modifiers |= ratatui::crossterm::event::KeyModifiers::ALT,
262
+ "shift" => modifiers |= ratatui::crossterm::event::KeyModifiers::SHIFT,
263
+ _ => {}
264
+ }
265
+ }
266
+ }
267
+
268
+ Ok(ratatui::crossterm::event::Event::Mouse(
269
+ ratatui::crossterm::event::MouseEvent {
270
+ kind,
271
+ column: x,
272
+ row: y,
273
+ modifiers,
274
+ },
275
+ ))
276
+ }
277
+
278
+ fn parse_resize_event(
279
+ data: magnus::RHash,
280
+ ruby: &magnus::Ruby,
281
+ ) -> Result<ratatui::crossterm::event::Event, Error> {
282
+ let width_val: Value = data.get(ruby.to_symbol("width")).ok_or_else(|| {
283
+ Error::new(
284
+ ruby.exception_arg_error(),
285
+ "Missing 'width' in resize event",
286
+ )
287
+ })?;
288
+ let width: u16 = u16::try_convert(width_val)?;
289
+
290
+ let height_val: Value = data.get(ruby.to_symbol("height")).ok_or_else(|| {
291
+ Error::new(
292
+ ruby.exception_arg_error(),
293
+ "Missing 'height' in resize event",
294
+ )
295
+ })?;
296
+ let height: u16 = u16::try_convert(height_val)?;
297
+
298
+ Ok(ratatui::crossterm::event::Event::Resize(width, height))
299
+ }
300
+
301
+ fn parse_paste_event(
302
+ data: magnus::RHash,
303
+ ruby: &magnus::Ruby,
304
+ ) -> Result<ratatui::crossterm::event::Event, Error> {
305
+ let content_val: Value = data.get(ruby.to_symbol("content")).ok_or_else(|| {
306
+ Error::new(
307
+ ruby.exception_arg_error(),
308
+ "Missing 'content' in paste event",
309
+ )
310
+ })?;
311
+ let content: String = String::try_convert(content_val)?;
312
+ Ok(ratatui::crossterm::event::Event::Paste(content))
313
+ }
314
+
315
+ pub fn clear_events() {
316
+ EVENT_QUEUE.with(|q| q.borrow_mut().clear());
317
+ }
318
+
319
+ /// Result of polling crossterm from outside the GVL.
320
+ ///
321
+ /// The blocking I/O (poll + read) happens without the Ruby GVL held,
322
+ /// so other Ruby threads can run concurrently. This enum carries the
323
+ /// result back across the GVL boundary for Ruby object construction.
324
+ #[derive(Debug)]
325
+ enum PollResult {
326
+ /// A crossterm event was read successfully.
327
+ Event(ratatui::crossterm::event::Event),
328
+ /// The timeout expired with no event available.
329
+ NoEvent,
330
+ /// An I/O error occurred during poll or read.
331
+ Error(String),
332
+ }
333
+
334
+ /// Data passed to the GVL-free polling callback.
335
+ ///
336
+ /// The `timeout` field controls the poll behavior:
337
+ /// - `Some(duration)`: poll with a timeout, return `NoEvent` if it expires.
338
+ /// - `None`: block indefinitely until an event arrives.
339
+ struct PollData {
340
+ timeout: Option<std::time::Duration>,
341
+ result: PollResult,
342
+ }
343
+
344
+ /// Polls crossterm for events without the Ruby GVL held.
345
+ ///
346
+ /// This is the work function passed to `rb_thread_call_without_gvl`.
347
+ /// It performs blocking I/O that would otherwise starve Ruby threads.
348
+ extern "C" fn poll_without_gvl(data: *mut c_void) -> *mut c_void {
349
+ // SAFETY: `data` is a valid pointer to `PollData`, passed by
350
+ // `poll_crossterm_without_gvl` via `rb_thread_call_without_gvl`. The
351
+ // pointer remains valid for the duration of this callback because the
352
+ // caller owns the `PollData` on the stack and waits for us to return.
353
+ let data = unsafe { &mut *data.cast::<PollData>() };
354
+
355
+ data.result = match data.timeout {
356
+ Some(duration) => match ratatui::crossterm::event::poll(duration) {
357
+ Ok(true) => match ratatui::crossterm::event::read() {
358
+ Ok(event) => PollResult::Event(event),
359
+ Err(e) => PollResult::Error(e.to_string()),
360
+ },
361
+ Ok(false) => PollResult::NoEvent,
362
+ Err(e) => PollResult::Error(e.to_string()),
363
+ },
364
+ None => match ratatui::crossterm::event::read() {
365
+ Ok(event) => PollResult::Event(event),
366
+ Err(e) => PollResult::Error(e.to_string()),
367
+ },
368
+ };
369
+
370
+ std::ptr::null_mut()
371
+ }
372
+
373
+ /// Polls crossterm with the GVL released, then converts the result
374
+ /// to a Ruby value after reacquiring the GVL.
375
+ fn poll_crossterm_without_gvl(
376
+ ruby: &magnus::Ruby,
377
+ timeout: Option<std::time::Duration>,
378
+ ) -> Result<Value, Error> {
379
+ let mut data = PollData {
380
+ timeout,
381
+ result: PollResult::NoEvent,
382
+ };
383
+
384
+ // SAFETY: `data` is a valid stack-local `PollData` whose lifetime
385
+ // spans this entire call. `poll_without_gvl` only reads/writes
386
+ // through the pointer while `rb_thread_call_without_gvl` blocks,
387
+ // and we don't access `data` again until that function returns.
388
+ unsafe {
389
+ rb_thread_call_without_gvl(
390
+ Some(poll_without_gvl),
391
+ (&raw mut data).cast::<c_void>(),
392
+ None,
393
+ std::ptr::null_mut(),
394
+ );
395
+ }
396
+
397
+ // GVL is now re-held — safe to create Ruby objects.
398
+ match data.result {
399
+ PollResult::Event(event) => handle_crossterm_event(event),
400
+ PollResult::NoEvent => Ok(ruby.qnil().into_value_with(ruby)),
401
+ PollResult::Error(msg) => Err(Error::new(ruby.exception_runtime_error(), msg)),
402
+ }
403
+ }
404
+
405
+ pub fn poll_event(ruby: &magnus::Ruby, timeout_val: Option<f64>) -> Result<Value, Error> {
406
+ let event = EVENT_QUEUE.with(|q| {
407
+ let mut queue = q.borrow_mut();
408
+ if queue.is_empty() {
409
+ None
410
+ } else {
411
+ Some(queue.remove(0))
412
+ }
413
+ });
414
+
415
+ if let Some(e) = event {
416
+ return handle_test_event(e);
417
+ }
418
+
419
+ let is_test_mode = crate::terminal::with_query(|q| q.is_test_mode()).unwrap_or(false);
420
+
421
+ if is_test_mode {
422
+ return Ok(ruby.qnil().into_value_with(ruby));
423
+ }
424
+
425
+ let timeout = timeout_val.map(std::time::Duration::from_secs_f64);
426
+ poll_crossterm_without_gvl(ruby, timeout)
427
+ }
428
+
429
+ fn handle_test_event(event: TestEvent) -> Result<Value, Error> {
430
+ match event {
431
+ TestEvent::Crossterm(e) => handle_crossterm_event(e),
432
+ TestEvent::Sync => handle_sync_event(),
433
+ }
434
+ }
435
+
436
+ fn handle_crossterm_event(event: ratatui::crossterm::event::Event) -> Result<Value, Error> {
437
+ match event {
438
+ ratatui::crossterm::event::Event::Key(key) => handle_key_event(key),
439
+ ratatui::crossterm::event::Event::Mouse(event) => handle_mouse_event(event),
440
+ ratatui::crossterm::event::Event::Resize(w, h) => handle_resize_event(w, h),
441
+ ratatui::crossterm::event::Event::Paste(content) => handle_paste_event(content),
442
+ ratatui::crossterm::event::Event::FocusGained => handle_focus_event("focus_gained"),
443
+ ratatui::crossterm::event::Event::FocusLost => handle_focus_event("focus_lost"),
444
+ }
445
+ }
446
+
447
+ fn handle_sync_event() -> Result<Value, Error> {
448
+ let ruby = magnus::Ruby::get().unwrap();
449
+ let hash = ruby.hash_new();
450
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("sync"))?;
451
+ Ok(hash.into_value_with(&ruby))
452
+ }
453
+
454
+ fn media_key_to_string(m: MediaKeyCode) -> &'static str {
455
+ MEDIA_KEY_MAPPINGS
456
+ .iter()
457
+ .find(|(_, code)| *code == m)
458
+ .map_or("unknown", |(s, _)| *s)
459
+ }
460
+
461
+ fn modifier_key_to_string(m: ModifierKeyCode) -> &'static str {
462
+ MODIFIER_KEY_MAPPINGS
463
+ .iter()
464
+ .find(|(_, code)| *code == m)
465
+ .map_or("unknown", |(s, _)| *s)
466
+ }
467
+
468
+ fn base_key_to_string(kc: KeyCode) -> Option<&'static str> {
469
+ BASE_KEY_MAPPINGS
470
+ .iter()
471
+ .find(|(_, code)| *code == kc)
472
+ .map(|(s, _)| *s)
473
+ }
474
+
475
+ fn handle_key_event(key: ratatui::crossterm::event::KeyEvent) -> Result<Value, Error> {
476
+ use ratatui::crossterm::event::KeyCode;
477
+
478
+ let ruby = magnus::Ruby::get().unwrap();
479
+ if key.kind != ratatui::crossterm::event::KeyEventKind::Press {
480
+ return Ok(ruby.qnil().into_value_with(&ruby));
481
+ }
482
+
483
+ let hash = ruby.hash_new();
484
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("key"))?;
485
+
486
+ // Determine the kind (category) of the key
487
+ let kind = match key.code {
488
+ KeyCode::Char(_)
489
+ | KeyCode::Enter
490
+ | KeyCode::Tab
491
+ | KeyCode::Backspace
492
+ | KeyCode::BackTab
493
+ | KeyCode::Up
494
+ | KeyCode::Down
495
+ | KeyCode::Left
496
+ | KeyCode::Right
497
+ | KeyCode::Home
498
+ | KeyCode::End
499
+ | KeyCode::PageUp
500
+ | KeyCode::PageDown
501
+ | KeyCode::Insert
502
+ | KeyCode::Delete
503
+ | KeyCode::Null => "standard",
504
+ KeyCode::F(_) => "function",
505
+ KeyCode::Media(_) => "media",
506
+ KeyCode::Modifier(_) => "modifier",
507
+ KeyCode::Esc
508
+ | KeyCode::CapsLock
509
+ | KeyCode::ScrollLock
510
+ | KeyCode::NumLock
511
+ | KeyCode::PrintScreen
512
+ | KeyCode::Pause
513
+ | KeyCode::Menu
514
+ | KeyCode::KeypadBegin => "system",
515
+ };
516
+
517
+ let code = if let KeyCode::Char(c) = key.code {
518
+ c.to_string()
519
+ } else if let KeyCode::F(n) = key.code {
520
+ format!("f{n}")
521
+ } else if let KeyCode::Media(m) = key.code {
522
+ media_key_to_string(m).to_string()
523
+ } else if let KeyCode::Modifier(m) = key.code {
524
+ modifier_key_to_string(m).to_string()
525
+ } else if let Some(s) = base_key_to_string(key.code) {
526
+ s.to_string()
527
+ } else {
528
+ "unknown".to_string()
529
+ };
530
+
531
+ hash.aset(ruby.to_symbol("code"), code)?;
532
+ hash.aset(ruby.to_symbol("kind"), ruby.to_symbol(kind))?;
533
+
534
+ let mut modifiers = Vec::new();
535
+ for (name, flag) in KEYBOARD_MODIFIER_MAPPINGS {
536
+ if key.modifiers.contains(*flag) {
537
+ modifiers.push(*name);
538
+ }
539
+ }
540
+ if !modifiers.is_empty() {
541
+ hash.aset(ruby.to_symbol("modifiers"), modifiers)?;
542
+ }
543
+ Ok(hash.into_value_with(&ruby))
544
+ }
545
+
546
+ fn handle_mouse_event(event: ratatui::crossterm::event::MouseEvent) -> Result<Value, Error> {
547
+ let ruby = magnus::Ruby::get().unwrap();
548
+ let hash = ruby.hash_new();
549
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("mouse"))?;
550
+ let (kind, button) = match event.kind {
551
+ ratatui::crossterm::event::MouseEventKind::Down(btn) => ("down", btn),
552
+ ratatui::crossterm::event::MouseEventKind::Up(btn) => ("up", btn),
553
+ ratatui::crossterm::event::MouseEventKind::Drag(btn) => ("drag", btn),
554
+ ratatui::crossterm::event::MouseEventKind::Moved => {
555
+ ("moved", ratatui::crossterm::event::MouseButton::Left)
556
+ }
557
+ ratatui::crossterm::event::MouseEventKind::ScrollDown => {
558
+ ("scroll_down", ratatui::crossterm::event::MouseButton::Left)
559
+ }
560
+ ratatui::crossterm::event::MouseEventKind::ScrollUp => {
561
+ ("scroll_up", ratatui::crossterm::event::MouseButton::Left)
562
+ }
563
+ ratatui::crossterm::event::MouseEventKind::ScrollLeft => {
564
+ ("scroll_left", ratatui::crossterm::event::MouseButton::Left)
565
+ }
566
+ ratatui::crossterm::event::MouseEventKind::ScrollRight => {
567
+ ("scroll_right", ratatui::crossterm::event::MouseButton::Left)
568
+ }
569
+ };
570
+ hash.aset(ruby.to_symbol("kind"), ruby.to_symbol(kind))?;
571
+ if matches!(
572
+ event.kind,
573
+ ratatui::crossterm::event::MouseEventKind::Down(_)
574
+ | ratatui::crossterm::event::MouseEventKind::Up(_)
575
+ | ratatui::crossterm::event::MouseEventKind::Drag(_)
576
+ ) {
577
+ let btn_sym = match button {
578
+ ratatui::crossterm::event::MouseButton::Left => "left",
579
+ ratatui::crossterm::event::MouseButton::Right => "right",
580
+ ratatui::crossterm::event::MouseButton::Middle => "middle",
581
+ };
582
+ hash.aset(ruby.to_symbol("button"), ruby.to_symbol(btn_sym))?;
583
+ } else {
584
+ hash.aset(ruby.to_symbol("button"), ruby.to_symbol("none"))?;
585
+ }
586
+ hash.aset(ruby.to_symbol("x"), event.column)?;
587
+ hash.aset(ruby.to_symbol("y"), event.row)?;
588
+ let mut modifiers = Vec::new();
589
+ if event
590
+ .modifiers
591
+ .contains(ratatui::crossterm::event::KeyModifiers::CONTROL)
592
+ {
593
+ modifiers.push("ctrl");
594
+ }
595
+ if event
596
+ .modifiers
597
+ .contains(ratatui::crossterm::event::KeyModifiers::ALT)
598
+ {
599
+ modifiers.push("alt");
600
+ }
601
+ if event
602
+ .modifiers
603
+ .contains(ratatui::crossterm::event::KeyModifiers::SHIFT)
604
+ {
605
+ modifiers.push("shift");
606
+ }
607
+ hash.aset(ruby.to_symbol("modifiers"), modifiers)?;
608
+ Ok(hash.into_value_with(&ruby))
609
+ }
610
+
611
+ fn handle_resize_event(w: u16, h: u16) -> Result<Value, Error> {
612
+ let ruby = magnus::Ruby::get().unwrap();
613
+ let hash = ruby.hash_new();
614
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("resize"))?;
615
+ hash.aset(ruby.to_symbol("width"), w)?;
616
+ hash.aset(ruby.to_symbol("height"), h)?;
617
+ Ok(hash.into_value_with(&ruby))
618
+ }
619
+
620
+ fn handle_paste_event(content: String) -> Result<Value, Error> {
621
+ let ruby = magnus::Ruby::get().unwrap();
622
+ let hash = ruby.hash_new();
623
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol("paste"))?;
624
+ hash.aset(ruby.to_symbol("content"), content)?;
625
+ Ok(hash.into_value_with(&ruby))
626
+ }
627
+
628
+ fn handle_focus_event(event_type: &str) -> Result<Value, Error> {
629
+ let ruby = magnus::Ruby::get().unwrap();
630
+ let hash = ruby.hash_new();
631
+ hash.aset(ruby.to_symbol("type"), ruby.to_symbol(event_type))?;
632
+ Ok(hash.into_value_with(&ruby))
633
+ }
634
+
635
+ #[cfg(test)]
636
+ mod tests {
637
+ use super::*;
638
+
639
+ #[test]
640
+ fn poll_data_defaults_to_no_event() {
641
+ let data = PollData {
642
+ timeout: Some(std::time::Duration::from_millis(0)),
643
+ result: PollResult::NoEvent,
644
+ };
645
+ assert!(matches!(data.result, PollResult::NoEvent));
646
+ }
647
+
648
+ #[test]
649
+ fn poll_data_accepts_none_timeout_for_indefinite_blocking() {
650
+ let data = PollData {
651
+ timeout: None,
652
+ result: PollResult::NoEvent,
653
+ };
654
+ assert!(data.timeout.is_none());
655
+ }
656
+
657
+ #[test]
658
+ fn poll_result_error_preserves_message() {
659
+ let result = PollResult::Error("connection reset".to_string());
660
+ match result {
661
+ PollResult::Error(msg) => assert_eq!(msg, "connection reset"),
662
+ _ => panic!("Expected PollResult::Error"),
663
+ }
664
+ }
665
+
666
+ #[test]
667
+ fn poll_result_event_wraps_crossterm_event() {
668
+ let key_event =
669
+ ratatui::crossterm::event::Event::Key(ratatui::crossterm::event::KeyEvent::new(
670
+ ratatui::crossterm::event::KeyCode::Char('q'),
671
+ ratatui::crossterm::event::KeyModifiers::empty(),
672
+ ));
673
+ let result = PollResult::Event(key_event.clone());
674
+ match result {
675
+ PollResult::Event(e) => assert_eq!(format!("{e:?}"), format!("{key_event:?}")),
676
+ _ => panic!("Expected PollResult::Event"),
677
+ }
678
+ }
679
+
680
+ #[test]
681
+ fn poll_without_gvl_returns_no_event_on_zero_timeout() {
682
+ // With a zero-duration timeout, poll returns immediately with no event.
683
+ // In headless environments (CI), crossterm may return an Error because
684
+ // there is no terminal to read from — that's also a valid outcome.
685
+ let mut data = PollData {
686
+ timeout: Some(std::time::Duration::from_millis(0)),
687
+ result: PollResult::Error("sentinel — should be overwritten".to_string()),
688
+ };
689
+
690
+ poll_without_gvl(&mut data as *mut PollData as *mut c_void);
691
+
692
+ // The callback must have overwritten the sentinel value.
693
+ match &data.result {
694
+ PollResult::Error(msg) if msg == "sentinel — should be overwritten" => {
695
+ panic!("poll_without_gvl did not write to data.result")
696
+ }
697
+ _ => {} // NoEvent, Event, or a *different* Error are all fine.
698
+ }
699
+ }
700
+ }