|
@@ -1,4 +1,5 @@
|
|
use crate::{
|
|
use crate::{
|
|
|
|
+ any_props::AnyProps,
|
|
arena::ElementId,
|
|
arena::ElementId,
|
|
innerlude::{DirtyScope, VComponent, VPlaceholder, VText},
|
|
innerlude::{DirtyScope, VComponent, VPlaceholder, VText},
|
|
mutations::Mutation,
|
|
mutations::Mutation,
|
|
@@ -141,32 +142,6 @@ impl<'b> VirtualDom {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- fn replace_placeholder(&mut self, l: &'b VPlaceholder, r: &'b [VNode<'b>]) {
|
|
|
|
- let m = self.create_children(r);
|
|
|
|
- let id = l.id.get().unwrap();
|
|
|
|
- self.mutations.push(Mutation::ReplaceWith { id, m });
|
|
|
|
- self.reclaim(id);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- fn node_to_placeholder(&mut self, l: &'b [VNode<'b>], r: &'b VPlaceholder) {
|
|
|
|
- // Remove the old nodes, except for one
|
|
|
|
- self.remove_nodes(&l[1..]);
|
|
|
|
-
|
|
|
|
- // Now create the new one
|
|
|
|
- let first = self.replace_inner(&l[0]);
|
|
|
|
-
|
|
|
|
- // Create the placeholder first, ensuring we get a dedicated ID for the placeholder
|
|
|
|
- let placeholder = self.next_element(&l[0], &[]);
|
|
|
|
- r.id.set(Some(placeholder));
|
|
|
|
- self.mutations
|
|
|
|
- .push(Mutation::CreatePlaceholder { id: placeholder });
|
|
|
|
-
|
|
|
|
- self.mutations
|
|
|
|
- .push(Mutation::ReplaceWith { id: first, m: 1 });
|
|
|
|
-
|
|
|
|
- self.try_reclaim(first);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
fn diff_vcomponent(
|
|
fn diff_vcomponent(
|
|
&mut self,
|
|
&mut self,
|
|
left: &'b VComponent<'b>,
|
|
left: &'b VComponent<'b>,
|
|
@@ -186,33 +161,27 @@ impl<'b> VirtualDom {
|
|
.root_node()
|
|
.root_node()
|
|
.extend_lifetime_ref()
|
|
.extend_lifetime_ref()
|
|
};
|
|
};
|
|
- let id = match head {
|
|
|
|
- RenderReturn::Sync(Ok(node)) => self.replace_inner(node),
|
|
|
|
|
|
+ let last = match head {
|
|
|
|
+ RenderReturn::Sync(Ok(node)) => self.find_last_element(node),
|
|
_ => todo!(),
|
|
_ => todo!(),
|
|
};
|
|
};
|
|
- self.mutations
|
|
|
|
- .push(Mutation::ReplaceWith { id, m: created });
|
|
|
|
- self.drop_scope(left.scope.get().unwrap());
|
|
|
|
|
|
+ self.mutations.push(Mutation::InsertAfter {
|
|
|
|
+ id: last,
|
|
|
|
+ m: created,
|
|
|
|
+ });
|
|
|
|
+ self.remove_component_node(left, true);
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
// Make sure the new vcomponent has the right scopeid associated to it
|
|
// Make sure the new vcomponent has the right scopeid associated to it
|
|
- let Some(scope_id) = left.scope.get() else {
|
|
|
|
- return;
|
|
|
|
- };
|
|
|
|
- // let scope_id = left.scope.get().unwrap_or_else(|| {
|
|
|
|
- // panic!(
|
|
|
|
- // "A component should always have a scope associated to it. {:?}\n {:#?}",
|
|
|
|
- // right.name,
|
|
|
|
- // std::backtrace::Backtrace::force_capture()
|
|
|
|
- // )
|
|
|
|
- // });
|
|
|
|
|
|
+ let scope_id = left.scope.get().unwrap();
|
|
|
|
|
|
right.scope.set(Some(scope_id));
|
|
right.scope.set(Some(scope_id));
|
|
|
|
|
|
// copy out the box for both
|
|
// copy out the box for both
|
|
let old = self.scopes[scope_id.0].props.as_ref();
|
|
let old = self.scopes[scope_id.0].props.as_ref();
|
|
- let new = right.props.take().unwrap();
|
|
|
|
|
|
+ let new: Box<dyn AnyProps> = right.props.take().unwrap();
|
|
|
|
+ let new: Box<dyn AnyProps> = unsafe { std::mem::transmute(new) };
|
|
|
|
|
|
// If the props are static, then we try to memoize by setting the new with the old
|
|
// If the props are static, then we try to memoize by setting the new with the old
|
|
// The target scopestate still has the reference to the old props, so there's no need to update anything
|
|
// The target scopestate still has the reference to the old props, so there's no need to update anything
|
|
@@ -222,7 +191,7 @@ impl<'b> VirtualDom {
|
|
}
|
|
}
|
|
|
|
|
|
// First, move over the props from the old to the new, dropping old props in the process
|
|
// First, move over the props from the old to the new, dropping old props in the process
|
|
- self.scopes[scope_id.0].props = unsafe { std::mem::transmute(new) };
|
|
|
|
|
|
+ self.scopes[scope_id.0].props = Some(new);
|
|
|
|
|
|
// Now run the component and diff it
|
|
// Now run the component and diff it
|
|
self.run_scope(scope_id);
|
|
self.run_scope(scope_id);
|
|
@@ -273,7 +242,7 @@ impl<'b> VirtualDom {
|
|
/// ```
|
|
/// ```
|
|
fn light_diff_templates(&mut self, left: &'b VNode<'b>, right: &'b VNode<'b>) {
|
|
fn light_diff_templates(&mut self, left: &'b VNode<'b>, right: &'b VNode<'b>) {
|
|
match matching_components(left, right) {
|
|
match matching_components(left, right) {
|
|
- None => self.replace(left, right),
|
|
|
|
|
|
+ None => self.replace(left, [right]),
|
|
Some(components) => components
|
|
Some(components) => components
|
|
.into_iter()
|
|
.into_iter()
|
|
.enumerate()
|
|
.enumerate()
|
|
@@ -298,116 +267,6 @@ impl<'b> VirtualDom {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- /// Remove all the top-level nodes, returning the firstmost root ElementId
|
|
|
|
- ///
|
|
|
|
- /// All IDs will be garbage collected
|
|
|
|
- fn replace_inner(&mut self, node: &'b VNode<'b>) -> ElementId {
|
|
|
|
- let id = match node.dynamic_root(0) {
|
|
|
|
- None => node.root_ids[0].get().unwrap(),
|
|
|
|
- Some(Text(t)) => t.id.get().unwrap(),
|
|
|
|
- Some(Placeholder(e)) => e.id.get().unwrap(),
|
|
|
|
- Some(Fragment(nodes)) => {
|
|
|
|
- let id = self.replace_inner(&nodes[0]);
|
|
|
|
- self.remove_nodes(&nodes[1..]);
|
|
|
|
- id
|
|
|
|
- }
|
|
|
|
- Some(Component(comp)) => {
|
|
|
|
- let scope = comp.scope.get().unwrap();
|
|
|
|
- match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
|
|
- RenderReturn::Sync(Ok(t)) => self.replace_inner(t),
|
|
|
|
- _ => todo!("cannot handle nonstandard nodes"),
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
-
|
|
|
|
- // Just remove the rest from the dom
|
|
|
|
- for (idx, _) in node.template.roots.iter().enumerate().skip(1) {
|
|
|
|
- self.remove_root_node(node, idx);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // Garabge collect all of the nodes since this gets used in replace
|
|
|
|
- self.clean_up_node(node);
|
|
|
|
-
|
|
|
|
- id
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /// Clean up the node, not generating mutations
|
|
|
|
- ///
|
|
|
|
- /// Simply walks through the dynamic nodes
|
|
|
|
- fn clean_up_node(&mut self, node: &'b VNode<'b>) {
|
|
|
|
- for (idx, dyn_node) in node.dynamic_nodes.iter().enumerate() {
|
|
|
|
- // Roots are cleaned up automatically?
|
|
|
|
- if node.template.node_paths[idx].len() == 1 {
|
|
|
|
- continue;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- match dyn_node {
|
|
|
|
- Component(comp) => {
|
|
|
|
- if let Some(scope) = comp.scope.take() {
|
|
|
|
- match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
|
|
- RenderReturn::Sync(Ok(t)) => self.clean_up_node(t),
|
|
|
|
- _ => todo!("cannot handle nonstandard nodes"),
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- Text(t) => {
|
|
|
|
- if let Some(id) = t.id.take() {
|
|
|
|
- self.reclaim(id)
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- Placeholder(t) => self.reclaim(t.id.take().unwrap()),
|
|
|
|
- Fragment(nodes) => nodes.iter().for_each(|node| self.clean_up_node(node)),
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // we clean up nodes with dynamic attributes, provided the node is unique and not a root node
|
|
|
|
- let mut id = None;
|
|
|
|
- for (idx, attr) in node.dynamic_attrs.iter().enumerate() {
|
|
|
|
- // We'll clean up the root nodes either way, so don't worry
|
|
|
|
- if node.template.attr_paths[idx].len() == 1 {
|
|
|
|
- continue;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- let next_id = attr.mounted_element.get();
|
|
|
|
-
|
|
|
|
- if id == Some(next_id) {
|
|
|
|
- continue;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- id = Some(next_id);
|
|
|
|
-
|
|
|
|
- self.reclaim(next_id);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- fn remove_root_node(&mut self, node: &'b VNode<'b>, idx: usize) {
|
|
|
|
- match node.dynamic_root(idx) {
|
|
|
|
- Some(Text(i)) => {
|
|
|
|
- let id = i.id.take().unwrap();
|
|
|
|
- self.mutations.push(Mutation::Remove { id });
|
|
|
|
- self.reclaim(id);
|
|
|
|
- }
|
|
|
|
- Some(Placeholder(e)) => {
|
|
|
|
- let id = e.id.take().unwrap();
|
|
|
|
- self.mutations.push(Mutation::Remove { id });
|
|
|
|
- self.reclaim(id);
|
|
|
|
- }
|
|
|
|
- Some(Fragment(nodes)) => self.remove_nodes(nodes),
|
|
|
|
- Some(Component(comp)) => {
|
|
|
|
- let scope = comp.scope.take().unwrap();
|
|
|
|
- match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
|
|
- RenderReturn::Sync(Ok(t)) => self.remove_node(t),
|
|
|
|
- _ => todo!("cannot handle nonstandard nodes"),
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
- None => {
|
|
|
|
- let id = node.root_ids[idx].get().unwrap();
|
|
|
|
- self.mutations.push(Mutation::Remove { id });
|
|
|
|
- self.reclaim(id);
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
fn diff_non_empty_fragment(&mut self, old: &'b [VNode<'b>], new: &'b [VNode<'b>]) {
|
|
fn diff_non_empty_fragment(&mut self, old: &'b [VNode<'b>], new: &'b [VNode<'b>]) {
|
|
let new_is_keyed = new[0].key.is_some();
|
|
let new_is_keyed = new[0].key.is_some();
|
|
let old_is_keyed = old[0].key.is_some();
|
|
let old_is_keyed = old[0].key.is_some();
|
|
@@ -658,7 +517,7 @@ impl<'b> VirtualDom {
|
|
if shared_keys.is_empty() {
|
|
if shared_keys.is_empty() {
|
|
if old.get(0).is_some() {
|
|
if old.get(0).is_some() {
|
|
self.remove_nodes(&old[1..]);
|
|
self.remove_nodes(&old[1..]);
|
|
- self.replace_many(&old[0], new);
|
|
|
|
|
|
+ self.replace(&old[0], new);
|
|
} else {
|
|
} else {
|
|
// I think this is wrong - why are we appending?
|
|
// I think this is wrong - why are we appending?
|
|
// only valid of the if there are no trailing elements
|
|
// only valid of the if there are no trailing elements
|
|
@@ -674,7 +533,7 @@ impl<'b> VirtualDom {
|
|
for child in old {
|
|
for child in old {
|
|
let key = child.key.unwrap();
|
|
let key = child.key.unwrap();
|
|
if !shared_keys.contains(&key) {
|
|
if !shared_keys.contains(&key) {
|
|
- self.remove_node(child);
|
|
|
|
|
|
+ self.remove_node(child, true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -777,54 +636,6 @@ impl<'b> VirtualDom {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- /// Remove these nodes from the dom
|
|
|
|
- /// Wont generate mutations for the inner nodes
|
|
|
|
- fn remove_nodes(&mut self, nodes: &'b [VNode<'b>]) {
|
|
|
|
- // note that we iterate in reverse to unlink lists of nodes in their rough index order
|
|
|
|
- nodes.iter().rev().for_each(|node| self.remove_node(node));
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- fn remove_node(&mut self, node: &'b VNode<'b>) {
|
|
|
|
- for (idx, _) in node.template.roots.iter().enumerate() {
|
|
|
|
- let id = match node.dynamic_root(idx) {
|
|
|
|
- Some(Text(t)) => t.id.take(),
|
|
|
|
- Some(Placeholder(t)) => t.id.take(),
|
|
|
|
- Some(Fragment(t)) => return self.remove_nodes(t),
|
|
|
|
- Some(Component(comp)) => {
|
|
|
|
- comp.scope.set(None);
|
|
|
|
- return self.remove_component(comp.scope.get().unwrap());
|
|
|
|
- }
|
|
|
|
- None => node.root_ids[idx].get(),
|
|
|
|
- }
|
|
|
|
- .unwrap();
|
|
|
|
-
|
|
|
|
- self.mutations.push(Mutation::Remove { id })
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- self.clean_up_node(node);
|
|
|
|
-
|
|
|
|
- for root in node.root_ids {
|
|
|
|
- let id = root.get().unwrap();
|
|
|
|
- if id.0 != 0 {
|
|
|
|
- self.reclaim(id);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- fn remove_component(&mut self, scope_id: ScopeId) {
|
|
|
|
- let height = self.scopes[scope_id.0].height;
|
|
|
|
- self.dirty_scopes.remove(&DirtyScope {
|
|
|
|
- height,
|
|
|
|
- id: scope_id,
|
|
|
|
- });
|
|
|
|
-
|
|
|
|
- // I promise, since we're descending down the tree, this is safe
|
|
|
|
- match unsafe { self.scopes[scope_id.0].root_node().extend_lifetime_ref() } {
|
|
|
|
- RenderReturn::Sync(Ok(t)) => self.remove_node(t),
|
|
|
|
- _ => todo!("cannot handle nonstandard nodes"),
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
/// Push all the real nodes on the stack
|
|
/// Push all the real nodes on the stack
|
|
fn push_all_real_nodes(&mut self, node: &'b VNode<'b>) -> usize {
|
|
fn push_all_real_nodes(&mut self, node: &'b VNode<'b>) -> usize {
|
|
node.template
|
|
node.template
|
|
@@ -832,44 +643,50 @@ impl<'b> VirtualDom {
|
|
.iter()
|
|
.iter()
|
|
.enumerate()
|
|
.enumerate()
|
|
.map(|(idx, _)| {
|
|
.map(|(idx, _)| {
|
|
- match node.dynamic_root(idx) {
|
|
|
|
- Some(Text(t)) => {
|
|
|
|
|
|
+ let node = match node.dynamic_root(idx) {
|
|
|
|
+ Some(node) => node,
|
|
|
|
+ None => {
|
|
|
|
+ self.mutations.push(Mutation::PushRoot {
|
|
|
|
+ id: node.root_ids[idx].get().unwrap(),
|
|
|
|
+ });
|
|
|
|
+ return 1;
|
|
|
|
+ }
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ match node {
|
|
|
|
+ Text(t) => {
|
|
self.mutations.push(Mutation::PushRoot {
|
|
self.mutations.push(Mutation::PushRoot {
|
|
id: t.id.get().unwrap(),
|
|
id: t.id.get().unwrap(),
|
|
});
|
|
});
|
|
1
|
|
1
|
|
}
|
|
}
|
|
- Some(Placeholder(t)) => {
|
|
|
|
|
|
+ Placeholder(t) => {
|
|
self.mutations.push(Mutation::PushRoot {
|
|
self.mutations.push(Mutation::PushRoot {
|
|
id: t.id.get().unwrap(),
|
|
id: t.id.get().unwrap(),
|
|
});
|
|
});
|
|
1
|
|
1
|
|
}
|
|
}
|
|
- Some(Fragment(nodes)) => nodes
|
|
|
|
|
|
+ Fragment(nodes) => nodes
|
|
.iter()
|
|
.iter()
|
|
.map(|node| self.push_all_real_nodes(node))
|
|
.map(|node| self.push_all_real_nodes(node))
|
|
.count(),
|
|
.count(),
|
|
|
|
|
|
- Some(Component(comp)) => {
|
|
|
|
|
|
+ Component(comp) => {
|
|
let scope = comp.scope.get().unwrap();
|
|
let scope = comp.scope.get().unwrap();
|
|
match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
RenderReturn::Sync(Ok(node)) => self.push_all_real_nodes(node),
|
|
RenderReturn::Sync(Ok(node)) => self.push_all_real_nodes(node),
|
|
_ => todo!(),
|
|
_ => todo!(),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- None => {
|
|
|
|
- self.mutations.push(Mutation::PushRoot {
|
|
|
|
- id: node.root_ids[idx].get().unwrap(),
|
|
|
|
- });
|
|
|
|
- 1
|
|
|
|
- }
|
|
|
|
- };
|
|
|
|
|
|
+ }
|
|
})
|
|
})
|
|
.count()
|
|
.count()
|
|
}
|
|
}
|
|
|
|
|
|
- fn create_children(&mut self, nodes: &'b [VNode<'b>]) -> usize {
|
|
|
|
- nodes.iter().fold(0, |acc, child| acc + self.create(child))
|
|
|
|
|
|
+ fn create_children(&mut self, nodes: impl IntoIterator<Item = &'b VNode<'b>>) -> usize {
|
|
|
|
+ nodes
|
|
|
|
+ .into_iter()
|
|
|
|
+ .fold(0, |acc, child| acc + self.create(child))
|
|
}
|
|
}
|
|
|
|
|
|
fn create_and_insert_before(&mut self, new: &'b [VNode<'b>], before: &'b VNode<'b>) {
|
|
fn create_and_insert_before(&mut self, new: &'b [VNode<'b>], before: &'b VNode<'b>) {
|
|
@@ -884,6 +701,134 @@ impl<'b> VirtualDom {
|
|
self.mutations.push(Mutation::InsertAfter { id, m })
|
|
self.mutations.push(Mutation::InsertAfter { id, m })
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ /// Simply replace a placeholder with a list of nodes
|
|
|
|
+ fn replace_placeholder(&mut self, l: &'b VPlaceholder, r: &'b [VNode<'b>]) {
|
|
|
|
+ let m = self.create_children(r);
|
|
|
|
+ let id = l.id.get().unwrap();
|
|
|
|
+ self.mutations.push(Mutation::ReplaceWith { id, m });
|
|
|
|
+ self.reclaim(id);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn replace(&mut self, left: &'b VNode<'b>, right: impl IntoIterator<Item = &'b VNode<'b>>) {
|
|
|
|
+ let m = self.create_children(right);
|
|
|
|
+
|
|
|
|
+ let id = self.find_last_element(left);
|
|
|
|
+
|
|
|
|
+ self.mutations.push(Mutation::InsertAfter { id, m });
|
|
|
|
+
|
|
|
|
+ self.remove_node(left, true);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn node_to_placeholder(&mut self, l: &'b [VNode<'b>], r: &'b VPlaceholder) {
|
|
|
|
+ // Create the placeholder first, ensuring we get a dedicated ID for the placeholder
|
|
|
|
+ let placeholder = self.next_element(&l[0], &[]);
|
|
|
|
+
|
|
|
|
+ r.id.set(Some(placeholder));
|
|
|
|
+
|
|
|
|
+ let id = self.find_last_element(&l[0]);
|
|
|
|
+
|
|
|
|
+ self.mutations
|
|
|
|
+ .push(Mutation::CreatePlaceholder { id: placeholder });
|
|
|
|
+
|
|
|
|
+ self.mutations.push(Mutation::InsertAfter { id, m: 1 });
|
|
|
|
+
|
|
|
|
+ self.remove_nodes(l);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /// Remove these nodes from the dom
|
|
|
|
+ /// Wont generate mutations for the inner nodes
|
|
|
|
+ fn remove_nodes(&mut self, nodes: &'b [VNode<'b>]) {
|
|
|
|
+ nodes.iter().for_each(|node| self.remove_node(node, true));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn remove_node(&mut self, node: &'b VNode<'b>, gen_muts: bool) {
|
|
|
|
+ // Clean up the roots, assuming we need to generate mutations for these
|
|
|
|
+ for (idx, _) in node.template.roots.iter().enumerate() {
|
|
|
|
+ if let Some(dy) = node.dynamic_root(idx) {
|
|
|
|
+ self.remove_dynamic_node(dy, gen_muts);
|
|
|
|
+ } else {
|
|
|
|
+ let id = node.root_ids[idx].get().unwrap();
|
|
|
|
+ if gen_muts {
|
|
|
|
+ self.mutations.push(Mutation::Remove { id });
|
|
|
|
+ }
|
|
|
|
+ self.reclaim(id);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ for (idx, dyn_node) in node.dynamic_nodes.iter().enumerate() {
|
|
|
|
+ // Roots are cleaned up automatically above
|
|
|
|
+ if node.template.node_paths[idx].len() == 1 {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ self.remove_dynamic_node(dyn_node, false);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // we clean up nodes with dynamic attributes, provided the node is unique and not a root node
|
|
|
|
+ let mut id = None;
|
|
|
|
+ for (idx, attr) in node.dynamic_attrs.iter().enumerate() {
|
|
|
|
+ // We'll clean up the root nodes either way, so don't worry
|
|
|
|
+ if node.template.attr_paths[idx].len() == 1 {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ let next_id = attr.mounted_element.get();
|
|
|
|
+
|
|
|
|
+ if id == Some(next_id) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ id = Some(next_id);
|
|
|
|
+
|
|
|
|
+ self.reclaim(next_id);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn remove_dynamic_node(&mut self, node: &DynamicNode, gen_muts: bool) {
|
|
|
|
+ match node {
|
|
|
|
+ Component(comp) => self.remove_component_node(comp, gen_muts),
|
|
|
|
+ Text(t) => self.remove_text_node(t),
|
|
|
|
+ Placeholder(t) => self.remove_placeholder(t),
|
|
|
|
+ Fragment(nodes) => nodes
|
|
|
|
+ .iter()
|
|
|
|
+ .for_each(|node| self.remove_node(node, gen_muts)),
|
|
|
|
+ };
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn remove_placeholder(&mut self, t: &VPlaceholder) {
|
|
|
|
+ if let Some(id) = t.id.take() {
|
|
|
|
+ self.reclaim(id)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn remove_text_node(&mut self, t: &VText) {
|
|
|
|
+ if let Some(id) = t.id.take() {
|
|
|
|
+ self.reclaim(id)
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn remove_component_node(&mut self, comp: &VComponent, gen_muts: bool) {
|
|
|
|
+ if let Some(scope) = comp.scope.take() {
|
|
|
|
+ match unsafe { self.scopes[scope.0].root_node().extend_lifetime_ref() } {
|
|
|
|
+ RenderReturn::Sync(Ok(t)) => self.remove_node(t, gen_muts),
|
|
|
|
+ _ => todo!("cannot handle nonstandard nodes"),
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ let props = self.scopes[scope.0].props.take();
|
|
|
|
+
|
|
|
|
+ self.dirty_scopes.remove(&DirtyScope {
|
|
|
|
+ height: self.scopes[scope.0].height,
|
|
|
|
+ id: scope,
|
|
|
|
+ });
|
|
|
|
+
|
|
|
|
+ *comp.props.borrow_mut() = unsafe { std::mem::transmute(props) };
|
|
|
|
+
|
|
|
|
+ // make sure to wipe any of its props and listeners
|
|
|
|
+ self.ensure_drop_safety(scope);
|
|
|
|
+ self.scopes.remove(scope.0);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
fn find_first_element(&self, node: &'b VNode<'b>) -> ElementId {
|
|
fn find_first_element(&self, node: &'b VNode<'b>) -> ElementId {
|
|
match node.dynamic_root(0) {
|
|
match node.dynamic_root(0) {
|
|
None => node.root_ids[0].get().unwrap(),
|
|
None => node.root_ids[0].get().unwrap(),
|
|
@@ -915,28 +860,6 @@ impl<'b> VirtualDom {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
-
|
|
|
|
- fn replace(&mut self, left: &'b VNode<'b>, right: &'b VNode<'b>) {
|
|
|
|
- let first = self.find_first_element(left);
|
|
|
|
- let id = self.replace_inner(left);
|
|
|
|
- let created = self.create(right);
|
|
|
|
- self.mutations.push(Mutation::ReplaceWith {
|
|
|
|
- id: first,
|
|
|
|
- m: created,
|
|
|
|
- });
|
|
|
|
- self.try_reclaim(id);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- fn replace_many(&mut self, left: &'b VNode<'b>, right: &'b [VNode<'b>]) {
|
|
|
|
- let first = self.find_first_element(left);
|
|
|
|
- let id = self.replace_inner(left);
|
|
|
|
- let created = self.create_children(right);
|
|
|
|
- self.mutations.push(Mutation::ReplaceWith {
|
|
|
|
- id: first,
|
|
|
|
- m: created,
|
|
|
|
- });
|
|
|
|
- self.try_reclaim(id);
|
|
|
|
- }
|
|
|
|
}
|
|
}
|
|
|
|
|
|
/// Are the templates the same?
|
|
/// Are the templates the same?
|