2017-03-26 04:56:31 +00:00
|
|
|
use owning_ref::{RcRef, OwningHandle};
|
2017-03-27 03:50:50 +00:00
|
|
|
use std::any::Any;
|
2015-05-23 17:33:29 +00:00
|
|
|
|
2017-03-26 04:56:31 +00:00
|
|
|
use std::cell::{RefCell, RefMut};
|
|
|
|
use std::rc::Rc;
|
2016-06-28 05:10:59 +00:00
|
|
|
use view::{Selector, View, ViewWrapper};
|
2015-05-23 17:33:29 +00:00
|
|
|
|
2017-03-26 04:56:31 +00:00
|
|
|
/// Wrapper around a view to provide interior mutability.
|
|
|
|
pub struct IdView<V: View> {
|
|
|
|
view: Rc<RefCell<V>>,
|
2015-05-23 17:33:29 +00:00
|
|
|
id: String,
|
|
|
|
}
|
|
|
|
|
2017-03-26 04:56:31 +00:00
|
|
|
/// Mutable reference to a view.
|
2017-03-27 03:50:50 +00:00
|
|
|
///
|
|
|
|
/// This behaves like a [`RefMut`], but without being tied to a lifetime.
|
|
|
|
///
|
|
|
|
/// [`RefMut`]: https://doc.rust-lang.org/std/cell/struct.RefMut.html
|
2017-03-26 04:56:31 +00:00
|
|
|
pub type ViewRef<V> = OwningHandle<RcRef<RefCell<V>>, RefMut<'static, V>>;
|
|
|
|
|
|
|
|
impl<V: View> IdView<V> {
|
|
|
|
/// Wraps `view` in a new `IdView`.
|
|
|
|
pub fn new<S: Into<String>>(id: S, view: V) -> Self {
|
2015-05-23 17:33:29 +00:00
|
|
|
IdView {
|
2017-03-26 04:56:31 +00:00
|
|
|
view: Rc::new(RefCell::new(view)),
|
|
|
|
id: id.into(),
|
2015-05-23 17:33:29 +00:00
|
|
|
}
|
|
|
|
}
|
2017-03-26 04:56:31 +00:00
|
|
|
|
|
|
|
/// Gets mutable access to the inner view.
|
2017-03-27 03:50:50 +00:00
|
|
|
///
|
|
|
|
/// This returns a `ViewRef<V>`, which implement `DerefMut<Target = V>`.
|
2017-03-26 04:56:31 +00:00
|
|
|
pub fn get_mut(&mut self) -> ViewRef<V> {
|
|
|
|
// TODO: return a standalone item (not tied to our lifetime)
|
|
|
|
// that bundles `self.view.clone()` and allow mutable reference to
|
|
|
|
// the inner view.
|
|
|
|
let cell_ref = RcRef::new(self.view.clone());
|
|
|
|
|
|
|
|
OwningHandle::new(cell_ref,
|
|
|
|
|x| unsafe { x.as_ref() }.unwrap().borrow_mut())
|
|
|
|
}
|
2015-05-23 17:33:29 +00:00
|
|
|
}
|
|
|
|
|
2017-03-26 04:56:31 +00:00
|
|
|
impl<T: View + 'static> ViewWrapper for IdView<T> {
|
|
|
|
type V = T;
|
|
|
|
|
|
|
|
fn with_view<F, R>(&self, f: F) -> Option<R>
|
|
|
|
where F: FnOnce(&Self::V) -> R
|
|
|
|
{
|
|
|
|
self.view
|
|
|
|
.try_borrow()
|
|
|
|
.ok()
|
|
|
|
.map(|v| f(&*v))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_view_mut<F, R>(&mut self, f: F) -> Option<R>
|
|
|
|
where F: FnOnce(&mut Self::V) -> R
|
|
|
|
{
|
|
|
|
self.view
|
|
|
|
.try_borrow_mut()
|
|
|
|
.ok()
|
|
|
|
.map(|mut v| f(&mut *v))
|
|
|
|
}
|
2015-05-23 17:33:29 +00:00
|
|
|
|
2017-03-27 03:50:50 +00:00
|
|
|
fn wrap_call_on_any<'a>(&mut self, selector: &Selector,
|
2017-03-26 04:56:31 +00:00
|
|
|
mut callback: Box<for<'b> FnMut(&'b mut Any) + 'a>) {
|
|
|
|
let result = match selector {
|
|
|
|
&Selector::Id(id) if id == self.id => callback(self),
|
2017-03-27 03:50:50 +00:00
|
|
|
s => self.view.borrow_mut().call_on_any(s, callback),
|
2017-03-26 04:56:31 +00:00
|
|
|
};
|
|
|
|
result
|
2015-05-23 17:33:29 +00:00
|
|
|
}
|
2017-03-25 21:50:52 +00:00
|
|
|
|
|
|
|
fn wrap_focus_view(&mut self, selector: &Selector) -> Result<(), ()> {
|
|
|
|
match selector {
|
|
|
|
&Selector::Id(id) if id == self.id => Ok(()),
|
2017-03-26 04:56:31 +00:00
|
|
|
s => self.view.borrow_mut().focus_view(s),
|
2017-03-25 21:50:52 +00:00
|
|
|
}
|
|
|
|
}
|
2015-05-23 17:33:29 +00:00
|
|
|
}
|