1use crate::style::StyleModifier;
12use crate::{
13 Button, Color32, Context, Frame, Id, InnerResponse, IntoAtoms, Layout, Popup,
14 PopupCloseBehavior, Response, Style, Ui, UiBuilder, UiKind, UiStack, UiStackInfo, Widget as _,
15};
16use emath::{Align, RectAlign, Vec2, vec2};
17use epaint::Stroke;
18
19pub fn menu_style(style: &mut Style) {
23 style.spacing.button_padding = vec2(2.0, 0.0);
24 style.visuals.widgets.active.bg_stroke = Stroke::NONE;
25 style.visuals.widgets.open.bg_stroke = Stroke::NONE;
26 style.visuals.widgets.hovered.bg_stroke = Stroke::NONE;
27 style.visuals.widgets.inactive.weak_bg_fill = Color32::TRANSPARENT;
28 style.visuals.widgets.inactive.bg_stroke = Stroke::NONE;
29}
30
31pub fn find_menu_root(ui: &Ui) -> &UiStack {
33 ui.stack()
34 .iter()
35 .find(|stack| {
36 stack.is_root_ui()
37 || [Some(UiKind::Popup), Some(UiKind::Menu)].contains(&stack.kind())
38 || stack.info.tags.contains(MenuConfig::MENU_CONFIG_TAG)
39 })
40 .expect("We should always find the root")
41}
42
43pub fn is_in_menu(ui: &Ui) -> bool {
48 for stack in ui.stack().iter() {
49 if let Some(config) = stack
50 .info
51 .tags
52 .get_downcast::<MenuConfig>(MenuConfig::MENU_CONFIG_TAG)
53 {
54 return !config.bar;
55 }
56 if [Some(UiKind::Popup), Some(UiKind::Menu)].contains(&stack.kind()) {
57 return true;
58 }
59 }
60 false
61}
62
63#[derive(Clone, Debug)]
65pub struct MenuConfig {
66 bar: bool,
68
69 pub close_behavior: PopupCloseBehavior,
71
72 pub style: StyleModifier,
76}
77
78impl Default for MenuConfig {
79 fn default() -> Self {
80 Self {
81 close_behavior: PopupCloseBehavior::default(),
82 bar: false,
83 style: menu_style.into(),
84 }
85 }
86}
87
88impl MenuConfig {
89 pub const MENU_CONFIG_TAG: &'static str = "egui_menu_config";
91
92 pub fn new() -> Self {
93 Self::default()
94 }
95
96 #[inline]
98 pub fn close_behavior(mut self, close_behavior: PopupCloseBehavior) -> Self {
99 self.close_behavior = close_behavior;
100 self
101 }
102
103 #[inline]
107 pub fn style(mut self, style: impl Into<StyleModifier>) -> Self {
108 self.style = style.into();
109 self
110 }
111
112 fn from_stack(stack: &UiStack) -> Self {
113 stack
114 .info
115 .tags
116 .get_downcast(Self::MENU_CONFIG_TAG)
117 .cloned()
118 .unwrap_or_default()
119 }
120
121 pub fn find(ui: &Ui) -> Self {
125 find_menu_root(ui)
126 .info
127 .tags
128 .get_downcast(Self::MENU_CONFIG_TAG)
129 .cloned()
130 .unwrap_or_default()
131 }
132}
133
134#[derive(Clone)]
136pub struct MenuState {
137 pub open_item: Option<Id>,
139 last_visible_pass: u64,
140}
141
142impl MenuState {
143 pub const ID: &'static str = "menu_state";
144
145 pub fn from_ui<R>(ui: &Ui, f: impl FnOnce(&mut Self, &UiStack) -> R) -> R {
147 let stack = find_menu_root(ui);
148 Self::from_id(ui.ctx(), stack.id, |state| f(state, stack))
149 }
150
151 pub fn from_id<R>(ctx: &Context, id: Id, f: impl FnOnce(&mut Self) -> R) -> R {
153 let pass_nr = ctx.cumulative_pass_nr();
154 ctx.data_mut(|data| {
155 let state_id = id.with(Self::ID);
156 let mut state = data.get_temp(state_id).unwrap_or(Self {
157 open_item: None,
158 last_visible_pass: pass_nr,
159 });
160 if state.last_visible_pass + 1 < pass_nr {
162 state.open_item = None;
163 }
164 if let Some(item) = state.open_item {
165 if data
166 .get_temp(item.with(Self::ID))
167 .is_none_or(|item: Self| item.last_visible_pass + 1 < pass_nr)
168 {
169 state.open_item = None;
171 }
172 }
173 let r = f(&mut state);
174 data.insert_temp(state_id, state);
175 r
176 })
177 }
178
179 pub fn mark_shown(ctx: &Context, id: Id) {
180 let pass_nr = ctx.cumulative_pass_nr();
181 Self::from_id(ctx, id, |state| {
182 state.last_visible_pass = pass_nr;
183 });
184 }
185
186 pub fn is_deepest_open_sub_menu(ctx: &Context, id: Id) -> bool {
190 let pass_nr = ctx.cumulative_pass_nr();
191 let open_item = Self::from_id(ctx, id, |state| state.open_item);
192 open_item.is_none_or(|submenu_id| {
194 Self::from_id(ctx, submenu_id, |state| state.last_visible_pass != pass_nr)
195 })
196 }
197}
198
199#[derive(Clone, Debug)]
218pub struct MenuBar {
219 config: MenuConfig,
220 style: StyleModifier,
221}
222
223#[deprecated = "Renamed to `egui::MenuBar`"]
224pub type Bar = MenuBar;
225
226impl Default for MenuBar {
227 fn default() -> Self {
228 Self {
229 config: MenuConfig::default(),
230 style: menu_style.into(),
231 }
232 }
233}
234
235impl MenuBar {
236 pub fn new() -> Self {
237 Self::default()
238 }
239
240 #[inline]
245 pub fn style(mut self, style: impl Into<StyleModifier>) -> Self {
246 self.style = style.into();
247 self
248 }
249
250 #[inline]
254 pub fn config(mut self, config: MenuConfig) -> Self {
255 self.config = config;
256 self
257 }
258
259 #[inline]
261 pub fn ui<R>(self, ui: &mut Ui, content: impl FnOnce(&mut Ui) -> R) -> InnerResponse<R> {
262 let Self { mut config, style } = self;
263 config.bar = true;
264 ui.horizontal(|ui| {
267 ui.scope_builder(
268 UiBuilder::new()
269 .layout(Layout::left_to_right(Align::Center))
270 .ui_stack_info(
271 UiStackInfo::new(UiKind::Menu)
272 .with_tag_value(MenuConfig::MENU_CONFIG_TAG, config),
273 ),
274 |ui| {
275 style.apply(ui.style_mut());
276
277 let height = ui.spacing().interact_size.y;
279 ui.set_min_size(vec2(ui.available_width(), height));
280
281 content(ui)
282 },
283 )
284 .inner
285 })
286 }
287}
288
289pub struct MenuButton<'a> {
295 pub button: Button<'a>,
296 pub config: Option<MenuConfig>,
297}
298
299impl<'a> MenuButton<'a> {
300 pub fn new(atoms: impl IntoAtoms<'a>) -> Self {
301 Self::from_button(Button::new(atoms.into_atoms()))
302 }
303
304 #[inline]
306 pub fn config(mut self, config: MenuConfig) -> Self {
307 self.config = Some(config);
308 self
309 }
310
311 #[inline]
313 pub fn from_button(button: Button<'a>) -> Self {
314 Self {
315 button,
316 config: None,
317 }
318 }
319
320 pub fn ui<R>(
322 self,
323 ui: &mut Ui,
324 content: impl FnOnce(&mut Ui) -> R,
325 ) -> (Response, Option<InnerResponse<R>>) {
326 let response = self.button.ui(ui);
327 let mut config = self.config.unwrap_or_else(|| MenuConfig::find(ui));
328 config.bar = false;
329 let inner = Popup::menu(&response)
330 .close_behavior(config.close_behavior)
331 .style(config.style.clone())
332 .info(
333 UiStackInfo::new(UiKind::Menu).with_tag_value(MenuConfig::MENU_CONFIG_TAG, config),
334 )
335 .show(content);
336 (response, inner)
337 }
338}
339
340pub struct SubMenuButton<'a> {
342 pub button: Button<'a>,
343 pub sub_menu: SubMenu,
344}
345
346impl<'a> SubMenuButton<'a> {
347 pub const RIGHT_ARROW: &'static str = "⏵";
349
350 pub fn new(atoms: impl IntoAtoms<'a>) -> Self {
351 Self::from_button(Button::new(atoms.into_atoms()).right_text("⏵"))
352 }
353
354 pub fn from_button(button: Button<'a>) -> Self {
359 Self {
360 button,
361 sub_menu: SubMenu::default(),
362 }
363 }
364
365 #[inline]
369 pub fn config(mut self, config: MenuConfig) -> Self {
370 self.sub_menu.config = Some(config);
371 self
372 }
373
374 pub fn ui<R>(
376 self,
377 ui: &mut Ui,
378 content: impl FnOnce(&mut Ui) -> R,
379 ) -> (Response, Option<InnerResponse<R>>) {
380 let my_id = ui.next_auto_id();
381 let open = MenuState::from_ui(ui, |state, _| {
382 state.open_item == Some(SubMenu::id_from_widget_id(my_id))
383 });
384 let inactive = ui.style().visuals.widgets.inactive;
385 if open {
387 ui.style_mut().visuals.widgets.inactive = ui.style().visuals.widgets.open;
388 }
389 let response = self.button.ui(ui);
390 ui.style_mut().visuals.widgets.inactive = inactive;
391
392 let popup_response = self.sub_menu.show(ui, &response, content);
393
394 (response, popup_response)
395 }
396}
397
398#[derive(Clone, Debug, Default)]
403pub struct SubMenu {
404 config: Option<MenuConfig>,
405}
406
407impl SubMenu {
408 pub fn new() -> Self {
409 Self::default()
410 }
411
412 #[inline]
416 pub fn config(mut self, config: MenuConfig) -> Self {
417 self.config = Some(config);
418 self
419 }
420
421 pub fn id_from_widget_id(widget_id: Id) -> Id {
423 widget_id.with("submenu")
424 }
425
426 pub fn show<R>(
431 self,
432 ui: &Ui,
433 button_response: &Response,
434 content: impl FnOnce(&mut Ui) -> R,
435 ) -> Option<InnerResponse<R>> {
436 let frame = Frame::menu(ui.style());
437
438 let id = Self::id_from_widget_id(button_response.id);
439
440 let (open_item, menu_id, parent_config) = MenuState::from_ui(ui, |state, stack| {
442 (state.open_item, stack.id, MenuConfig::from_stack(stack))
443 });
444
445 let mut menu_config = self.config.unwrap_or_else(|| parent_config.clone());
446 menu_config.bar = false;
447
448 let menu_root_response = ui
449 .ctx()
450 .read_response(menu_id)
451 .unwrap();
453
454 let hover_pos = ui.ctx().pointer_hover_pos();
455
456 let menu_rect = menu_root_response.rect - frame.total_margin();
458 let is_hovering_menu = hover_pos.is_some_and(|pos| {
459 ui.ctx().layer_id_at(pos) == Some(menu_root_response.layer_id)
460 && menu_rect.contains(pos)
461 });
462
463 let is_any_open = open_item.is_some();
464 let mut is_open = open_item == Some(id);
465 let mut set_open = None;
466
467 let button_rect = button_response
470 .rect
471 .expand2(ui.style().spacing.item_spacing / 2.0);
472
473 let is_hovered = hover_pos.is_some_and(|pos| button_rect.contains(pos));
476
477 let should_open =
479 ui.is_enabled() && (button_response.clicked() || (is_hovered && !is_any_open));
480 if should_open {
481 set_open = Some(true);
482 is_open = true;
483 MenuState::from_id(ui.ctx(), menu_id, |state| {
485 state.open_item = None;
486 });
487 }
488
489 let gap = frame.total_margin().sum().x / 2.0 + 2.0;
490
491 let mut response = button_response.clone();
492 let expand = Vec2::new(0.0, frame.total_margin().sum().y / 2.0);
494 response.interact_rect = response.interact_rect.expand2(expand);
495
496 let popup_response = Popup::from_response(&response)
497 .id(id)
498 .open(is_open)
499 .align(RectAlign::RIGHT_START)
500 .layout(Layout::top_down_justified(Align::Min))
501 .gap(gap)
502 .style(menu_config.style.clone())
503 .frame(frame)
504 .close_behavior(PopupCloseBehavior::IgnoreClicks)
506 .info(
507 UiStackInfo::new(UiKind::Menu)
508 .with_tag_value(MenuConfig::MENU_CONFIG_TAG, menu_config.clone()),
509 )
510 .show(|ui| {
511 if button_response.clicked() || button_response.is_pointer_button_down_on() {
513 ui.ctx().move_to_top(ui.layer_id());
514 }
515 content(ui)
516 });
517
518 if let Some(popup_response) = &popup_response {
519 let is_deepest_submenu = MenuState::is_deepest_open_sub_menu(ui.ctx(), id);
521
522 let clicked_outside = is_deepest_submenu
528 && popup_response.response.clicked_elsewhere()
529 && menu_root_response.clicked_elsewhere();
530
531 let submenu_button_clicked = button_response.clicked();
536
537 let clicked_inside = is_deepest_submenu
538 && !submenu_button_clicked
539 && response.ctx.input(|i| i.pointer.any_click())
540 && hover_pos.is_some_and(|pos| popup_response.response.interact_rect.contains(pos));
541
542 let click_close = match menu_config.close_behavior {
543 PopupCloseBehavior::CloseOnClick => clicked_outside || clicked_inside,
544 PopupCloseBehavior::CloseOnClickOutside => clicked_outside,
545 PopupCloseBehavior::IgnoreClicks => false,
546 };
547
548 if click_close {
549 set_open = Some(false);
550 ui.close();
551 }
552
553 let is_moving_towards_rect = ui.input(|i| {
554 i.pointer
555 .is_moving_towards_rect(&popup_response.response.rect)
556 });
557 if is_moving_towards_rect {
558 ui.ctx().request_repaint();
561 }
562 let hovering_other_menu_entry = is_open
563 && !is_hovered
564 && !popup_response.response.contains_pointer()
565 && !is_moving_towards_rect
566 && is_hovering_menu;
567
568 let close_called = popup_response.response.should_close();
569
570 if close_called {
572 ui.close();
573 }
574
575 if hovering_other_menu_entry {
576 set_open = Some(false);
577 }
578
579 if ui.will_parent_close() {
580 ui.data_mut(|data| data.remove_by_type::<MenuState>());
581 }
582 }
583
584 if let Some(set_open) = set_open {
585 MenuState::from_id(ui.ctx(), menu_id, |state| {
586 state.open_item = set_open.then_some(id);
587 });
588 }
589
590 popup_response
591 }
592}