123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179 |
- use std::cell::{RefCell, RefMut};
- use std::fmt::Display;
- use std::{cell::UnsafeCell, rc::Rc};
- use crate::heuristics::*;
- use crate::innerlude::*;
- use futures_util::stream::FuturesUnordered;
- use fxhash::{FxHashMap, FxHashSet};
- use slab::Slab;
- use smallvec::SmallVec;
- // slotmap::new_key_type! {
- // // A dedicated key type for the all the scopes
- // pub struct ScopeId;
- // }
- // #[cfg(feature = "serialize", serde::Serialize)]
- // #[cfg(feature = "serialize", serde::Serialize)]
- #[derive(serde::Serialize, serde::Deserialize, Copy, Clone, PartialEq, Eq, Hash, Debug)]
- pub struct ScopeId(pub usize);
- #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
- pub struct ElementId(pub usize);
- impl Display for ElementId {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- write!(f, "{}", self.0)
- }
- }
- impl ElementId {
- pub fn as_u64(self) -> u64 {
- self.0 as u64
- }
- }
- type Shared<T> = Rc<RefCell<T>>;
- type TaskReceiver = futures_channel::mpsc::UnboundedReceiver<EventTrigger>;
- type TaskSender = futures_channel::mpsc::UnboundedSender<EventTrigger>;
- /// These are resources shared among all the components and the virtualdom itself
- #[derive(Clone)]
- pub struct SharedResources {
- pub components: Rc<UnsafeCell<Slab<Scope>>>,
- pub(crate) heuristics: Shared<HeuristicsEngine>,
- ///
- pub task_sender: TaskSender,
- pub task_receiver: Shared<TaskReceiver>,
- pub async_tasks: Shared<FuturesUnordered<FiberTask>>,
- /// We use a SlotSet to keep track of the keys that are currently being used.
- /// However, we don't store any specific data since the "mirror"
- pub raw_elements: Rc<RefCell<Slab<()>>>,
- pub task_setter: Rc<dyn Fn(ScopeId)>,
- }
- impl SharedResources {
- pub fn new() -> Self {
- // preallocate 2000 elements and 20 scopes to avoid dynamic allocation
- let components: Rc<UnsafeCell<Slab<Scope>>> =
- Rc::new(UnsafeCell::new(Slab::with_capacity(100)));
- // elements are super cheap - the value takes no space
- let raw_elements = Slab::with_capacity(2000);
- // let pending_events = Rc::new(RefCell::new(Vec::new()));
- let (sender, receiver) = futures_channel::mpsc::unbounded();
- let heuristics = HeuristicsEngine::new();
- // we allocate this task setter once to save us from having to allocate later
- let task_setter = {
- let queue = sender.clone();
- let components = components.clone();
- Rc::new(move |idx: ScopeId| {
- let comps = unsafe { &*components.get() };
- if let Some(scope) = comps.get(idx.0) {
- queue
- .unbounded_send(EventTrigger::new(
- VirtualEvent::ScheduledUpdate {
- height: scope.height,
- },
- idx,
- None,
- EventPriority::High,
- ))
- .expect("The event queu receiver should *never* be dropped");
- }
- }) as Rc<dyn Fn(ScopeId)>
- };
- Self {
- components,
- async_tasks: Rc::new(RefCell::new(FuturesUnordered::new())),
- task_receiver: Rc::new(RefCell::new(receiver)),
- task_sender: sender,
- heuristics: Rc::new(RefCell::new(heuristics)),
- raw_elements: Rc::new(RefCell::new(raw_elements)),
- task_setter,
- }
- }
- /// this is unsafe because the caller needs to track which other scopes it's already using
- pub unsafe fn get_scope(&self, idx: ScopeId) -> Option<&Scope> {
- let inner = &*self.components.get();
- inner.get(idx.0)
- }
- /// this is unsafe because the caller needs to track which other scopes it's already using
- pub unsafe fn get_scope_mut(&self, idx: ScopeId) -> Option<&mut Scope> {
- let inner = &mut *self.components.get();
- inner.get_mut(idx.0)
- }
- pub fn with_scope<'b, O: 'static>(
- &'b self,
- _id: ScopeId,
- _f: impl FnOnce(&'b mut Scope) -> O,
- ) -> Result<O> {
- todo!()
- }
- // return a bumpframe with a lifetime attached to the arena borrow
- // this is useful for merging lifetimes
- pub fn with_scope_vnode<'b>(
- &self,
- _id: ScopeId,
- _f: impl FnOnce(&mut Scope) -> &VNode<'b>,
- ) -> Result<&VNode<'b>> {
- todo!()
- }
- pub fn try_remove(&self, id: ScopeId) -> Result<Scope> {
- let inner = unsafe { &mut *self.components.get() };
- Ok(inner.remove(id.0))
- // .try_remove(id.0)
- // .ok_or_else(|| Error::FatalInternal("Scope not found"))
- }
- pub fn reserve_node(&self) -> ElementId {
- ElementId(self.raw_elements.borrow_mut().insert(()))
- }
- /// return the id, freeing the space of the original node
- pub fn collect_garbage(&self, id: ElementId) {
- self.raw_elements.borrow_mut().remove(id.0);
- }
- pub fn insert_scope_with_key(&self, f: impl FnOnce(ScopeId) -> Scope) -> ScopeId {
- let g = unsafe { &mut *self.components.get() };
- let entry = g.vacant_entry();
- let id = ScopeId(entry.key());
- entry.insert(f(id));
- id
- }
- pub fn schedule_update(&self) -> Rc<dyn Fn(ScopeId)> {
- self.task_setter.clone()
- }
- pub fn submit_task(&self, task: FiberTask) -> TaskHandle {
- self.async_tasks.borrow_mut().push(task);
- TaskHandle {}
- }
- }
- pub struct TaskHandle {}
- impl TaskHandle {
- pub fn toggle(&self) {}
- pub fn start(&self) {}
- pub fn stop(&self) {}
- pub fn restart(&self) {}
- }
|