implement HasMouseData for WheelEvent (#2728)

This commit is contained in:
Adam Kundrát 2024-07-29 21:28:06 +02:00 committed by GitHub
parent 23dfe39bdf
commit 4e338accad
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 171 additions and 8 deletions

View file

@ -1,10 +1,17 @@
use dioxus_core::Event;
use std::fmt::Formatter;
use crate::geometry::WheelDelta;
use crate::geometry::*;
use crate::input_data::{MouseButton, MouseButtonSet};
use crate::prelude::*;
use super::HasMouseData;
/// A synthetic event that wraps a web-style
/// [`WheelEvent`](https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent)
pub type WheelEvent = Event<WheelData>;
/// Data associated with a [WheelEvent]
pub struct WheelData {
inner: Box<dyn HasWheelData>,
}
@ -19,6 +26,10 @@ impl std::fmt::Debug for WheelData {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.debug_struct("WheelData")
.field("delta", &self.delta())
.field("coordinates", &self.coordinates())
.field("modifiers", &self.modifiers())
.field("held_buttons", &self.held_buttons())
.field("trigger_button", &self.trigger_button())
.finish()
}
}
@ -30,7 +41,6 @@ impl PartialEq for WheelData {
}
impl WheelData {
/// Create a new WheelData
pub fn new(inner: impl HasWheelData + 'static) -> Self {
Self {
inner: Box::new(inner),
@ -45,7 +55,48 @@ impl WheelData {
/// Downcast this event to a concrete event type
pub fn downcast<T: 'static>(&self) -> Option<&T> {
self.inner.as_any().downcast_ref::<T>()
HasWheelData::as_any(&*self.inner).downcast_ref::<T>()
}
}
impl InteractionLocation for WheelData {
fn client_coordinates(&self) -> ClientPoint {
self.inner.client_coordinates()
}
fn page_coordinates(&self) -> PagePoint {
self.inner.page_coordinates()
}
fn screen_coordinates(&self) -> ScreenPoint {
self.inner.screen_coordinates()
}
}
impl InteractionElementOffset for WheelData {
fn element_coordinates(&self) -> ElementPoint {
self.inner.element_coordinates()
}
fn coordinates(&self) -> Coordinates {
self.inner.coordinates()
}
}
impl ModifiersInteraction for WheelData {
fn modifiers(&self) -> Modifiers {
self.inner.modifiers()
}
}
impl PointerInteraction for WheelData {
fn held_buttons(&self) -> MouseButtonSet {
self.inner.held_buttons()
}
// 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
fn trigger_button(&self) -> Option<MouseButton> {
self.inner.trigger_button()
}
}
@ -53,6 +104,9 @@ impl WheelData {
/// A serialized version of WheelData
#[derive(serde::Serialize, serde::Deserialize, Debug, PartialEq, Clone)]
pub struct SerializedWheelData {
#[serde(flatten)]
pub mouse: crate::point_interaction::SerializedPointInteraction,
pub delta_mode: u32,
pub delta_x: f64,
pub delta_y: f64,
@ -62,14 +116,15 @@ pub struct SerializedWheelData {
#[cfg(feature = "serialize")]
impl SerializedWheelData {
/// Create a new SerializedWheelData
pub fn new(delta: WheelDelta) -> Self {
let delta_mode = match delta {
pub fn new(wheel: &WheelData) -> Self {
let delta_mode = match wheel.delta() {
WheelDelta::Pixels(_) => 0,
WheelDelta::Lines(_) => 1,
WheelDelta::Pages(_) => 2,
};
let delta_raw = delta.strip_units();
let delta_raw = wheel.delta().strip_units();
Self {
mouse: crate::point_interaction::SerializedPointInteraction::from(wheel),
delta_mode,
delta_x: delta_raw.x,
delta_y: delta_raw.y,
@ -81,7 +136,7 @@ impl SerializedWheelData {
#[cfg(feature = "serialize")]
impl From<&WheelData> for SerializedWheelData {
fn from(data: &WheelData) -> Self {
Self::new(data.delta())
Self::new(data)
}
}
@ -96,6 +151,57 @@ impl HasWheelData for SerializedWheelData {
}
}
#[cfg(feature = "serialize")]
impl HasMouseData for SerializedWheelData {
fn as_any(&self) -> &dyn std::any::Any {
self
}
}
#[cfg(feature = "serialize")]
impl InteractionLocation for SerializedWheelData {
fn client_coordinates(&self) -> ClientPoint {
self.mouse.client_coordinates()
}
fn page_coordinates(&self) -> PagePoint {
self.mouse.page_coordinates()
}
fn screen_coordinates(&self) -> ScreenPoint {
self.mouse.screen_coordinates()
}
}
#[cfg(feature = "serialize")]
impl InteractionElementOffset for SerializedWheelData {
fn element_coordinates(&self) -> ElementPoint {
self.mouse.element_coordinates()
}
fn coordinates(&self) -> Coordinates {
self.mouse.coordinates()
}
}
#[cfg(feature = "serialize")]
impl ModifiersInteraction for SerializedWheelData {
fn modifiers(&self) -> Modifiers {
self.mouse.modifiers()
}
}
#[cfg(feature = "serialize")]
impl PointerInteraction for SerializedWheelData {
fn held_buttons(&self) -> MouseButtonSet {
self.mouse.held_buttons()
}
fn trigger_button(&self) -> Option<MouseButton> {
self.mouse.trigger_button()
}
}
#[cfg(feature = "serialize")]
impl serde::Serialize for WheelData {
fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
@ -120,7 +226,7 @@ impl_event![
onwheel
];
pub trait HasWheelData: std::any::Any {
pub trait HasWheelData: HasMouseData + std::any::Any {
/// The amount of wheel movement
fn delta(&self) -> WheelDelta;

View file

@ -378,6 +378,63 @@ impl HasWheelData for WheelEvent {
}
}
impl HasMouseData for WheelEvent {
fn as_any(&self) -> &dyn std::any::Any {
self
}
}
impl InteractionLocation for WheelEvent {
fn client_coordinates(&self) -> ClientPoint {
ClientPoint::new(self.client_x().into(), self.client_y().into())
}
fn screen_coordinates(&self) -> ScreenPoint {
ScreenPoint::new(self.screen_x().into(), self.screen_y().into())
}
fn page_coordinates(&self) -> PagePoint {
PagePoint::new(self.page_x().into(), self.page_y().into())
}
}
impl InteractionElementOffset for WheelEvent {
fn element_coordinates(&self) -> ElementPoint {
ElementPoint::new(self.offset_x().into(), self.offset_y().into())
}
}
impl ModifiersInteraction for WheelEvent {
fn modifiers(&self) -> Modifiers {
let mut modifiers = Modifiers::empty();
if self.alt_key() {
modifiers.insert(Modifiers::ALT);
}
if self.ctrl_key() {
modifiers.insert(Modifiers::CONTROL);
}
if self.meta_key() {
modifiers.insert(Modifiers::META);
}
if self.shift_key() {
modifiers.insert(Modifiers::SHIFT);
}
modifiers
}
}
impl PointerInteraction for WheelEvent {
fn held_buttons(&self) -> crate::input_data::MouseButtonSet {
decode_mouse_button_set(self.buttons())
}
fn trigger_button(&self) -> Option<MouseButton> {
Some(MouseButton::from_web_code(self.button()))
}
}
impl HasAnimationData for AnimationEvent {
fn animation_name(&self) -> String {
self.animation_name()