2022-10-28 04:58:47 +00:00
|
|
|
use crate::{
|
|
|
|
any_props::AnyProps,
|
|
|
|
arena::ElementId,
|
|
|
|
bump_frame::BumpFrame,
|
2022-11-04 03:56:31 +00:00
|
|
|
factory::RenderReturn,
|
2022-11-08 06:55:22 +00:00
|
|
|
innerlude::{SuspenseId, SuspenseLeaf},
|
2022-11-09 18:58:11 +00:00
|
|
|
scheduler::RcWake,
|
2022-11-02 01:42:29 +00:00
|
|
|
scopes::{ScopeId, ScopeState},
|
2022-11-09 03:39:37 +00:00
|
|
|
virtual_dom::VirtualDom,
|
2022-10-28 04:58:47 +00:00
|
|
|
};
|
2022-11-09 18:58:11 +00:00
|
|
|
use futures_util::FutureExt;
|
|
|
|
use std::{
|
|
|
|
mem,
|
|
|
|
pin::Pin,
|
|
|
|
rc::Rc,
|
|
|
|
task::{Context, Poll},
|
|
|
|
};
|
2022-10-28 04:58:47 +00:00
|
|
|
|
|
|
|
impl VirtualDom {
|
2022-11-16 00:05:22 +00:00
|
|
|
pub(super) fn new_scope(&mut self, props: *mut dyn AnyProps<'static>) -> &mut ScopeState {
|
2022-11-02 01:42:29 +00:00
|
|
|
let parent = self.acquire_current_scope_raw();
|
|
|
|
let container = self.acquire_current_container();
|
2022-10-28 04:58:47 +00:00
|
|
|
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());
|
2022-10-28 04:58:47 +00:00
|
|
|
|
|
|
|
entry.insert(ScopeState {
|
|
|
|
parent,
|
|
|
|
container,
|
2022-11-02 01:42:29 +00:00
|
|
|
id,
|
2022-10-28 04:58:47 +00:00
|
|
|
height,
|
|
|
|
props,
|
2022-11-08 06:55:22 +00:00
|
|
|
placeholder: None.into(),
|
2022-10-28 04:58:47 +00:00
|
|
|
node_arena_1: BumpFrame::new(50),
|
|
|
|
node_arena_2: BumpFrame::new(50),
|
2022-11-09 18:58:11 +00:00
|
|
|
spawned_tasks: Default::default(),
|
2022-10-28 04:58:47 +00:00
|
|
|
render_cnt: Default::default(),
|
|
|
|
hook_arena: Default::default(),
|
2022-11-19 23:24:28 +00:00
|
|
|
hook_list: Default::default(),
|
2022-10-28 04:58:47 +00:00
|
|
|
hook_idx: Default::default(),
|
2022-11-02 01:42:29 +00:00
|
|
|
shared_contexts: Default::default(),
|
2022-11-09 18:58:11 +00:00
|
|
|
tasks: self.scheduler.clone(),
|
|
|
|
})
|
2022-11-02 01:42:29 +00:00
|
|
|
}
|
|
|
|
|
2022-11-09 18:58:11 +00:00
|
|
|
fn acquire_current_container(&self) -> ElementId {
|
2022-11-02 01:42:29 +00:00
|
|
|
self.element_stack
|
|
|
|
.last()
|
|
|
|
.copied()
|
|
|
|
.expect("Always have a container")
|
2022-10-28 04:58:47 +00:00
|
|
|
}
|
|
|
|
|
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))
|
|
|
|
}
|
|
|
|
|
2022-11-09 18:58:11 +00:00
|
|
|
pub(crate) unsafe fn run_scope_extend<'a>(
|
|
|
|
&mut self,
|
|
|
|
scope_id: ScopeId,
|
|
|
|
) -> &'a RenderReturn<'a> {
|
|
|
|
unsafe { self.run_scope(scope_id).extend_lifetime_ref() }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn run_scope(&mut self, scope_id: ScopeId) -> &RenderReturn {
|
2022-11-18 06:31:14 +00:00
|
|
|
println!("run_scope: {:?}", scope_id);
|
|
|
|
|
2022-11-08 06:55:22 +00:00
|
|
|
let mut new_nodes = unsafe {
|
|
|
|
let scope = &mut self.scopes[scope_id.0];
|
2022-11-18 06:31:14 +00:00
|
|
|
println!("run_scope: scope: {:?}", scope.render_cnt.get());
|
2022-11-08 06:55:22 +00:00
|
|
|
scope.hook_idx.set(0);
|
2022-10-28 04:58:47 +00:00
|
|
|
|
2022-11-09 18:58:11 +00:00
|
|
|
// safety: due to how we traverse the tree, we know that the scope is not currently aliased
|
2022-11-08 06:55:22 +00:00
|
|
|
let props: &mut dyn AnyProps = mem::transmute(&mut *scope.props);
|
2022-11-09 18:58:11 +00:00
|
|
|
props.render(scope).extend_lifetime()
|
2022-11-03 09:11:04 +00:00
|
|
|
};
|
2022-10-28 04:58:47 +00:00
|
|
|
|
2022-11-08 06:55:22 +00:00
|
|
|
// immediately resolve futures that can be resolved
|
|
|
|
if let RenderReturn::Async(task) = &mut new_nodes {
|
2022-11-09 18:58:11 +00:00
|
|
|
let mut leaves = self.scheduler.leaves.borrow_mut();
|
2022-11-08 06:55:22 +00:00
|
|
|
|
|
|
|
let entry = leaves.vacant_entry();
|
2022-11-09 18:58:11 +00:00
|
|
|
let suspense_id = SuspenseId(entry.key());
|
2022-11-08 06:55:22 +00:00
|
|
|
|
|
|
|
let leaf = Rc::new(SuspenseLeaf {
|
|
|
|
scope_id,
|
|
|
|
task: task.as_mut(),
|
2022-11-09 03:39:37 +00:00
|
|
|
id: suspense_id,
|
2022-11-09 18:58:11 +00:00
|
|
|
tx: self.scheduler.sender.clone(),
|
|
|
|
notified: Default::default(),
|
2022-11-08 06:55:22 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
let waker = leaf.waker();
|
|
|
|
let mut cx = Context::from_waker(&waker);
|
2022-11-09 18:58:11 +00:00
|
|
|
|
|
|
|
// safety: the task is already pinned in the bump arena
|
2022-11-08 06:55:22 +00:00
|
|
|
let mut pinned = unsafe { Pin::new_unchecked(task.as_mut()) };
|
|
|
|
|
2022-11-09 18:58:11 +00:00
|
|
|
// Keep polling until either we get a value or the future is not ready
|
2022-11-08 06:55:22 +00:00
|
|
|
loop {
|
|
|
|
match pinned.poll_unpin(&mut cx) {
|
|
|
|
// If nodes are produced, then set it and we can break
|
|
|
|
Poll::Ready(nodes) => {
|
|
|
|
new_nodes = RenderReturn::Sync(nodes);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no nodes are produced but the future woke up immediately, then try polling it again
|
|
|
|
// This circumvents things like yield_now, but is important is important when rendering
|
|
|
|
// components that are just a stream of immediately ready futures
|
2022-11-09 18:58:11 +00:00
|
|
|
_ if leaf.notified.get() => {
|
|
|
|
leaf.notified.set(false);
|
2022-11-08 06:55:22 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no nodes are produced, then we need to wait for the future to be woken up
|
|
|
|
// Insert the future into fiber leaves and break
|
|
|
|
_ => {
|
|
|
|
entry.insert(leaf);
|
2022-11-09 18:58:11 +00:00
|
|
|
self.collected_leaves.push(suspense_id);
|
2022-11-08 06:55:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
2022-11-04 05:30:26 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-11-18 06:31:14 +00:00
|
|
|
/*
|
|
|
|
todo: use proper mutability here
|
|
|
|
|
|
|
|
right now we're aliasing the scope, which is not allowed
|
|
|
|
*/
|
|
|
|
|
2022-11-08 06:55:22 +00:00
|
|
|
let scope = &mut self.scopes[scope_id.0];
|
2022-11-18 06:31:14 +00:00
|
|
|
let frame = scope.current_frame();
|
2022-10-28 04:58:47 +00:00
|
|
|
|
|
|
|
// set the head of the bump frame
|
2022-11-08 06:55:22 +00:00
|
|
|
let alloced = frame.bump.alloc(new_nodes);
|
2022-10-28 04:58:47 +00:00
|
|
|
frame.node.set(alloced);
|
|
|
|
|
2022-11-18 06:31:14 +00:00
|
|
|
// And move the render generation forward by one
|
|
|
|
scope.render_cnt.set(scope.render_cnt.get() + 1);
|
|
|
|
|
2022-10-28 04:58:47 +00:00
|
|
|
// rebind the lifetime now that its stored internally
|
2022-11-08 06:55:22 +00:00
|
|
|
unsafe { mem::transmute(alloced) }
|
2022-10-28 04:58:47 +00:00
|
|
|
}
|
|
|
|
}
|