123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212 |
- //! This module contains the stateful DiffMachine and all methods to diff VNodes, their properties, and their children.
- //!
- //! The [`DiffMachine`] calculates the diffs between the old and new frames, updates the new nodes, and generates a set
- //! of mutations for the RealDom to apply.
- //!
- //! ## Notice:
- //!
- //! The inspiration and code for this module was originally taken from Dodrio (@fitzgen) and then modified to support
- //! Components, Fragments, Suspense, SubTree memoization, incremental diffing, cancellation, NodeRefs, pausing, priority
- //! scheduling, and additional batching operations.
- //!
- //! ## Implementation Details:
- //!
- //! ### IDs for elements
- //! --------------------
- //! All nodes are addressed by their IDs. The RealDom provides an imperative interface for making changes to these nodes.
- //! We don't necessarily require that DOM changes happen instantly during the diffing process, so the implementor may choose
- //! to batch nodes if it is more performant for their application. The element IDs are indices into the internal element
- //! array. The expectation is that implementors will use the ID as an index into a Vec of real nodes, allowing for passive
- //! garbage collection as the VirtualDOM replaces old nodes.
- //!
- //! When new vnodes are created through `cx.render`, they won't know which real node they correspond to. During diffing,
- //! we always make sure to copy over the ID. If we don't do this properly, the ElementId will be populated incorrectly
- //! and brick the user's page.
- //!
- //! ### Fragment Support
- //! --------------------
- //! Fragments (nodes without a parent) are supported through a combination of "replace with" and anchor vnodes. Fragments
- //! can be particularly challenging when they are empty, so the anchor node lets us "reserve" a spot for the empty
- //! fragment to be replaced with when it is no longer empty. This is guaranteed by logic in the NodeFactory - it is
- //! impossible to craft a fragment with 0 elements - they must always have at least a single placeholder element. Adding
- //! "dummy" nodes _is_ inefficient, but it makes our diffing algorithm faster and the implementation is completely up to
- //! the platform.
- //!
- //! Other implementations either don't support fragments or use a "child + sibling" pattern to represent them. Our code is
- //! vastly simpler and more performant when we can just create a placeholder element while the fragment has no children.
- //!
- //! ### Suspense
- //! ------------
- //! Dioxus implements Suspense slightly differently than React. In React, each fiber is manually progressed until it runs
- //! into a promise-like value. React will then work on the next "ready" fiber, checking back on the previous fiber once
- //! it has finished its new work. In Dioxus, we use a similar approach, but try to completely render the tree before
- //! switching sub-fibers. Instead, each future is submitted into a futures-queue and the node is manually loaded later on.
- //! Due to the frequent calls to "yield_now" we can get the pure "fetch-as-you-render" behavior of React Fiber.
- //!
- //! We're able to use this approach because we use placeholder nodes - futures that aren't ready still get submitted to
- //! DOM, but as a placeholder.
- //!
- //! Right now, the "suspense" queue is intertwined with hooks. In the future, we should allow any future to drive attributes
- //! and contents, without the need for the "use_suspense" hook. In the interim, this is the quickest way to get Suspense working.
- //!
- //! ## Subtree Memoization
- //! -----------------------
- //! We also employ "subtree memoization" which saves us from having to check trees which hold no dynamic content. We can
- //! detect if a subtree is "static" by checking if its children are "static". Since we dive into the tree depth-first, the
- //! calls to "create" propagate this information upwards. Structures like the one below are entirely static:
- //! ```rust
- //! rsx!( div { class: "hello world", "this node is entirely static" } )
- //! ```
- //! Because the subtrees won't be diffed, their "real node" data will be stale (invalid), so it's up to the reconciler to
- //! track nodes created in a scope and clean up all relevant data. Support for this is currently WIP and depends on comp-time
- //! hashing of the subtree from the rsx! macro. We do a very limited form of static analysis via static string pointers as
- //! a way of short-circuiting the most expensive checks.
- //!
- //! ## Bloom Filter and Heuristics
- //! ------------------------------
- //! For all components, we employ some basic heuristics to speed up allocations and pre-size bump arenas. The heuristics are
- //! currently very rough, but will get better as time goes on. The information currently tracked includes the size of a
- //! bump arena after first render, the number of hooks, and the number of nodes in the tree.
- //!
- //! ## Garbage Collection
- //! ---------------------
- //! Dioxus uses a passive garbage collection system to clean up old nodes once the work has been completed. This garbage
- //! collection is done internally once the main diffing work is complete. After the "garbage" is collected, Dioxus will then
- //! start to re-use old keys for new nodes. This results in a passive memory management system that is very efficient.
- //!
- //! The IDs used by the key/map are just an index into a Vec. This means that Dioxus will drive the key allocation strategy
- //! so the client only needs to maintain a simple list of nodes. By default, Dioxus will not manually clean up old nodes
- //! for the client. As new nodes are created, old nodes will be over-written.
- //!
- //! ## Further Reading and Thoughts
- //! ----------------------------
- //! There are more ways of increasing diff performance here that are currently not implemented.
- //! - Strong memoization of subtrees.
- //! - Guided diffing.
- //! - Certain web-dom-specific optimizations.
- //!
- //! More info on how to improve this diffing algorithm:
- //! - https://hacks.mozilla.org/2019/03/fast-bump-allocated-virtual-doms-with-rust-and-wasm/
- use crate::innerlude::*;
- use futures_channel::mpsc::UnboundedSender;
- use fxhash::{FxHashMap, FxHashSet};
- use slab::Slab;
- use DomEdit::*;
- /// Our DiffMachine is an iterative tree differ.
- ///
- /// It uses techniques of a stack machine to allow pausing and restarting of the diff algorithm. This
- /// was originally implemented using recursive techniques, but Rust lacks the ability to call async functions recursively,
- /// meaning we could not "pause" the original diffing algorithm.
- ///
- /// Instead, we use a traditional stack machine approach to diff and create new nodes. The diff algorithm periodically
- /// calls "yield_now" which allows the machine to pause and return control to the caller. The caller can then wait for
- /// the next period of idle time, preventing our diff algorithm from blocking the main thread.
- ///
- /// Funnily enough, this stack machine's entire job is to create instructions for another stack machine to execute. It's
- /// stack machines all the way down!
- pub struct DiffState<'bump> {
- scopes: &'bump ScopeArena,
- pub mutations: Mutations<'bump>,
- pub(crate) stack: DiffStack<'bump>,
- pub seen_scopes: FxHashSet<ScopeId>,
- pub force_diff: bool,
- }
- impl<'bump> DiffState<'bump> {
- pub(crate) fn new(scopes: &'bump ScopeArena) -> Self {
- Self {
- scopes,
- mutations: Mutations::new(),
- stack: DiffStack::new(),
- seen_scopes: Default::default(),
- force_diff: false,
- }
- }
- }
- impl<'bump> DiffState<'bump> {
- pub fn diff_scope(&mut self, id: &ScopeId) {
- let (old, new) = (self.scopes.wip_head(id), self.scopes.fin_head(id));
- self.stack.push(DiffInstruction::Diff { old, new });
- self.work(|| false);
- }
- /// Progress the diffing for this "fiber"
- ///
- /// This method implements a depth-first iterative tree traversal.
- ///
- /// We do depth-first to maintain high cache locality (nodes were originally generated recursively).
- ///
- /// Returns a `bool` indicating that the work completed properly.
- pub fn work(&mut self, mut deadline_expired: impl FnMut() -> bool) -> bool {
- while let Some(instruction) = self.stack.pop() {
- match instruction {
- DiffInstruction::Diff { old, new } => self.diff_node(old, new),
- DiffInstruction::Create { node } => self.create_node(node),
- DiffInstruction::Mount { and } => self.mount(and),
- DiffInstruction::PrepareMove { node } => {
- let num_on_stack = self.push_all_nodes(node);
- self.stack.add_child_count(num_on_stack);
- }
- DiffInstruction::PopScope => self.stack.pop_off_scope(),
- };
- if deadline_expired() {
- log::debug!("Deadline expired before we could finished!");
- return false;
- }
- }
- true
- }
- // recursively push all the nodes of a tree onto the stack and return how many are there
- fn push_all_nodes(&mut self, node: &'bump VNode<'bump>) -> usize {
- match node {
- VNode::Text(_) | VNode::Anchor(_) | VNode::Suspended(_) => {
- self.mutations.push_root(node.mounted_id());
- 1
- }
- VNode::Linked(linked) => {
- todo!("load linked");
- 0
- // let num_on_stack = linked.children.iter().map(|child| {
- // self.push_all_nodes( child)
- // }).sum();
- // self.mutations.push_root(node.mounted_id());
- // num_on_stack + 1
- }
- VNode::Fragment(_) | VNode::Component(_) => {
- //
- let mut added = 0;
- for child in node.children() {
- added += self.push_all_nodes(child);
- }
- added
- }
- VNode::Element(el) => {
- let mut num_on_stack = 0;
- for child in el.children.iter() {
- num_on_stack += self.push_all_nodes(child);
- }
- self.mutations.push_root(el.dom_id.get().unwrap());
- num_on_stack + 1
- }
- }
- }
- fn mount(&mut self, and: MountType<'bump>) {
- let nodes_created = self.stack.pop_nodes_created();
- match and {
- // add the nodes from this virtual list to the parent
- // used by fragments and components
- MountType::Absorb => {
- self.stack.add_child_count(nodes_created);
- }
- MountType::Replace { old } => {
- if let Some(old_id) = old.try_mounted_id() {
- self.mutations.replace_with(old_id, nodes_created as u32);
- self.remove_nodes(Some(old), true);
- } else {
- if let Some(id) = self.find_first_element_id(old) {
- self.mutations.replace_with(id, nodes_created as u32);
- }
- self.remove_nodes(Some(old), true);
- }
- }
- MountType::Append => {
- self.mutations.edits.push(AppendChildren {
- many: nodes_created as u32,
- });
- }
- MountType::InsertAfter { other_node } => {
- let root = self.find_last_element(other_node).unwrap();
- self.mutations.insert_after(root, nodes_created as u32);
- }
- MountType::InsertBefore { other_node } => {
- let root = self.find_first_element_id(other_node).unwrap();
- self.mutations.insert_before(root, nodes_created as u32);
- }
- }
- }
- // =================================
- // Tools for creating new nodes
- // =================================
- fn create_node(&mut self, node: &'bump VNode<'bump>) {
- match node {
- VNode::Text(vtext) => self.create_text_node(vtext, node),
- VNode::Suspended(suspended) => self.create_suspended_node(suspended, node),
- VNode::Anchor(anchor) => self.create_anchor_node(anchor, node),
- VNode::Element(element) => self.create_element_node(element, node),
- VNode::Fragment(frag) => self.create_fragment_node(frag),
- VNode::Component(component) => self.create_component_node(component),
- VNode::Linked(linked) => self.create_linked_node(linked),
- }
- }
- fn create_text_node(&mut self, vtext: &'bump VText<'bump>, node: &'bump VNode<'bump>) {
- let real_id = self.scopes.reserve_node(node);
- self.mutations.create_text_node(vtext.text, real_id);
- vtext.dom_id.set(Some(real_id));
- self.stack.add_child_count(1);
- }
- fn create_suspended_node(&mut self, suspended: &'bump VSuspended, node: &'bump VNode<'bump>) {
- let real_id = self.scopes.reserve_node(node);
- self.mutations.create_placeholder(real_id);
- suspended.dom_id.set(Some(real_id));
- self.stack.add_child_count(1);
- self.attach_suspended_node_to_scope(suspended);
- }
- fn create_anchor_node(&mut self, anchor: &'bump VAnchor, node: &'bump VNode<'bump>) {
- let real_id = self.scopes.reserve_node(node);
- self.mutations.create_placeholder(real_id);
- anchor.dom_id.set(Some(real_id));
- self.stack.add_child_count(1);
- }
- fn create_element_node(&mut self, element: &'bump VElement<'bump>, node: &'bump VNode<'bump>) {
- let VElement {
- tag_name,
- listeners,
- attributes,
- children,
- namespace,
- dom_id,
- ..
- } = element;
- let real_id = self.scopes.reserve_node(node);
- dom_id.set(Some(real_id));
- self.mutations.create_element(tag_name, *namespace, real_id);
- self.stack.add_child_count(1);
- if let Some(cur_scope_id) = self.stack.current_scope() {
- let scope = self.scopes.get_scope(&cur_scope_id).unwrap();
- for listener in *listeners {
- self.attach_listener_to_scope(listener, scope);
- listener.mounted_node.set(Some(real_id));
- self.mutations.new_event_listener(listener, cur_scope_id);
- }
- } else {
- log::warn!("create element called with no scope on the stack - this is an error for a live dom");
- }
- for attr in *attributes {
- self.mutations.set_attribute(attr, real_id.as_u64());
- }
- if !children.is_empty() {
- self.stack.create_children(children, MountType::Append);
- }
- }
- fn create_fragment_node(&mut self, frag: &'bump VFragment<'bump>) {
- self.stack.create_children(frag.children, MountType::Absorb);
- }
- fn create_component_node(&mut self, vcomponent: &'bump VComponent<'bump>) {
- // let caller = vcomponent.caller;
- let parent_idx = self.stack.current_scope().unwrap();
- let shared: UnboundedSender<SchedulerMsg> = todo!();
- // let shared: UnboundedSender<SchedulerMsg> = self.sender.clone();
- // Insert a new scope into our component list
- let parent_scope = self.scopes.get_scope(&parent_idx).unwrap();
- let new_idx: ScopeId = todo!();
- // self
- // .new_with_key(fc_ptr, vcomp, parent_scope, height, subtree, sender);
- // .(|new_idx| {
- // // ScopeInner::new(
- // vcomponent,
- // new_idx,
- // Some(parent_idx),
- // parent_scope.height + 1,
- // parent_scope.subtree(),
- // shared,
- // // )
- // });
- // Actually initialize the caller's slot with the right address
- vcomponent.associated_scope.set(Some(new_idx));
- if !vcomponent.can_memoize {
- let cur_scope = self.scopes.get_scope(&parent_idx).unwrap();
- let extended = unsafe { std::mem::transmute(vcomponent) };
- cur_scope.items.get_mut().borrowed_props.push(extended);
- }
- // TODO:
- // add noderefs to current noderef list Noderefs
- // add effects to current effect list Effects
- let new_component = self.scopes.get_scope(&new_idx).unwrap();
- log::debug!(
- "initializing component {:?} with height {:?}",
- new_idx,
- parent_scope.height + 1
- );
- // Run the scope for one iteration to initialize it
- //
- todo!("run scope");
- // if new_component.run_scope(self) {
- // // Take the node that was just generated from running the component
- // let nextnode = new_component.frames.fin_head();
- // self.stack.create_component(new_idx, nextnode);
- // //
- // /*
- // tree_item {
- // }
- // */
- // if new_component.is_subtree_root.get() {
- // self.stack.push_subtree();
- // }
- // }
- // Finally, insert this scope as a seen node.
- self.seen_scopes.insert(new_idx);
- }
- fn create_linked_node(&mut self, link: &'bump NodeLink) {
- todo!()
- }
- // =================================
- // Tools for diffing nodes
- // =================================
- pub fn diff_node(&mut self, old_node: &'bump VNode<'bump>, new_node: &'bump VNode<'bump>) {
- use VNode::*;
- match (old_node, new_node) {
- // Check the most common cases first
- (Text(old), Text(new)) => self.diff_text_nodes(old, new),
- (Component(old), Component(new)) => {
- self.diff_component_nodes(old_node, new_node, old, new)
- }
- (Fragment(old), Fragment(new)) => self.diff_fragment_nodes(old, new),
- (Anchor(old), Anchor(new)) => new.dom_id.set(old.dom_id.get()),
- (Suspended(old), Suspended(new)) => self.diff_suspended_nodes(old, new),
- (Element(old), Element(new)) => self.diff_element_nodes(old, new, old_node, new_node),
- (Linked(old), Linked(new)) => self.diff_linked_nodes(old, new),
- // Anything else is just a basic replace and create
- (
- Linked(_) | Component(_) | Fragment(_) | Text(_) | Element(_) | Anchor(_)
- | Suspended(_),
- Linked(_) | Component(_) | Fragment(_) | Text(_) | Element(_) | Anchor(_)
- | Suspended(_),
- ) => self
- .stack
- .create_node(new_node, MountType::Replace { old: old_node }),
- }
- }
- fn diff_text_nodes(&mut self, old: &'bump VText<'bump>, new: &'bump VText<'bump>) {
- if let Some(root) = old.dom_id.get() {
- if old.text != new.text {
- self.mutations.set_text(new.text, root.as_u64());
- }
- new.dom_id.set(Some(root));
- }
- }
- fn diff_element_nodes(
- &mut self,
- old: &'bump VElement<'bump>,
- new: &'bump VElement<'bump>,
- old_node: &'bump VNode<'bump>,
- new_node: &'bump VNode<'bump>,
- ) {
- let root = old.dom_id.get().unwrap();
- // If the element type is completely different, the element needs to be re-rendered completely
- // This is an optimization React makes due to how users structure their code
- //
- // This case is rather rare (typically only in non-keyed lists)
- if new.tag_name != old.tag_name || new.namespace != old.namespace {
- // maybe make this an instruction?
- // issue is that we need the "vnode" but this method only has the velement
- self.stack.push_nodes_created(0);
- self.stack.push(DiffInstruction::Mount {
- and: MountType::Replace { old: old_node },
- });
- self.create_element_node(new, new_node);
- return;
- }
- new.dom_id.set(Some(root));
- // todo: attributes currently rely on the element on top of the stack, but in theory, we only need the id of the
- // element to modify its attributes.
- // it would result in fewer instructions if we just set the id directly.
- // it would also clean up this code some, but that's not very important anyways
- // Diff Attributes
- //
- // It's extraordinarily rare to have the number/order of attributes change
- // In these cases, we just completely erase the old set and make a new set
- //
- // TODO: take a more efficient path than this
- if old.attributes.len() == new.attributes.len() {
- for (old_attr, new_attr) in old.attributes.iter().zip(new.attributes.iter()) {
- if old_attr.value != new_attr.value || new_attr.is_volatile {
- self.mutations.set_attribute(new_attr, root.as_u64());
- }
- }
- } else {
- for attribute in old.attributes {
- self.mutations.remove_attribute(attribute, root.as_u64());
- }
- for attribute in new.attributes {
- self.mutations.set_attribute(attribute, root.as_u64())
- }
- }
- // Diff listeners
- //
- // It's extraordinarily rare to have the number/order of listeners change
- // In the cases where the listeners change, we completely wipe the data attributes and add new ones
- //
- // We also need to make sure that all listeners are properly attached to the parent scope (fix_listener)
- //
- // TODO: take a more efficient path than this
- if let Some(cur_scope_id) = self.stack.current_scope() {
- let scope = self.scopes.get_scope(&cur_scope_id).unwrap();
- if old.listeners.len() == new.listeners.len() {
- for (old_l, new_l) in old.listeners.iter().zip(new.listeners.iter()) {
- if old_l.event != new_l.event {
- self.mutations
- .remove_event_listener(old_l.event, root.as_u64());
- self.mutations.new_event_listener(new_l, cur_scope_id);
- }
- new_l.mounted_node.set(old_l.mounted_node.get());
- self.attach_listener_to_scope(new_l, scope);
- }
- } else {
- for listener in old.listeners {
- self.mutations
- .remove_event_listener(listener.event, root.as_u64());
- }
- for listener in new.listeners {
- listener.mounted_node.set(Some(root));
- self.mutations.new_event_listener(listener, cur_scope_id);
- self.attach_listener_to_scope(listener, scope);
- }
- }
- }
- if old.children.is_empty() && !new.children.is_empty() {
- self.mutations.edits.push(PushRoot {
- root: root.as_u64(),
- });
- self.stack.create_children(new.children, MountType::Append);
- } else {
- self.diff_children(old.children, new.children);
- }
- }
- fn diff_component_nodes(
- &mut self,
- old_node: &'bump VNode<'bump>,
- new_node: &'bump VNode<'bump>,
- old: &'bump VComponent<'bump>,
- new: &'bump VComponent<'bump>,
- ) {
- let scope_addr = old.associated_scope.get().unwrap();
- // Make sure we're dealing with the same component (by function pointer)
- if old.user_fc == new.user_fc {
- log::debug!("Diffing component {:?} - {:?}", new.user_fc, scope_addr);
- //
- self.stack.scope_stack.push(scope_addr);
- // Make sure the new component vnode is referencing the right scope id
- new.associated_scope.set(Some(scope_addr));
- // make sure the component's caller function is up to date
- let scope = self.scopes.get_scope(&scope_addr).unwrap();
- let mut items = scope.items.borrow_mut();
- // React doesn't automatically memoize, but we do.
- let props_are_the_same = todo!("reworking component memoization");
- // let props_are_the_same = todo!("reworking component memoization");
- // let props_are_the_same = old.comparator.unwrap();
- // if self.cfg.force_diff || !props_are_the_same(new) {
- // let succeeded = scope.run_scope(self);
- // if succeeded {
- // self.diff_node(scope.frames.wip_head(), scope.frames.fin_head());
- // }
- // }
- self.stack.scope_stack.pop();
- } else {
- self.stack
- .create_node(new_node, MountType::Replace { old: old_node });
- }
- }
- fn diff_fragment_nodes(&mut self, old: &'bump VFragment<'bump>, new: &'bump VFragment<'bump>) {
- // This is the case where options or direct vnodes might be used.
- // In this case, it's faster to just skip ahead to their diff
- if old.children.len() == 1 && new.children.len() == 1 {
- self.diff_node(&old.children[0], &new.children[0]);
- return;
- }
- debug_assert!(!old.children.is_empty());
- debug_assert!(!new.children.is_empty());
- self.diff_children(old.children, new.children);
- }
- fn diff_suspended_nodes(&mut self, old: &'bump VSuspended, new: &'bump VSuspended) {
- new.dom_id.set(old.dom_id.get());
- self.attach_suspended_node_to_scope(new);
- }
- fn diff_linked_nodes(&mut self, old: &'bump NodeLink, new: &'bump NodeLink) {
- todo!();
- // new.dom_id.set(old.dom_id.get());
- // self.attach_linked_node_to_scope( new);
- }
- // =============================================
- // Utilities for creating new diff instructions
- // =============================================
- // Diff the given set of old and new children.
- //
- // The parent must be on top of the change list stack when this function is
- // entered:
- //
- // [... parent]
- //
- // the change list stack is in the same state when this function returns.
- //
- // If old no anchors are provided, then it's assumed that we can freely append to the parent.
- //
- // Remember, non-empty lists does not mean that there are real elements, just that there are virtual elements.
- //
- // Fragment nodes cannot generate empty children lists, so we can assume that when a list is empty, it belongs only
- // to an element, and appending makes sense.
- fn diff_children(&mut self, old: &'bump [VNode<'bump>], new: &'bump [VNode<'bump>]) {
- // Remember, fragments can never be empty (they always have a single child)
- match (old, new) {
- ([], []) => {}
- ([], _) => {
- // we need to push the
- self.stack.create_children(new, MountType::Append);
- }
- (_, []) => {
- self.remove_nodes(old, true);
- }
- ([VNode::Anchor(old_anchor)], [VNode::Anchor(new_anchor)]) => {
- old_anchor.dom_id.set(new_anchor.dom_id.get());
- }
- ([VNode::Anchor(_)], _) => {
- self.stack
- .create_children(new, MountType::Replace { old: &old[0] });
- }
- (_, [VNode::Anchor(_)]) => {
- self.replace_and_create_many_with_one(old, &new[0]);
- }
- _ => {
- let new_is_keyed = new[0].key().is_some();
- let old_is_keyed = old[0].key().is_some();
- debug_assert!(
- new.iter().all(|n| n.key().is_some() == new_is_keyed),
- "all siblings must be keyed or all siblings must be non-keyed"
- );
- debug_assert!(
- old.iter().all(|o| o.key().is_some() == old_is_keyed),
- "all siblings must be keyed or all siblings must be non-keyed"
- );
- if new_is_keyed && old_is_keyed {
- self.diff_keyed_children(old, new);
- } else {
- self.diff_non_keyed_children(old, new);
- }
- }
- }
- }
- // Diff children that are not keyed.
- //
- // The parent must be on the top of the change list stack when entering this
- // function:
- //
- // [... parent]
- //
- // the change list stack is in the same state when this function returns.
- fn diff_non_keyed_children(&mut self, old: &'bump [VNode<'bump>], new: &'bump [VNode<'bump>]) {
- // Handled these cases in `diff_children` before calling this function.
- debug_assert!(!new.is_empty());
- debug_assert!(!old.is_empty());
- for (new, old) in new.iter().zip(old.iter()).rev() {
- self.stack.push(DiffInstruction::Diff { new, old });
- }
- use std::cmp::Ordering;
- match old.len().cmp(&new.len()) {
- Ordering::Greater => self.remove_nodes(&old[new.len()..], true),
- Ordering::Less => {
- self.stack.create_children(
- &new[old.len()..],
- MountType::InsertAfter {
- other_node: old.last().unwrap(),
- },
- );
- }
- Ordering::Equal => {
- // nothing - they're the same size
- }
- }
- }
- // Diffing "keyed" children.
- //
- // With keyed children, we care about whether we delete, move, or create nodes
- // versus mutate existing nodes in place. Presumably there is some sort of CSS
- // transition animation that makes the virtual DOM diffing algorithm
- // observable. By specifying keys for nodes, we know which virtual DOM nodes
- // must reuse (or not reuse) the same physical DOM nodes.
- //
- // This is loosely based on Inferno's keyed patching implementation. However, we
- // have to modify the algorithm since we are compiling the diff down into change
- // list instructions that will be executed later, rather than applying the
- // changes to the DOM directly as we compare virtual DOMs.
- //
- // https://github.com/infernojs/inferno/blob/36fd96/packages/inferno/src/DOM/patching.ts#L530-L739
- //
- // The stack is empty upon entry.
- fn diff_keyed_children(&mut self, old: &'bump [VNode<'bump>], new: &'bump [VNode<'bump>]) {
- if cfg!(debug_assertions) {
- let mut keys = fxhash::FxHashSet::default();
- let mut assert_unique_keys = |children: &'bump [VNode<'bump>]| {
- keys.clear();
- for child in children {
- let key = child.key();
- debug_assert!(
- key.is_some(),
- "if any sibling is keyed, all siblings must be keyed"
- );
- keys.insert(key);
- }
- debug_assert_eq!(
- children.len(),
- keys.len(),
- "keyed siblings must each have a unique key"
- );
- };
- assert_unique_keys(old);
- assert_unique_keys(new);
- }
- // First up, we diff all the nodes with the same key at the beginning of the
- // children.
- //
- // `shared_prefix_count` is the count of how many nodes at the start of
- // `new` and `old` share the same keys.
- let (left_offset, right_offset) = match self.diff_keyed_ends(old, new) {
- Some(count) => count,
- None => return,
- };
- // log::debug!(
- // "Left offset, right offset, {}, {}",
- // left_offset,
- // right_offset,
- // );
- // log::debug!("stack before lo is {:#?}", self.stack.instructions);
- // Ok, we now hopefully have a smaller range of children in the middle
- // within which to re-order nodes with the same keys, remove old nodes with
- // now-unused keys, and create new nodes with fresh keys.
- let old_middle = &old[left_offset..(old.len() - right_offset)];
- let new_middle = &new[left_offset..(new.len() - right_offset)];
- debug_assert!(
- !((old_middle.len() == new_middle.len()) && old_middle.is_empty()),
- "keyed children must have the same number of children"
- );
- if new_middle.is_empty() {
- // remove the old elements
- self.remove_nodes(old_middle, true);
- } else if old_middle.is_empty() {
- // there were no old elements, so just create the new elements
- // we need to find the right "foothold" though - we shouldn't use the "append" at all
- if left_offset == 0 {
- // insert at the beginning of the old list
- let foothold = &old[old.len() - right_offset];
- self.stack.create_children(
- new_middle,
- MountType::InsertBefore {
- other_node: foothold,
- },
- );
- } else if right_offset == 0 {
- // insert at the end the old list
- let foothold = old.last().unwrap();
- self.stack.create_children(
- new_middle,
- MountType::InsertAfter {
- other_node: foothold,
- },
- );
- } else {
- // inserting in the middle
- let foothold = &old[left_offset - 1];
- self.stack.create_children(
- new_middle,
- MountType::InsertAfter {
- other_node: foothold,
- },
- );
- }
- } else {
- self.diff_keyed_middle(old_middle, new_middle);
- }
- log::debug!("stack after km is {:#?}", self.stack.instructions);
- }
- /// Diff both ends of the children that share keys.
- ///
- /// Returns a left offset and right offset of that indicates a smaller section to pass onto the middle diffing.
- ///
- /// If there is no offset, then this function returns None and the diffing is complete.
- fn diff_keyed_ends(
- &mut self,
- old: &'bump [VNode<'bump>],
- new: &'bump [VNode<'bump>],
- ) -> Option<(usize, usize)> {
- let mut left_offset = 0;
- for (old, new) in old.iter().zip(new.iter()) {
- // abort early if we finally run into nodes with different keys
- if old.key() != new.key() {
- break;
- }
- self.stack.push(DiffInstruction::Diff { old, new });
- left_offset += 1;
- }
- // If that was all of the old children, then create and append the remaining
- // new children and we're finished.
- if left_offset == old.len() {
- self.stack.create_children(
- &new[left_offset..],
- MountType::InsertAfter {
- other_node: old.last().unwrap(),
- },
- );
- return None;
- }
- // And if that was all of the new children, then remove all of the remaining
- // old children and we're finished.
- if left_offset == new.len() {
- self.remove_nodes(&old[left_offset..], true);
- return None;
- }
- // if the shared prefix is less than either length, then we need to walk backwards
- let mut right_offset = 0;
- for (old, new) in old.iter().rev().zip(new.iter().rev()) {
- // abort early if we finally run into nodes with different keys
- if old.key() != new.key() {
- break;
- }
- self.diff_node(old, new);
- right_offset += 1;
- }
- Some((left_offset, right_offset))
- }
- // The most-general, expensive code path for keyed children diffing.
- //
- // We find the longest subsequence within `old` of children that are relatively
- // ordered the same way in `new` (via finding a longest-increasing-subsequence
- // of the old child's index within `new`). The children that are elements of
- // this subsequence will remain in place, minimizing the number of DOM moves we
- // will have to do.
- //
- // Upon entry to this function, the change list stack must be empty.
- //
- // This function will load the appropriate nodes onto the stack and do diffing in place.
- //
- // Upon exit from this function, it will be restored to that same self.
- fn diff_keyed_middle(&mut self, old: &'bump [VNode<'bump>], new: &'bump [VNode<'bump>]) {
- /*
- 1. Map the old keys into a numerical ordering based on indices.
- 2. Create a map of old key to its index
- 3. Map each new key to the old key, carrying over the old index.
- - IE if we have ABCD becomes BACD, our sequence would be 1,0,2,3
- - if we have ABCD to ABDE, our sequence would be 0,1,3,MAX because E doesn't exist
- now, we should have a list of integers that indicates where in the old list the new items map to.
- 4. Compute the LIS of this list
- - this indicates the longest list of new children that won't need to be moved.
- 5. Identify which nodes need to be removed
- 6. Identify which nodes will need to be diffed
- 7. Going along each item in the new list, create it and insert it before the next closest item in the LIS.
- - if the item already existed, just move it to the right place.
- 8. Finally, generate instructions to remove any old children.
- 9. Generate instructions to finally diff children that are the same between both
- */
- // 0. Debug sanity checks
- // Should have already diffed the shared-key prefixes and suffixes.
- debug_assert_ne!(new.first().map(|n| n.key()), old.first().map(|o| o.key()));
- debug_assert_ne!(new.last().map(|n| n.key()), old.last().map(|o| o.key()));
- // 1. Map the old keys into a numerical ordering based on indices.
- // 2. Create a map of old key to its index
- // IE if the keys were A B C, then we would have (A, 1) (B, 2) (C, 3).
- let old_key_to_old_index = old
- .iter()
- .enumerate()
- .map(|(i, o)| (o.key().unwrap(), i))
- .collect::<FxHashMap<_, _>>();
- let mut shared_keys = FxHashSet::default();
- // 3. Map each new key to the old key, carrying over the old index.
- let new_index_to_old_index = new
- .iter()
- .map(|node| {
- let key = node.key().unwrap();
- if let Some(&index) = old_key_to_old_index.get(&key) {
- shared_keys.insert(key);
- index
- } else {
- u32::MAX as usize
- }
- })
- .collect::<Vec<_>>();
- // If none of the old keys are reused by the new children, then we remove all the remaining old children and
- // create the new children afresh.
- if shared_keys.is_empty() {
- log::debug!(
- "no shared keys, replacing and creating many with many, {:#?}, {:#?}",
- old,
- new
- );
- log::debug!("old_key_to_old_index, {:#?}", old_key_to_old_index);
- log::debug!("new_index_to_old_index, {:#?}", new_index_to_old_index);
- log::debug!("shared_keys, {:#?}", shared_keys);
- self.replace_and_create_many_with_many(old, new);
- return;
- }
- // 4. Compute the LIS of this list
- let mut lis_sequence = Vec::default();
- lis_sequence.reserve(new_index_to_old_index.len());
- let mut predecessors = vec![0; new_index_to_old_index.len()];
- let mut starts = vec![0; new_index_to_old_index.len()];
- longest_increasing_subsequence::lis_with(
- &new_index_to_old_index,
- &mut lis_sequence,
- |a, b| a < b,
- &mut predecessors,
- &mut starts,
- );
- // the lis comes out backwards, I think. can't quite tell.
- lis_sequence.sort_unstable();
- // if a new node gets u32 max and is at the end, then it might be part of our LIS (because u32 max is a valid LIS)
- if lis_sequence.last().map(|f| new_index_to_old_index[*f]) == Some(u32::MAX as usize) {
- lis_sequence.pop();
- }
- let apply = |new_idx, new_node: &'bump VNode<'bump>, stack: &mut DiffStack<'bump>| {
- let old_index = new_index_to_old_index[new_idx];
- if old_index == u32::MAX as usize {
- stack.create_node(new_node, MountType::Absorb);
- } else {
- // this function should never take LIS indices
- stack.push(DiffInstruction::PrepareMove { node: new_node });
- stack.push(DiffInstruction::Diff {
- new: new_node,
- old: &old[old_index],
- });
- }
- };
- // add mount instruction for the last items not covered by the lis
- let first_lis = *lis_sequence.first().unwrap();
- if first_lis > 0 {
- self.stack.push_nodes_created(0);
- self.stack.push(DiffInstruction::Mount {
- and: MountType::InsertBefore {
- other_node: &new[first_lis],
- },
- });
- for (idx, new_node) in new[..first_lis].iter().enumerate().rev() {
- apply(idx, new_node, &mut self.stack);
- }
- }
- // for each spacing, generate a mount instruction
- let mut lis_iter = lis_sequence.iter().rev();
- let mut last = *lis_iter.next().unwrap();
- for next in lis_iter {
- if last - next > 1 {
- self.stack.push_nodes_created(0);
- self.stack.push(DiffInstruction::Mount {
- and: MountType::InsertBefore {
- other_node: &new[last],
- },
- });
- for (idx, new_node) in new[(next + 1)..last].iter().enumerate().rev() {
- apply(idx + next + 1, new_node, &mut self.stack);
- }
- }
- last = *next;
- }
- // add mount instruction for the first items not covered by the lis
- let last = *lis_sequence.last().unwrap();
- if last < (new.len() - 1) {
- self.stack.push_nodes_created(0);
- self.stack.push(DiffInstruction::Mount {
- and: MountType::InsertAfter {
- other_node: &new[last],
- },
- });
- for (idx, new_node) in new[(last + 1)..].iter().enumerate().rev() {
- apply(idx + last + 1, new_node, &mut self.stack);
- }
- }
- for idx in lis_sequence.iter().rev() {
- self.stack.push(DiffInstruction::Diff {
- new: &new[*idx],
- old: &old[new_index_to_old_index[*idx]],
- });
- }
- }
- // =====================
- // Utilities
- // =====================
- fn find_last_element(&mut self, vnode: &'bump VNode<'bump>) -> Option<ElementId> {
- let mut search_node = Some(vnode);
- loop {
- match &search_node.take().unwrap() {
- VNode::Text(t) => break t.dom_id.get(),
- VNode::Element(t) => break t.dom_id.get(),
- VNode::Suspended(t) => break t.dom_id.get(),
- VNode::Anchor(t) => break t.dom_id.get(),
- VNode::Linked(_) => {
- todo!()
- }
- VNode::Fragment(frag) => {
- search_node = frag.children.last();
- }
- VNode::Component(el) => {
- let scope_id = el.associated_scope.get().unwrap();
- // let scope = self.scopes.get_scope(&scope_id).unwrap();
- search_node = Some(self.scopes.root_node(&scope_id));
- }
- }
- }
- }
- fn find_first_element_id(&mut self, vnode: &'bump VNode<'bump>) -> Option<ElementId> {
- let mut search_node = Some(vnode);
- loop {
- match &search_node.take().unwrap() {
- // the ones that have a direct id
- VNode::Fragment(frag) => {
- search_node = Some(&frag.children[0]);
- }
- VNode::Component(el) => {
- let scope_id = el.associated_scope.get().unwrap();
- // let scope = self.scopes.get_scope(&scope_id).unwrap();
- search_node = Some(self.scopes.root_node(&scope_id));
- }
- VNode::Linked(link) => {
- todo!("linked")
- }
- VNode::Text(t) => break t.dom_id.get(),
- VNode::Element(t) => break t.dom_id.get(),
- VNode::Suspended(t) => break t.dom_id.get(),
- VNode::Anchor(t) => break t.dom_id.get(),
- }
- }
- }
- fn replace_and_create_many_with_one(
- &mut self,
- old: &'bump [VNode<'bump>],
- new: &'bump VNode<'bump>,
- ) {
- if let Some(first_old) = old.get(0) {
- self.remove_nodes(&old[1..], true);
- self.stack
- .create_node(new, MountType::Replace { old: first_old });
- } else {
- self.stack.create_node(new, MountType::Append {});
- }
- }
- /// schedules nodes for garbage collection and pushes "remove" to the mutation stack
- /// remove can happen whenever
- fn remove_nodes(
- &mut self,
- nodes: impl IntoIterator<Item = &'bump VNode<'bump>>,
- gen_muts: bool,
- ) {
- // or cache the vec on the diff machine
- for node in nodes {
- match node {
- VNode::Text(t) => {
- let id = t.dom_id.get().unwrap();
- self.scopes.collect_garbage(id);
- if gen_muts {
- self.mutations.remove(id.as_u64());
- }
- }
- VNode::Suspended(s) => {
- let id = s.dom_id.get().unwrap();
- self.scopes.collect_garbage(id);
- if gen_muts {
- self.mutations.remove(id.as_u64());
- }
- }
- VNode::Anchor(a) => {
- let id = a.dom_id.get().unwrap();
- self.scopes.collect_garbage(id);
- if gen_muts {
- self.mutations.remove(id.as_u64());
- }
- }
- VNode::Element(e) => {
- let id = e.dom_id.get().unwrap();
- if gen_muts {
- self.mutations.remove(id.as_u64());
- }
- self.remove_nodes(e.children, false);
- }
- VNode::Fragment(f) => {
- self.remove_nodes(f.children, gen_muts);
- }
- VNode::Linked(l) => {
- todo!()
- }
- VNode::Component(c) => {
- let scope_id = c.associated_scope.get().unwrap();
- // let scope = self.scopes.get_scope(&scope_id).unwrap();
- let root = self.scopes.root_node(&scope_id);
- self.remove_nodes(Some(root), gen_muts);
- log::debug!("Destroying scope {:?}", scope_id);
- let mut s = self.scopes.try_remove(&scope_id).unwrap();
- s.hooks.clear_hooks();
- }
- }
- }
- }
- /// Remove all the old nodes and replace them with newly created new nodes.
- ///
- /// The new nodes *will* be created - don't create them yourself!
- fn replace_and_create_many_with_many(
- &mut self,
- old: &'bump [VNode<'bump>],
- new: &'bump [VNode<'bump>],
- ) {
- if let Some(first_old) = old.get(0) {
- self.remove_nodes(&old[1..], true);
- self.stack
- .create_children(new, MountType::Replace { old: first_old })
- } else {
- self.stack.create_children(new, MountType::Append {});
- }
- }
- /// Adds a listener closure to a scope during diff.
- fn attach_listener_to_scope(&mut self, listener: &'bump Listener<'bump>, scope: &Scope) {
- let long_listener = unsafe { std::mem::transmute(listener) };
- scope.items.borrow_mut().listeners.push(long_listener)
- }
- fn attach_suspended_node_to_scope(&mut self, suspended: &'bump VSuspended) {
- if let Some(scope) = self
- .stack
- .current_scope()
- .and_then(|id| self.scopes.get_scope(&id))
- {
- // safety: this lifetime is managed by the logic on scope
- let extended = unsafe { std::mem::transmute(suspended) };
- scope
- .items
- .borrow_mut()
- .suspended_nodes
- .insert(suspended.task_id, extended);
- }
- }
- }
|