dioxus/packages/core/src/scope_arena.rs

74 lines
2.2 KiB
Rust
Raw Normal View History

use crate::{
any_props::AnyProps,
arena::ElementId,
bump_frame::BumpFrame,
2022-11-02 01:42:29 +00:00
nodes::VNode,
scopes::{ScopeId, ScopeState},
virtualdom::VirtualDom,
};
impl VirtualDom {
2022-11-02 01:42:29 +00:00
pub fn new_scope(&mut self, props: *mut dyn AnyProps) -> ScopeId {
let parent = self.acquire_current_scope_raw();
let container = self.acquire_current_container();
let entry = self.scopes.vacant_entry();
let height = unsafe { parent.map(|f| (*f).height).unwrap_or(0) + 1 };
2022-11-02 01:42:29 +00:00
let id = ScopeId(entry.key());
entry.insert(ScopeState {
parent,
container,
2022-11-02 01:42:29 +00:00
id,
height,
props,
2022-11-02 01:42:29 +00:00
tasks: self.pending_futures.clone(),
node_arena_1: BumpFrame::new(50),
node_arena_2: BumpFrame::new(50),
render_cnt: Default::default(),
hook_arena: Default::default(),
hook_vals: Default::default(),
hook_idx: Default::default(),
2022-11-02 01:42:29 +00:00
shared_contexts: Default::default(),
});
2022-11-02 01:42:29 +00:00
id
}
pub fn acquire_current_container(&self) -> ElementId {
self.element_stack
.last()
.copied()
.expect("Always have a container")
}
2022-11-02 01:42:29 +00:00
fn acquire_current_scope_raw(&mut self) -> Option<*mut ScopeState> {
self.scope_stack
.last()
.copied()
.and_then(|id| self.scopes.get_mut(id.0).map(|f| f as *mut ScopeState))
}
pub fn run_scope<'a>(&'a mut self, id: ScopeId) -> &'a VNode<'a> {
let scope = &mut self.scopes[id.0];
scope.hook_idx.set(0);
2022-11-02 01:42:29 +00:00
let props = unsafe { &mut *scope.props };
let res = props.render(scope).unwrap();
let res: VNode<'static> = unsafe { std::mem::transmute(res) };
let frame = match scope.render_cnt % 2 {
0 => &mut scope.node_arena_1,
1 => &mut scope.node_arena_2,
_ => unreachable!(),
};
// set the head of the bump frame
let alloced = frame.bump.alloc(res);
frame.node.set(alloced);
// rebind the lifetime now that its stored internally
unsafe { std::mem::transmute(alloced) }
}
}