use crate::{ any_props::AnyProps, bump_frame::BumpFrame, innerlude::DirtyScope, innerlude::{SuspenseId, SuspenseLeaf}, nodes::RenderReturn, scheduler::RcWake, scopes::{ScopeId, ScopeState}, virtual_dom::VirtualDom, }; use bumpalo::Bump; use futures_util::FutureExt; use std::{ mem, pin::Pin, rc::Rc, task::{Context, Poll}, }; impl VirtualDom { pub(super) fn new_scope( &mut self, props: Box>, name: &'static str, ) -> &ScopeState { let parent = self.acquire_current_scope_raw(); let entry = self.scopes.vacant_entry(); let height = unsafe { parent.map(|f| (*f).height + 1).unwrap_or(0) }; let id = ScopeId(entry.key()); entry.insert(Box::new(ScopeState { parent, id, height, props: Some(props), name, placeholder: Default::default(), node_arena_1: BumpFrame::new(0), node_arena_2: BumpFrame::new(0), spawned_tasks: Default::default(), render_cnt: Default::default(), hook_arena: Default::default(), hook_list: Default::default(), hook_idx: Default::default(), shared_contexts: Default::default(), tasks: self.scheduler.clone(), })) } 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() as *mut _)) } pub(crate) fn run_scope(&mut self, scope_id: ScopeId) -> &RenderReturn { // Cycle to the next frame and then reset it // This breaks any latent references, invalidating every pointer referencing into it. // Remove all the outdated listeners self.ensure_drop_safety(scope_id); let mut new_nodes = unsafe { let scope = self.scopes[scope_id.0].as_mut(); // if this frame hasn't been intialized yet, we can guess the size of the next frame to be more efficient if scope.previous_frame().bump.allocated_bytes() == 0 { scope.previous_frame_mut().bump = Bump::with_capacity(scope.current_frame().bump.allocated_bytes()); } else { scope.previous_frame_mut().bump.reset(); } // Make sure to reset the hook counter so we give out hooks in the right order scope.hook_idx.set(0); // safety: due to how we traverse the tree, we know that the scope is not currently aliased let props = scope.props.as_ref().unwrap().as_ref(); let props: &dyn AnyProps = mem::transmute(props); props.render(scope).extend_lifetime() }; // immediately resolve futures that can be resolved if let RenderReturn::Async(task) = &mut new_nodes { let mut leaves = self.scheduler.leaves.borrow_mut(); let entry = leaves.vacant_entry(); let suspense_id = SuspenseId(entry.key()); let leaf = Rc::new(SuspenseLeaf { scope_id, task: task.as_mut(), id: suspense_id, tx: self.scheduler.sender.clone(), notified: Default::default(), }); let waker = leaf.waker(); let mut cx = Context::from_waker(&waker); // safety: the task is already pinned in the bump arena let mut pinned = unsafe { Pin::new_unchecked(task.as_mut()) }; // Keep polling until either we get a value or the future is not ready 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 _ if leaf.notified.get() => { leaf.notified.set(false); 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); self.collected_leaves.push(suspense_id); break; } }; } }; let scope = &self.scopes[scope_id.0]; // We write on top of the previous frame and then make it the current by pushing the generation forward let frame = scope.previous_frame(); // set the new head of the bump frame let allocated = &*frame.bump.alloc(new_nodes); frame.node.set(allocated); // And move the render generation forward by one scope.render_cnt.set(scope.render_cnt.get() + 1); // remove this scope from dirty scopes self.dirty_scopes.remove(&DirtyScope { height: scope.height, id: scope.id, }); // rebind the lifetime now that its stored internally unsafe { mem::transmute(allocated) } } }