123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- use crate::dom::WebsysDom;
- use dioxus_core::{
- AttributeValue, DynamicNode, ElementId, ScopeState, TemplateNode, VNode, VPlaceholder, VText,
- VirtualDom,
- };
- use dioxus_html::event_bubbles;
- use wasm_bindgen::JsCast;
- use web_sys::{Comment, Node};
- #[derive(Debug, Copy, Clone)]
- pub enum RehydrationError {
- NodeTypeMismatch,
- NodeNotFound,
- VNodeNotInitialized,
- }
- use RehydrationError::*;
- fn set_node(hydrated: &mut Vec<bool>, id: ElementId, node: Node) {
- let idx = id.0;
- if idx >= hydrated.len() {
- hydrated.resize(idx + 1, false);
- }
- if !hydrated[idx] {
- dioxus_interpreter_js::set_node(idx as u32, node);
- hydrated[idx] = true;
- }
- }
- impl WebsysDom {
- // we're streaming in patches, but the nodes already exist
- // so we're just going to write the correct IDs to the node and load them in
- pub fn rehydrate(&mut self, dom: &VirtualDom) -> Result<(), RehydrationError> {
- let mut root = self
- .root
- .clone()
- .dyn_into::<Node>()
- .map_err(|_| NodeTypeMismatch)?
- .first_child()
- .ok_or(NodeNotFound);
- let root_scope = dom.base_scope();
- let mut hydrated = vec![true];
- let mut last_node_was_static_text = false;
- // Recursively rehydrate the dom from the VirtualDom
- self.rehydrate_scope(
- root_scope,
- &mut root,
- &mut hydrated,
- dom,
- &mut last_node_was_static_text,
- )?;
- self.interpreter.flush();
- Ok(())
- }
- fn rehydrate_scope(
- &mut self,
- scope: &ScopeState,
- current_child: &mut Result<Node, RehydrationError>,
- hydrated: &mut Vec<bool>,
- dom: &VirtualDom,
- last_node_was_static_text: &mut bool,
- ) -> Result<(), RehydrationError> {
- let vnode = match scope.root_node() {
- dioxus_core::RenderReturn::Ready(ready) => ready,
- _ => return Err(VNodeNotInitialized),
- };
- self.rehydrate_vnode(
- current_child,
- hydrated,
- dom,
- vnode,
- last_node_was_static_text,
- )
- }
- fn rehydrate_vnode(
- &mut self,
- current_child: &mut Result<Node, RehydrationError>,
- hydrated: &mut Vec<bool>,
- dom: &VirtualDom,
- vnode: &VNode,
- last_node_was_static_text: &mut bool,
- ) -> Result<(), RehydrationError> {
- for (i, root) in vnode.template.get().roots.iter().enumerate() {
- // make sure we set the root node ids even if the node is not dynamic
- set_node(
- hydrated,
- *vnode.root_ids.borrow().get(i).ok_or(VNodeNotInitialized)?,
- current_child.clone()?,
- );
- self.rehydrate_template_node(
- current_child,
- hydrated,
- dom,
- vnode,
- root,
- last_node_was_static_text,
- )?;
- }
- Ok(())
- }
- fn rehydrate_template_node(
- &mut self,
- current_child: &mut Result<Node, RehydrationError>,
- hydrated: &mut Vec<bool>,
- dom: &VirtualDom,
- vnode: &VNode,
- node: &TemplateNode,
- last_node_was_static_text: &mut bool,
- ) -> Result<(), RehydrationError> {
- log::trace!("rehydrate template node: {:?}", node);
- if let Ok(current_child) = current_child {
- if log::log_enabled!(log::Level::Trace) {
- web_sys::console::log_1(¤t_child.clone().into());
- }
- }
- match node {
- TemplateNode::Element {
- children, attrs, ..
- } => {
- let mut mounted_id = None;
- for attr in *attrs {
- if let dioxus_core::TemplateAttribute::Dynamic { id } = attr {
- let attribute = &vnode.dynamic_attrs[*id];
- let value = &attribute.value;
- let id = attribute.mounted_element.get();
- mounted_id = Some(id);
- let name = attribute.name;
- if let AttributeValue::Listener(_) = value {
- let event_name = &name[2..];
- self.interpreter.new_event_listener(
- event_name,
- id.0 as u32,
- event_bubbles(event_name) as u8,
- );
- }
- }
- }
- if let Some(id) = mounted_id {
- set_node(hydrated, id, current_child.clone()?);
- }
- if !children.is_empty() {
- let mut children_current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .first_child()
- .ok_or(NodeNotFound)?
- .dyn_into::<Node>()
- .map_err(|_| NodeTypeMismatch);
- for child in *children {
- self.rehydrate_template_node(
- &mut children_current_child,
- hydrated,
- dom,
- vnode,
- child,
- last_node_was_static_text,
- )?;
- }
- }
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- *last_node_was_static_text = false;
- }
- TemplateNode::Text { .. } => {
- // if the last node was static text, it got merged with this one
- if !*last_node_was_static_text {
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- }
- *last_node_was_static_text = true;
- }
- TemplateNode::Dynamic { id } | TemplateNode::DynamicText { id } => {
- self.rehydrate_dynamic_node(
- current_child,
- hydrated,
- dom,
- &vnode.dynamic_nodes[*id],
- last_node_was_static_text,
- )?;
- }
- }
- Ok(())
- }
- fn rehydrate_dynamic_node(
- &mut self,
- current_child: &mut Result<Node, RehydrationError>,
- hydrated: &mut Vec<bool>,
- dom: &VirtualDom,
- dynamic: &DynamicNode,
- last_node_was_static_text: &mut bool,
- ) -> Result<(), RehydrationError> {
- log::trace!("rehydrate dynamic node: {:?}", dynamic);
- if let Ok(current_child) = current_child {
- if log::log_enabled!(log::Level::Trace) {
- web_sys::console::log_1(¤t_child.clone().into());
- }
- }
- match dynamic {
- dioxus_core::DynamicNode::Text(VText { id, .. }) => {
- // skip comment separator before node
- if cfg!(debug_assertions) {
- assert!(current_child
- .as_mut()
- .map_err(|e| *e)?
- .has_type::<Comment>());
- }
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- set_node(
- hydrated,
- id.get().ok_or(VNodeNotInitialized)?,
- current_child.clone()?,
- );
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- // skip comment separator after node
- if cfg!(debug_assertions) {
- assert!(current_child
- .as_mut()
- .map_err(|e| *e)?
- .has_type::<Comment>());
- }
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- *last_node_was_static_text = false;
- }
- dioxus_core::DynamicNode::Placeholder(VPlaceholder { id, .. }) => {
- set_node(
- hydrated,
- id.get().ok_or(VNodeNotInitialized)?,
- current_child.clone()?,
- );
- *current_child = current_child
- .as_mut()
- .map_err(|e| *e)?
- .next_sibling()
- .ok_or(NodeNotFound);
- *last_node_was_static_text = false;
- }
- dioxus_core::DynamicNode::Component(comp) => {
- let scope = comp.scope.get().ok_or(VNodeNotInitialized)?;
- self.rehydrate_scope(
- dom.get_scope(scope).unwrap(),
- current_child,
- hydrated,
- dom,
- last_node_was_static_text,
- )?;
- }
- dioxus_core::DynamicNode::Fragment(fragment) => {
- for vnode in *fragment {
- self.rehydrate_vnode(
- current_child,
- hydrated,
- dom,
- vnode,
- last_node_was_static_text,
- )?;
- }
- }
- }
- Ok(())
- }
- }
|