use std::cmp::max; use vec::Vec2; use view::{View,SizeRequest,DimensionRequest}; use event::EventResult; use printer::Printer; /// Simple stack of views. /// Only the top-most view is active and can receive input. pub struct StackView { layers: Vec, } struct Layer { view: Box, size: Vec2, } impl StackView { /// Creates a new empty StackView pub fn new() -> Self { StackView { layers: Vec::new(), } } /// Add new view on top of the stack. pub fn add_layer(&mut self, view: T) { self.layers.push(Layer { view: Box::new(view), size: Vec2::new(0,0), }); } /// Remove the top-most layer. pub fn pop_layer(&mut self) { self.layers.pop(); } } impl View for StackView { fn draw(&self, printer: &Printer, focused: bool) { match self.layers.last() { None => (), Some(v) => { // Center the view let view_size = Vec2::min(printer.size, v.size); let offset = (printer.size - view_size) / 2; v.view.draw(&printer.sub_printer(offset, v.size), focused); }, } } fn on_key_event(&mut self, ch: i32) -> EventResult { match self.layers.last_mut() { None => EventResult::Ignored, Some(v) => v.view.on_key_event(ch), } } fn layout(&mut self, size: Vec2) { let req = SizeRequest { w: DimensionRequest::AtMost(size.x), h: DimensionRequest::AtMost(size.y), }; for layer in self.layers.iter_mut() { layer.size = layer.view.get_min_size(req); layer.view.layout(layer.size); } } fn get_min_size(&self, size: SizeRequest) -> Vec2 { // The min size is the max of all children's let mut s = Vec2::new(1,1); for layer in self.layers.iter() { let vs = layer.view.get_min_size(size); s.x = max(s.x, vs.x); s.y = max(s.y, vs.y); } s } fn take_focus(&mut self) -> bool { match self.layers.last_mut() { None => false, Some(mut v) => v.view.take_focus() } } }