cursive/src/view/view_wrapper.rs

195 lines
5.1 KiB
Rust
Raw Normal View History

2016-10-02 22:22:29 +00:00
use Printer;
use direction::Direction;
2016-10-02 22:22:29 +00:00
use event::{Event, EventResult};
use std::any::Any;
2015-05-19 02:41:35 +00:00
use vec::Vec2;
use view::{Selector, View};
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.
2016-09-01 18:56:11 +00:00
///
/// You can use the [`wrap_impl!`] macro to define `with_view` and
/// `with_view_mut` for you.
2016-09-01 18:56:11 +00:00
///
/// [`wrap_impl!`]: ../macro.wrap_impl.html
2017-12-12 02:54:40 +00:00
pub trait ViewWrapper: 'static {
/// Type that this view wraps.
type V: View + ?Sized;
2017-09-25 01:50:45 +00:00
/// Runs a function on the inner view, returning the result.
///
2017-09-25 01:50:45 +00:00
/// Returns `None` if the inner view is unavailable. This should only
/// happen with some views if they are already borrowed by another call.
fn with_view<F, R>(&self, f: F) -> Option<R>
where
F: FnOnce(&Self::V) -> R;
2016-07-11 02:11:21 +00:00
2017-09-25 01:50:45 +00:00
/// Runs a function on the inner view, returning the result.
///
2017-09-25 01:50:45 +00:00
/// Returns `None` if the inner view is unavailable. This should only
/// happen with some views if they are already borrowed by another call.
fn with_view_mut<F, R>(&mut self, f: F) -> Option<R>
where
F: FnOnce(&mut Self::V) -> R;
2015-05-19 02:41:35 +00:00
/// Attempts to retrieve the inner view.
fn into_inner(self) -> Result<Self::V, Self>
where
Self: Sized,
Self::V: Sized,
{
Err(self)
}
/// Wraps the `draw` method.
fn wrap_draw(&self, printer: &Printer) {
self.with_view(|v| v.draw(printer));
2015-05-19 02:41:35 +00:00
}
/// Wraps the `required_size` method.
fn wrap_required_size(&mut self, req: Vec2) -> Vec2 {
self.with_view_mut(|v| v.required_size(req))
.unwrap_or_else(Vec2::zero)
2015-05-19 02:41:35 +00:00
}
/// Wraps the `on_event` method.
fn wrap_on_event(&mut self, ch: Event) -> EventResult {
2017-10-12 23:38:55 +00:00
self.with_view_mut(|v| v.on_event(ch))
.unwrap_or(EventResult::Ignored)
2015-05-19 02:41:35 +00:00
}
/// Wraps the `layout` method.
2015-05-19 02:41:35 +00:00
fn wrap_layout(&mut self, size: Vec2) {
self.with_view_mut(|v| v.layout(size));
2015-05-19 02:41:35 +00:00
}
2015-05-20 00:31:52 +00:00
/// Wraps the `take_focus` method.
fn wrap_take_focus(&mut self, source: Direction) -> bool {
2017-10-12 23:38:55 +00:00
self.with_view_mut(|v| v.take_focus(source))
.unwrap_or(false)
2015-05-20 00:31:52 +00:00
}
2015-05-23 17:33:29 +00:00
/// Wraps the `find` method.
2017-10-12 23:38:55 +00:00
fn wrap_call_on_any<'a>(
2017-12-30 22:03:42 +00:00
&mut self, selector: &Selector, callback: Box<FnMut(&mut Any) + 'a>
2017-10-12 23:38:55 +00:00
) {
self.with_view_mut(|v| v.call_on_any(selector, callback));
2015-05-23 17:33:29 +00:00
}
2017-03-25 21:50:52 +00:00
/// Wraps the `focus_view` method.
fn wrap_focus_view(&mut self, selector: &Selector) -> Result<(), ()> {
2017-10-12 23:38:55 +00:00
self.with_view_mut(|v| v.focus_view(selector))
.unwrap_or(Err(()))
2017-03-25 21:50:52 +00:00
}
/// Wraps the `needs_relayout` method.
fn wrap_needs_relayout(&self) -> bool {
self.with_view(|v| v.needs_relayout()).unwrap_or(true)
}
2015-05-19 02:41:35 +00:00
}
2017-09-25 01:50:45 +00:00
// Some types easily implement ViewWrapper.
// This includes Box<T: View>
use std::ops::{Deref, DerefMut};
2017-12-30 22:03:42 +00:00
impl<U: View + ?Sized, T: Deref<Target = U> + DerefMut + 'static> ViewWrapper
for T {
type V = U;
fn with_view<F, R>(&self, f: F) -> Option<R>
where
F: FnOnce(&Self::V) -> R,
{
Some(f(self.deref()))
}
fn with_view_mut<F, R>(&mut self, f: F) -> Option<R>
where
F: FnOnce(&mut Self::V) -> R,
{
Some(f(self.deref_mut()))
}
}
2017-09-25 01:50:45 +00:00
// The main point of implementing ViewWrapper is to have View for free.
2016-06-25 23:36:22 +00:00
impl<T: ViewWrapper> View for T {
fn draw(&self, printer: &Printer) {
self.wrap_draw(printer);
2015-05-19 02:41:35 +00:00
}
fn required_size(&mut self, req: Vec2) -> Vec2 {
self.wrap_required_size(req)
2015-05-19 02:41:35 +00:00
}
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, source: Direction) -> bool {
self.wrap_take_focus(source)
2015-05-20 00:31:52 +00:00
}
2015-05-23 17:33:29 +00:00
2017-10-12 23:38:55 +00:00
fn call_on_any<'a>(
2017-12-30 22:03:42 +00:00
&mut self, selector: &Selector, callback: Box<FnMut(&mut Any) + 'a>
2017-10-12 23:38:55 +00:00
) {
self.wrap_call_on_any(selector, callback)
2015-05-23 17:33:29 +00:00
}
fn needs_relayout(&self) -> bool {
self.wrap_needs_relayout()
}
2017-03-25 21:50:52 +00:00
fn focus_view(&mut self, selector: &Selector) -> Result<(), ()> {
self.wrap_focus_view(selector)
}
2015-05-19 02:41:35 +00:00
}
2016-09-01 18:56:11 +00:00
/// Convenient macro to implement the [`ViewWrapper`] trait.
///
/// It defines the `with_view` and `with_view_mut` implementations,
/// as well as the `type V` declaration.
///
2016-09-01 18:56:11 +00:00
/// [`ViewWrapper`]: view/trait.ViewWrapper.html
2015-05-20 00:31:52 +00:00
///
/// # Examples
///
/// ```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> {
/// wrap_impl!(self.view: T);
2015-05-20 00:31:52 +00:00
/// }
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: $t:ty) => {
type V = $t;
fn with_view<F, R>(&self, f: F) -> Option<R>
where F: FnOnce(&Self::V) -> R
{
Some(f(&self.$v))
}
fn with_view_mut<F, R>(&mut self, f: F) -> Option<R>
where F: FnOnce(&mut Self::V) -> R
{
Some(f(&mut self.$v))
}
fn into_inner(self) -> Result<Self::V, Self> where Self::V: Sized {
Ok(self.$v)
}
};
}