Skip to main content

gtk4/
callback_action.rs

1// Take a look at the license at the top of the repository in the LICENSE file.
2
3use glib::translate::*;
4use std::boxed::Box as Box_;
5
6use crate::{CallbackAction, Widget, ffi};
7
8impl CallbackAction {
9    /// Create a custom action that calls the given @callback when
10    /// activated.
11    /// ## `callback`
12    /// the callback
13    ///   to call when the action is activated
14    ///
15    /// # Returns
16    ///
17    /// A new shortcut action
18    #[doc(alias = "gtk_callback_action_new")]
19    pub fn new<P: Fn(&Widget, Option<&glib::Variant>) -> glib::Propagation + 'static>(
20        callback: P,
21    ) -> CallbackAction {
22        assert_initialized_main_thread!();
23        let callback_data: Box_<P> = Box_::new(callback);
24        unsafe extern "C" fn callback_func<
25            P: Fn(&Widget, Option<&glib::Variant>) -> glib::Propagation + 'static,
26        >(
27            widget: *mut ffi::GtkWidget,
28            args: *mut glib::ffi::GVariant,
29            user_data: glib::ffi::gpointer,
30        ) -> glib::ffi::gboolean {
31            unsafe {
32                let widget = from_glib_borrow(widget);
33                let args: Borrowed<Option<glib::Variant>> = from_glib_borrow(args);
34                let callback = &*(user_data as *mut P);
35                (*callback)(&widget, args.as_ref().as_ref()).into_glib()
36            }
37        }
38        let callback = Some(callback_func::<P> as _);
39        unsafe extern "C" fn destroy_func<
40            P: Fn(&Widget, Option<&glib::Variant>) -> glib::Propagation + 'static,
41        >(
42            data: glib::ffi::gpointer,
43        ) {
44            unsafe {
45                let _callback = Box_::from_raw(data as *mut P);
46            }
47        }
48        let destroy_call2 = Some(destroy_func::<P> as _);
49        let super_callback0: Box_<P> = callback_data;
50        unsafe {
51            from_glib_full(ffi::gtk_callback_action_new(
52                callback,
53                Box_::into_raw(super_callback0) as *mut _,
54                destroy_call2,
55            ))
56        }
57    }
58}