use leptos::html::ElementDescriptor;
use leptos::*;
use std::ops::Deref;
/// Used as an argument type to make it easily possible to pass either
/// * a `web_sys` element that implements `EventTarget`,
/// * an `Option` where `T` is the web_sys element,
/// * a `Signal` where `T` is the web_sys element,
/// * a `Signal>` where `T` is the web_sys element,
/// * a `NodeRef`
/// into a function. Used for example in [`use_event_listener`].
pub enum EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
Static(Option),
Dynamic(Signal>),
}
impl Default for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn default() -> Self {
Self::Static(None)
}
}
impl Clone for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn clone(&self) -> Self {
match self {
Self::Static(t) => Self::Static(t.clone()),
Self::Dynamic(s) => Self::Dynamic(*s),
}
}
}
impl SignalGet> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn get(&self) -> Option {
match self {
Self::Static(t) => t.clone(),
Self::Dynamic(s) => s.get(),
}
}
fn try_get(&self) -> Option> {
match self {
Self::Static(t) => Some(t.clone()),
Self::Dynamic(s) => s.try_get(),
}
}
}
impl SignalWith> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn with(&self, f: impl FnOnce(&Option) -> O) -> O {
match self {
Self::Static(t) => f(t),
Self::Dynamic(s) => s.with(f),
}
}
fn try_with(&self, f: impl FnOnce(&Option) -> O) -> Option {
match self {
Self::Static(t) => Some(f(t)),
Self::Dynamic(s) => s.try_with(f),
}
}
}
impl SignalWithUntracked> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn with_untracked(&self, f: impl FnOnce(&Option) -> O) -> O {
match self {
Self::Static(t) => f(t),
Self::Dynamic(s) => s.with_untracked(f),
}
}
fn try_with_untracked(&self, f: impl FnOnce(&Option) -> O) -> Option {
match self {
Self::Static(t) => Some(f(t)),
Self::Dynamic(s) => s.try_with_untracked(f),
}
}
}
impl SignalGetUntracked> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn get_untracked(&self) -> Option {
match self {
Self::Static(t) => t.clone(),
Self::Dynamic(s) => s.get_untracked(),
}
}
fn try_get_untracked(&self) -> Option> {
match self {
Self::Static(t) => Some(t.clone()),
Self::Dynamic(s) => s.try_get_untracked(),
}
}
}
impl From<(Scope, T)> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn from(value: (Scope, T)) -> Self {
EventTargetMaybeSignal::Static(Some(value.1))
}
}
impl From<(Scope, Option)> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn from(target: (Scope, Option)) -> Self {
EventTargetMaybeSignal::Static(target.1)
}
}
macro_rules! impl_from_signal_option {
($ty:ty) => {
impl From<(Scope, $ty)> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn from(target: (Scope, $ty)) -> Self {
EventTargetMaybeSignal::Dynamic(target.1.into())
}
}
};
}
impl_from_signal_option!(Signal>);
impl_from_signal_option!(ReadSignal >);
impl_from_signal_option!(RwSignal >);
impl_from_signal_option!(Memo >);
macro_rules! impl_from_signal {
($ty:ty) => {
impl From<(Scope, $ty)> for EventTargetMaybeSignal
where
T: Into + Clone + 'static,
{
fn from(target: (Scope, $ty)) -> Self {
let (cx, signal) = target;
EventTargetMaybeSignal::Dynamic(Signal::derive(cx, move || Some(signal.get())))
}
}
};
}
impl_from_signal!(Signal);
impl_from_signal!(ReadSignal);
impl_from_signal!(RwSignal);
impl_from_signal!(Memo);
impl From<(Scope, NodeRef)> for EventTargetMaybeSignal
where
R: ElementDescriptor + Clone + 'static,
{
fn from(target: (Scope, NodeRef)) -> Self {
let (cx, node_ref) = target;
EventTargetMaybeSignal::Dynamic(Signal::derive(cx, move || {
node_ref.get().map(move |el| {
let el = el.into_any();
let el: web_sys::EventTarget = el.deref().clone().into();
el
})
}))
}
}