dioxus/packages/core/src/create.rs

400 lines
16 KiB
Rust
Raw Normal View History

use std::cell::Cell;
use crate::innerlude::{VComponent, VText};
use crate::mutations::Mutation;
2022-11-02 01:42:29 +00:00
use crate::mutations::Mutation::*;
use crate::nodes::VNode;
2022-11-03 08:24:20 +00:00
use crate::nodes::{DynamicNode, TemplateNode};
2022-11-09 03:39:37 +00:00
use crate::virtual_dom::VirtualDom;
2022-12-03 00:24:49 +00:00
use crate::{AttributeValue, ElementId, RenderReturn, ScopeId, SuspenseContext};
2022-11-29 21:31:04 +00:00
impl<'b> VirtualDom {
2022-11-22 01:00:34 +00:00
/// Create a new template [`VNode`] and write it to the [`Mutations`] buffer.
///
/// This method pushes the ScopeID to the internal scopestack and returns the number of nodes created.
2022-11-23 03:59:56 +00:00
pub(crate) fn create_scope(&mut self, scope: ScopeId, template: &'b VNode<'b>) -> usize {
2022-11-09 18:58:11 +00:00
self.scope_stack.push(scope);
2022-11-23 03:59:56 +00:00
let out = self.create(template);
2022-11-09 18:58:11 +00:00
self.scope_stack.pop();
2022-11-22 01:00:34 +00:00
2022-11-09 18:58:11 +00:00
out
}
/// Create this template and write its mutations
2022-11-23 03:59:56 +00:00
pub(crate) fn create(&mut self, template: &'b VNode<'b>) -> usize {
2022-11-09 18:58:11 +00:00
// The best renderers will have templates prehydrated and registered
// Just in case, let's create the template using instructions anyways
2022-12-03 00:24:49 +00:00
if !self.templates.contains_key(&template.template.name) {
2022-11-23 03:59:56 +00:00
self.register_template(template);
}
2022-11-09 18:58:11 +00:00
// Walk the roots, creating nodes and assigning IDs
// todo: adjust dynamic nodes to be in the order of roots and then leaves (ie BFS)
2022-11-16 09:13:39 +00:00
let mut dynamic_attrs = template.template.attr_paths.iter().enumerate().peekable();
let mut dynamic_nodes = template.template.node_paths.iter().enumerate().peekable();
let cur_scope = self.scope_stack.last().copied().unwrap();
let mut on_stack = 0;
2022-11-16 09:13:39 +00:00
for (root_idx, root) in template.template.roots.iter().enumerate() {
2022-11-24 07:15:01 +00:00
// We might need to generate an ID for the root node
on_stack += match root {
2022-11-16 07:22:41 +00:00
TemplateNode::DynamicText(id) | TemplateNode::Dynamic(id) => {
2022-11-16 19:48:47 +00:00
match &template.dynamic_nodes[*id] {
2022-11-24 07:15:01 +00:00
// a dynamic text node doesn't replace a template node, instead we create it on the fly
2022-11-20 23:58:05 +00:00
DynamicNode::Text(VText { id: slot, value }) => {
2022-11-18 04:00:39 +00:00
let id = self.next_element(template, template.template.node_paths[*id]);
2022-11-16 09:13:39 +00:00
slot.set(id);
2022-11-29 21:31:04 +00:00
// Safety: we promise not to re-alias this text later on after committing it to the mutation
let unbounded_text = unsafe { std::mem::transmute(*value) };
self.mutations.push(CreateTextNode {
value: unbounded_text,
id,
});
2022-11-16 09:13:39 +00:00
1
}
2022-11-24 07:15:01 +00:00
DynamicNode::Placeholder(slot) => {
2022-11-24 07:15:01 +00:00
let id = self.next_element(template, template.template.node_paths[*id]);
slot.set(id);
self.mutations.push(CreatePlaceholder { id });
1
}
DynamicNode::Fragment(_) | DynamicNode::Component { .. } => {
2022-11-24 07:15:01 +00:00
self.create_dynamic_node(template, &template.dynamic_nodes[*id], *id)
}
2022-11-16 09:13:39 +00:00
}
}
2022-11-24 07:15:01 +00:00
TemplateNode::Element { .. } | TemplateNode::Text(_) => {
2022-11-27 14:38:40 +00:00
let this_id = self.next_root(template, root_idx);
2022-11-24 07:15:01 +00:00
template.root_ids[root_idx].set(this_id);
self.mutations.push(LoadTemplate {
2022-12-03 00:24:49 +00:00
name: template.template.name,
2022-11-24 07:15:01 +00:00
index: root_idx,
id: this_id,
});
2022-11-24 07:15:01 +00:00
// we're on top of a node that has a dynamic attribute for a descendant
// Set that attribute now before the stack gets in a weird state
while let Some((mut attr_id, path)) =
dynamic_attrs.next_if(|(_, p)| p[0] == root_idx as u8)
{
// if attribute is on a root node, then we've already created the element
// Else, it's deep in the template and we should create a new id for it
let id = match path.len() {
1 => this_id,
2022-11-27 05:22:39 +00:00
_ => {
let id = self
.next_element(template, template.template.attr_paths[attr_id]);
self.mutations.push(Mutation::AssignId {
path: &path[1..],
id,
});
id
}
2022-11-24 07:15:01 +00:00
};
loop {
let attribute = template.dynamic_attrs.get(attr_id).unwrap();
attribute.mounted_element.set(id);
2022-11-29 21:31:04 +00:00
// Safety: we promise not to re-alias this text later on after committing it to the mutation
let unbounded_name = unsafe { std::mem::transmute(attribute.name) };
2022-11-24 07:15:01 +00:00
match &attribute.value {
2022-11-29 21:31:04 +00:00
AttributeValue::Text(value) => {
// Safety: we promise not to re-alias this text later on after committing it to the mutation
let unbounded_value = unsafe { std::mem::transmute(*value) };
self.mutations.push(SetAttribute {
name: unbounded_name,
value: unbounded_value,
ns: attribute.namespace,
id,
})
}
2022-11-24 07:15:01 +00:00
AttributeValue::Bool(value) => {
self.mutations.push(SetBoolAttribute {
2022-11-29 21:31:04 +00:00
name: unbounded_name,
2022-11-24 07:15:01 +00:00
value: *value,
id,
})
}
AttributeValue::Listener(_) => {
self.mutations.push(NewEventListener {
// all listeners start with "on"
2022-11-30 22:21:10 +00:00
name: &unbounded_name[2..],
2022-11-24 07:15:01 +00:00
scope: cur_scope,
id,
})
}
AttributeValue::Float(_) => todo!(),
AttributeValue::Int(_) => todo!(),
AttributeValue::Any(_) => todo!(),
AttributeValue::None => todo!(),
}
2022-11-24 07:15:01 +00:00
// Only push the dynamic attributes forward if they match the current path (same element)
match dynamic_attrs.next_if(|(_, p)| *p == path) {
Some((next_attr_id, _)) => attr_id = next_attr_id,
None => break,
}
}
2022-11-03 00:29:18 +00:00
}
2022-11-16 19:48:47 +00:00
2022-11-24 07:15:01 +00:00
// We're on top of a node that has a dynamic child for a descendant
// Skip any node that's a root
let mut start = None;
let mut end = None;
// Collect all the dynamic nodes below this root
// We assign the start and end of the range of dynamic nodes since they area ordered in terms of tree path
//
// [0]
// [1, 1] <---|
// [1, 1, 1] <---| these are the range of dynamic nodes below root 1
// [1, 1, 2] <---|
// [2]
//
// We collect each range and then create them and replace the placeholder in the template
while let Some((idx, p)) =
dynamic_nodes.next_if(|(_, p)| p[0] == root_idx as u8)
{
if p.len() == 1 {
continue;
}
2022-11-16 19:48:47 +00:00
2022-11-24 07:15:01 +00:00
if start.is_none() {
start = Some(idx);
}
2022-11-16 19:48:47 +00:00
2022-11-24 07:15:01 +00:00
end = Some(idx);
}
2022-11-16 09:13:39 +00:00
2022-11-24 07:15:01 +00:00
//
if let (Some(start), Some(end)) = (start, end) {
for idx in start..=end {
let node = &template.dynamic_nodes[idx];
let m = self.create_dynamic_node(template, node, idx);
if m > 0 {
self.mutations.push(ReplacePlaceholder {
m,
path: &template.template.node_paths[idx][1..],
});
}
}
2022-11-16 19:48:47 +00:00
}
2022-11-24 07:15:01 +00:00
// elements create only one node :-)
1
2022-11-16 19:48:47 +00:00
}
2022-11-24 07:15:01 +00:00
};
}
on_stack
}
2022-11-22 01:00:34 +00:00
/// Insert a new template into the VirtualDom's template registry
2022-11-23 03:59:56 +00:00
fn register_template(&mut self, template: &'b VNode<'b>) {
2022-11-23 05:32:26 +00:00
// First, make sure we mark the template as seen, regardless if we process it
self.templates
2022-12-03 00:24:49 +00:00
.insert(template.template.name, template.template);
2022-11-23 05:32:26 +00:00
// If it's all dynamic nodes, then we don't need to register it
// Quickly run through and see if it's all just dynamic nodes
let dynamic_roots = template
.template
.roots
.iter()
.filter(|root| {
matches!(
root,
TemplateNode::Dynamic(_) | TemplateNode::DynamicText(_)
)
})
.count();
if dynamic_roots == template.template.roots.len() {
return;
}
2022-12-01 05:46:15 +00:00
self.mutations.templates.push(template.template);
}
2022-11-23 03:59:56 +00:00
pub(crate) fn create_dynamic_node(
&mut self,
2022-11-23 03:59:56 +00:00
template: &'b VNode<'b>,
node: &'b DynamicNode<'b>,
2022-11-03 08:24:20 +00:00
idx: usize,
) -> usize {
2022-11-22 01:00:34 +00:00
use DynamicNode::*;
match node {
2022-11-23 03:59:56 +00:00
Text(text) => self.create_dynamic_text(template, text, idx),
Fragment(frag) => self.create_fragment(frag),
Placeholder(frag) => self.create_placeholder(frag, template, idx),
2022-11-23 03:59:56 +00:00
Component(component) => self.create_component_node(template, component, idx),
2022-11-22 01:00:34 +00:00
}
}
2022-11-02 01:42:29 +00:00
2022-11-23 03:59:56 +00:00
fn create_dynamic_text(
2022-11-22 01:00:34 +00:00
&mut self,
2022-11-23 03:59:56 +00:00
template: &'b VNode<'b>,
text: &'b VText<'b>,
2022-11-22 01:00:34 +00:00
idx: usize,
) -> usize {
// Allocate a dynamic element reference for this text node
let new_id = self.next_element(template, template.template.node_paths[idx]);
2022-11-16 00:05:22 +00:00
2022-11-22 01:00:34 +00:00
// Make sure the text node is assigned to the correct element
text.id.set(new_id);
2022-11-16 00:05:22 +00:00
2022-11-29 21:31:04 +00:00
// Safety: we promise not to re-alias this text later on after committing it to the mutation
let value = unsafe { std::mem::transmute(text.value) };
2022-11-22 01:00:34 +00:00
// Add the mutation to the list
2022-11-23 03:59:56 +00:00
self.mutations.push(HydrateText {
2022-11-22 01:00:34 +00:00
id: new_id,
path: &template.template.node_paths[idx][1..],
2022-11-29 21:31:04 +00:00
value,
2022-11-22 01:00:34 +00:00
});
2022-11-04 05:30:26 +00:00
2022-11-22 01:00:34 +00:00
// Since we're hydrating an existing node, we don't create any new nodes
0
}
2022-11-16 19:48:47 +00:00
pub(crate) fn create_placeholder(
2022-11-22 01:00:34 +00:00
&mut self,
slot: &Cell<ElementId>,
2022-11-23 03:59:56 +00:00
template: &'b VNode<'b>,
2022-11-22 01:00:34 +00:00
idx: usize,
) -> usize {
// Allocate a dynamic element reference for this text node
let id = self.next_element(template, template.template.node_paths[idx]);
2022-11-16 19:48:47 +00:00
// Make sure the text node is assigned to the correct element
slot.set(id);
2022-11-16 19:48:47 +00:00
// Assign the ID to the existing node in the template
self.mutations.push(AssignId {
path: &template.template.node_paths[idx][1..],
id,
});
2022-11-16 19:48:47 +00:00
// Since the placeholder is already in the DOM, we don't create any new nodes
0
}
2022-11-16 19:48:47 +00:00
pub(crate) fn create_fragment(&mut self, nodes: &'b [VNode<'b>]) -> usize {
nodes.iter().fold(0, |acc, child| acc + self.create(child))
2022-11-22 01:00:34 +00:00
}
2022-11-08 06:55:22 +00:00
2022-11-29 21:31:04 +00:00
pub(super) fn create_component_node(
2022-11-22 01:00:34 +00:00
&mut self,
2022-11-23 03:59:56 +00:00
template: &'b VNode<'b>,
component: &'b VComponent<'b>,
2022-11-22 01:00:34 +00:00
idx: usize,
) -> usize {
2022-11-29 21:31:04 +00:00
let props = component
.props
.replace(None)
.expect("Props to always exist when a component is being created");
2022-11-09 03:39:37 +00:00
2022-11-29 21:31:04 +00:00
let unbounded_props = unsafe { std::mem::transmute(props) };
2022-11-02 01:42:29 +00:00
2022-11-29 21:31:04 +00:00
let scope = self.new_scope(unbounded_props).id;
2022-11-22 01:00:34 +00:00
component.scope.set(Some(scope));
2022-11-04 00:34:42 +00:00
2022-11-22 01:00:34 +00:00
let return_nodes = unsafe { self.run_scope(scope).extend_lifetime_ref() };
use RenderReturn::*;
2022-11-16 07:59:19 +00:00
2022-11-22 01:00:34 +00:00
match return_nodes {
2022-11-29 21:31:04 +00:00
Sync(Ok(t)) => self.mount_component(scope, template, t, idx),
2022-11-23 02:38:27 +00:00
Sync(Err(_e)) => todo!("Propogate error upwards"),
2022-11-23 03:59:56 +00:00
Async(_) => self.mount_component_placeholder(template, idx, scope),
}
}
2022-11-22 01:00:34 +00:00
2022-11-24 07:15:01 +00:00
fn mount_component(
&mut self,
scope: ScopeId,
parent: &'b VNode<'b>,
new: &'b VNode<'b>,
idx: usize,
) -> usize {
2022-11-22 18:05:13 +00:00
// Keep track of how many mutations are in the buffer in case we need to split them out if a suspense boundary
// is encountered
2022-12-01 05:46:15 +00:00
let mutations_to_this_point = self.mutations.edits.len();
2022-11-22 01:00:34 +00:00
// Create the component's root element
2022-11-24 07:15:01 +00:00
let created = self.create_scope(scope, new);
2022-11-22 01:00:34 +00:00
2022-11-22 18:05:13 +00:00
// If there are no suspense leaves below us, then just don't bother checking anything suspense related
if self.collected_leaves.is_empty() {
return created;
2022-11-22 01:00:34 +00:00
}
2022-11-22 18:05:13 +00:00
// If running the scope has collected some leaves and *this* component is a boundary, then handle the suspense
let boundary = match self.scopes[scope.0].has_context::<SuspenseContext>() {
2022-12-05 22:16:54 +00:00
Some(boundary) => boundary,
2022-11-22 18:05:13 +00:00
_ => return created,
};
// Since this is a boundary, use its placeholder within the template as the placeholder for the suspense tree
2022-11-24 07:15:01 +00:00
let new_id = self.next_element(new, parent.template.node_paths[idx]);
2022-11-22 18:05:13 +00:00
// Now connect everything to the boundary
self.scopes[scope.0].placeholder.set(Some(new_id));
// This involves breaking off the mutations to this point, and then creating a new placeholder for the boundary
// Note that we break off dynamic mutations only - since static mutations aren't rendered immediately
let split_off = unsafe {
std::mem::transmute::<Vec<Mutation>, Vec<Mutation>>(
2022-12-01 05:46:15 +00:00
self.mutations.edits.split_off(mutations_to_this_point),
2022-11-22 18:05:13 +00:00
)
};
2022-12-01 05:46:15 +00:00
boundary.mutations.borrow_mut().edits.extend(split_off);
2022-11-22 18:05:13 +00:00
boundary.created_on_stack.set(created);
boundary
.waiting_on
.borrow_mut()
.extend(self.collected_leaves.drain(..));
// Now assign the placeholder in the DOM
2022-11-23 03:59:56 +00:00
self.mutations.push(AssignId {
2022-11-22 18:05:13 +00:00
id: new_id,
2022-11-24 07:15:01 +00:00
path: &parent.template.node_paths[idx][1..],
2022-11-22 18:05:13 +00:00
});
0
2022-11-22 01:00:34 +00:00
}
/// Take the rendered nodes from a component and handle them if they were async
///
/// IE simply assign an ID to the placeholder
2022-11-22 18:05:13 +00:00
fn mount_component_placeholder(
2022-11-22 01:00:34 +00:00
&mut self,
template: &VNode,
idx: usize,
scope: ScopeId,
) -> usize {
let new_id = self.next_element(template, template.template.node_paths[idx]);
// Set the placeholder of the scope
self.scopes[scope.0].placeholder.set(Some(new_id));
// Since the placeholder is already in the DOM, we don't create any new nodes
2022-11-23 03:59:56 +00:00
self.mutations.push(AssignId {
2022-11-22 01:00:34 +00:00
id: new_id,
path: &template.template.node_paths[idx][1..],
});
0
}
}