newt 0.7.2

Rust bindings for the Newt console UI library
Documentation
//
// Copyright (C) 2019 Robert Gill <rtgill82@gmail.com>
//
// This file is a part of newt-rs.
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License version 2.1 as published by the Free Software Foundation.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
//

use crate::component::Component;
use crate::private::funcs::*;
use newt_sys::*;

///
/// A callback called when a newt `Component` is activated.
///
/// ## Example
/// ```rust no_run
/// extern crate newt;
/// use newt::Callback;
/// use newt::prelude::*;
///
/// pub fn main() {
///     // Receives the new value when the Callback is activated.
///     let mut value: i32 = 0;
///
///     newt::init().unwrap();
///     newt::cls();
///     newt::centered_window(20, 6, Some("Callback Test")).unwrap();
///
///     let cb1 = Checkbox::new(3, 1, "Check 1", None, None);
///     let cb2 = Checkbox::new(3, 2, "Check 2", None, None);
///     let ok = CompactButton::new(7, 4, "Ok");
///
///     let mut form = Form::new(None, 0);
///     form.add_components(&[&cb1, &cb2, &ok]).unwrap();
///
///     // Closure `f` borrows `value` as mutable so create a new subscope here
///     // allowing `value` to be borrowed immutably when printing the result
///     // later.
///     {
///         // Create closure to be called by Callback
///         let mut f = |_c: &dyn Component, data: Option<&i32>| {
///             value = *data.unwrap();
///         };
///
///         // Create Callback with first Checkbox using `5` as data.
///         let mut callback = Callback::new(&cb1, Some(5), &mut f);
///         // Add second Checkbox using `10` as data.
///         callback.add_component(&cb2, Some(10));
///
///         form.run().unwrap();
///         newt::finished();
///     }
///
///     // `value` will be 0, 5, or 10 depending on the last Checkbox "clicked".
///     println!("value = {}", value);
/// }
/// ```
///
pub struct Callback<'a, F: 'a, T: 'a>
where F: FnMut(&dyn Component, Option<&T>)
{
    function: F,
    components: Vec<(&'a dyn Component, Option<T>)>
}

impl<'a, F: 'a, T: 'a> Callback<'a, F, T>
where F: FnMut(&dyn Component, Option<&T>)
{
    ///
    /// Create a new `Callback` using the function or closure `function` and
    /// associate it with `component`.
    ///
    /// * `component` - The `Component` associated with the callback.
    /// * `data` - Optional user data to pass to the function.
    /// * `function` - The function or closure to call when the
    ///   `Component` is activated.
    ///
    pub fn new(component: &'a dyn Component, data: Option<T>, function: F)
      -> Box<Callback<'a, F, T>> {

        unsafe {
            let cb = Box::new(Callback {
                function,
                components: vec![(component, data)]
            });
            newt_set_callback(component.co(), cb.as_ref());
            cb
        }
    }

    ///
    /// Associate another component with the `Callback`.
    ///
    /// * `component` - Another `Component` to associate with the callback.
    /// * `data` - Optional user data to pass to the function.
    ///
    pub fn add_component(&mut self, component: &'a dyn Component,
                         data: Option<T>)
    {
        unsafe {
            self.components.push((component, data));
            newt_set_callback(component.co(), self);
        }
    }

    pub(crate) fn call(&mut self, co: newtComponent) {
        for (component, data) in self.components.iter() {
            if component.co() == co {
                (self.function)(*component, data.as_ref());
            }
        }
    }
}

impl<'a, F: 'a, T: 'a> Drop for Callback<'a, F, T>
where F: FnMut(&dyn Component, Option<&T>)
{
    fn drop(&mut self) {
        unsafe {
            for (component, _data) in self.components.iter() {
                newt_unset_callback(*component)
            }
        }
    }
}