use cursive; use cursive::{ direction::Direction, event::{Event, EventResult, MouseButton, MouseEvent}, impl_enabled, style::PaletteStyle, view::{CannotFocus, View}, Cursive, Printer, Vec2, With, }; use std::sync::Arc; type Callback = dyn Fn(&mut Cursive, bool) + Send + Sync; /// Checkable box. /// /// # Examples /// /// ```rust /// # use cursive_core as cursive; /// use cursive::traits::Nameable; /// use cursive::views::Checkbox; /// /// let checkbox = Checkbox::new().checked().with_name("check"); /// ``` pub struct Checkbox { checked: bool, enabled: bool, on_change: Option>, } impl Checkbox { impl_enabled!(self.enabled); /// Creates a new, unchecked checkbox. pub fn new() -> Self { Checkbox { checked: false, enabled: true, on_change: None, } } /// Sets a callback to be used when the state changes. #[cursive::callback_helpers] pub fn set_on_change( &mut self, on_change: F, ) { self.on_change = Some(Arc::new(on_change)); } /// Sets a callback to be used when the state changes. /// /// Chainable variant. #[must_use] pub fn on_change( self, on_change: F, ) -> Self { self.with(|s| s.set_on_change(on_change)) } /// Toggles the checkbox state. pub fn toggle(&mut self) -> EventResult { let checked = !self.checked; self.set_checked(checked) } /// Check the checkbox. pub fn check(&mut self) -> EventResult { self.set_checked(true) } /// Check the checkbox. /// /// Chainable variant. #[must_use] pub fn checked(self) -> Self { self.with(|s| { s.check(); }) } /// Returns `true` if the checkbox is checked. /// /// # Examples /// /// ``` /// use cursive_core::views::Checkbox; /// /// let mut checkbox = Checkbox::new().checked(); /// assert!(checkbox.is_checked()); /// /// checkbox.uncheck(); /// assert!(!checkbox.is_checked()); /// ``` pub fn is_checked(&self) -> bool { self.checked } /// Uncheck the checkbox. pub fn uncheck(&mut self) -> EventResult { self.set_checked(false) } /// Uncheck the checkbox. /// /// Chainable variant. #[must_use] pub fn unchecked(self) -> Self { self.with(|s| { s.uncheck(); }) } /// Sets the checkbox state. pub fn set_checked(&mut self, checked: bool) -> EventResult { self.checked = checked; if let Some(ref on_change) = self.on_change { let on_change = Arc::clone(on_change); EventResult::with_cb(move |s| on_change(s, checked)) } else { EventResult::Consumed(None) } } /// Set the checkbox state. /// /// Chainable variant. #[must_use] pub fn with_checked(self, is_checked: bool) -> Self { self.with(|s| { s.set_checked(is_checked); }) } fn draw_internal(&self, printer: &Printer) { printer.print((0, 0), "[ ]"); if self.checked { printer.print((1, 0), "X"); } } } impl View for Checkbox { fn required_size(&mut self, _: Vec2) -> Vec2 { Vec2::new(3, 1) } fn take_focus(&mut self, _: Direction) -> Result { self.enabled.then(EventResult::consumed).ok_or(CannotFocus) } fn draw(&self, printer: &Printer) { if self.enabled && printer.enabled { printer.with_selection(printer.focused, |printer| self.draw_internal(printer)); } else { printer.with_style(PaletteStyle::Secondary, |printer| { self.draw_internal(printer) }); } } fn on_event(&mut self, event: Event) -> EventResult { if !self.enabled { return EventResult::Ignored; } match event { Event::Char(' ') => self.toggle(), Event::Mouse { event: MouseEvent::Release(MouseButton::Left), position, offset, } if position.fits_in_rect(offset, (3, 1)) => self.toggle(), _ => EventResult::Ignored, } } } #[cursive::blueprint(Checkbox::new())] struct Blueprint { on_change: Option<_>, checked: Option, enabled: Option, }