dioxus_html/events/
touch.rsuse dioxus_core::Event;
use keyboard_types::Modifiers;
use crate::geometry::*;
use crate::prelude::{InteractionLocation, ModifiersInteraction};
pub type TouchEvent = Event<TouchData>;
pub struct TouchData {
    inner: Box<dyn HasTouchData>,
}
impl<E: HasTouchData> From<E> for TouchData {
    fn from(e: E) -> Self {
        Self { inner: Box::new(e) }
    }
}
impl std::fmt::Debug for TouchData {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("TouchData")
            .field("modifiers", &self.modifiers())
            .field("touches", &self.touches())
            .field("touches_changed", &self.touches_changed())
            .field("target_touches", &self.target_touches())
            .finish()
    }
}
impl PartialEq for TouchData {
    fn eq(&self, other: &Self) -> bool {
        self.modifiers() == other.modifiers()
    }
}
impl TouchData {
    pub fn new(inner: impl HasTouchData + 'static) -> Self {
        Self {
            inner: Box::new(inner),
        }
    }
    pub fn touches(&self) -> Vec<TouchPoint> {
        self.inner.touches()
    }
    pub fn touches_changed(&self) -> Vec<TouchPoint> {
        self.inner.touches_changed()
    }
    pub fn target_touches(&self) -> Vec<TouchPoint> {
        self.inner.target_touches()
    }
    #[inline(always)]
    pub fn downcast<T: 'static>(&self) -> Option<&T> {
        self.inner.as_any().downcast_ref::<T>()
    }
}
impl ModifiersInteraction for TouchData {
    fn modifiers(&self) -> Modifiers {
        self.inner.modifiers()
    }
}
#[cfg(feature = "serialize")]
#[derive(serde::Serialize, serde::Deserialize, Debug, PartialEq, Clone)]
pub struct SerializedTouchData {
    alt_key: bool,
    ctrl_key: bool,
    meta_key: bool,
    shift_key: bool,
    touches: Vec<SerializedTouchPoint>,
    changed_touches: Vec<SerializedTouchPoint>,
    target_touches: Vec<SerializedTouchPoint>,
}
#[cfg(feature = "serialize")]
impl From<&TouchData> for SerializedTouchData {
    fn from(data: &TouchData) -> Self {
        let modifiers = data.modifiers();
        Self {
            alt_key: modifiers.contains(Modifiers::ALT),
            ctrl_key: modifiers.contains(Modifiers::CONTROL),
            meta_key: modifiers.contains(Modifiers::META),
            shift_key: modifiers.contains(Modifiers::SHIFT),
            touches: data.touches().iter().map(|t| t.into()).collect(),
            changed_touches: data.touches_changed().iter().map(|t| t.into()).collect(),
            target_touches: data.target_touches().iter().map(|t| t.into()).collect(),
        }
    }
}
#[cfg(feature = "serialize")]
impl ModifiersInteraction for SerializedTouchData {
    fn modifiers(&self) -> Modifiers {
        let mut modifiers = Modifiers::default();
        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
    }
}
#[cfg(feature = "serialize")]
impl HasTouchData for SerializedTouchData {
    fn touches(&self) -> Vec<TouchPoint> {
        self.touches.clone().into_iter().map(Into::into).collect()
    }
    fn touches_changed(&self) -> Vec<TouchPoint> {
        self.changed_touches
            .clone()
            .into_iter()
            .map(Into::into)
            .collect()
    }
    fn target_touches(&self) -> Vec<TouchPoint> {
        self.target_touches
            .clone()
            .into_iter()
            .map(Into::into)
            .collect()
    }
    fn as_any(&self) -> &dyn std::any::Any {
        self
    }
}
#[cfg(feature = "serialize")]
impl serde::Serialize for TouchData {
    fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
        SerializedTouchData::from(self).serialize(serializer)
    }
}
#[cfg(feature = "serialize")]
impl<'de> serde::Deserialize<'de> for TouchData {
    fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
        let data = SerializedTouchData::deserialize(deserializer)?;
        Ok(Self {
            inner: Box::new(data),
        })
    }
}
pub trait HasTouchData: ModifiersInteraction + std::any::Any {
    fn touches(&self) -> Vec<TouchPoint>;
    fn touches_changed(&self) -> Vec<TouchPoint>;
    fn target_touches(&self) -> Vec<TouchPoint>;
    fn as_any(&self) -> &dyn std::any::Any;
}
pub struct TouchPoint {
    inner: Box<dyn HasTouchPointData>,
}
impl<E: HasTouchPointData> From<E> for TouchPoint {
    fn from(e: E) -> Self {
        Self { inner: Box::new(e) }
    }
}
impl TouchPoint {
    pub fn new(inner: impl HasTouchPointData + 'static) -> Self {
        Self {
            inner: Box::new(inner),
        }
    }
    fn identifier(&self) -> i32 {
        self.inner.identifier()
    }
    fn force(&self) -> f64 {
        self.inner.force()
    }
    fn radius(&self) -> ScreenPoint {
        self.inner.radius()
    }
    fn rotation(&self) -> f64 {
        self.inner.rotation()
    }
    #[inline(always)]
    pub fn downcast<T: 'static>(&self) -> Option<&T> {
        self.inner.as_any().downcast_ref::<T>()
    }
}
impl std::fmt::Debug for TouchPoint {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("TouchPoint")
            .field("client_coordinates", &self.client_coordinates())
            .field("page_coordinates", &self.page_coordinates())
            .field("screen_coordinates", &self.screen_coordinates())
            .field("identifier", &self.identifier())
            .field("force", &self.force())
            .field("radius", &self.radius())
            .field("rotation", &self.rotation())
            .finish()
    }
}
impl InteractionLocation for TouchPoint {
    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()
    }
}
pub trait HasTouchPointData: InteractionLocation + std::any::Any {
    fn identifier(&self) -> i32;
    fn force(&self) -> f64;
    fn radius(&self) -> ScreenPoint;
    fn rotation(&self) -> f64;
    fn as_any(&self) -> &dyn std::any::Any;
}
#[cfg(feature = "serialize")]
#[derive(serde::Serialize, serde::Deserialize, Debug, PartialEq, Clone)]
struct SerializedTouchPoint {
    identifier: i32,
    client_x: f64,
    client_y: f64,
    page_x: f64,
    page_y: f64,
    screen_x: f64,
    screen_y: f64,
    force: f64,
    radius_x: f64,
    radius_y: f64,
    rotation_angle: f64,
}
#[cfg(feature = "serialize")]
impl From<&TouchPoint> for SerializedTouchPoint {
    fn from(point: &TouchPoint) -> Self {
        let client_coordinates = point.client_coordinates();
        let page_coordinates = point.page_coordinates();
        let screen_coordinates = point.screen_coordinates();
        Self {
            identifier: point.identifier(),
            client_x: client_coordinates.x,
            client_y: client_coordinates.y,
            page_x: page_coordinates.x,
            page_y: page_coordinates.y,
            screen_x: screen_coordinates.x,
            screen_y: screen_coordinates.y,
            force: point.force(),
            radius_x: point.radius().x,
            radius_y: point.radius().y,
            rotation_angle: point.rotation(),
        }
    }
}
#[cfg(feature = "serialize")]
impl HasTouchPointData for SerializedTouchPoint {
    fn identifier(&self) -> i32 {
        self.identifier
    }
    fn force(&self) -> f64 {
        self.force
    }
    fn radius(&self) -> ScreenPoint {
        ScreenPoint::new(self.radius_x, self.radius_y)
    }
    fn rotation(&self) -> f64 {
        self.rotation_angle
    }
    fn as_any(&self) -> &dyn std::any::Any {
        self
    }
}
#[cfg(feature = "serialize")]
impl InteractionLocation for SerializedTouchPoint {
    fn client_coordinates(&self) -> ClientPoint {
        ClientPoint::new(self.client_x, self.client_y)
    }
    fn screen_coordinates(&self) -> ScreenPoint {
        ScreenPoint::new(self.screen_x, self.screen_y)
    }
    fn page_coordinates(&self) -> PagePoint {
        PagePoint::new(self.page_x, self.page_y)
    }
}
impl_event! {
    TouchData;
    ontouchstart
    ontouchmove
    ontouchend
    ontouchcancel
}