egui/memory/
mod.rs

1#![warn(missing_docs)] // Let's keep this file well-documented.` to memory.rs
2
3use std::num::NonZeroUsize;
4
5use ahash::{HashMap, HashSet};
6use epaint::emath::TSTransform;
7
8use crate::{
9    EventFilter, Id, IdMap, LayerId, Order, Pos2, Rangef, RawInput, Rect, Style, Vec2, ViewportId,
10    ViewportIdMap, ViewportIdSet, area, vec2,
11};
12
13mod theme;
14pub use theme::{Theme, ThemePreference};
15
16// ----------------------------------------------------------------------------
17
18/// The data that egui persists between frames.
19///
20/// This includes window positions and sizes,
21/// how far the user has scrolled in a [`ScrollArea`](crate::ScrollArea) etc.
22///
23/// If you want this to persist when closing your app, you should serialize [`Memory`] and store it.
24/// For this you need to enable the `persistence`.
25///
26/// If you want to store data for your widgets, you should look at [`Memory::data`]
27#[derive(Clone, Debug)]
28#[cfg_attr(feature = "persistence", derive(serde::Deserialize, serde::Serialize))]
29#[cfg_attr(feature = "persistence", serde(default))]
30pub struct Memory {
31    /// Global egui options.
32    pub options: Options,
33
34    /// This map stores some superficial state for all widgets with custom [`Id`]s.
35    ///
36    /// This includes storing whether a [`crate::CollapsingHeader`] is open, how far scrolled a
37    /// [`crate::ScrollArea`] is, where the cursor in a [`crate::TextEdit`] is, etc.
38    ///
39    /// This is NOT meant to store any important data. Store that in your own structures!
40    ///
41    /// Each read clones the data, so keep your values cheap to clone.
42    /// If you want to store a lot of data, you should wrap it in `Arc<Mutex<…>>` so it is cheap to clone.
43    ///
44    /// This will be saved between different program runs if you use the `persistence` feature.
45    ///
46    /// To store a state common for all your widgets (a singleton), use [`Id::NULL`] as the key.
47    pub data: crate::util::IdTypeMap,
48
49    // ------------------------------------------
50    /// Can be used to cache computations from one frame to another.
51    ///
52    /// This is for saving CPU time when you have something that may take 1-100ms to compute.
53    /// Very slow operations (>100ms) should instead be done async (i.e. in another thread)
54    /// so as not to lock the UI thread.
55    ///
56    /// ```
57    /// use egui::cache::{ComputerMut, FrameCache};
58    ///
59    /// #[derive(Default)]
60    /// struct CharCounter {}
61    /// impl ComputerMut<&str, usize> for CharCounter {
62    ///     fn compute(&mut self, s: &str) -> usize {
63    ///         s.chars().count() // you probably want to cache something more expensive than this
64    ///     }
65    /// }
66    /// type CharCountCache<'a> = FrameCache<usize, CharCounter>;
67    ///
68    /// # let mut ctx = egui::Context::default();
69    /// ctx.memory_mut(|mem| {
70    ///     let cache = mem.caches.cache::<CharCountCache<'_>>();
71    ///     assert_eq!(cache.get("hello"), 5);
72    /// });
73    /// ```
74    #[cfg_attr(feature = "persistence", serde(skip))]
75    pub caches: crate::cache::CacheStorage,
76
77    // ------------------------------------------
78    /// new fonts that will be applied at the start of the next frame
79    #[cfg_attr(feature = "persistence", serde(skip))]
80    pub(crate) new_font_definitions: Option<epaint::text::FontDefinitions>,
81
82    /// add new font that will be applied at the start of the next frame
83    #[cfg_attr(feature = "persistence", serde(skip))]
84    pub(crate) add_fonts: Vec<epaint::text::FontInsert>,
85
86    // Current active viewport
87    #[cfg_attr(feature = "persistence", serde(skip))]
88    pub(crate) viewport_id: ViewportId,
89
90    #[cfg_attr(feature = "persistence", serde(skip))]
91    everything_is_visible: bool,
92
93    /// Transforms per layer.
94    ///
95    /// Instead of using this directly, use:
96    /// * [`crate::Context::set_transform_layer`]
97    /// * [`crate::Context::layer_transform_to_global`]
98    /// * [`crate::Context::layer_transform_from_global`]
99    pub to_global: HashMap<LayerId, TSTransform>,
100
101    // -------------------------------------------------
102    // Per-viewport:
103    areas: ViewportIdMap<Areas>,
104
105    #[cfg_attr(feature = "persistence", serde(skip))]
106    pub(crate) interactions: ViewportIdMap<InteractionState>,
107
108    #[cfg_attr(feature = "persistence", serde(skip))]
109    pub(crate) focus: ViewportIdMap<Focus>,
110
111    /// Which popup-window is open on a viewport (if any)?
112    /// Could be a combo box, color picker, menu, etc.
113    /// Optionally stores the position of the popup (usually this would be the position where
114    /// the user clicked).
115    /// If position is [`None`], the popup position will be calculated based on some configuration
116    /// (e.g. relative to some other widget).
117    #[cfg_attr(feature = "persistence", serde(skip))]
118    popups: ViewportIdMap<OpenPopup>,
119}
120
121impl Default for Memory {
122    fn default() -> Self {
123        let mut slf = Self {
124            options: Default::default(),
125            data: Default::default(),
126            caches: Default::default(),
127            new_font_definitions: Default::default(),
128            interactions: Default::default(),
129            focus: Default::default(),
130            viewport_id: Default::default(),
131            areas: Default::default(),
132            to_global: Default::default(),
133            popups: Default::default(),
134            everything_is_visible: Default::default(),
135            add_fonts: Default::default(),
136        };
137        slf.interactions.entry(slf.viewport_id).or_default();
138        slf.areas.entry(slf.viewport_id).or_default();
139        slf
140    }
141}
142
143/// A direction in which to move the keyboard focus.
144#[derive(Clone, Copy, Debug, Default, PartialEq, Eq)]
145pub enum FocusDirection {
146    /// Select the widget closest above the current focused widget.
147    Up,
148
149    /// Select the widget to the right of the current focused widget.
150    Right,
151
152    /// Select the widget below the current focused widget.
153    Down,
154
155    /// Select the widget to the left of the current focused widget.
156    Left,
157
158    /// Select the previous widget that had focus.
159    Previous,
160
161    /// Select the next widget that wants focus.
162    Next,
163
164    /// Don't change focus.
165    #[default]
166    None,
167}
168
169impl FocusDirection {
170    fn is_cardinal(&self) -> bool {
171        match self {
172            Self::Up | Self::Right | Self::Down | Self::Left => true,
173
174            Self::Previous | Self::Next | Self::None => false,
175        }
176    }
177}
178
179// ----------------------------------------------------------------------------
180
181/// Some global options that you can read and write.
182///
183/// See also [`crate::style::DebugOptions`].
184#[derive(Clone, Debug, PartialEq)]
185#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
186#[cfg_attr(feature = "serde", serde(default))]
187pub struct Options {
188    /// The default style for new [`Ui`](crate::Ui):s in dark mode.
189    #[cfg_attr(feature = "serde", serde(skip))]
190    pub dark_style: std::sync::Arc<Style>,
191
192    /// The default style for new [`Ui`](crate::Ui):s in light mode.
193    #[cfg_attr(feature = "serde", serde(skip))]
194    pub light_style: std::sync::Arc<Style>,
195
196    /// Preference for selection between dark and light [`crate::Context::style`]
197    /// as the active style used by all subsequent windows, panels, etc.
198    ///
199    /// Default: `ThemePreference::System`.
200    pub theme_preference: ThemePreference,
201
202    /// Which theme to use in case [`Self::theme_preference`] is [`ThemePreference::System`]
203    /// and egui fails to detect the system theme.
204    ///
205    /// Default: [`crate::Theme::Dark`].
206    pub fallback_theme: Theme,
207
208    /// The current system theme, used to choose between
209    /// dark and light style in case [`Self::theme_preference`] is [`ThemePreference::System`].
210    #[cfg_attr(feature = "serde", serde(skip))]
211    pub(crate) system_theme: Option<Theme>,
212
213    /// Global zoom factor of the UI.
214    ///
215    /// This is used to calculate the `pixels_per_point`
216    /// for the UI as `pixels_per_point = zoom_fator * native_pixels_per_point`.
217    ///
218    /// The default is 1.0. Increase it to make all UI elements larger.
219    ///
220    /// You should call [`crate::Context::set_zoom_factor`]
221    /// instead of modifying this directly!
222    pub zoom_factor: f32,
223
224    /// If `true`, egui will change the scale of the ui ([`crate::Context::zoom_factor`]) when the user
225    /// presses Cmd+Plus, Cmd+Minus or Cmd+0, just like in a browser.
226    ///
227    /// This is `true` by default.
228    ///
229    /// On the web-backend of `eframe` this is set to false by default,
230    /// so that the zoom shortcuts are handled exclusively by the browser,
231    /// which will change the `native_pixels_per_point` (`devicePixelRatio`).
232    /// You can still zoom egui independently by calling [`crate::Context::set_zoom_factor`],
233    /// which will be applied on top of the browsers global zoom.
234    #[cfg_attr(feature = "serde", serde(skip))]
235    pub zoom_with_keyboard: bool,
236
237    /// Controls the tessellator.
238    pub tessellation_options: epaint::TessellationOptions,
239
240    /// If any widget moves or changes id, repaint everything.
241    ///
242    /// It is recommended you keep this OFF, as it may
243    /// lead to endless repaints for an unknown reason. See
244    /// (<https://github.com/rerun-io/rerun/issues/5018>).
245    pub repaint_on_widget_change: bool,
246
247    /// Maximum number of passes to run in one frame.
248    ///
249    /// Set to `1` for pure single-pass immediate mode.
250    /// Set to something larger than `1` to allow multi-pass when needed.
251    ///
252    /// Default is `2`. This means sometimes a frame will cost twice as much,
253    /// but usually only rarely (e.g. when showing a new panel for the first time).
254    ///
255    /// egui will usually only ever run one pass, even if `max_passes` is large.
256    ///
257    /// If this is `1`, [`crate::Context::request_discard`] will be ignored.
258    ///
259    /// Multi-pass is supported by [`crate::Context::run`].
260    ///
261    /// See [`crate::Context::request_discard`] for more.
262    pub max_passes: NonZeroUsize,
263
264    /// This is a signal to any backend that we want the [`crate::PlatformOutput::events`] read out loud.
265    ///
266    /// The only change to egui is that labels can be focused by pressing tab.
267    ///
268    /// Screen readers are an experimental feature of egui, and not supported on all platforms.
269    /// `eframe` only supports it on web.
270    ///
271    /// Consider using [AccessKit](https://github.com/AccessKit/accesskit) instead,
272    /// which is supported by `eframe`.
273    pub screen_reader: bool,
274
275    /// Check reusing of [`Id`]s, and show a visual warning on screen when one is found.
276    ///
277    /// By default this is `true` in debug builds.
278    pub warn_on_id_clash: bool,
279
280    /// Options related to input state handling.
281    pub input_options: crate::input_state::InputOptions,
282
283    /// If `true`, `egui` will discard the loaded image data after
284    /// the texture is loaded onto the GPU to reduce memory usage.
285    ///
286    /// In modern GPU rendering, the texture data is not required after the texture is loaded.
287    ///
288    /// This is beneficial when using a large number or resolution of images and there is no need to
289    /// retain the image data, potentially saving a significant amount of memory.
290    ///
291    /// The drawback is that it becomes impossible to serialize the loaded images or render in non-GPU systems.
292    ///
293    /// Default is `false`.
294    pub reduce_texture_memory: bool,
295}
296
297impl Default for Options {
298    fn default() -> Self {
299        Self {
300            dark_style: std::sync::Arc::new(Theme::Dark.default_style()),
301            light_style: std::sync::Arc::new(Theme::Light.default_style()),
302            theme_preference: Default::default(),
303            fallback_theme: Theme::Dark,
304            system_theme: None,
305            zoom_factor: 1.0,
306            zoom_with_keyboard: true,
307            tessellation_options: Default::default(),
308            repaint_on_widget_change: false,
309            max_passes: NonZeroUsize::new(2).unwrap(),
310            screen_reader: false,
311            warn_on_id_clash: cfg!(debug_assertions),
312
313            // Input:
314            input_options: Default::default(),
315            reduce_texture_memory: false,
316        }
317    }
318}
319
320impl Options {
321    // Needs to be pub because we need to set the system_theme early in the eframe glow renderer.
322    #[doc(hidden)]
323    pub fn begin_pass(&mut self, new_raw_input: &RawInput) {
324        self.system_theme = new_raw_input.system_theme;
325    }
326
327    /// The currently active theme (may depend on the system theme).
328    pub(crate) fn theme(&self) -> Theme {
329        match self.theme_preference {
330            ThemePreference::Dark => Theme::Dark,
331            ThemePreference::Light => Theme::Light,
332            ThemePreference::System => self.system_theme.unwrap_or(self.fallback_theme),
333        }
334    }
335
336    pub(crate) fn style(&self) -> &std::sync::Arc<Style> {
337        match self.theme() {
338            Theme::Dark => &self.dark_style,
339            Theme::Light => &self.light_style,
340        }
341    }
342
343    pub(crate) fn style_mut(&mut self) -> &mut std::sync::Arc<Style> {
344        match self.theme() {
345            Theme::Dark => &mut self.dark_style,
346            Theme::Light => &mut self.light_style,
347        }
348    }
349}
350
351impl Options {
352    /// Show the options in the ui.
353    pub fn ui(&mut self, ui: &mut crate::Ui) {
354        let theme = self.theme();
355
356        let Self {
357            dark_style, // covered above
358            light_style,
359            theme_preference,
360            fallback_theme: _,
361            system_theme: _,
362            zoom_factor,
363            zoom_with_keyboard,
364            tessellation_options,
365            repaint_on_widget_change,
366            max_passes,
367            screen_reader: _, // needs to come from the integration
368            warn_on_id_clash,
369            input_options,
370            reduce_texture_memory,
371        } = self;
372
373        use crate::Widget as _;
374        use crate::containers::CollapsingHeader;
375
376        CollapsingHeader::new("⚙ Options")
377            .default_open(false)
378            .show(ui, |ui| {
379                ui.horizontal(|ui| {
380                    ui.label("Max passes:");
381                    ui.add(crate::DragValue::new(max_passes).range(0..=10));
382                });
383
384                ui.checkbox(
385                    repaint_on_widget_change,
386                    "Repaint if any widget moves or changes id",
387                );
388
389                ui.horizontal(|ui| {
390                    ui.label("Zoom factor:");
391                    ui.add(crate::DragValue::new(zoom_factor).range(0.10..=10.0));
392                });
393
394                ui.checkbox(
395                    zoom_with_keyboard,
396                    "Zoom with keyboard (Cmd +, Cmd -, Cmd 0)",
397                );
398
399                ui.checkbox(warn_on_id_clash, "Warn if two widgets have the same Id");
400
401                ui.checkbox(reduce_texture_memory, "Reduce texture memory");
402            });
403
404        CollapsingHeader::new("🎑 Style")
405            .default_open(true)
406            .show(ui, |ui| {
407                theme_preference.radio_buttons(ui);
408
409                let style = std::sync::Arc::make_mut(match theme {
410                    Theme::Dark => dark_style,
411                    Theme::Light => light_style,
412                });
413                style.ui(ui);
414            });
415
416        CollapsingHeader::new("✒ Painting")
417            .default_open(false)
418            .show(ui, |ui| {
419                tessellation_options.ui(ui);
420                ui.vertical_centered(|ui| {
421                    crate::reset_button(ui, tessellation_options, "Reset paint settings");
422                });
423            });
424
425        CollapsingHeader::new("🖱 Input")
426            .default_open(false)
427            .show(ui, |ui| {
428                input_options.ui(ui);
429            });
430
431        ui.vertical_centered(|ui| crate::reset_button(ui, self, "Reset all"));
432    }
433}
434
435// ----------------------------------------------------------------------------
436
437/// The state of the interaction in egui,
438/// i.e. what is being dragged.
439///
440/// Say there is a button in a scroll area.
441/// If the user clicks the button, the button should click.
442/// If the user drags the button we should scroll the scroll area.
443/// Therefore, when the mouse is pressed, we register both the button
444/// and the scroll area (as `click_id`/`drag_id`).
445/// If the user releases the button without moving the mouse, we register it as a click on `click_id`.
446/// If the cursor moves too much, we clear the `click_id` and start passing move events to `drag_id`.
447#[derive(Clone, Debug, Default)]
448pub(crate) struct InteractionState {
449    /// A widget interested in clicks that has a mouse press on it.
450    pub potential_click_id: Option<Id>,
451
452    /// A widget interested in drags that has a mouse press on it.
453    ///
454    /// Note that this is set as soon as the mouse is pressed,
455    /// so the widget may not yet be marked as "dragged"
456    /// as that can only happen after the mouse has moved a bit
457    /// (at least if the widget is interesated in both clicks and drags).
458    pub potential_drag_id: Option<Id>,
459}
460
461/// Keeps tracks of what widget has keyboard focus
462#[derive(Clone, Debug, Default)]
463pub(crate) struct Focus {
464    /// The widget with keyboard focus (i.e. a text input field).
465    focused_widget: Option<FocusWidget>,
466
467    /// The ID of a widget that had keyboard focus during the previous frame.
468    id_previous_frame: Option<Id>,
469
470    /// The ID of a widget to give the focus to in the next frame.
471    id_next_frame: Option<Id>,
472
473    #[cfg(feature = "accesskit")]
474    id_requested_by_accesskit: Option<accesskit::NodeId>,
475
476    /// If set, the next widget that is interested in focus will automatically get it.
477    /// Probably because the user pressed Tab.
478    give_to_next: bool,
479
480    /// The last widget interested in focus.
481    last_interested: Option<Id>,
482
483    /// Set when looking for widget with navigational keys like arrows, tab, shift+tab.
484    focus_direction: FocusDirection,
485
486    /// The top-most modal layer from the previous frame.
487    top_modal_layer: Option<LayerId>,
488
489    /// The top-most modal layer from the current frame.
490    top_modal_layer_current_frame: Option<LayerId>,
491
492    /// A cache of widget IDs that are interested in focus with their corresponding rectangles.
493    focus_widgets_cache: IdMap<Rect>,
494}
495
496/// The widget with focus.
497#[derive(Clone, Copy, Debug)]
498struct FocusWidget {
499    pub id: Id,
500    pub filter: EventFilter,
501}
502
503impl FocusWidget {
504    pub fn new(id: Id) -> Self {
505        Self {
506            id,
507            filter: Default::default(),
508        }
509    }
510}
511
512impl InteractionState {
513    /// Are we currently clicking or dragging an egui widget?
514    pub fn is_using_pointer(&self) -> bool {
515        self.potential_click_id.is_some() || self.potential_drag_id.is_some()
516    }
517}
518
519impl Focus {
520    /// Which widget currently has keyboard focus?
521    pub fn focused(&self) -> Option<Id> {
522        self.focused_widget.as_ref().map(|w| w.id)
523    }
524
525    fn begin_pass(&mut self, new_input: &crate::data::input::RawInput) {
526        self.id_previous_frame = self.focused();
527        if let Some(id) = self.id_next_frame.take() {
528            self.focused_widget = Some(FocusWidget::new(id));
529        }
530        let event_filter = self.focused_widget.map(|w| w.filter).unwrap_or_default();
531
532        #[cfg(feature = "accesskit")]
533        {
534            self.id_requested_by_accesskit = None;
535        }
536
537        self.focus_direction = FocusDirection::None;
538
539        for event in &new_input.events {
540            if !event_filter.matches(event)
541                && let crate::Event::Key {
542                    key,
543                    pressed: true,
544                    modifiers,
545                    ..
546                } = event
547                && let Some(cardinality) = match key {
548                    crate::Key::ArrowUp => Some(FocusDirection::Up),
549                    crate::Key::ArrowRight => Some(FocusDirection::Right),
550                    crate::Key::ArrowDown => Some(FocusDirection::Down),
551                    crate::Key::ArrowLeft => Some(FocusDirection::Left),
552
553                    crate::Key::Tab => {
554                        if modifiers.shift {
555                            Some(FocusDirection::Previous)
556                        } else {
557                            Some(FocusDirection::Next)
558                        }
559                    }
560                    crate::Key::Escape => {
561                        self.focused_widget = None;
562                        Some(FocusDirection::None)
563                    }
564                    _ => None,
565                }
566            {
567                self.focus_direction = cardinality;
568            }
569
570            #[cfg(feature = "accesskit")]
571            {
572                if let crate::Event::AccessKitActionRequest(accesskit::ActionRequest {
573                    action: accesskit::Action::Focus,
574                    target,
575                    data: None,
576                }) = event
577                {
578                    self.id_requested_by_accesskit = Some(*target);
579                }
580            }
581        }
582    }
583
584    pub(crate) fn end_pass(&mut self, used_ids: &IdMap<Rect>) {
585        if self.focus_direction.is_cardinal()
586            && let Some(found_widget) = self.find_widget_in_direction(used_ids)
587        {
588            self.focused_widget = Some(FocusWidget::new(found_widget));
589        }
590
591        if let Some(focused_widget) = self.focused_widget {
592            // Allow calling `request_focus` one frame and not using it until next frame
593            let recently_gained_focus = self.id_previous_frame != Some(focused_widget.id);
594
595            if !recently_gained_focus && !used_ids.contains_key(&focused_widget.id) {
596                // Dead-mans-switch: the widget with focus has disappeared!
597                self.focused_widget = None;
598            }
599        }
600
601        self.top_modal_layer = self.top_modal_layer_current_frame.take();
602    }
603
604    pub(crate) fn had_focus_last_frame(&self, id: Id) -> bool {
605        self.id_previous_frame == Some(id)
606    }
607
608    fn interested_in_focus(&mut self, id: Id) {
609        #[cfg(feature = "accesskit")]
610        {
611            if self.id_requested_by_accesskit == Some(id.accesskit_id()) {
612                self.focused_widget = Some(FocusWidget::new(id));
613                self.id_requested_by_accesskit = None;
614                self.give_to_next = false;
615                self.reset_focus();
616            }
617        }
618
619        // The rect is updated at the end of the frame.
620        self.focus_widgets_cache
621            .entry(id)
622            .or_insert(Rect::EVERYTHING);
623
624        if self.give_to_next && !self.had_focus_last_frame(id) {
625            self.focused_widget = Some(FocusWidget::new(id));
626            self.give_to_next = false;
627        } else if self.focused() == Some(id) {
628            if self.focus_direction == FocusDirection::Next {
629                self.focused_widget = None;
630                self.give_to_next = true;
631                self.reset_focus();
632            } else if self.focus_direction == FocusDirection::Previous {
633                self.id_next_frame = self.last_interested; // frame-delay so gained_focus works
634                self.reset_focus();
635            }
636        } else if self.focus_direction == FocusDirection::Next
637            && self.focused_widget.is_none()
638            && !self.give_to_next
639        {
640            // nothing has focus and the user pressed tab - give focus to the first widgets that wants it:
641            self.focused_widget = Some(FocusWidget::new(id));
642            self.reset_focus();
643        } else if self.focus_direction == FocusDirection::Previous
644            && self.focused_widget.is_none()
645            && !self.give_to_next
646        {
647            // nothing has focus and the user pressed Shift+Tab - give focus to the last widgets that wants it:
648            self.focused_widget = self.last_interested.map(FocusWidget::new);
649            self.reset_focus();
650        }
651
652        self.last_interested = Some(id);
653    }
654
655    fn set_modal_layer(&mut self, layer_id: LayerId) {
656        self.top_modal_layer_current_frame = Some(layer_id);
657    }
658
659    pub(crate) fn top_modal_layer(&self) -> Option<LayerId> {
660        self.top_modal_layer
661    }
662
663    fn reset_focus(&mut self) {
664        self.focus_direction = FocusDirection::None;
665    }
666
667    fn find_widget_in_direction(&mut self, new_rects: &IdMap<Rect>) -> Option<Id> {
668        // NOTE: `new_rects` here include some widgets _not_ interested in focus.
669
670        /// * negative if `a` is left of `b`
671        /// * positive if `a` is right of `b`
672        /// * zero if the ranges overlap significantly
673        fn range_diff(a: Rangef, b: Rangef) -> f32 {
674            let has_significant_overlap = a.intersection(b).span() >= 0.5 * b.span().min(a.span());
675            if has_significant_overlap {
676                0.0
677            } else {
678                a.center() - b.center()
679            }
680        }
681
682        let current_focused = self.focused_widget?;
683
684        // In what direction we are looking for the next widget.
685        let search_direction = match self.focus_direction {
686            FocusDirection::Up => Vec2::UP,
687            FocusDirection::Right => Vec2::RIGHT,
688            FocusDirection::Down => Vec2::DOWN,
689            FocusDirection::Left => Vec2::LEFT,
690            _ => {
691                return None;
692            }
693        };
694
695        // Update cache with new rects
696        self.focus_widgets_cache.retain(|id, old_rect| {
697            if let Some(new_rect) = new_rects.get(id) {
698                *old_rect = *new_rect;
699                true // Keep the item
700            } else {
701                false // Remove the item
702            }
703        });
704
705        let current_rect = self.focus_widgets_cache.get(&current_focused.id)?;
706
707        let mut best_score = f32::INFINITY;
708        let mut best_id = None;
709
710        // iteration order should only matter in case of a tie, and that should be very rare
711        #[expect(clippy::iter_over_hash_type)]
712        for (candidate_id, candidate_rect) in &self.focus_widgets_cache {
713            if *candidate_id == current_focused.id {
714                continue;
715            }
716
717            // There is a lot of room for improvement here.
718            let to_candidate = vec2(
719                range_diff(candidate_rect.x_range(), current_rect.x_range()),
720                range_diff(candidate_rect.y_range(), current_rect.y_range()),
721            );
722
723            let acos_angle = to_candidate.normalized().dot(search_direction);
724
725            // Only interested in widgets that fall in a 90° cone (±45°)
726            // of the search direction.
727            let is_in_search_cone = 0.5_f32.sqrt() <= acos_angle;
728            if is_in_search_cone {
729                let distance = to_candidate.length();
730
731                // There is a lot of room for improvement here.
732                let score = distance / (acos_angle * acos_angle);
733
734                if score < best_score {
735                    best_score = score;
736                    best_id = Some(*candidate_id);
737                }
738            }
739        }
740
741        best_id
742    }
743}
744
745impl Memory {
746    pub(crate) fn begin_pass(&mut self, new_raw_input: &RawInput, viewports: &ViewportIdSet) {
747        profiling::function_scope!();
748
749        self.viewport_id = new_raw_input.viewport_id;
750
751        // Cleanup
752        self.interactions.retain(|id, _| viewports.contains(id));
753        self.areas.retain(|id, _| viewports.contains(id));
754        self.popups.retain(|id, _| viewports.contains(id));
755
756        self.areas.entry(self.viewport_id).or_default();
757
758        // self.interactions  is handled elsewhere
759
760        self.options.begin_pass(new_raw_input);
761
762        self.focus
763            .entry(self.viewport_id)
764            .or_default()
765            .begin_pass(new_raw_input);
766    }
767
768    pub(crate) fn end_pass(&mut self, used_ids: &IdMap<Rect>) {
769        self.caches.update();
770        self.areas_mut().end_pass();
771        self.focus_mut().end_pass(used_ids);
772
773        // Clean up abandoned popups.
774        if let Some(popup) = self.popups.get_mut(&self.viewport_id) {
775            if popup.open_this_frame {
776                popup.open_this_frame = false;
777            } else {
778                self.popups.remove(&self.viewport_id);
779            }
780        }
781    }
782
783    pub(crate) fn set_viewport_id(&mut self, viewport_id: ViewportId) {
784        self.viewport_id = viewport_id;
785    }
786
787    /// Access memory of the [`Area`](crate::containers::area::Area)s, such as `Window`s.
788    pub fn areas(&self) -> &Areas {
789        self.areas
790            .get(&self.viewport_id)
791            .expect("Memory broken: no area for the current viewport")
792    }
793
794    /// Access memory of the [`Area`](crate::containers::area::Area)s, such as `Window`s.
795    pub fn areas_mut(&mut self) -> &mut Areas {
796        self.areas.entry(self.viewport_id).or_default()
797    }
798
799    /// Top-most layer at the given position.
800    pub fn layer_id_at(&self, pos: Pos2) -> Option<LayerId> {
801        let layer_id = self.areas().layer_id_at(pos, &self.to_global)?;
802        if self.is_above_modal_layer(layer_id) {
803            Some(layer_id)
804        } else {
805            self.top_modal_layer()
806        }
807    }
808
809    /// The currently set transform of a layer.
810    #[deprecated = "Use `Context::layer_transform_to_global` instead"]
811    pub fn layer_transforms(&self, layer_id: LayerId) -> Option<TSTransform> {
812        self.to_global.get(&layer_id).copied()
813    }
814
815    /// An iterator over all layers. Back-to-front, top is last.
816    pub fn layer_ids(&self) -> impl ExactSizeIterator<Item = LayerId> + '_ {
817        self.areas().order().iter().copied()
818    }
819
820    /// Check if the layer had focus last frame.
821    /// returns `true` if the layer had focus last frame, but not this one.
822    pub fn had_focus_last_frame(&self, id: Id) -> bool {
823        self.focus().and_then(|f| f.id_previous_frame) == Some(id)
824    }
825
826    /// Check if the layer lost focus last frame.
827    /// returns `true` if the layer lost focus last frame, but not this one.
828    pub(crate) fn lost_focus(&self, id: Id) -> bool {
829        self.had_focus_last_frame(id) && !self.has_focus(id)
830    }
831
832    /// Check if the layer gained focus this frame.
833    /// returns `true` if the layer gained focus this frame, but not last one.
834    pub(crate) fn gained_focus(&self, id: Id) -> bool {
835        !self.had_focus_last_frame(id) && self.has_focus(id)
836    }
837
838    /// Does this widget have keyboard focus?
839    ///
840    /// This function does not consider whether the UI as a whole (e.g. window)
841    /// has the keyboard focus. That makes this function suitable for deciding
842    /// widget state that should not be disrupted if the user moves away from
843    /// the window and back.
844    #[inline(always)]
845    pub fn has_focus(&self, id: Id) -> bool {
846        self.focused() == Some(id)
847    }
848
849    /// Which widget has keyboard focus?
850    pub fn focused(&self) -> Option<Id> {
851        self.focus()?.focused()
852    }
853
854    /// Set an event filter for a widget.
855    ///
856    /// This allows you to control whether the widget will loose focus
857    /// when the user presses tab, arrow keys, or escape.
858    ///
859    /// You must first give focus to the widget before calling this.
860    pub fn set_focus_lock_filter(&mut self, id: Id, event_filter: EventFilter) {
861        if self.had_focus_last_frame(id)
862            && self.has_focus(id)
863            && let Some(focused) = &mut self.focus_mut().focused_widget
864            && focused.id == id
865        {
866            focused.filter = event_filter;
867        }
868    }
869
870    /// Give keyboard focus to a specific widget.
871    /// See also [`crate::Response::request_focus`].
872    #[inline(always)]
873    pub fn request_focus(&mut self, id: Id) {
874        self.focus_mut().focused_widget = Some(FocusWidget::new(id));
875    }
876
877    /// Surrender keyboard focus for a specific widget.
878    /// See also [`crate::Response::surrender_focus`].
879    #[inline(always)]
880    pub fn surrender_focus(&mut self, id: Id) {
881        let focus = self.focus_mut();
882        if focus.focused() == Some(id) {
883            focus.focused_widget = None;
884        }
885    }
886
887    /// Move keyboard focus in a specific direction.
888    pub fn move_focus(&mut self, direction: FocusDirection) {
889        self.focus_mut().focus_direction = direction;
890    }
891
892    /// Returns true if
893    /// - this layer is the top-most modal layer or above it
894    /// - there is no modal layer
895    pub fn is_above_modal_layer(&self, layer_id: LayerId) -> bool {
896        if let Some(modal_layer) = self.focus().and_then(|f| f.top_modal_layer) {
897            matches!(
898                self.areas().compare_order(layer_id, modal_layer),
899                std::cmp::Ordering::Equal | std::cmp::Ordering::Greater
900            )
901        } else {
902            true
903        }
904    }
905
906    /// Does this layer allow interaction?
907    /// Returns true if
908    ///  - the layer is not behind a modal layer
909    ///  - the [`Order`] allows interaction
910    pub fn allows_interaction(&self, layer_id: LayerId) -> bool {
911        let is_above_modal_layer = self.is_above_modal_layer(layer_id);
912        let ordering_allows_interaction = layer_id.order.allow_interaction();
913        is_above_modal_layer && ordering_allows_interaction
914    }
915
916    /// Register this widget as being interested in getting keyboard focus.
917    /// This will allow the user to select it with tab and shift-tab.
918    /// This is normally done automatically when handling interactions,
919    /// but it is sometimes useful to pre-register interest in focus,
920    /// e.g. before deciding which type of underlying widget to use,
921    /// as in the [`crate::DragValue`] widget, so a widget can be focused
922    /// and rendered correctly in a single frame.
923    ///
924    /// Pass in the `layer_id` of the layer that the widget is in.
925    #[inline(always)]
926    pub fn interested_in_focus(&mut self, id: Id, layer_id: LayerId) {
927        if !self.allows_interaction(layer_id) {
928            return;
929        }
930        self.focus_mut().interested_in_focus(id);
931    }
932
933    /// Limit focus to widgets on the given layer and above.
934    /// If this is called multiple times per frame, the top layer wins.
935    pub fn set_modal_layer(&mut self, layer_id: LayerId) {
936        if let Some(current) = self.focus().and_then(|f| f.top_modal_layer_current_frame)
937            && matches!(
938                self.areas().compare_order(layer_id, current),
939                std::cmp::Ordering::Less
940            )
941        {
942            return;
943        }
944
945        self.focus_mut().set_modal_layer(layer_id);
946    }
947
948    /// Get the top modal layer (from the previous frame).
949    pub fn top_modal_layer(&self) -> Option<LayerId> {
950        self.focus()?.top_modal_layer()
951    }
952
953    /// Stop editing the active [`TextEdit`](crate::TextEdit) (if any).
954    #[inline(always)]
955    pub fn stop_text_input(&mut self) {
956        self.focus_mut().focused_widget = None;
957    }
958
959    /// Forget window positions, sizes etc.
960    /// Can be used to auto-layout windows.
961    pub fn reset_areas(&mut self) {
962        #[expect(clippy::iter_over_hash_type)]
963        for area in self.areas.values_mut() {
964            *area = Default::default();
965        }
966    }
967
968    /// Obtain the previous rectangle of an area.
969    pub fn area_rect(&self, id: impl Into<Id>) -> Option<Rect> {
970        self.areas().get(id.into()).map(|state| state.rect())
971    }
972
973    pub(crate) fn interaction(&self) -> &InteractionState {
974        self.interactions
975            .get(&self.viewport_id)
976            .expect("Failed to get interaction")
977    }
978
979    pub(crate) fn interaction_mut(&mut self) -> &mut InteractionState {
980        self.interactions.entry(self.viewport_id).or_default()
981    }
982
983    pub(crate) fn focus(&self) -> Option<&Focus> {
984        self.focus.get(&self.viewport_id)
985    }
986
987    pub(crate) fn focus_mut(&mut self) -> &mut Focus {
988        self.focus.entry(self.viewport_id).or_default()
989    }
990}
991
992/// State of an open popup.
993#[derive(Clone, Copy, Debug)]
994struct OpenPopup {
995    /// Id of the popup.
996    id: Id,
997
998    /// Optional position of the popup.
999    pos: Option<Pos2>,
1000
1001    /// Whether this popup was still open this frame. Otherwise it's considered abandoned and `Memory::popup` will be cleared.
1002    open_this_frame: bool,
1003}
1004
1005impl OpenPopup {
1006    /// Create a new `OpenPopup`.
1007    fn new(id: Id, pos: Option<Pos2>) -> Self {
1008        Self {
1009            id,
1010            pos,
1011            open_this_frame: true,
1012        }
1013    }
1014}
1015
1016/// ## Deprecated popup API
1017/// Use [`crate::Popup`] instead.
1018impl Memory {
1019    /// Is the given popup open?
1020    #[deprecated = "Use Popup::is_id_open instead"]
1021    pub fn is_popup_open(&self, popup_id: Id) -> bool {
1022        self.popups
1023            .get(&self.viewport_id)
1024            .is_some_and(|state| state.id == popup_id)
1025            || self.everything_is_visible()
1026    }
1027
1028    /// Is any popup open?
1029    #[deprecated = "Use Popup::is_any_open instead"]
1030    pub fn any_popup_open(&self) -> bool {
1031        self.popups.contains_key(&self.viewport_id) || self.everything_is_visible()
1032    }
1033
1034    /// Open the given popup and close all others.
1035    ///
1036    /// Note that you must call `keep_popup_open` on subsequent frames as long as the popup is open.
1037    #[deprecated = "Use Popup::open_id instead"]
1038    pub fn open_popup(&mut self, popup_id: Id) {
1039        self.popups
1040            .insert(self.viewport_id, OpenPopup::new(popup_id, None));
1041    }
1042
1043    /// Popups must call this every frame while open.
1044    ///
1045    /// This is needed because in some cases popups can go away without `close_popup` being
1046    /// called. For example, when a context menu is open and the underlying widget stops
1047    /// being rendered.
1048    #[deprecated = "Use Popup::show instead"]
1049    pub fn keep_popup_open(&mut self, popup_id: Id) {
1050        if let Some(state) = self.popups.get_mut(&self.viewport_id)
1051            && state.id == popup_id
1052        {
1053            state.open_this_frame = true;
1054        }
1055    }
1056
1057    /// Open the popup and remember its position.
1058    #[deprecated = "Use Popup with PopupAnchor::Position instead"]
1059    pub fn open_popup_at(&mut self, popup_id: Id, pos: impl Into<Option<Pos2>>) {
1060        self.popups
1061            .insert(self.viewport_id, OpenPopup::new(popup_id, pos.into()));
1062    }
1063
1064    /// Get the position for this popup.
1065    #[deprecated = "Use Popup::position_of_id instead"]
1066    pub fn popup_position(&self, id: Id) -> Option<Pos2> {
1067        let state = self.popups.get(&self.viewport_id)?;
1068        if state.id == id { state.pos } else { None }
1069    }
1070
1071    /// Close any currently open popup.
1072    #[deprecated = "Use Popup::close_all instead"]
1073    pub fn close_all_popups(&mut self) {
1074        self.popups.clear();
1075    }
1076
1077    /// Close the given popup, if it is open.
1078    ///
1079    /// See also [`Self::close_all_popups`] if you want to close any / all currently open popups.
1080    #[deprecated = "Use Popup::close_id instead"]
1081    pub fn close_popup(&mut self, popup_id: Id) {
1082        #[expect(deprecated)]
1083        if self.is_popup_open(popup_id) {
1084            self.popups.remove(&self.viewport_id);
1085        }
1086    }
1087
1088    /// Toggle the given popup between closed and open.
1089    ///
1090    /// Note: At most, only one popup can be open at a time.
1091    #[deprecated = "Use Popup::toggle_id instead"]
1092    pub fn toggle_popup(&mut self, popup_id: Id) {
1093        #[expect(deprecated)]
1094        if self.is_popup_open(popup_id) {
1095            self.close_popup(popup_id);
1096        } else {
1097            self.open_popup(popup_id);
1098        }
1099    }
1100}
1101
1102impl Memory {
1103    /// If true, all windows, menus, tooltips, etc., will be visible at once.
1104    ///
1105    /// This is useful for testing, benchmarking, pre-caching, etc.
1106    ///
1107    /// Experimental feature!
1108    #[inline(always)]
1109    pub fn everything_is_visible(&self) -> bool {
1110        self.everything_is_visible
1111    }
1112
1113    /// If true, all windows, menus, tooltips etc are to be visible at once.
1114    ///
1115    /// This is useful for testing, benchmarking, pre-caching, etc.
1116    ///
1117    /// Experimental feature!
1118    pub fn set_everything_is_visible(&mut self, value: bool) {
1119        self.everything_is_visible = value;
1120    }
1121}
1122
1123// ----------------------------------------------------------------------------
1124
1125/// Map containing the index of each layer in the order list, for quick lookups.
1126type OrderMap = HashMap<LayerId, usize>;
1127
1128/// Keeps track of [`Area`](crate::containers::area::Area)s, which are free-floating [`Ui`](crate::Ui)s.
1129/// These [`Area`](crate::containers::area::Area)s can be in any [`Order`].
1130#[derive(Clone, Debug, Default)]
1131#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
1132#[cfg_attr(feature = "serde", serde(default))]
1133pub struct Areas {
1134    areas: IdMap<area::AreaState>,
1135
1136    visible_areas_last_frame: ahash::HashSet<LayerId>,
1137    visible_areas_current_frame: ahash::HashSet<LayerId>,
1138
1139    // ----------------------------
1140    // Everything below this is general to all layers, not just areas.
1141    // TODO(emilk): move this to a separate struct.
1142    /// Back-to-front,  top is last.
1143    order: Vec<LayerId>,
1144
1145    /// Inverse of [`Self::order`], calculated at the end of the frame.
1146    order_map: OrderMap,
1147
1148    /// When an area wants to be on top, it is assigned here.
1149    /// This is used to reorder the layers at the end of the frame.
1150    /// If several layers want to be on top, they will keep their relative order.
1151    /// This means closing three windows and then reopening them all in one frame
1152    /// results in them being sent to the top and keeping their previous internal order.
1153    wants_to_be_on_top: ahash::HashSet<LayerId>,
1154
1155    /// The sublayers that each layer has.
1156    ///
1157    /// The parent sublayer is moved directly above the child sublayers in the ordering.
1158    sublayers: ahash::HashMap<LayerId, HashSet<LayerId>>,
1159}
1160
1161impl Areas {
1162    pub(crate) fn count(&self) -> usize {
1163        self.areas.len()
1164    }
1165
1166    pub(crate) fn get(&self, id: Id) -> Option<&area::AreaState> {
1167        self.areas.get(&id)
1168    }
1169
1170    /// All layers back-to-front, top is last.
1171    pub(crate) fn order(&self) -> &[LayerId] {
1172        &self.order
1173    }
1174
1175    /// Compare the order of two layers, based on the order list from last frame.
1176    ///
1177    /// May return [`std::cmp::Ordering::Equal`] if the layers are not in the order list.
1178    pub(crate) fn compare_order(&self, a: LayerId, b: LayerId) -> std::cmp::Ordering {
1179        // Sort by layer `order` first and use `order_map` to resolve disputes.
1180        // If `order_map` only contains one layer ID, then the other one will be
1181        // lower because `None < Some(x)`.
1182        match a.order.cmp(&b.order) {
1183            std::cmp::Ordering::Equal => self.order_map.get(&a).cmp(&self.order_map.get(&b)),
1184            cmp => cmp,
1185        }
1186    }
1187
1188    pub(crate) fn set_state(&mut self, layer_id: LayerId, state: area::AreaState) {
1189        self.visible_areas_current_frame.insert(layer_id);
1190        self.areas.insert(layer_id.id, state);
1191        if !self.order.contains(&layer_id) {
1192            self.order.push(layer_id);
1193        }
1194    }
1195
1196    /// Top-most layer at the given position.
1197    pub fn layer_id_at(
1198        &self,
1199        pos: Pos2,
1200        layer_to_global: &HashMap<LayerId, TSTransform>,
1201    ) -> Option<LayerId> {
1202        for layer in self.order.iter().rev() {
1203            if self.is_visible(layer)
1204                && let Some(state) = self.areas.get(&layer.id)
1205            {
1206                let mut rect = state.rect();
1207                if state.interactable {
1208                    if let Some(to_global) = layer_to_global.get(layer) {
1209                        rect = *to_global * rect;
1210                    }
1211
1212                    if rect.contains(pos) {
1213                        return Some(*layer);
1214                    }
1215                }
1216            }
1217        }
1218        None
1219    }
1220
1221    pub fn visible_last_frame(&self, layer_id: &LayerId) -> bool {
1222        self.visible_areas_last_frame.contains(layer_id)
1223    }
1224
1225    pub fn is_visible(&self, layer_id: &LayerId) -> bool {
1226        self.visible_areas_last_frame.contains(layer_id)
1227            || self.visible_areas_current_frame.contains(layer_id)
1228    }
1229
1230    pub fn visible_layer_ids(&self) -> ahash::HashSet<LayerId> {
1231        self.visible_areas_last_frame
1232            .iter()
1233            .copied()
1234            .chain(self.visible_areas_current_frame.iter().copied())
1235            .collect()
1236    }
1237
1238    pub(crate) fn visible_windows(&self) -> impl Iterator<Item = (LayerId, &area::AreaState)> {
1239        self.visible_layer_ids()
1240            .into_iter()
1241            .filter(|layer| layer.order == crate::Order::Middle)
1242            .filter(|&layer| !self.is_sublayer(&layer))
1243            .filter_map(|layer| Some((layer, self.get(layer.id)?)))
1244    }
1245
1246    pub fn move_to_top(&mut self, layer_id: LayerId) {
1247        self.visible_areas_current_frame.insert(layer_id);
1248        self.wants_to_be_on_top.insert(layer_id);
1249
1250        if !self.order.contains(&layer_id) {
1251            self.order.push(layer_id);
1252        }
1253    }
1254
1255    /// Mark the `child` layer as a sublayer of `parent`.
1256    ///
1257    /// Sublayers are moved directly above the parent layer at the end of the frame. This is mainly
1258    /// intended for adding a new [Area](crate::Area) inside a [Window](crate::Window).
1259    ///
1260    /// This currently only supports one level of nesting. If `parent` is a sublayer of another
1261    /// layer, the behavior is unspecified.
1262    ///
1263    /// The two layers must have the same [`LayerId::order`].
1264    pub fn set_sublayer(&mut self, parent: LayerId, child: LayerId) {
1265        debug_assert_eq!(
1266            parent.order, child.order,
1267            "DEBUG ASSERT: Trying to set sublayers across layers of different order ({:?}, {:?}), which is currently undefined behavior in egui",
1268            parent.order, child.order
1269        );
1270
1271        self.sublayers.entry(parent).or_default().insert(child);
1272
1273        // Make sure the layers are in the order list:
1274        if !self.order.contains(&parent) {
1275            self.order.push(parent);
1276        }
1277        if !self.order.contains(&child) {
1278            self.order.push(child);
1279        }
1280    }
1281
1282    pub fn top_layer_id(&self, order: Order) -> Option<LayerId> {
1283        self.order
1284            .iter()
1285            .filter(|layer| layer.order == order && !self.is_sublayer(layer))
1286            .next_back()
1287            .copied()
1288    }
1289
1290    /// If this layer is the sublayer of another layer, return the parent.
1291    pub fn parent_layer(&self, layer_id: LayerId) -> Option<LayerId> {
1292        self.sublayers.iter().find_map(|(parent, children)| {
1293            if children.contains(&layer_id) {
1294                Some(*parent)
1295            } else {
1296                None
1297            }
1298        })
1299    }
1300
1301    /// All the child layers of this layer.
1302    pub fn child_layers(&self, layer_id: LayerId) -> impl Iterator<Item = LayerId> + '_ {
1303        self.sublayers.get(&layer_id).into_iter().flatten().copied()
1304    }
1305
1306    pub(crate) fn is_sublayer(&self, layer: &LayerId) -> bool {
1307        self.parent_layer(*layer).is_some()
1308    }
1309
1310    pub(crate) fn end_pass(&mut self) {
1311        let Self {
1312            visible_areas_last_frame,
1313            visible_areas_current_frame,
1314            order,
1315            wants_to_be_on_top,
1316            sublayers,
1317            ..
1318        } = self;
1319
1320        std::mem::swap(visible_areas_last_frame, visible_areas_current_frame);
1321        visible_areas_current_frame.clear();
1322
1323        order.sort_by_key(|layer| (layer.order, wants_to_be_on_top.contains(layer)));
1324        wants_to_be_on_top.clear();
1325
1326        // For all layers with sublayers, put the sublayers directly after the parent layer:
1327        // (it doesn't matter in which order we replace parents with their children)
1328        #[expect(clippy::iter_over_hash_type)]
1329        for (parent, children) in std::mem::take(sublayers) {
1330            let mut moved_layers = vec![parent]; // parent first…
1331
1332            order.retain(|l| {
1333                if children.contains(l) {
1334                    moved_layers.push(*l); // …followed by children
1335                    false
1336                } else {
1337                    true
1338                }
1339            });
1340            let Some(parent_pos) = order.iter().position(|l| l == &parent) else {
1341                continue;
1342            };
1343            order.splice(parent_pos..=parent_pos, moved_layers); // replace the parent with itself and its children
1344        }
1345
1346        self.order_map = self
1347            .order
1348            .iter()
1349            .enumerate()
1350            .map(|(i, id)| (*id, i))
1351            .collect();
1352    }
1353}
1354
1355// ----------------------------------------------------------------------------
1356
1357#[test]
1358fn memory_impl_send_sync() {
1359    fn assert_send_sync<T: Send + Sync>() {}
1360    assert_send_sync::<Memory>();
1361}
1362
1363#[test]
1364fn order_map_total_ordering() {
1365    let mut layers = [
1366        LayerId::new(Order::Tooltip, Id::new("a")),
1367        LayerId::new(Order::Background, Id::new("b")),
1368        LayerId::new(Order::Background, Id::new("c")),
1369        LayerId::new(Order::Tooltip, Id::new("d")),
1370        LayerId::new(Order::Background, Id::new("e")),
1371        LayerId::new(Order::Background, Id::new("f")),
1372        LayerId::new(Order::Tooltip, Id::new("g")),
1373    ];
1374    let mut areas = Areas::default();
1375
1376    // skip some of the layers
1377    for &layer in &layers[3..] {
1378        areas.set_state(layer, crate::AreaState::default());
1379    }
1380    areas.end_pass(); // sort layers
1381
1382    // Sort layers
1383    layers.sort_by(|&a, &b| areas.compare_order(a, b));
1384
1385    // Assert that `areas.compare_order()` forms a total ordering
1386    let mut equivalence_classes = vec![0];
1387    let mut i = 0;
1388    for l in layers.windows(2) {
1389        assert!(l[0].order <= l[1].order, "does not follow LayerId.order");
1390        if areas.compare_order(l[0], l[1]) != std::cmp::Ordering::Equal {
1391            i += 1;
1392        }
1393        equivalence_classes.push(i);
1394    }
1395    assert_eq!(layers.len(), equivalence_classes.len());
1396    for (&l1, c1) in std::iter::zip(&layers, &equivalence_classes) {
1397        for (&l2, c2) in std::iter::zip(&layers, &equivalence_classes) {
1398            assert_eq!(
1399                c1.cmp(c2),
1400                areas.compare_order(l1, l2),
1401                "not a total ordering",
1402            );
1403        }
1404    }
1405}