mirror of
https://github.com/DioxusLabs/dioxus
synced 2025-03-08 01:07:43 +00:00
165 lines
4 KiB
Rust
165 lines
4 KiB
Rust
use std::{
|
|
cell::{Cell, RefCell},
|
|
rc::Rc,
|
|
};
|
|
|
|
/// A wrapper around some generic data that handles the event's state
|
|
///
|
|
///
|
|
/// Prevent this event from continuing to bubble up the tree to parent elements.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```rust, ignore
|
|
/// rsx! {
|
|
/// button {
|
|
/// onclick: move |evt: Event<MouseData>| {
|
|
/// evt.cancel_bubble();
|
|
///
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
pub struct Event<T: 'static + ?Sized> {
|
|
/// The data associated with this event
|
|
pub data: Rc<T>,
|
|
pub(crate) propogates: Rc<Cell<bool>>,
|
|
}
|
|
|
|
impl<T> Event<T> {
|
|
/// Prevent this event from continuing to bubble up the tree to parent elements.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```rust, ignore
|
|
/// rsx! {
|
|
/// button {
|
|
/// onclick: move |evt: Event<MouseData>| {
|
|
/// evt.cancel_bubble();
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
#[deprecated = "use stop_propogation instead"]
|
|
pub fn cancel_bubble(&self) {
|
|
self.propogates.set(false);
|
|
}
|
|
|
|
/// Prevent this event from continuing to bubble up the tree to parent elements.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```rust, ignore
|
|
/// rsx! {
|
|
/// button {
|
|
/// onclick: move |evt: Event<MouseData>| {
|
|
/// evt.cancel_bubble();
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
pub fn stop_propogation(&self) {
|
|
self.propogates.set(false);
|
|
}
|
|
|
|
/// Get a reference to the inner data from this event
|
|
///
|
|
/// ```rust, ignore
|
|
/// rsx! {
|
|
/// button {
|
|
/// onclick: move |evt: Event<MouseData>| {
|
|
/// let data = evt.inner.clone();
|
|
/// cx.spawn(async move {
|
|
/// println!("{:?}", data);
|
|
/// });
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
pub fn inner(&self) -> &Rc<T> {
|
|
&self.data
|
|
}
|
|
}
|
|
|
|
impl<T: ?Sized> Clone for Event<T> {
|
|
fn clone(&self) -> Self {
|
|
Self {
|
|
propogates: self.propogates.clone(),
|
|
data: self.data.clone(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> std::ops::Deref for Event<T> {
|
|
type Target = Rc<T>;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.data
|
|
}
|
|
}
|
|
|
|
impl<T: std::fmt::Debug> std::fmt::Debug for Event<T> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.debug_struct("UiEvent")
|
|
.field("bubble_state", &self.propogates)
|
|
.field("data", &self.data)
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
/// The callback type generated by the `rsx!` macro when an `on` field is specified for components.
|
|
///
|
|
/// This makes it possible to pass `move |evt| {}` style closures into components as property fields.
|
|
///
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```rust, ignore
|
|
/// rsx!{
|
|
/// MyComponent { onclick: move |evt| log::info!("clicked") }
|
|
/// }
|
|
///
|
|
/// #[derive(Props)]
|
|
/// struct MyProps<'a> {
|
|
/// onclick: EventHandler<'a, MouseEvent>,
|
|
/// }
|
|
///
|
|
/// fn MyComponent(cx: Scope<'a, MyProps<'a>>) -> Element {
|
|
/// cx.render(rsx!{
|
|
/// button {
|
|
/// onclick: move |evt| cx.props.onclick.call(evt),
|
|
/// }
|
|
/// })
|
|
/// }
|
|
///
|
|
/// ```
|
|
pub struct EventHandler<'bump, T = ()> {
|
|
pub(super) callback: RefCell<Option<ExternalListenerCallback<'bump, T>>>,
|
|
}
|
|
|
|
impl<T> Default for EventHandler<'_, T> {
|
|
fn default() -> Self {
|
|
Self {
|
|
callback: Default::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
type ExternalListenerCallback<'bump, T> = bumpalo::boxed::Box<'bump, dyn FnMut(T) + 'bump>;
|
|
|
|
impl<T> EventHandler<'_, T> {
|
|
/// Call this event handler with the appropriate event type
|
|
///
|
|
/// This borrows the event using a RefCell. Recursively calling a listener will cause a panic.
|
|
pub fn call(&self, event: T) {
|
|
if let Some(callback) = self.callback.borrow_mut().as_mut() {
|
|
callback(event);
|
|
}
|
|
}
|
|
|
|
/// Forcibly drop the internal handler callback, releasing memory
|
|
///
|
|
/// This will force any future calls to "call" to not doing anything
|
|
pub fn release(&self) {
|
|
self.callback.replace(None);
|
|
}
|
|
}
|