use crate::event::AnyCb;
use crate::view::{Selector, View, ViewWrapper};
use crate::Vec2;
use crate::With;
pub struct HideableView<V> {
view: V,
visible: bool,
invalidated: bool,
}
new_default!(HideableView<V: Default>);
impl<V> HideableView<V> {
pub fn new(view: V) -> Self {
HideableView {
view,
visible: true,
invalidated: true,
}
}
pub fn set_visible(&mut self, visible: bool) {
self.visible = visible;
self.invalidate();
}
pub fn hide(&mut self) {
self.set_visible(false);
}
pub fn unhide(&mut self) {
self.set_visible(true);
}
#[must_use]
pub fn hidden(self) -> Self {
self.with(Self::hide)
}
#[must_use]
pub fn visible(self, flag: bool) -> Self {
self.with(|s| s.set_visible(flag))
}
pub fn is_visible(&self) -> bool {
self.visible
}
fn invalidate(&mut self) {
self.invalidated = true;
}
inner_getters!(self.view: V);
}
impl<V: View> ViewWrapper for HideableView<V> {
type V = V;
fn with_view<F, R>(&self, f: F) -> Option<R>
where
F: FnOnce(&Self::V) -> R,
{
if self.visible {
Some(f(&self.view))
} else {
None
}
}
fn with_view_mut<F, R>(&mut self, f: F) -> Option<R>
where
F: FnOnce(&mut Self::V) -> R,
{
if self.visible {
Some(f(&mut self.view))
} else {
None
}
}
fn wrap_call_on_any(&mut self, selector: &Selector, callback: AnyCb) {
self.view.call_on_any(selector, callback)
}
fn into_inner(self) -> Result<Self::V, Self>
where
Self: Sized,
Self::V: Sized,
{
Ok(self.view)
}
fn wrap_layout(&mut self, size: Vec2) {
self.invalidated = false;
self.with_view_mut(|v| v.layout(size));
}
fn wrap_needs_relayout(&self) -> bool {
self.invalidated || (self.visible && self.view.needs_relayout())
}
}
#[crate::blueprint(HideableView::new(view))]
struct Blueprint {
view: crate::views::BoxedView,
visible: Option<bool>,
}
crate::manual_blueprint!(with hideable, |config, context| {
let visible: Option<bool> = context.resolve(&config["visible"])?;
Ok(move |view| HideableView::new(view).visible(visible.unwrap_or(true)))
});