use std::collections::HashMap; #[derive(Debug)] pub struct AlreadyHandledError(()); pub enum HandlerMap { Leaf(T), Branch(HashMap>), } impl Default for HandlerMap { fn default() -> Self { Self::Branch(HashMap::new()) } } impl HandlerMap { pub fn insert<'a>( &mut self, mut path: impl Iterator, handler: T, ) -> Result<(), AlreadyHandledError> { match self { Self::Leaf(_) => Err(AlreadyHandledError(())), Self::Branch(children) => { if let Some(component) = path.next() { let node = children.entry(component.to_owned()).or_default(); node.insert(path, handler) } else if children.is_empty() { *self = Self::Leaf(handler); Ok(()) } else { Err(AlreadyHandledError(())) } } } } pub fn get<'a>(&self, mut path: impl Iterator) -> Option<&T> { match self { Self::Leaf(t) => Some(t), Self::Branch(m) => { let component = path.next()?; m.get(component)?.get(path) } } } pub fn remove<'a>(&mut self, mut path: impl Iterator) -> Option { match self { Self::Leaf(_) => match std::mem::take(self) { Self::Leaf(t) => Some(t), _ => unreachable!(), }, Self::Branch(map) => { let component = path.next()?; let next = map.get_mut(component)?; let result = next.remove(path); match &*next { Self::Branch(b) if b.is_empty() => { map.remove(component); } _ => (), } result } } } } pub struct SubscriberMap { subscribed: Vec, children: HashMap>, } impl Default for SubscriberMap { fn default() -> Self { Self { subscribed: Vec::new(), children: HashMap::new(), } } } impl SubscriberMap { pub fn insert<'a>(&mut self, mut path: impl Iterator, handler: T) { if let Some(component) = path.next() { self.children .entry(component.to_owned()) .or_default() .insert(path, handler); } else { self.subscribed.push(handler); } } pub fn is_empty(&self) -> bool { self.children.is_empty() && self.subscribed.is_empty() } pub fn retain<'a>( &mut self, mut path: impl Iterator, fun: &mut impl FnMut(&T) -> bool, ) { self.subscribed.retain(|x| fun(x)); if let Some(next) = path.next() { if let Some(y) = self.children.get_mut(next) { y.retain(path, fun); if y.is_empty() { self.children.remove(next); } } } } }