cursive/src/view/view_wrapper.rs

130 lines
3.0 KiB
Rust
Raw Normal View History

2015-05-23 17:33:29 +00:00
use std::any::Any;
2015-05-19 02:41:35 +00:00
use vec::Vec2;
2016-03-15 22:37:57 +00:00
use view::{View, SizeRequest, Selector};
2015-05-19 02:41:35 +00:00
use printer::Printer;
2016-03-15 22:37:57 +00:00
use event::{Event, EventResult};
2015-05-19 02:41:35 +00:00
2015-05-23 17:33:29 +00:00
/// Generic wrapper around a view.
///
/// Default implementation forwards all calls to the child view.
/// Overrides some methods as desired.
2015-05-19 02:41:35 +00:00
pub trait ViewWrapper {
2015-05-20 00:31:52 +00:00
/// Get an immutable reference to the wrapped view, so that we can forward some calls to it.
2015-05-19 02:41:35 +00:00
fn get_view(&self) -> &View;
2015-05-20 00:31:52 +00:00
/// Get a mutable reference to the wrapped view, for the mutable methods.
2015-05-19 02:41:35 +00:00
fn get_view_mut(&mut self) -> &mut View;
2015-05-20 00:31:52 +00:00
/// Wraps the draw method.
fn wrap_draw(&mut self, printer: &Printer) {
self.get_view_mut().draw(printer);
2015-05-19 02:41:35 +00:00
}
2015-05-20 00:31:52 +00:00
/// Wraps the get_min_size method.
2015-05-19 02:41:35 +00:00
fn wrap_get_min_size(&self, req: SizeRequest) -> Vec2 {
self.get_view().get_min_size(req)
}
/// Wraps the on_event method.
fn wrap_on_event(&mut self, ch: Event) -> EventResult {
self.get_view_mut().on_event(ch)
2015-05-19 02:41:35 +00:00
}
2015-05-20 00:31:52 +00:00
/// Wraps the layout method
2015-05-19 02:41:35 +00:00
fn wrap_layout(&mut self, size: Vec2) {
self.get_view_mut().layout(size);
}
2015-05-20 00:31:52 +00:00
/// Wraps the take_focus method
fn wrap_take_focus(&mut self) -> bool {
self.get_view_mut().take_focus()
}
2015-05-23 17:33:29 +00:00
fn wrap_find(&mut self, selector: &Selector) -> Option<&mut Any> {
self.get_view_mut().find(selector)
}
2015-05-19 02:41:35 +00:00
}
2016-06-25 23:36:22 +00:00
impl<T: ViewWrapper> View for T {
fn draw(&mut self, printer: &Printer) {
self.wrap_draw(printer);
2015-05-19 02:41:35 +00:00
}
fn get_min_size(&self, req: SizeRequest) -> Vec2 {
self.wrap_get_min_size(req)
}
fn on_event(&mut self, ch: Event) -> EventResult {
self.wrap_on_event(ch)
2015-05-19 02:41:35 +00:00
}
fn layout(&mut self, size: Vec2) {
self.wrap_layout(size);
}
2015-05-20 00:31:52 +00:00
fn take_focus(&mut self) -> bool {
self.wrap_take_focus()
}
2015-05-23 17:33:29 +00:00
fn find(&mut self, selector: &Selector) -> Option<&mut Any> {
self.wrap_find(selector)
}
2015-05-19 02:41:35 +00:00
}
2015-05-20 00:31:52 +00:00
/// Convenient macro to implement to two methods required for the ViewWrapper trait.
///
/// # Examples
///
/// If the wrapped view is in a box, just name it in the macro:
///
/// ```no_run
2015-06-04 18:40:35 +00:00
/// # #[macro_use] extern crate cursive;
/// # use cursive::view::{View,ViewWrapper};
2015-05-20 00:31:52 +00:00
/// struct BoxFooView {
/// content: Box<View>,
/// }
///
2015-06-04 18:40:35 +00:00
/// impl ViewWrapper for BoxFooView {
2015-05-20 00:31:52 +00:00
/// wrap_impl!(self.content);
/// }
2015-06-04 18:40:35 +00:00
/// # fn main() { }
2015-05-20 00:31:52 +00:00
/// ```
///
/// If the content is directly a view, reference it:
///
/// ```no_run
2015-06-04 18:40:35 +00:00
/// # #[macro_use] extern crate cursive;
/// # use cursive::view::{View,ViewWrapper};
2015-05-20 00:31:52 +00:00
/// struct FooView<T: View> {
/// view: T,
/// }
///
2015-06-04 18:40:35 +00:00
/// impl <T: View> ViewWrapper for FooView<T> {
2015-05-20 00:31:52 +00:00
/// wrap_impl!(&self.view);
/// }
2015-06-04 18:40:35 +00:00
/// # fn main() { }
2015-05-20 00:31:52 +00:00
/// ```
#[macro_export]
macro_rules! wrap_impl {
(&self.$v:ident) => {
fn get_view(&self) -> &View {
&self.$v
}
fn get_view_mut(&mut self) -> &mut View {
&mut self.$v
}
};
(self.$v:ident) => {
fn get_view(&self) -> &View {
&*self.$v
}
fn get_view_mut(&mut self) -> &mut View {
&mut *self.$v
}
};
}