1use presentar_core::{
7 widget::{LayoutResult, TextStyle},
8 Brick, BrickAssertion, BrickBudget, BrickVerification, Canvas, Color, Constraints, Event, Key,
9 Point, Rect, Size, TypeId, Widget,
10};
11use serde::{Deserialize, Serialize};
12use std::any::Any;
13use std::time::Duration;
14
15#[derive(Debug, Clone, Serialize, Deserialize)]
17pub enum MenuItem {
18 Action {
20 label: String,
22 action: String,
24 disabled: bool,
26 shortcut: Option<String>,
28 },
29 Checkbox {
31 label: String,
33 action: String,
35 checked: bool,
37 disabled: bool,
39 },
40 Separator,
42 Submenu {
44 label: String,
46 items: Vec<Self>,
48 disabled: bool,
50 },
51}
52
53impl MenuItem {
54 #[must_use]
56 pub fn action(label: impl Into<String>, action: impl Into<String>) -> Self {
57 Self::Action {
58 label: label.into(),
59 action: action.into(),
60 disabled: false,
61 shortcut: None,
62 }
63 }
64
65 #[must_use]
67 pub fn checkbox(label: impl Into<String>, action: impl Into<String>, checked: bool) -> Self {
68 Self::Checkbox {
69 label: label.into(),
70 action: action.into(),
71 checked,
72 disabled: false,
73 }
74 }
75
76 #[must_use]
78 pub const fn separator() -> Self {
79 Self::Separator
80 }
81
82 #[must_use]
84 pub fn submenu(label: impl Into<String>, items: Vec<Self>) -> Self {
85 Self::Submenu {
86 label: label.into(),
87 items,
88 disabled: false,
89 }
90 }
91
92 #[must_use]
94 pub fn disabled(mut self, disabled: bool) -> Self {
95 match &mut self {
96 Self::Action { disabled: d, .. }
97 | Self::Checkbox { disabled: d, .. }
98 | Self::Submenu { disabled: d, .. } => *d = disabled,
99 Self::Separator => {}
100 }
101 self
102 }
103
104 #[must_use]
106 pub fn shortcut(mut self, shortcut: impl Into<String>) -> Self {
107 if let Self::Action { shortcut: s, .. } = &mut self {
108 *s = Some(shortcut.into());
109 }
110 self
111 }
112
113 #[must_use]
115 pub fn is_selectable(&self) -> bool {
116 match self {
117 Self::Action { disabled, .. }
118 | Self::Checkbox { disabled, .. }
119 | Self::Submenu { disabled, .. } => !disabled,
120 Self::Separator => false,
121 }
122 }
123
124 #[must_use]
126 pub const fn height(&self) -> f32 {
127 match self {
128 Self::Separator => 9.0, _ => 32.0,
130 }
131 }
132}
133
134#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Serialize, Deserialize)]
136pub enum MenuTrigger {
137 #[default]
139 Click,
140 Hover,
142 ContextMenu,
144}
145
146#[derive(Serialize, Deserialize)]
148pub struct Menu {
149 pub items: Vec<MenuItem>,
151 pub open: bool,
153 pub trigger: MenuTrigger,
155 pub width: f32,
157 pub background_color: Color,
159 pub hover_color: Color,
161 pub text_color: Color,
163 pub disabled_color: Color,
165 test_id_value: Option<String>,
167 #[serde(skip)]
169 bounds: Rect,
170 #[serde(skip)]
172 panel_bounds: Rect,
173 #[serde(skip)]
175 highlighted_index: Option<usize>,
176 #[serde(skip)]
178 open_submenu: Option<usize>,
179 #[serde(skip)]
181 trigger_widget: Option<Box<dyn Widget>>,
182}
183
184impl Default for Menu {
185 fn default() -> Self {
186 Self {
187 items: Vec::new(),
188 open: false,
189 trigger: MenuTrigger::Click,
190 width: 200.0,
191 background_color: Color::WHITE,
192 hover_color: Color::rgba(0.0, 0.0, 0.0, 0.1),
193 text_color: Color::BLACK,
194 disabled_color: Color::rgb(0.6, 0.6, 0.6),
195 test_id_value: None,
196 bounds: Rect::default(),
197 panel_bounds: Rect::default(),
198 highlighted_index: None,
199 open_submenu: None,
200 trigger_widget: None,
201 }
202 }
203}
204
205impl Menu {
206 #[must_use]
208 pub fn new() -> Self {
209 Self::default()
210 }
211
212 #[must_use]
214 pub fn items(mut self, items: Vec<MenuItem>) -> Self {
215 self.items = items;
216 self
217 }
218
219 #[must_use]
221 pub fn item(mut self, item: MenuItem) -> Self {
222 self.items.push(item);
223 self
224 }
225
226 #[must_use]
228 pub const fn trigger(mut self, trigger: MenuTrigger) -> Self {
229 self.trigger = trigger;
230 self
231 }
232
233 #[must_use]
235 pub const fn width(mut self, width: f32) -> Self {
236 self.width = width;
237 self
238 }
239
240 #[must_use]
242 pub const fn background_color(mut self, color: Color) -> Self {
243 self.background_color = color;
244 self
245 }
246
247 #[must_use]
249 pub const fn hover_color(mut self, color: Color) -> Self {
250 self.hover_color = color;
251 self
252 }
253
254 #[must_use]
256 pub const fn text_color(mut self, color: Color) -> Self {
257 self.text_color = color;
258 self
259 }
260
261 pub fn trigger_widget(mut self, widget: impl Widget + 'static) -> Self {
263 self.trigger_widget = Some(Box::new(widget));
264 self
265 }
266
267 #[must_use]
269 pub fn with_test_id(mut self, id: impl Into<String>) -> Self {
270 self.test_id_value = Some(id.into());
271 self
272 }
273
274 pub fn show(&mut self) {
276 self.open = true;
277 self.highlighted_index = None;
278 }
279
280 pub fn hide(&mut self) {
282 self.open = false;
283 self.highlighted_index = None;
284 self.open_submenu = None;
285 }
286
287 pub fn toggle(&mut self) {
289 if self.open {
290 self.hide();
291 } else {
292 self.show();
293 }
294 }
295
296 #[must_use]
298 pub const fn is_open(&self) -> bool {
299 self.open
300 }
301
302 #[must_use]
304 pub const fn highlighted_index(&self) -> Option<usize> {
305 self.highlighted_index
306 }
307
308 fn calculate_menu_height(&self) -> f32 {
310 let padding = 8.0; let items_height: f32 = self.items.iter().map(MenuItem::height).sum();
312 items_height + padding * 2.0
313 }
314
315 fn next_selectable(&self, from: Option<usize>, forward: bool) -> Option<usize> {
317 if self.items.is_empty() {
318 return None;
319 }
320
321 let start = from.map_or_else(
322 || if forward { 0 } else { self.items.len() - 1 },
323 |i| {
324 if forward {
325 if i + 1 >= self.items.len() {
326 0
327 } else {
328 i + 1
329 }
330 } else if i == 0 {
331 self.items.len() - 1
332 } else {
333 i - 1
334 }
335 },
336 );
337
338 let mut idx = start;
339 for _ in 0..self.items.len() {
340 if self.items[idx].is_selectable() {
341 return Some(idx);
342 }
343 if forward {
344 idx = if idx + 1 >= self.items.len() {
345 0
346 } else {
347 idx + 1
348 };
349 } else {
350 idx = if idx == 0 {
351 self.items.len() - 1
352 } else {
353 idx - 1
354 };
355 }
356 }
357
358 None
359 }
360
361 fn item_at_position(&self, y: f32) -> Option<usize> {
363 let relative_y = y - self.panel_bounds.y - 8.0; if relative_y < 0.0 {
365 return None;
366 }
367
368 let mut current_y = 0.0;
369 for (i, item) in self.items.iter().enumerate() {
370 let height = item.height();
371 if relative_y >= current_y && relative_y < current_y + height {
372 return Some(i);
373 }
374 current_y += height;
375 }
376
377 None
378 }
379}
380
381impl Widget for Menu {
382 fn type_id(&self) -> TypeId {
383 TypeId::of::<Self>()
384 }
385
386 fn measure(&self, constraints: Constraints) -> Size {
387 if let Some(ref trigger) = self.trigger_widget {
389 trigger.measure(constraints)
390 } else {
391 Size::new(self.width.min(constraints.max_width), 32.0)
392 }
393 }
394
395 fn layout(&mut self, bounds: Rect) -> LayoutResult {
396 self.bounds = bounds;
397
398 if let Some(ref mut trigger) = self.trigger_widget {
400 trigger.layout(bounds);
401 }
402
403 if self.open {
405 let menu_height = self.calculate_menu_height();
406 self.panel_bounds =
407 Rect::new(bounds.x, bounds.y + bounds.height, self.width, menu_height);
408 }
409
410 LayoutResult {
411 size: bounds.size(),
412 }
413 }
414
415 #[allow(clippy::too_many_lines)]
416 fn paint(&self, canvas: &mut dyn Canvas) {
417 if let Some(ref trigger) = self.trigger_widget {
419 trigger.paint(canvas);
420 }
421
422 if !self.open {
423 return;
424 }
425
426 let shadow_bounds = Rect::new(
428 self.panel_bounds.x + 2.0,
429 self.panel_bounds.y + 2.0,
430 self.panel_bounds.width,
431 self.panel_bounds.height,
432 );
433 canvas.fill_rect(shadow_bounds, Color::rgba(0.0, 0.0, 0.0, 0.1));
434 canvas.fill_rect(self.panel_bounds, self.background_color);
435
436 let mut y = self.panel_bounds.y + 8.0; let text_style = TextStyle {
439 size: 14.0,
440 color: self.text_color,
441 ..Default::default()
442 };
443 let disabled_style = TextStyle {
444 size: 14.0,
445 color: self.disabled_color,
446 ..Default::default()
447 };
448
449 for (i, item) in self.items.iter().enumerate() {
450 let height = item.height();
451
452 match item {
453 MenuItem::Action {
454 label,
455 disabled,
456 shortcut,
457 ..
458 } => {
459 if self.highlighted_index == Some(i) && !disabled {
461 let hover_rect =
462 Rect::new(self.panel_bounds.x, y, self.panel_bounds.width, height);
463 canvas.fill_rect(hover_rect, self.hover_color);
464 }
465
466 let style = if *disabled {
468 &disabled_style
469 } else {
470 &text_style
471 };
472 canvas.draw_text(
473 label,
474 Point::new(self.panel_bounds.x + 12.0, y + 20.0),
475 style,
476 );
477
478 if let Some(ref shortcut) = shortcut {
480 let shortcut_style = TextStyle {
481 size: 12.0,
482 color: self.disabled_color,
483 ..Default::default()
484 };
485 canvas.draw_text(
486 shortcut,
487 Point::new(
488 self.panel_bounds.x + self.panel_bounds.width - 60.0,
489 y + 20.0,
490 ),
491 &shortcut_style,
492 );
493 }
494 }
495 MenuItem::Checkbox {
496 label,
497 checked,
498 disabled,
499 ..
500 } => {
501 if self.highlighted_index == Some(i) && !disabled {
503 let hover_rect =
504 Rect::new(self.panel_bounds.x, y, self.panel_bounds.width, height);
505 canvas.fill_rect(hover_rect, self.hover_color);
506 }
507
508 let check_text = if *checked { "✓" } else { " " };
510 let style = if *disabled {
511 &disabled_style
512 } else {
513 &text_style
514 };
515 canvas.draw_text(
516 check_text,
517 Point::new(self.panel_bounds.x + 12.0, y + 20.0),
518 style,
519 );
520
521 canvas.draw_text(
523 label,
524 Point::new(self.panel_bounds.x + 32.0, y + 20.0),
525 style,
526 );
527 }
528 MenuItem::Separator => {
529 let line_y = y + 4.0;
530 canvas.draw_line(
531 Point::new(self.panel_bounds.x + 8.0, line_y),
532 Point::new(self.panel_bounds.x + self.panel_bounds.width - 8.0, line_y),
533 Color::rgb(0.9, 0.9, 0.9),
534 1.0,
535 );
536 }
537 MenuItem::Submenu {
538 label, disabled, ..
539 } => {
540 if self.highlighted_index == Some(i) && !disabled {
542 let hover_rect =
543 Rect::new(self.panel_bounds.x, y, self.panel_bounds.width, height);
544 canvas.fill_rect(hover_rect, self.hover_color);
545 }
546
547 let style = if *disabled {
549 &disabled_style
550 } else {
551 &text_style
552 };
553 canvas.draw_text(
554 label,
555 Point::new(self.panel_bounds.x + 12.0, y + 20.0),
556 style,
557 );
558
559 canvas.draw_text(
561 "›",
562 Point::new(
563 self.panel_bounds.x + self.panel_bounds.width - 20.0,
564 y + 20.0,
565 ),
566 style,
567 );
568 }
569 }
570
571 y += height;
572 }
573 }
574
575 #[allow(clippy::too_many_lines)]
576 fn event(&mut self, event: &Event) -> Option<Box<dyn Any + Send>> {
577 match event {
578 Event::MouseDown { position, .. } => {
579 let on_trigger = position.x >= self.bounds.x
581 && position.x <= self.bounds.x + self.bounds.width
582 && position.y >= self.bounds.y
583 && position.y <= self.bounds.y + self.bounds.height;
584
585 if on_trigger && self.trigger == MenuTrigger::Click {
586 self.toggle();
587 return Some(Box::new(MenuToggled { open: self.open }));
588 }
589
590 if self.open {
592 let on_menu = position.x >= self.panel_bounds.x
593 && position.x <= self.panel_bounds.x + self.panel_bounds.width
594 && position.y >= self.panel_bounds.y
595 && position.y <= self.panel_bounds.y + self.panel_bounds.height;
596
597 if on_menu {
598 if let Some(idx) = self.item_at_position(position.y) {
599 if let Some(item) = self.items.get_mut(idx) {
600 match item {
601 MenuItem::Action {
602 action, disabled, ..
603 } if !*disabled => {
604 let action_id = action.clone();
605 self.hide();
606 return Some(Box::new(MenuItemSelected {
607 action: action_id,
608 }));
609 }
610 MenuItem::Checkbox {
611 action,
612 checked,
613 disabled,
614 ..
615 } if !*disabled => {
616 *checked = !*checked;
617 let action_id = action.clone();
618 let is_checked = *checked;
619 return Some(Box::new(MenuCheckboxToggled {
620 action: action_id,
621 checked: is_checked,
622 }));
623 }
624 MenuItem::Submenu { disabled, .. } if !*disabled => {
625 self.open_submenu = Some(idx);
626 }
627 _ => {}
628 }
629 }
630 }
631 } else {
632 self.hide();
634 return Some(Box::new(MenuClosed));
635 }
636 }
637 }
638 Event::MouseMove { position } => {
639 if self.open {
640 let on_menu = position.x >= self.panel_bounds.x
641 && position.x <= self.panel_bounds.x + self.panel_bounds.width
642 && position.y >= self.panel_bounds.y
643 && position.y <= self.panel_bounds.y + self.panel_bounds.height;
644
645 if on_menu {
646 self.highlighted_index = self.item_at_position(position.y);
647 } else {
648 self.highlighted_index = None;
649 }
650 }
651 }
652 Event::KeyDown { key } if self.open => match key {
653 Key::Escape => {
654 self.hide();
655 return Some(Box::new(MenuClosed));
656 }
657 Key::Up => {
658 self.highlighted_index = self.next_selectable(self.highlighted_index, false);
659 }
660 Key::Down => {
661 self.highlighted_index = self.next_selectable(self.highlighted_index, true);
662 }
663 Key::Enter | Key::Space => {
664 if let Some(idx) = self.highlighted_index {
665 if let Some(item) = self.items.get_mut(idx) {
666 match item {
667 MenuItem::Action {
668 action, disabled, ..
669 } if !*disabled => {
670 let action_id = action.clone();
671 self.hide();
672 return Some(Box::new(MenuItemSelected { action: action_id }));
673 }
674 MenuItem::Checkbox {
675 action,
676 checked,
677 disabled,
678 ..
679 } if !*disabled => {
680 *checked = !*checked;
681 let action_id = action.clone();
682 let is_checked = *checked;
683 return Some(Box::new(MenuCheckboxToggled {
684 action: action_id,
685 checked: is_checked,
686 }));
687 }
688 _ => {}
689 }
690 }
691 }
692 }
693 _ => {}
694 },
695 _ => {}
696 }
697
698 None
699 }
700
701 fn children(&self) -> &[Box<dyn Widget>] {
702 &[]
703 }
704
705 fn children_mut(&mut self) -> &mut [Box<dyn Widget>] {
706 &mut []
707 }
708
709 fn is_focusable(&self) -> bool {
710 true
711 }
712
713 fn test_id(&self) -> Option<&str> {
714 self.test_id_value.as_deref()
715 }
716
717 fn bounds(&self) -> Rect {
718 self.bounds
719 }
720}
721
722impl Brick for Menu {
724 fn brick_name(&self) -> &'static str {
725 "Menu"
726 }
727
728 fn assertions(&self) -> &[BrickAssertion] {
729 &[BrickAssertion::MaxLatencyMs(16)]
730 }
731
732 fn budget(&self) -> BrickBudget {
733 BrickBudget::uniform(16)
734 }
735
736 fn verify(&self) -> BrickVerification {
737 BrickVerification {
738 passed: self.assertions().to_vec(),
739 failed: vec![],
740 verification_time: Duration::from_micros(10),
741 }
742 }
743
744 fn to_html(&self) -> String {
745 r#"<div class="brick-menu"></div>"#.to_string()
746 }
747
748 fn to_css(&self) -> String {
749 ".brick-menu { display: block; position: relative; }".to_string()
750 }
751
752 fn test_id(&self) -> Option<&str> {
753 self.test_id_value.as_deref()
754 }
755}
756
757#[derive(Debug, Clone)]
759pub struct MenuToggled {
760 pub open: bool,
762}
763
764#[derive(Debug, Clone)]
766pub struct MenuItemSelected {
767 pub action: String,
769}
770
771#[derive(Debug, Clone)]
773pub struct MenuCheckboxToggled {
774 pub action: String,
776 pub checked: bool,
778}
779
780#[derive(Debug, Clone)]
782pub struct MenuClosed;
783
784#[cfg(test)]
785mod tests {
786 use super::*;
787
788 #[test]
793 fn test_menu_item_action() {
794 let item = MenuItem::action("Cut", "edit.cut");
795 match item {
796 MenuItem::Action {
797 label,
798 action,
799 disabled,
800 shortcut,
801 } => {
802 assert_eq!(label, "Cut");
803 assert_eq!(action, "edit.cut");
804 assert!(!disabled);
805 assert!(shortcut.is_none());
806 }
807 _ => panic!("Expected Action"),
808 }
809 }
810
811 #[test]
812 fn test_menu_item_action_with_shortcut() {
813 let item = MenuItem::action("Cut", "edit.cut").shortcut("Ctrl+X");
814 match item {
815 MenuItem::Action { shortcut, .. } => {
816 assert_eq!(shortcut, Some("Ctrl+X".to_string()));
817 }
818 _ => panic!("Expected Action"),
819 }
820 }
821
822 #[test]
823 fn test_menu_item_checkbox() {
824 let item = MenuItem::checkbox("Show Grid", "view.grid", true);
825 match item {
826 MenuItem::Checkbox {
827 label,
828 checked,
829 disabled,
830 ..
831 } => {
832 assert_eq!(label, "Show Grid");
833 assert!(checked);
834 assert!(!disabled);
835 }
836 _ => panic!("Expected Checkbox"),
837 }
838 }
839
840 #[test]
841 fn test_menu_item_separator() {
842 let item = MenuItem::separator();
843 assert!(matches!(item, MenuItem::Separator));
844 }
845
846 #[test]
847 fn test_menu_item_submenu() {
848 let items = vec![MenuItem::action("Sub 1", "sub.1")];
849 let item = MenuItem::submenu("More", items);
850 match item {
851 MenuItem::Submenu {
852 label,
853 items,
854 disabled,
855 } => {
856 assert_eq!(label, "More");
857 assert_eq!(items.len(), 1);
858 assert!(!disabled);
859 }
860 _ => panic!("Expected Submenu"),
861 }
862 }
863
864 #[test]
865 fn test_menu_item_disabled() {
866 let item = MenuItem::action("Cut", "edit.cut").disabled(true);
867 match item {
868 MenuItem::Action { disabled, .. } => assert!(disabled),
869 _ => panic!("Expected Action"),
870 }
871 }
872
873 #[test]
874 fn test_menu_item_is_selectable() {
875 assert!(MenuItem::action("Cut", "edit.cut").is_selectable());
876 assert!(!MenuItem::action("Cut", "edit.cut")
877 .disabled(true)
878 .is_selectable());
879 assert!(!MenuItem::separator().is_selectable());
880 assert!(MenuItem::checkbox("Show", "show", false).is_selectable());
881 }
882
883 #[test]
884 fn test_menu_item_height() {
885 assert_eq!(MenuItem::action("Cut", "edit.cut").height(), 32.0);
886 assert_eq!(MenuItem::separator().height(), 9.0);
887 }
888
889 #[test]
894 fn test_menu_new() {
895 let menu = Menu::new();
896 assert!(menu.items.is_empty());
897 assert!(!menu.open);
898 assert_eq!(menu.trigger, MenuTrigger::Click);
899 }
900
901 #[test]
902 fn test_menu_builder() {
903 let menu = Menu::new()
904 .items(vec![
905 MenuItem::action("Cut", "cut"),
906 MenuItem::separator(),
907 MenuItem::action("Paste", "paste"),
908 ])
909 .trigger(MenuTrigger::Hover)
910 .width(250.0);
911
912 assert_eq!(menu.items.len(), 3);
913 assert_eq!(menu.trigger, MenuTrigger::Hover);
914 assert_eq!(menu.width, 250.0);
915 }
916
917 #[test]
918 fn test_menu_add_item() {
919 let menu = Menu::new()
920 .item(MenuItem::action("Cut", "cut"))
921 .item(MenuItem::action("Copy", "copy"));
922 assert_eq!(menu.items.len(), 2);
923 }
924
925 #[test]
926 fn test_menu_show_hide() {
927 let mut menu = Menu::new();
928 assert!(!menu.is_open());
929
930 menu.show();
931 assert!(menu.is_open());
932
933 menu.hide();
934 assert!(!menu.is_open());
935 }
936
937 #[test]
938 fn test_menu_toggle() {
939 let mut menu = Menu::new();
940
941 menu.toggle();
942 assert!(menu.is_open());
943
944 menu.toggle();
945 assert!(!menu.is_open());
946 }
947
948 #[test]
949 fn test_menu_calculate_height() {
950 let menu = Menu::new().items(vec![
951 MenuItem::action("Cut", "cut"),
952 MenuItem::separator(),
953 MenuItem::action("Paste", "paste"),
954 ]);
955 assert_eq!(menu.calculate_menu_height(), 89.0);
957 }
958
959 #[test]
960 fn test_menu_measure() {
961 let menu = Menu::new().width(200.0);
962 let size = menu.measure(Constraints::loose(Size::new(300.0, 400.0)));
963 assert_eq!(size.width, 200.0);
964 }
965
966 #[test]
967 fn test_menu_layout() {
968 let mut menu = Menu::new().width(200.0);
969 menu.open = true;
970 menu.items = vec![MenuItem::action("Cut", "cut")];
971
972 let result = menu.layout(Rect::new(10.0, 20.0, 100.0, 32.0));
973 assert_eq!(result.size, Size::new(100.0, 32.0));
974 assert_eq!(menu.panel_bounds.x, 10.0);
975 assert_eq!(menu.panel_bounds.y, 52.0); }
977
978 #[test]
979 fn test_menu_type_id() {
980 let menu = Menu::new();
981 assert_eq!(Widget::type_id(&menu), TypeId::of::<Menu>());
982 }
983
984 #[test]
985 fn test_menu_is_focusable() {
986 let menu = Menu::new();
987 assert!(menu.is_focusable());
988 }
989
990 #[test]
991 fn test_menu_test_id() {
992 let menu = Menu::new().with_test_id("my-menu");
993 assert_eq!(Widget::test_id(&menu), Some("my-menu"));
994 }
995
996 #[test]
997 fn test_menu_highlighted_index() {
998 let mut menu = Menu::new();
999 assert!(menu.highlighted_index().is_none());
1000
1001 menu.highlighted_index = Some(2);
1002 assert_eq!(menu.highlighted_index(), Some(2));
1003 }
1004
1005 #[test]
1006 fn test_menu_next_selectable() {
1007 let menu = Menu::new().items(vec![
1008 MenuItem::action("Cut", "cut"),
1009 MenuItem::separator(),
1010 MenuItem::action("Paste", "paste"),
1011 ]);
1012
1013 assert_eq!(menu.next_selectable(None, true), Some(0));
1015
1016 assert_eq!(menu.next_selectable(Some(0), true), Some(2)); assert_eq!(menu.next_selectable(Some(2), false), Some(0)); }
1022
1023 #[test]
1024 fn test_menu_escape_closes() {
1025 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1026 menu.show();
1027
1028 let result = menu.event(&Event::KeyDown { key: Key::Escape });
1029 assert!(result.is_some());
1030 assert!(!menu.is_open());
1031 }
1032
1033 #[test]
1034 fn test_menu_arrow_navigation() {
1035 let mut menu = Menu::new().items(vec![
1036 MenuItem::action("Cut", "cut"),
1037 MenuItem::action("Copy", "copy"),
1038 ]);
1039 menu.show();
1040
1041 menu.event(&Event::KeyDown { key: Key::Down });
1042 assert_eq!(menu.highlighted_index, Some(0));
1043
1044 menu.event(&Event::KeyDown { key: Key::Down });
1045 assert_eq!(menu.highlighted_index, Some(1));
1046 }
1047
1048 #[test]
1053 fn test_menu_toggled_message() {
1054 let msg = MenuToggled { open: true };
1055 assert!(msg.open);
1056 }
1057
1058 #[test]
1059 fn test_menu_item_selected_message() {
1060 let msg = MenuItemSelected {
1061 action: "edit.cut".to_string(),
1062 };
1063 assert_eq!(msg.action, "edit.cut");
1064 }
1065
1066 #[test]
1067 fn test_menu_checkbox_toggled_message() {
1068 let msg = MenuCheckboxToggled {
1069 action: "view.grid".to_string(),
1070 checked: true,
1071 };
1072 assert_eq!(msg.action, "view.grid");
1073 assert!(msg.checked);
1074 }
1075
1076 #[test]
1077 fn test_menu_closed_message() {
1078 let _msg = MenuClosed;
1079 }
1080
1081 #[test]
1086 fn test_menu_shortcut_on_non_action() {
1087 let item = MenuItem::checkbox("Show", "show", false).shortcut("Ctrl+S");
1089 match item {
1090 MenuItem::Checkbox { .. } => {} _ => panic!("Expected Checkbox"),
1092 }
1093 }
1094
1095 #[test]
1096 fn test_menu_disabled_checkbox() {
1097 let item = MenuItem::checkbox("Show", "show", true).disabled(true);
1098 match item {
1099 MenuItem::Checkbox { disabled, .. } => assert!(disabled),
1100 _ => panic!("Expected Checkbox"),
1101 }
1102 }
1103
1104 #[test]
1105 fn test_menu_disabled_submenu() {
1106 let item = MenuItem::submenu("More", vec![]).disabled(true);
1107 match item {
1108 MenuItem::Submenu { disabled, .. } => assert!(disabled),
1109 _ => panic!("Expected Submenu"),
1110 }
1111 }
1112
1113 #[test]
1114 fn test_menu_disabled_separator_no_op() {
1115 let item = MenuItem::separator().disabled(true);
1117 assert!(matches!(item, MenuItem::Separator));
1118 }
1119
1120 #[test]
1121 fn test_menu_submenu_not_selectable_when_disabled() {
1122 let item = MenuItem::submenu("More", vec![]).disabled(true);
1123 assert!(!item.is_selectable());
1124 }
1125
1126 #[test]
1127 fn test_menu_context_menu_trigger() {
1128 let menu = Menu::new().trigger(MenuTrigger::ContextMenu);
1129 assert_eq!(menu.trigger, MenuTrigger::ContextMenu);
1130 }
1131
1132 #[test]
1133 fn test_menu_hover_trigger() {
1134 let menu = Menu::new().trigger(MenuTrigger::Hover);
1135 assert_eq!(menu.trigger, MenuTrigger::Hover);
1136 }
1137
1138 #[test]
1139 fn test_menu_background_color() {
1140 let menu = Menu::new().background_color(Color::RED);
1141 assert_eq!(menu.background_color, Color::RED);
1142 }
1143
1144 #[test]
1145 fn test_menu_hover_color() {
1146 let menu = Menu::new().hover_color(Color::BLUE);
1147 assert_eq!(menu.hover_color, Color::BLUE);
1148 }
1149
1150 #[test]
1151 fn test_menu_text_color() {
1152 let menu = Menu::new().text_color(Color::GREEN);
1153 assert_eq!(menu.text_color, Color::GREEN);
1154 }
1155
1156 #[test]
1157 fn test_menu_next_selectable_empty() {
1158 let menu = Menu::new();
1159 assert!(menu.next_selectable(None, true).is_none());
1160 assert!(menu.next_selectable(None, false).is_none());
1161 }
1162
1163 #[test]
1164 fn test_menu_next_selectable_all_disabled() {
1165 let menu = Menu::new().items(vec![
1166 MenuItem::separator(),
1167 MenuItem::action("Cut", "cut").disabled(true),
1168 MenuItem::separator(),
1169 ]);
1170 assert!(menu.next_selectable(None, true).is_none());
1171 }
1172
1173 #[test]
1174 fn test_menu_next_selectable_wrap_forward() {
1175 let menu = Menu::new().items(vec![MenuItem::action("A", "a"), MenuItem::action("B", "b")]);
1176 assert_eq!(menu.next_selectable(Some(1), true), Some(0));
1178 }
1179
1180 #[test]
1181 fn test_menu_next_selectable_wrap_backward() {
1182 let menu = Menu::new().items(vec![MenuItem::action("A", "a"), MenuItem::action("B", "b")]);
1183 assert_eq!(menu.next_selectable(Some(0), false), Some(1));
1185 }
1186
1187 #[test]
1188 fn test_menu_children_empty() {
1189 let menu = Menu::new();
1190 assert!(menu.children().is_empty());
1191 }
1192
1193 #[test]
1194 fn test_menu_children_mut_empty() {
1195 let mut menu = Menu::new();
1196 assert!(menu.children_mut().is_empty());
1197 }
1198
1199 #[test]
1200 fn test_menu_bounds() {
1201 let mut menu = Menu::new();
1202 menu.layout(Rect::new(10.0, 20.0, 200.0, 32.0));
1203 assert_eq!(menu.bounds(), Rect::new(10.0, 20.0, 200.0, 32.0));
1204 }
1205
1206 #[test]
1207 fn test_menu_trigger_default() {
1208 assert_eq!(MenuTrigger::default(), MenuTrigger::Click);
1209 }
1210
1211 #[test]
1212 fn test_menu_event_closed_returns_none() {
1213 let mut menu = Menu::new();
1214 let result = menu.event(&Event::KeyDown { key: Key::Down });
1216 assert!(result.is_none());
1217 }
1218
1219 #[test]
1220 fn test_menu_enter_selects_item() {
1221 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1222 menu.show();
1223 menu.highlighted_index = Some(0);
1224 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1225
1226 let result = menu.event(&Event::KeyDown { key: Key::Enter });
1227 assert!(result.is_some());
1228 assert!(!menu.is_open());
1229 }
1230
1231 #[test]
1232 fn test_menu_space_selects_item() {
1233 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1234 menu.show();
1235 menu.highlighted_index = Some(0);
1236 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1237
1238 let result = menu.event(&Event::KeyDown { key: Key::Space });
1239 assert!(result.is_some());
1240 }
1241
1242 #[test]
1243 fn test_menu_enter_on_checkbox_toggles() {
1244 let mut menu = Menu::new().items(vec![MenuItem::checkbox("Show", "show", false)]);
1245 menu.show();
1246 menu.highlighted_index = Some(0);
1247 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1248
1249 let result = menu.event(&Event::KeyDown { key: Key::Enter });
1250 assert!(result.is_some());
1251 assert!(menu.is_open());
1253 }
1254
1255 #[test]
1256 fn test_menu_enter_on_disabled_does_nothing() {
1257 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut").disabled(true)]);
1258 menu.show();
1259 menu.highlighted_index = Some(0);
1260 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1261
1262 let result = menu.event(&Event::KeyDown { key: Key::Enter });
1263 assert!(result.is_none());
1264 assert!(menu.is_open()); }
1266
1267 #[test]
1268 fn test_menu_up_arrow_navigation() {
1269 let mut menu =
1270 Menu::new().items(vec![MenuItem::action("A", "a"), MenuItem::action("B", "b")]);
1271 menu.show();
1272 menu.highlighted_index = Some(1);
1273
1274 menu.event(&Event::KeyDown { key: Key::Up });
1275 assert_eq!(menu.highlighted_index, Some(0));
1276 }
1277
1278 #[test]
1283 fn test_menu_brick_name() {
1284 let menu = Menu::new();
1285 assert_eq!(menu.brick_name(), "Menu");
1286 }
1287
1288 #[test]
1289 fn test_menu_brick_assertions() {
1290 let menu = Menu::new();
1291 let assertions = menu.assertions();
1292 assert!(!assertions.is_empty());
1293 assert!(matches!(assertions[0], BrickAssertion::MaxLatencyMs(16)));
1294 }
1295
1296 #[test]
1297 fn test_menu_brick_budget() {
1298 let menu = Menu::new();
1299 let budget = menu.budget();
1300 assert!(budget.layout_ms > 0);
1302 assert!(budget.paint_ms > 0);
1303 }
1304
1305 #[test]
1306 fn test_menu_brick_verify() {
1307 let menu = Menu::new();
1308 let verification = menu.verify();
1309 assert!(!verification.passed.is_empty());
1310 assert!(verification.failed.is_empty());
1311 }
1312
1313 #[test]
1314 fn test_menu_brick_to_html() {
1315 let menu = Menu::new();
1316 let html = menu.to_html();
1317 assert!(html.contains("brick-menu"));
1318 }
1319
1320 #[test]
1321 fn test_menu_brick_to_css() {
1322 let menu = Menu::new();
1323 let css = menu.to_css();
1324 assert!(css.contains(".brick-menu"));
1325 assert!(css.contains("display: block"));
1326 assert!(css.contains("position: relative"));
1327 }
1328
1329 #[test]
1330 fn test_menu_brick_test_id() {
1331 let menu = Menu::new().with_test_id("my-menu");
1332 assert_eq!(Brick::test_id(&menu), Some("my-menu"));
1333 }
1334
1335 #[test]
1336 fn test_menu_brick_test_id_none() {
1337 let menu = Menu::new();
1338 assert!(Brick::test_id(&menu).is_none());
1339 }
1340
1341 #[test]
1346 fn test_menu_item_at_position_valid() {
1347 let mut menu = Menu::new().items(vec![
1348 MenuItem::action("Cut", "cut"),
1349 MenuItem::action("Copy", "copy"),
1350 MenuItem::action("Paste", "paste"),
1351 ]);
1352 menu.open = true;
1353 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1354
1355 let item = menu.item_at_position(menu.panel_bounds.y + 8.0 + 10.0);
1358 assert_eq!(item, Some(0));
1359
1360 let item = menu.item_at_position(menu.panel_bounds.y + 8.0 + 40.0);
1361 assert_eq!(item, Some(1));
1362
1363 let item = menu.item_at_position(menu.panel_bounds.y + 8.0 + 72.0);
1364 assert_eq!(item, Some(2));
1365 }
1366
1367 #[test]
1368 fn test_menu_item_at_position_above_menu() {
1369 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1370 menu.open = true;
1371 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1372
1373 let item = menu.item_at_position(menu.panel_bounds.y - 10.0);
1375 assert!(item.is_none());
1376 }
1377
1378 #[test]
1379 fn test_menu_item_at_position_below_items() {
1380 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1381 menu.open = true;
1382 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1383
1384 let item = menu.item_at_position(menu.panel_bounds.y + 500.0);
1386 assert!(item.is_none());
1387 }
1388
1389 #[test]
1394 fn test_menu_click_on_trigger_opens() {
1395 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1396 menu.layout(Rect::new(10.0, 10.0, 200.0, 32.0));
1397
1398 let result = menu.event(&Event::MouseDown {
1400 position: Point::new(50.0, 20.0),
1401 button: presentar_core::MouseButton::Left,
1402 });
1403
1404 assert!(result.is_some());
1405 assert!(menu.is_open());
1406 }
1407
1408 #[test]
1409 fn test_menu_click_outside_closes() {
1410 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1411 menu.show();
1412 menu.layout(Rect::new(10.0, 10.0, 200.0, 32.0));
1413
1414 let result = menu.event(&Event::MouseDown {
1416 position: Point::new(500.0, 500.0),
1417 button: presentar_core::MouseButton::Left,
1418 });
1419
1420 assert!(result.is_some());
1421 assert!(!menu.is_open());
1422 }
1423
1424 #[test]
1425 fn test_menu_click_on_action_item() {
1426 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1427 menu.show();
1428 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1429
1430 let click_y = menu.panel_bounds.y + 8.0 + 16.0; let result = menu.event(&Event::MouseDown {
1433 position: Point::new(menu.panel_bounds.x + 50.0, click_y),
1434 button: presentar_core::MouseButton::Left,
1435 });
1436
1437 assert!(result.is_some());
1438 assert!(!menu.is_open()); }
1440
1441 #[test]
1442 fn test_menu_click_on_checkbox_item() {
1443 let mut menu = Menu::new().items(vec![MenuItem::checkbox("Show Grid", "show", false)]);
1444 menu.show();
1445 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1446
1447 let click_y = menu.panel_bounds.y + 8.0 + 16.0;
1449 let result = menu.event(&Event::MouseDown {
1450 position: Point::new(menu.panel_bounds.x + 50.0, click_y),
1451 button: presentar_core::MouseButton::Left,
1452 });
1453
1454 assert!(result.is_some());
1455 if let MenuItem::Checkbox { checked, .. } = &menu.items[0] {
1457 assert!(*checked);
1458 } else {
1459 panic!("Expected Checkbox item");
1460 }
1461 }
1462
1463 #[test]
1464 fn test_menu_click_on_disabled_action() {
1465 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut").disabled(true)]);
1466 menu.show();
1467 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1468
1469 let click_y = menu.panel_bounds.y + 8.0 + 16.0;
1471 let result = menu.event(&Event::MouseDown {
1472 position: Point::new(menu.panel_bounds.x + 50.0, click_y),
1473 button: presentar_core::MouseButton::Left,
1474 });
1475
1476 assert!(result.is_none()); assert!(menu.is_open()); }
1479
1480 #[test]
1481 fn test_menu_click_on_submenu_opens_it() {
1482 let mut menu = Menu::new().items(vec![MenuItem::submenu(
1483 "More",
1484 vec![MenuItem::action("Sub", "sub")],
1485 )]);
1486 menu.show();
1487 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1488
1489 let click_y = menu.panel_bounds.y + 8.0 + 16.0;
1491 let result = menu.event(&Event::MouseDown {
1492 position: Point::new(menu.panel_bounds.x + 50.0, click_y),
1493 button: presentar_core::MouseButton::Left,
1494 });
1495
1496 assert!(result.is_none()); assert_eq!(menu.open_submenu, Some(0));
1498 }
1499
1500 #[test]
1501 fn test_menu_mouse_move_updates_highlight() {
1502 let mut menu = Menu::new().items(vec![
1503 MenuItem::action("Cut", "cut"),
1504 MenuItem::action("Copy", "copy"),
1505 ]);
1506 menu.show();
1507 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1508
1509 let y1 = menu.panel_bounds.y + 8.0 + 16.0;
1511 menu.event(&Event::MouseMove {
1512 position: Point::new(menu.panel_bounds.x + 50.0, y1),
1513 });
1514 assert_eq!(menu.highlighted_index, Some(0));
1515
1516 let y2 = menu.panel_bounds.y + 8.0 + 48.0;
1518 menu.event(&Event::MouseMove {
1519 position: Point::new(menu.panel_bounds.x + 50.0, y2),
1520 });
1521 assert_eq!(menu.highlighted_index, Some(1));
1522 }
1523
1524 #[test]
1525 fn test_menu_mouse_move_outside_clears_highlight() {
1526 let mut menu = Menu::new().items(vec![MenuItem::action("Cut", "cut")]);
1527 menu.show();
1528 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1529 menu.highlighted_index = Some(0);
1530
1531 menu.event(&Event::MouseMove {
1533 position: Point::new(500.0, 500.0),
1534 });
1535 assert!(menu.highlighted_index.is_none());
1536 }
1537
1538 #[test]
1543 fn test_menu_up_from_none_selects_last() {
1544 let mut menu =
1545 Menu::new().items(vec![MenuItem::action("A", "a"), MenuItem::action("B", "b")]);
1546 menu.show();
1547
1548 menu.event(&Event::KeyDown { key: Key::Up });
1549 assert_eq!(menu.highlighted_index, Some(1)); }
1551
1552 #[test]
1553 fn test_menu_down_from_last_wraps_to_first() {
1554 let mut menu =
1555 Menu::new().items(vec![MenuItem::action("A", "a"), MenuItem::action("B", "b")]);
1556 menu.show();
1557 menu.highlighted_index = Some(1);
1558
1559 menu.event(&Event::KeyDown { key: Key::Down });
1560 assert_eq!(menu.highlighted_index, Some(0)); }
1562
1563 #[test]
1564 fn test_menu_up_skips_separator() {
1565 let mut menu = Menu::new().items(vec![
1566 MenuItem::action("A", "a"),
1567 MenuItem::separator(),
1568 MenuItem::action("B", "b"),
1569 ]);
1570 menu.show();
1571 menu.highlighted_index = Some(2);
1572
1573 menu.event(&Event::KeyDown { key: Key::Up });
1574 assert_eq!(menu.highlighted_index, Some(0)); }
1576
1577 #[test]
1578 fn test_menu_down_skips_disabled() {
1579 let mut menu = Menu::new().items(vec![
1580 MenuItem::action("A", "a"),
1581 MenuItem::action("B", "b").disabled(true),
1582 MenuItem::action("C", "c"),
1583 ]);
1584 menu.show();
1585 menu.highlighted_index = Some(0);
1586
1587 menu.event(&Event::KeyDown { key: Key::Down });
1588 assert_eq!(menu.highlighted_index, Some(2)); }
1590
1591 #[test]
1592 fn test_menu_other_key_does_nothing() {
1593 let mut menu = Menu::new().items(vec![MenuItem::action("A", "a")]);
1594 menu.show();
1595 menu.highlighted_index = Some(0);
1596
1597 let result = menu.event(&Event::KeyDown { key: Key::Tab });
1598 assert!(result.is_none());
1599 assert_eq!(menu.highlighted_index, Some(0));
1600 }
1601
1602 #[test]
1603 fn test_menu_enter_on_separator_does_nothing() {
1604 let mut menu = Menu::new().items(vec![MenuItem::separator(), MenuItem::action("A", "a")]);
1605 menu.show();
1606 menu.highlighted_index = Some(0); let result = menu.event(&Event::KeyDown { key: Key::Enter });
1609 assert!(result.is_none());
1610 assert!(menu.is_open());
1611 }
1612
1613 #[test]
1614 fn test_menu_enter_on_submenu_does_nothing() {
1615 let mut menu = Menu::new().items(vec![MenuItem::submenu(
1616 "More",
1617 vec![MenuItem::action("Sub", "sub")],
1618 )]);
1619 menu.show();
1620 menu.highlighted_index = Some(0);
1621
1622 let result = menu.event(&Event::KeyDown { key: Key::Enter });
1623 assert!(result.is_none());
1624 assert!(menu.is_open());
1625 }
1626
1627 #[test]
1628 fn test_menu_space_on_checkbox_toggles() {
1629 let mut menu = Menu::new().items(vec![MenuItem::checkbox("Show", "show", false)]);
1630 menu.show();
1631 menu.highlighted_index = Some(0);
1632 menu.layout(Rect::new(0.0, 0.0, 200.0, 32.0));
1633
1634 let result = menu.event(&Event::KeyDown { key: Key::Space });
1635 assert!(result.is_some());
1636 if let MenuItem::Checkbox { checked, .. } = &menu.items[0] {
1638 assert!(*checked);
1639 }
1640 }
1641
1642 #[test]
1647 fn test_menu_item_height_action() {
1648 let item = MenuItem::action("Test", "test");
1649 assert_eq!(item.height(), 32.0);
1650 }
1651
1652 #[test]
1653 fn test_menu_item_height_checkbox() {
1654 let item = MenuItem::checkbox("Test", "test", false);
1655 assert_eq!(item.height(), 32.0);
1656 }
1657
1658 #[test]
1659 fn test_menu_item_height_submenu() {
1660 let item = MenuItem::submenu("More", vec![]);
1661 assert_eq!(item.height(), 32.0);
1662 }
1663
1664 #[test]
1665 fn test_menu_item_is_selectable_submenu() {
1666 let item = MenuItem::submenu("More", vec![]);
1667 assert!(item.is_selectable());
1668 }
1669
1670 #[test]
1671 fn test_menu_item_is_selectable_disabled_checkbox() {
1672 let item = MenuItem::checkbox("Test", "test", false).disabled(true);
1673 assert!(!item.is_selectable());
1674 }
1675
1676 #[test]
1681 fn test_menu_trigger_hover_no_click_open() {
1682 let mut menu = Menu::new()
1683 .trigger(MenuTrigger::Hover)
1684 .items(vec![MenuItem::action("Cut", "cut")]);
1685 menu.layout(Rect::new(10.0, 10.0, 200.0, 32.0));
1686
1687 let result = menu.event(&Event::MouseDown {
1689 position: Point::new(50.0, 20.0),
1690 button: presentar_core::MouseButton::Left,
1691 });
1692
1693 assert!(result.is_none());
1694 assert!(!menu.is_open());
1695 }
1696
1697 #[test]
1698 fn test_menu_trigger_context_menu_no_click_open() {
1699 let mut menu = Menu::new()
1700 .trigger(MenuTrigger::ContextMenu)
1701 .items(vec![MenuItem::action("Cut", "cut")]);
1702 menu.layout(Rect::new(10.0, 10.0, 200.0, 32.0));
1703
1704 let result = menu.event(&Event::MouseDown {
1706 position: Point::new(50.0, 20.0),
1707 button: presentar_core::MouseButton::Left,
1708 });
1709
1710 assert!(result.is_none());
1711 assert!(!menu.is_open());
1712 }
1713
1714 #[test]
1719 fn test_menu_toggled_clone() {
1720 let msg = MenuToggled { open: true };
1721 let cloned = msg.clone();
1722 assert_eq!(cloned.open, msg.open);
1723 }
1724
1725 #[test]
1726 fn test_menu_item_selected_clone() {
1727 let msg = MenuItemSelected {
1728 action: "test".to_string(),
1729 };
1730 let cloned = msg.clone();
1731 assert_eq!(cloned.action, msg.action);
1732 }
1733
1734 #[test]
1735 fn test_menu_checkbox_toggled_clone() {
1736 let msg = MenuCheckboxToggled {
1737 action: "test".to_string(),
1738 checked: true,
1739 };
1740 let cloned = msg.clone();
1741 assert_eq!(cloned.action, msg.action);
1742 assert_eq!(cloned.checked, msg.checked);
1743 }
1744
1745 #[test]
1746 fn test_menu_closed_clone() {
1747 let msg = MenuClosed;
1748 let _cloned = msg.clone();
1749 }
1750
1751 #[test]
1756 fn test_menu_default() {
1757 let menu = Menu::default();
1758 assert!(menu.items.is_empty());
1759 assert!(!menu.open);
1760 assert_eq!(menu.trigger, MenuTrigger::Click);
1761 assert_eq!(menu.width, 200.0);
1762 }
1763
1764 #[test]
1765 fn test_menu_trigger_eq() {
1766 assert_eq!(MenuTrigger::Click, MenuTrigger::Click);
1767 assert_ne!(MenuTrigger::Click, MenuTrigger::Hover);
1768 assert_ne!(MenuTrigger::Hover, MenuTrigger::ContextMenu);
1769 }
1770
1771 #[test]
1772 fn test_menu_hide_clears_submenu() {
1773 let mut menu = Menu::new().items(vec![MenuItem::submenu(
1774 "More",
1775 vec![MenuItem::action("Sub", "sub")],
1776 )]);
1777 menu.show();
1778 menu.open_submenu = Some(0);
1779
1780 menu.hide();
1781 assert!(!menu.is_open());
1782 assert!(menu.open_submenu.is_none());
1783 assert!(menu.highlighted_index.is_none());
1784 }
1785
1786 #[test]
1787 fn test_menu_debug() {
1788 let item = MenuItem::action("Test", "test");
1789 let debug_str = format!("{:?}", item);
1790 assert!(debug_str.contains("Test"));
1791 }
1792
1793 #[test]
1794 fn test_menu_toggled_debug() {
1795 let msg = MenuToggled { open: true };
1796 let debug_str = format!("{:?}", msg);
1797 assert!(debug_str.contains("true"));
1798 }
1799}