pub use derive_refineable::Refineable; pub trait Refineable: Clone { type Refinement: Refineable + Default; fn refine(&mut self, refinement: &Self::Refinement); fn refined(self, refinement: Self::Refinement) -> Self; fn from_cascade(cascade: &Cascade) -> Self where Self: Default + Sized, { Self::default().refined(cascade.merged()) } } pub struct Cascade(Vec>); impl Default for Cascade { fn default() -> Self { Self(vec![Some(Default::default())]) } } #[derive(Copy, Clone)] pub struct CascadeSlot(usize); impl Cascade { pub fn reserve(&mut self) -> CascadeSlot { self.0.push(None); CascadeSlot(self.0.len() - 1) } pub fn base(&mut self) -> &mut S::Refinement { self.0[0].as_mut().unwrap() } pub fn set(&mut self, slot: CascadeSlot, refinement: Option) { self.0[slot.0] = refinement } pub fn merged(&self) -> S::Refinement { let mut merged = self.0[0].clone().unwrap(); for refinement in self.0.iter().skip(1).flatten() { merged.refine(refinement); } merged } }