|
@@ -1,105 +1,62 @@
|
|
|
use crate::dom::WebsysDom;
|
|
|
-use dioxus_core::{
|
|
|
- AttributeValue, DynamicNode, ElementId, ScopeState, TemplateNode, VNode, VirtualDom,
|
|
|
-};
|
|
|
-use dioxus_html::event_bubbles;
|
|
|
-use wasm_bindgen::JsCast;
|
|
|
-use web_sys::{Comment, Node};
|
|
|
+use dioxus_core::{DynamicNode, ElementId, ScopeState, TemplateNode, VNode, VirtualDom};
|
|
|
|
|
|
-#[derive(Debug, Copy, Clone)]
|
|
|
+#[derive(Debug)]
|
|
|
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;
|
|
|
- }
|
|
|
-}
|
|
|
+use RehydrationError::*;
|
|
|
|
|
|
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 ids = Vec::new();
|
|
|
+ let mut to_mount = Vec::new();
|
|
|
|
|
|
- let mut hydrated = vec![true];
|
|
|
+ // Recursively rehydrate the dom from the VirtualDom
|
|
|
+ self.rehydrate_scope(root_scope, dom, &mut ids, &mut to_mount)?;
|
|
|
|
|
|
- let mut last_node_was_static_text = false;
|
|
|
+ dioxus_interpreter_js::hydrate(ids);
|
|
|
|
|
|
- // Recursively rehydrate the dom from the VirtualDom
|
|
|
- self.rehydrate_scope(
|
|
|
- root_scope,
|
|
|
- &mut root,
|
|
|
- &mut hydrated,
|
|
|
- dom,
|
|
|
- &mut last_node_was_static_text,
|
|
|
- )?;
|
|
|
+ for id in to_mount {
|
|
|
+ self.send_mount_event(id);
|
|
|
+ }
|
|
|
|
|
|
- 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,
|
|
|
+ ids: &mut Vec<u32>,
|
|
|
+ to_mount: &mut Vec<ElementId>,
|
|
|
) -> 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,
|
|
|
- )
|
|
|
+ self.rehydrate_vnode(dom, vnode, ids, to_mount)
|
|
|
}
|
|
|
|
|
|
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,
|
|
|
+ ids: &mut Vec<u32>,
|
|
|
+ to_mount: &mut Vec<ElementId>,
|
|
|
) -> 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,
|
|
|
+ ids,
|
|
|
+ to_mount,
|
|
|
+ Some(*vnode.root_ids.borrow().get(i).ok_or(VNodeNotInitialized)?),
|
|
|
)?;
|
|
|
}
|
|
|
Ok(())
|
|
@@ -107,191 +64,75 @@ impl WebsysDom {
|
|
|
|
|
|
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,
|
|
|
+ ids: &mut Vec<u32>,
|
|
|
+ to_mount: &mut Vec<ElementId>,
|
|
|
+ root_id: Option<ElementId>,
|
|
|
) -> Result<(), RehydrationError> {
|
|
|
tracing::trace!("rehydrate template node: {:?}", node);
|
|
|
- if let Ok(current_child) = current_child {
|
|
|
- if tracing::event_enabled!(tracing::Level::TRACE) {
|
|
|
- web_sys::console::log_1(¤t_child.clone().into());
|
|
|
- }
|
|
|
- }
|
|
|
match node {
|
|
|
TemplateNode::Element {
|
|
|
children, attrs, ..
|
|
|
} => {
|
|
|
- let mut mounted_id = None;
|
|
|
- let mut should_send_mount_event = true;
|
|
|
+ let mut mounted_id = root_id;
|
|
|
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();
|
|
|
mounted_id = Some(id);
|
|
|
- let name = attribute.name;
|
|
|
- if let AttributeValue::Listener(_) = value {
|
|
|
- let event_name = &name[2..];
|
|
|
- match event_name {
|
|
|
- "mounted" => should_send_mount_event = true,
|
|
|
- _ => {
|
|
|
- self.interpreter.new_event_listener(
|
|
|
- event_name,
|
|
|
- id.0 as u32,
|
|
|
- event_bubbles(event_name) as u8,
|
|
|
- );
|
|
|
- }
|
|
|
+ if let dioxus_core::AttributeValue::Listener(_) = attribute.value {
|
|
|
+ if attribute.name == "onmounted" {
|
|
|
+ to_mount.push(id);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
if let Some(id) = mounted_id {
|
|
|
- set_node(hydrated, id, current_child.clone()?);
|
|
|
- if should_send_mount_event {
|
|
|
- self.send_mount_event(id);
|
|
|
- }
|
|
|
+ ids.push(id.0 as u32);
|
|
|
}
|
|
|
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,
|
|
|
- )?;
|
|
|
+ self.rehydrate_template_node(dom, vnode, child, ids, to_mount, None)?;
|
|
|
}
|
|
|
}
|
|
|
- *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,
|
|
|
- )?;
|
|
|
+ self.rehydrate_dynamic_node(dom, &vnode.dynamic_nodes[*id], ids, to_mount)?;
|
|
|
}
|
|
|
+ _ => {}
|
|
|
}
|
|
|
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,
|
|
|
+ ids: &mut Vec<u32>,
|
|
|
+ to_mount: &mut Vec<ElementId>,
|
|
|
) -> Result<(), RehydrationError> {
|
|
|
tracing::trace!("rehydrate dynamic node: {:?}", dynamic);
|
|
|
- if let Ok(current_child) = current_child {
|
|
|
- if tracing::event_enabled!(tracing::Level::TRACE) {
|
|
|
- web_sys::console::log_1(¤t_child.clone().into());
|
|
|
- }
|
|
|
- }
|
|
|
match dynamic {
|
|
|
dioxus_core::DynamicNode::Text(text) => {
|
|
|
- let id = text.mounted_element();
|
|
|
- // 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.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;
|
|
|
+ ids.push(text.mounted_element().ok_or(VNodeNotInitialized)?.0 as u32);
|
|
|
}
|
|
|
dioxus_core::DynamicNode::Placeholder(placeholder) => {
|
|
|
- set_node(
|
|
|
- hydrated,
|
|
|
- placeholder.mounted_element().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;
|
|
|
+ ids.push(placeholder.mounted_element().ok_or(VNodeNotInitialized)?.0 as u32);
|
|
|
}
|
|
|
dioxus_core::DynamicNode::Component(comp) => {
|
|
|
let scope = comp.mounted_scope().ok_or(VNodeNotInitialized)?;
|
|
|
self.rehydrate_scope(
|
|
|
- dom.get_scope(scope).unwrap(),
|
|
|
- current_child,
|
|
|
- hydrated,
|
|
|
+ dom.get_scope(scope).ok_or(VNodeNotInitialized)?,
|
|
|
dom,
|
|
|
- last_node_was_static_text,
|
|
|
+ ids,
|
|
|
+ to_mount,
|
|
|
)?;
|
|
|
}
|
|
|
dioxus_core::DynamicNode::Fragment(fragment) => {
|
|
|
for vnode in *fragment {
|
|
|
- self.rehydrate_vnode(
|
|
|
- current_child,
|
|
|
- hydrated,
|
|
|
- dom,
|
|
|
- vnode,
|
|
|
- last_node_was_static_text,
|
|
|
- )?;
|
|
|
+ self.rehydrate_vnode(dom, vnode, ids, to_mount)?;
|
|
|
}
|
|
|
}
|
|
|
}
|