dioxus_html/events/
drag.rs

1use crate::geometry::{ClientPoint, Coordinates, ElementPoint, PagePoint, ScreenPoint};
2use crate::input_data::{MouseButton, MouseButtonSet};
3use crate::prelude::*;
4
5use dioxus_core::Event;
6use keyboard_types::Modifiers;
7
8use crate::HasMouseData;
9
10pub type DragEvent = Event<DragData>;
11
12/// The DragEvent interface is a DOM event that represents a drag and drop interaction. The user initiates a drag by
13/// placing a pointer device (such as a mouse) on the touch surface and then dragging the pointer to a new location
14/// (such as another DOM element). Applications are free to interpret a drag and drop interaction in an
15/// application-specific way.
16pub struct DragData {
17    inner: Box<dyn HasDragData>,
18}
19
20impl<E: HasDragData + 'static> From<E> for DragData {
21    fn from(e: E) -> Self {
22        Self { inner: Box::new(e) }
23    }
24}
25
26impl std::fmt::Debug for DragData {
27    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
28        f.debug_struct("DragData")
29            .field("coordinates", &self.coordinates())
30            .field("modifiers", &self.modifiers())
31            .field("held_buttons", &self.held_buttons())
32            .field("trigger_button", &self.trigger_button())
33            .finish()
34    }
35}
36
37impl PartialEq for DragData {
38    fn eq(&self, other: &Self) -> bool {
39        self.coordinates() == other.coordinates()
40            && self.modifiers() == other.modifiers()
41            && self.held_buttons() == other.held_buttons()
42            && self.trigger_button() == other.trigger_button()
43    }
44}
45
46impl DragData {
47    /// Create a new DragData
48    pub fn new(inner: impl HasDragData + 'static) -> Self {
49        Self {
50            inner: Box::new(inner),
51        }
52    }
53
54    /// Downcast this event data to a specific type
55    #[inline(always)]
56    pub fn downcast<T: 'static>(&self) -> Option<&T> {
57        HasDragData::as_any(&*self.inner).downcast_ref::<T>()
58    }
59}
60
61impl crate::HasFileData for DragData {
62    fn files(&self) -> Option<std::sync::Arc<dyn crate::file_data::FileEngine>> {
63        self.inner.files()
64    }
65}
66
67impl InteractionLocation for DragData {
68    fn client_coordinates(&self) -> ClientPoint {
69        self.inner.client_coordinates()
70    }
71
72    fn page_coordinates(&self) -> PagePoint {
73        self.inner.page_coordinates()
74    }
75
76    fn screen_coordinates(&self) -> ScreenPoint {
77        self.inner.screen_coordinates()
78    }
79}
80
81impl InteractionElementOffset for DragData {
82    fn element_coordinates(&self) -> ElementPoint {
83        self.inner.element_coordinates()
84    }
85
86    fn coordinates(&self) -> Coordinates {
87        self.inner.coordinates()
88    }
89}
90
91impl ModifiersInteraction for DragData {
92    fn modifiers(&self) -> Modifiers {
93        self.inner.modifiers()
94    }
95}
96
97impl PointerInteraction for DragData {
98    fn held_buttons(&self) -> MouseButtonSet {
99        self.inner.held_buttons()
100    }
101
102    // todo the following is kind of bad; should we just return None when the trigger_button is unreliable (and frankly irrelevant)? i guess we would need the event_type here
103    fn trigger_button(&self) -> Option<MouseButton> {
104        self.inner.trigger_button()
105    }
106}
107
108#[cfg(feature = "serialize")]
109/// A serialized version of DragData
110#[derive(serde::Serialize, serde::Deserialize, Debug, PartialEq, Clone)]
111pub struct SerializedDragData {
112    pub mouse: crate::point_interaction::SerializedPointInteraction,
113
114    #[serde(default)]
115    files: Option<crate::file_data::SerializedFileEngine>,
116}
117
118#[cfg(feature = "serialize")]
119impl SerializedDragData {
120    fn new(drag: &DragData) -> Self {
121        Self {
122            mouse: crate::point_interaction::SerializedPointInteraction::from(drag),
123            files: None,
124        }
125    }
126}
127
128#[cfg(feature = "serialize")]
129impl HasDragData for SerializedDragData {
130    fn as_any(&self) -> &dyn std::any::Any {
131        self
132    }
133}
134
135#[cfg(feature = "serialize")]
136impl crate::file_data::HasFileData for SerializedDragData {
137    fn files(&self) -> Option<std::sync::Arc<dyn crate::file_data::FileEngine>> {
138        self.files
139            .as_ref()
140            .map(|files| std::sync::Arc::new(files.clone()) as _)
141    }
142}
143
144#[cfg(feature = "serialize")]
145impl HasMouseData for SerializedDragData {
146    fn as_any(&self) -> &dyn std::any::Any {
147        self
148    }
149}
150
151#[cfg(feature = "serialize")]
152impl InteractionLocation for SerializedDragData {
153    fn client_coordinates(&self) -> ClientPoint {
154        self.mouse.client_coordinates()
155    }
156
157    fn page_coordinates(&self) -> PagePoint {
158        self.mouse.page_coordinates()
159    }
160
161    fn screen_coordinates(&self) -> ScreenPoint {
162        self.mouse.screen_coordinates()
163    }
164}
165
166#[cfg(feature = "serialize")]
167impl InteractionElementOffset for SerializedDragData {
168    fn element_coordinates(&self) -> ElementPoint {
169        self.mouse.element_coordinates()
170    }
171
172    fn coordinates(&self) -> Coordinates {
173        self.mouse.coordinates()
174    }
175}
176
177#[cfg(feature = "serialize")]
178impl ModifiersInteraction for SerializedDragData {
179    fn modifiers(&self) -> Modifiers {
180        self.mouse.modifiers()
181    }
182}
183
184#[cfg(feature = "serialize")]
185impl PointerInteraction for SerializedDragData {
186    fn held_buttons(&self) -> MouseButtonSet {
187        self.mouse.held_buttons()
188    }
189
190    fn trigger_button(&self) -> Option<MouseButton> {
191        self.mouse.trigger_button()
192    }
193}
194
195#[cfg(feature = "serialize")]
196impl serde::Serialize for DragData {
197    fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
198        SerializedDragData::new(self).serialize(serializer)
199    }
200}
201
202#[cfg(feature = "serialize")]
203impl<'de> serde::Deserialize<'de> for DragData {
204    fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
205        let data = SerializedDragData::deserialize(deserializer)?;
206        Ok(Self {
207            inner: Box::new(data),
208        })
209    }
210}
211
212/// A trait for any object that has the data for a drag event
213pub trait HasDragData: HasMouseData + crate::HasFileData {
214    /// return self as Any
215    fn as_any(&self) -> &dyn std::any::Any;
216}
217
218impl_event! {
219    DragData;
220
221    /// ondrag
222    ondrag
223
224    /// ondragend
225    ondragend
226
227    /// ondragenter
228    ondragenter
229
230    /// ondragexit
231    ondragexit
232
233    /// ondragleave
234    ondragleave
235
236    /// ondragover
237    ondragover
238
239    /// ondragstart
240    ondragstart
241
242    /// ondrop
243    ondrop
244}