1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use std::{any::Any, rc::Rc};
use wasm_bindgen::closure::{Closure, WasmClosure};

use crate::{DynFn, FnOfArgs, IntoPropValue};

pub struct AnyFn<F>(pub Rc<F>)
where
    F: ?Sized + DynFn;

impl<F: ?Sized + DynFn> AnyFn<F> {
    #[inline]
    pub fn inner(&self) -> &Rc<F> {
        &self.0
    }

    #[inline]
    pub fn into_inner(self) -> Rc<F> {
        self.0
    }

    #[inline]
    pub fn new<
        TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>,
        RF: crate::IntoRc<TFunc>,
    >(
        func: RF,
    ) -> Self {
        let func: Rc<TFunc> = func.into_rc();
        let func = func.into_rc_dyn_fn();
        Self(func)
    }
}

impl<F: ?Sized + DynFn> std::ops::Deref for AnyFn<F> {
    type Target = Rc<F>;

    #[inline]
    fn deref(&self) -> &Self::Target {
        &self.0
    }
}

impl<F: ?Sized + DynFn> AsRef<Rc<F>> for AnyFn<F> {
    #[inline]
    fn as_ref(&self) -> &Rc<F> {
        &self.0
    }
}

impl<F: ?Sized + DynFn> Clone for AnyFn<F> {
    #[inline]
    fn clone(&self) -> Self {
        Self(self.0.clone())
    }
}

impl<F: ?Sized + DynFn> std::fmt::Debug for AnyFn<F> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "AnyFn<{}>", std::any::type_name::<F>())
    }
}

impl<F: ?Sized + DynFn> crate::SafeIntoJsRuntime for AnyFn<F>
where
    F: 'static + WasmClosure,
{
    fn safe_into_js_runtime(self) -> crate::PassedToJsRuntime {
        let boxed = self.0.rc_into_box_dyn_fn();

        let closure = Closure::wrap(boxed);
        let js_value = closure.as_ref().clone();
        crate::PassedToJsRuntime {
            js_value,
            to_persist: Some(Box::new(closure) as Box<dyn Any>),
        }
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<AnyFn<F>> for Rc<TFunc>
{
    fn into_prop_value(self) -> AnyFn<F> {
        AnyFn(self.into_rc_dyn_fn())
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<Option<AnyFn<F>>> for Rc<TFunc>
{
    fn into_prop_value(self) -> Option<AnyFn<F>> {
        Some(self.into_prop_value())
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<Option<AnyFn<F>>> for Option<Rc<TFunc>>
{
    fn into_prop_value(self) -> Option<AnyFn<F>> {
        self.map(IntoPropValue::into_prop_value)
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<AnyFn<F>> for &Rc<TFunc>
{
    fn into_prop_value(self) -> AnyFn<F> {
        AnyFn(Rc::clone(self).into_rc_dyn_fn())
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<Option<AnyFn<F>>> for &Rc<TFunc>
{
    fn into_prop_value(self) -> Option<AnyFn<F>> {
        Some(self.into_prop_value())
    }
}

impl<F: ?Sized + DynFn, TFunc: 'static + ?Sized + FnOfArgs<F::ArgsTuple, DynFn = F>>
    IntoPropValue<Option<AnyFn<F>>> for Option<&Rc<TFunc>>
{
    fn into_prop_value(self) -> Option<AnyFn<F>> {
        self.map(IntoPropValue::into_prop_value)
    }
}