use crate::{innerlude::CapturedPanic, ComponentFunction, Element};
use std::{any::Any, panic::AssertUnwindSafe};
pub(crate) type BoxedAnyProps = Box<dyn AnyProps>;
pub(crate) trait AnyProps: 'static {
fn render(&self) -> Element;
fn memoize(&mut self, other: &dyn Any) -> bool;
fn props(&self) -> &dyn Any;
fn props_mut(&mut self) -> &mut dyn Any;
fn duplicate(&self) -> BoxedAnyProps;
}
pub(crate) struct VProps<F: ComponentFunction<P, M>, P, M> {
render_fn: F,
memo: fn(&mut P, &P) -> bool,
props: P,
name: &'static str,
phantom: std::marker::PhantomData<M>,
}
impl<F: ComponentFunction<P, M>, P: Clone, M> Clone for VProps<F, P, M> {
fn clone(&self) -> Self {
Self {
render_fn: self.render_fn.clone(),
memo: self.memo,
props: self.props.clone(),
name: self.name,
phantom: std::marker::PhantomData,
}
}
}
impl<F: ComponentFunction<P, M> + Clone, P: Clone + 'static, M: 'static> VProps<F, P, M> {
pub fn new(
render_fn: F,
memo: fn(&mut P, &P) -> bool,
props: P,
name: &'static str,
) -> VProps<F, P, M> {
VProps {
render_fn,
memo,
props,
name,
phantom: std::marker::PhantomData,
}
}
}
impl<F: ComponentFunction<P, M> + Clone, P: Clone + 'static, M: 'static> AnyProps
for VProps<F, P, M>
{
fn memoize(&mut self, other: &dyn Any) -> bool {
match other.downcast_ref::<P>() {
Some(other) => (self.memo)(&mut self.props, other),
None => false,
}
}
fn props(&self) -> &dyn Any {
&self.props
}
fn props_mut(&mut self) -> &mut dyn Any {
&mut self.props
}
fn render(&self) -> Element {
fn render_inner(_name: &str, res: Result<Element, Box<dyn Any + Send>>) -> Element {
match res {
Ok(node) => node,
Err(err) => {
#[cfg(not(target_arch = "wasm32"))]
{
tracing::error!("Panic while rendering component `{_name}`: {err:?}");
}
Element::Err(CapturedPanic(err).into())
}
}
}
render_inner(
self.name,
std::panic::catch_unwind(AssertUnwindSafe(move || {
self.render_fn.rebuild(self.props.clone())
})),
)
}
fn duplicate(&self) -> BoxedAnyProps {
Box::new(Self {
render_fn: self.render_fn.clone(),
memo: self.memo,
props: self.props.clone(),
name: self.name,
phantom: std::marker::PhantomData,
})
}
}