egui/emigui/src/memory.rs

89 lines
2.8 KiB
Rust
Raw Normal View History

use std::collections::{HashMap, HashSet};
2020-04-17 13:29:48 +00:00
use crate::{
2020-05-10 11:14:52 +00:00
containers::{area, collapsing_header, resize, scroll_area},
Id, Layer, Pos2, Rect,
};
2020-04-21 12:47:17 +00:00
#[derive(Clone, Debug, Default, serde_derive::Deserialize, serde_derive::Serialize)]
#[serde(default)]
2020-04-17 13:29:48 +00:00
pub struct Memory {
/// The widget being interacted with (e.g. dragged, in case of a slider).
#[serde(skip)]
2020-04-17 13:29:48 +00:00
pub(crate) active_id: Option<Id>,
2020-04-29 19:25:49 +00:00
/// The widget with keyboard focus (i.e. a text input field).
#[serde(skip)]
2020-04-29 19:25:49 +00:00
pub(crate) kb_focus_id: Option<Id>,
2020-04-21 18:48:31 +00:00
// states of various types of widgets
pub(crate) collapsing_headers: HashMap<Id, collapsing_header::State>,
2020-04-22 16:25:02 +00:00
pub(crate) scroll_areas: HashMap<Id, scroll_area::State>,
2020-04-25 12:37:39 +00:00
pub(crate) resize: HashMap<Id, resize::State>,
2020-04-17 13:29:48 +00:00
2020-05-10 11:14:52 +00:00
area: HashMap<Id, area::State>,
2020-04-17 13:29:48 +00:00
/// Top is last
2020-05-10 12:27:02 +00:00
area_order: Vec<Layer>,
area_visible_last_frame: HashSet<Layer>,
area_visible_current_frame: HashSet<Layer>,
2020-04-17 13:29:48 +00:00
}
impl Memory {
2020-05-10 11:14:52 +00:00
pub(crate) fn get_area(&mut self, id: Id) -> Option<area::State> {
self.area.get(&id).cloned()
2020-04-19 21:34:34 +00:00
}
2020-05-10 12:27:02 +00:00
pub(crate) fn area_order(&self) -> &[Layer] {
2020-05-10 11:14:52 +00:00
&self.area_order
}
2020-05-10 12:27:02 +00:00
pub(crate) fn set_area_state(&mut self, layer: Layer, state: area::State) {
self.area_visible_current_frame.insert(layer);
let did_insert = self.area.insert(layer.id, state).is_none();
2020-04-19 21:34:34 +00:00
if did_insert {
2020-05-10 12:27:02 +00:00
self.area_order.push(layer);
self.area_order.sort_by_key(|layer| layer.order);
2020-04-19 21:34:34 +00:00
}
2020-04-17 13:29:48 +00:00
}
/// TODO: call once at the start of the frame for the current mouse pos
2020-05-10 12:27:02 +00:00
pub fn layer_at(&self, pos: Pos2) -> Option<Layer> {
for layer in self.area_order.iter().rev() {
if self.is_area_visible(layer) {
if let Some(state) = self.area.get(&layer.id) {
if state.interactable {
let rect = Rect::from_min_size(state.pos, state.size);
if rect.contains(pos) {
return Some(*layer);
}
}
2020-04-17 13:29:48 +00:00
}
}
}
2020-05-10 12:27:02 +00:00
None
2020-04-17 13:29:48 +00:00
}
2020-05-10 12:27:02 +00:00
pub fn is_area_visible(&self, layer: &Layer) -> bool {
self.area_visible_last_frame.contains(layer)
|| self.area_visible_current_frame.contains(layer)
}
pub fn move_area_to_top(&mut self, layer: Layer) {
self.area_visible_current_frame.insert(layer);
if self.area_order.last() == Some(&layer) {
return; // common case early-out
}
2020-05-10 12:27:02 +00:00
if let Some(index) = self.area_order.iter().position(|x| *x == layer) {
2020-05-10 11:14:52 +00:00
self.area_order.remove(index);
}
2020-05-10 12:27:02 +00:00
self.area_order.push(layer);
self.area_order.sort_by_key(|layer| layer.order);
}
pub(crate) fn begin_frame(&mut self) {
2020-05-10 11:14:52 +00:00
self.area_visible_last_frame = std::mem::take(&mut self.area_visible_current_frame);
2020-04-17 13:29:48 +00:00
}
}