presentar_widgets/
menu.rs

1//! Menu and dropdown widgets for action lists.
2//!
3//! Provides hierarchical menus with keyboard navigation, submenus,
4//! and various item types (actions, checkable, separators).
5
6use 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/// Menu item variant.
16#[derive(Debug, Clone, Serialize, Deserialize)]
17pub enum MenuItem {
18    /// Action item with label
19    Action {
20        /// Item label
21        label: String,
22        /// Unique action ID
23        action: String,
24        /// Whether item is disabled
25        disabled: bool,
26        /// Optional keyboard shortcut display
27        shortcut: Option<String>,
28    },
29    /// Checkable item
30    Checkbox {
31        /// Item label
32        label: String,
33        /// Action ID
34        action: String,
35        /// Whether checked
36        checked: bool,
37        /// Whether disabled
38        disabled: bool,
39    },
40    /// Separator line
41    Separator,
42    /// Submenu
43    Submenu {
44        /// Submenu label
45        label: String,
46        /// Child items
47        items: Vec<Self>,
48        /// Whether disabled
49        disabled: bool,
50    },
51}
52
53impl MenuItem {
54    /// Create a new action item.
55    #[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    /// Create a new checkbox item.
66    #[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    /// Create a separator.
77    #[must_use]
78    pub const fn separator() -> Self {
79        Self::Separator
80    }
81
82    /// Create a submenu.
83    #[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    /// Set disabled state.
93    #[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    /// Set shortcut (for action items).
105    #[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    /// Check if this item is selectable (not separator, not disabled).
114    #[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    /// Get item height.
125    #[must_use]
126    pub const fn height(&self) -> f32 {
127        match self {
128            Self::Separator => 9.0, // 1px line + 4px padding each side
129            _ => 32.0,
130        }
131    }
132}
133
134/// Menu trigger mode.
135#[derive(Debug, Clone, Copy, PartialEq, Eq, Default, Serialize, Deserialize)]
136pub enum MenuTrigger {
137    /// Click to open
138    #[default]
139    Click,
140    /// Hover to open
141    Hover,
142    /// Right-click (context menu)
143    ContextMenu,
144}
145
146/// Menu widget for dropdown actions.
147#[derive(Serialize, Deserialize)]
148pub struct Menu {
149    /// Menu items
150    pub items: Vec<MenuItem>,
151    /// Whether menu is open
152    pub open: bool,
153    /// Trigger mode
154    pub trigger: MenuTrigger,
155    /// Menu width
156    pub width: f32,
157    /// Background color
158    pub background_color: Color,
159    /// Hover color
160    pub hover_color: Color,
161    /// Text color
162    pub text_color: Color,
163    /// Disabled text color
164    pub disabled_color: Color,
165    /// Test ID
166    test_id_value: Option<String>,
167    /// Cached bounds
168    #[serde(skip)]
169    bounds: Rect,
170    /// Panel bounds (dropdown area)
171    #[serde(skip)]
172    panel_bounds: Rect,
173    /// Currently highlighted index
174    #[serde(skip)]
175    highlighted_index: Option<usize>,
176    /// Open submenu index
177    #[serde(skip)]
178    open_submenu: Option<usize>,
179    /// Trigger widget
180    #[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    /// Create a new menu.
207    #[must_use]
208    pub fn new() -> Self {
209        Self::default()
210    }
211
212    /// Set menu items.
213    #[must_use]
214    pub fn items(mut self, items: Vec<MenuItem>) -> Self {
215        self.items = items;
216        self
217    }
218
219    /// Add a single item.
220    #[must_use]
221    pub fn item(mut self, item: MenuItem) -> Self {
222        self.items.push(item);
223        self
224    }
225
226    /// Set trigger mode.
227    #[must_use]
228    pub const fn trigger(mut self, trigger: MenuTrigger) -> Self {
229        self.trigger = trigger;
230        self
231    }
232
233    /// Set menu width.
234    #[must_use]
235    pub const fn width(mut self, width: f32) -> Self {
236        self.width = width;
237        self
238    }
239
240    /// Set background color.
241    #[must_use]
242    pub const fn background_color(mut self, color: Color) -> Self {
243        self.background_color = color;
244        self
245    }
246
247    /// Set hover color.
248    #[must_use]
249    pub const fn hover_color(mut self, color: Color) -> Self {
250        self.hover_color = color;
251        self
252    }
253
254    /// Set text color.
255    #[must_use]
256    pub const fn text_color(mut self, color: Color) -> Self {
257        self.text_color = color;
258        self
259    }
260
261    /// Set the trigger widget.
262    pub fn trigger_widget(mut self, widget: impl Widget + 'static) -> Self {
263        self.trigger_widget = Some(Box::new(widget));
264        self
265    }
266
267    /// Set test ID.
268    #[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    /// Open the menu.
275    pub fn show(&mut self) {
276        self.open = true;
277        self.highlighted_index = None;
278    }
279
280    /// Close the menu.
281    pub fn hide(&mut self) {
282        self.open = false;
283        self.highlighted_index = None;
284        self.open_submenu = None;
285    }
286
287    /// Toggle the menu.
288    pub fn toggle(&mut self) {
289        if self.open {
290            self.hide();
291        } else {
292            self.show();
293        }
294    }
295
296    /// Check if menu is open.
297    #[must_use]
298    pub const fn is_open(&self) -> bool {
299        self.open
300    }
301
302    /// Get highlighted index.
303    #[must_use]
304    pub const fn highlighted_index(&self) -> Option<usize> {
305        self.highlighted_index
306    }
307
308    /// Calculate total menu height.
309    fn calculate_menu_height(&self) -> f32 {
310        let padding = 8.0; // Top and bottom padding
311        let items_height: f32 = self.items.iter().map(MenuItem::height).sum();
312        items_height + padding * 2.0
313    }
314
315    /// Find next selectable item.
316    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    /// Get item at y position.
362    fn item_at_position(&self, y: f32) -> Option<usize> {
363        let relative_y = y - self.panel_bounds.y - 8.0; // Subtract padding
364        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        // Measure just the trigger area
388        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        // Layout trigger widget
399        if let Some(ref mut trigger) = self.trigger_widget {
400            trigger.layout(bounds);
401        }
402
403        // Calculate menu panel bounds (below trigger)
404        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        // Paint trigger
418        if let Some(ref trigger) = self.trigger_widget {
419            trigger.paint(canvas);
420        }
421
422        if !self.open {
423            return;
424        }
425
426        // Paint menu background with shadow
427        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        // Paint items
437        let mut y = self.panel_bounds.y + 8.0; // Top padding
438        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                    // Hover background
460                    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                    // Label
467                    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                    // Shortcut
479                    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                    // Hover background
502                    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                    // Checkbox indicator
509                    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                    // Label
522                    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                    // Hover background
541                    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                    // Label
548                    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                    // Arrow indicator
560                    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                // Check if click is on trigger
580                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                // Check if click is on menu item
591                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                        // Click outside menu - close
633                        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
722// PROBAR-SPEC-009: Brick Architecture - Tests define interface
723impl 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/// Message when menu is toggled.
758#[derive(Debug, Clone)]
759pub struct MenuToggled {
760    /// Whether menu is now open
761    pub open: bool,
762}
763
764/// Message when menu item is selected.
765#[derive(Debug, Clone)]
766pub struct MenuItemSelected {
767    /// Action ID of selected item
768    pub action: String,
769}
770
771/// Message when menu checkbox is toggled.
772#[derive(Debug, Clone)]
773pub struct MenuCheckboxToggled {
774    /// Action ID
775    pub action: String,
776    /// New checked state
777    pub checked: bool,
778}
779
780/// Message when menu is closed.
781#[derive(Debug, Clone)]
782pub struct MenuClosed;
783
784#[cfg(test)]
785mod tests {
786    use super::*;
787
788    // =========================================================================
789    // MenuItem Tests
790    // =========================================================================
791
792    #[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    // =========================================================================
890    // Menu Tests
891    // =========================================================================
892
893    #[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        // 2 actions (32px each) + 1 separator (9px) + padding (16px) = 89px
956        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); // Below trigger
976    }
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        // Forward from None
1014        assert_eq!(menu.next_selectable(None, true), Some(0));
1015
1016        // Forward from 0
1017        assert_eq!(menu.next_selectable(Some(0), true), Some(2)); // Skips separator
1018
1019        // Backward from 2
1020        assert_eq!(menu.next_selectable(Some(2), false), Some(0)); // Skips separator
1021    }
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    // =========================================================================
1049    // Message Tests
1050    // =========================================================================
1051
1052    #[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    // =========================================================================
1082    // Additional Coverage Tests
1083    // =========================================================================
1084
1085    #[test]
1086    fn test_menu_shortcut_on_non_action() {
1087        // shortcut() should do nothing on checkbox items
1088        let item = MenuItem::checkbox("Show", "show", false).shortcut("Ctrl+S");
1089        match item {
1090            MenuItem::Checkbox { .. } => {} // Still checkbox, shortcut not applied
1091            _ => 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        // Calling disabled on separator should be a no-op
1116        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        // From last item, should wrap to first
1177        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        // From first item, should wrap to last
1184        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        // Event on closed menu should return None
1215        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        // Menu stays open for checkbox
1252        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()); // Still open
1265    }
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    // =========================================================================
1279    // Brick Trait Tests
1280    // =========================================================================
1281
1282    #[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        // Verify budget has reasonable values
1301        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    // =========================================================================
1342    // Item at Position Tests
1343    // =========================================================================
1344
1345    #[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        // Item 0 starts at y = panel_bounds.y + 8.0
1356        // Each action item is 32px tall
1357        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        // Y position above the menu
1374        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        // Y position far below the item
1385        let item = menu.item_at_position(menu.panel_bounds.y + 500.0);
1386        assert!(item.is_none());
1387    }
1388
1389    // =========================================================================
1390    // Mouse Event Tests
1391    // =========================================================================
1392
1393    #[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        // Click inside trigger area
1399        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        // Click far outside menu area
1415        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        // Click on the action item
1431        let click_y = menu.panel_bounds.y + 8.0 + 16.0; // Middle of first item
1432        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()); // Menu closes after action
1439    }
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        // Click on the checkbox item
1448        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        // Checkbox should be toggled
1456        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        // Click on the disabled item
1470        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()); // No action taken
1477        assert!(menu.is_open()); // Menu stays open
1478    }
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        // Click on the submenu item
1490        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()); // No message, just opens submenu
1497        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        // Move over first item
1510        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        // Move over second item
1517        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        // Move outside menu
1532        menu.event(&Event::MouseMove {
1533            position: Point::new(500.0, 500.0),
1534        });
1535        assert!(menu.highlighted_index.is_none());
1536    }
1537
1538    // =========================================================================
1539    // Keyboard Navigation Edge Cases
1540    // =========================================================================
1541
1542    #[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)); // Wraps to last
1550    }
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)); // Wraps to first
1561    }
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)); // Skips separator
1575    }
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)); // Skips disabled
1589    }
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); // Manually set to separator (shouldn't happen in practice)
1607
1608        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        // Checkbox should be toggled
1637        if let MenuItem::Checkbox { checked, .. } = &menu.items[0] {
1638            assert!(*checked);
1639        }
1640    }
1641
1642    // =========================================================================
1643    // MenuItem Methods Tests
1644    // =========================================================================
1645
1646    #[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    // =========================================================================
1677    // Menu Trigger Tests
1678    // =========================================================================
1679
1680    #[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        // Click with hover trigger should not toggle
1688        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        // Regular click with context trigger should not toggle
1705        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    // =========================================================================
1715    // Message Clone Tests
1716    // =========================================================================
1717
1718    #[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    // =========================================================================
1752    // Default Trait Tests
1753    // =========================================================================
1754
1755    #[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}