dioxus/packages/core/src/scope.rs

143 lines
5 KiB
Rust
Raw Normal View History

use crate::context::hooks::Hook;
2021-02-12 21:11:33 +00:00
use crate::innerlude::*;
2021-02-07 22:38:17 +00:00
use crate::nodes::VNode;
use bumpalo::Bump;
use generational_arena::Index;
2021-02-07 22:38:17 +00:00
use std::{
any::TypeId, borrow::Borrow, cell::RefCell, future::Future, marker::PhantomData,
sync::atomic::AtomicUsize, todo,
2021-02-07 22:38:17 +00:00
};
2021-02-08 00:14:04 +00:00
/// Every component in Dioxus is represented by a `Scope`.
///
/// Scopes contain the state for hooks, the component's props, and other lifecycle information.
///
/// Scopes are allocated in a generational arena. As components are mounted/unmounted, they will replace slots of dead components.
/// The actual contents of the hooks, though, will be allocated with the standard allocator. These should not allocate as frequently.
2021-02-07 22:38:17 +00:00
pub struct Scope {
// TODO @Jon
2021-02-08 00:14:04 +00:00
// These hooks are actually references into the hook arena
// These two could be combined with "OwningRef" to remove unsafe usage
// could also use ourborous
2021-02-12 21:11:33 +00:00
pub hooks: RefCell<Vec<*mut Hook>>,
pub hook_arena: typed_arena::Arena<Hook>,
2021-02-08 00:14:04 +00:00
// Map to the parent
2021-02-12 21:11:33 +00:00
pub parent: Option<Index>,
// todo, do better with the active frame stuff
2021-02-12 21:11:33 +00:00
pub frames: [Bump; 2],
// somehow build this vnode with a lifetime tied to self
// This root node has "static" lifetime, but it's really not static.
// It's goverened by the oldest of the two frames and is switched every time a new render occurs
// Use this node as if it were static is unsafe, and needs to be fixed with ourborous or owning ref
// ! do not copy this reference are things WILL break !
pub root_node: *mut VNode<'static>,
pub active_frame: ActiveFrame,
// IE Which listeners need to be woken up?
2021-02-12 21:11:33 +00:00
pub listeners: Vec<Box<dyn Fn()>>,
//
2021-02-12 21:11:33 +00:00
pub props_type: TypeId,
pub caller: *const i32,
2021-02-07 22:38:17 +00:00
}
pub enum ActiveFrame {
First,
Second,
}
impl ActiveFrame {
fn next(&mut self) {
match self {
ActiveFrame::First => *self = ActiveFrame::Second,
ActiveFrame::Second => *self = ActiveFrame::First,
}
}
}
2021-02-07 22:38:17 +00:00
impl Scope {
// create a new scope from a function
pub(crate) fn new<T: 'static>(f: FC<T>, parent: Option<Index>) -> Self {
2021-02-07 22:38:17 +00:00
// Capture the props type
let props_type = TypeId::of::<T>();
let hook_arena = typed_arena::Arena::new();
2021-02-07 22:38:17 +00:00
let hooks = RefCell::new(Vec::new());
// Capture the caller
2021-02-07 22:38:17 +00:00
let caller = f as *const i32;
// Create the two buffers the componetn will render into
// There will always be an "old" and "new"
let frames = [Bump::new(), Bump::new()];
let listeners = Vec::new();
let active_frame = ActiveFrame::First;
let new = frames[0].alloc(VNode::Text(VText::new("")));
let cur_node = new as *mut _;
2021-02-07 22:38:17 +00:00
Self {
hook_arena,
2021-02-07 22:38:17 +00:00
hooks,
props_type,
caller,
active_frame,
listeners,
parent,
frames,
root_node: cur_node,
2021-02-07 22:38:17 +00:00
}
}
/// Create a new context and run the component with references from the Virtual Dom
/// This function downcasts the function pointer based on the stored props_type
///
2021-02-12 21:11:33 +00:00
/// Props is ?Sized because we borrow the props and don't need to know the size. P (sized) is used as a marker (unsized)
pub(crate) fn run<'a, P: Properties + ?Sized>(&self, props: &'a P) {
let bump = match self.active_frame {
// If the active frame is the first, then we need to bump into the second
ActiveFrame::First => &self.frames[1],
// If the active frame is the second, then we need to bump into the first
ActiveFrame::Second => &self.frames[0],
}; // n.b, there might be a better way of doing this active frame stuff - perhaps swapping
let ctx = Context {
2021-02-12 05:29:46 +00:00
scope: &*self,
2021-02-07 22:38:17 +00:00
_p: PhantomData {},
2021-02-08 00:14:04 +00:00
arena: &self.hook_arena,
2021-02-07 22:38:17 +00:00
hooks: &self.hooks,
idx: 0.into(),
bump,
};
2021-02-07 22:38:17 +00:00
/*
SAFETY ALERT
This particular usage of transmute is outlined in its docs https://doc.rust-lang.org/std/mem/fn.transmute.html
We hide the generic bound on the function item by casting it to raw pointer. When the function is actually called,
we transmute the function back using the props as reference.
2021-02-07 22:38:17 +00:00
2021-02-12 21:11:33 +00:00
we could do a better check to make sure that the TypeID is correct before casting
--
This is safe because we check that the generic type matches before casting.
*/
let caller = unsafe { std::mem::transmute::<*const i32, FC<P>>(self.caller) };
let new_nodes = caller(ctx, props);
let old_nodes: &mut VNode<'static> = unsafe { &mut *self.root_node };
2021-02-12 21:11:33 +00:00
// TODO: Iterate through the new nodes
// move any listeners into ourself
// perform the diff, dumping into the mutable change list
// this doesnt perform any "diff compression" where an event and a re-render
// crate::diff::diff(old_nodes, &new_nodes);
todo!()
}
2021-02-07 22:38:17 +00:00
}