wait.rs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. use futures_util::FutureExt;
  2. use std::{
  3. rc::Rc,
  4. task::{Context, Poll},
  5. };
  6. use crate::{
  7. innerlude::{Mutation, Mutations, SuspenseContext},
  8. nodes::RenderReturn,
  9. ScopeId, TaskId, VNode, VirtualDom,
  10. };
  11. use super::SuspenseId;
  12. impl VirtualDom {
  13. /// Handle notifications by tasks inside the scheduler
  14. ///
  15. /// This is precise, meaning we won't poll every task, just tasks that have woken up as notified to use by the
  16. /// queue
  17. pub(crate) fn handle_task_wakeup(&mut self, id: TaskId) {
  18. let mut tasks = self.scheduler.tasks.borrow_mut();
  19. let task = match tasks.get(id.0) {
  20. Some(task) => task,
  21. // The task was removed from the scheduler, so we can just ignore it
  22. None => return,
  23. };
  24. let mut cx = Context::from_waker(&task.waker);
  25. // If the task completes...
  26. if task.task.borrow_mut().as_mut().poll(&mut cx).is_ready() {
  27. // Remove it from the scope so we dont try to double drop it when the scope dropes
  28. let scope = &self.scopes[task.scope];
  29. scope.spawned_tasks.borrow_mut().remove(&id);
  30. // Remove it from the scheduler
  31. tasks.try_remove(id.0);
  32. }
  33. }
  34. pub(crate) fn acquire_suspense_boundary(&self, id: ScopeId) -> Rc<SuspenseContext> {
  35. self.scopes[id]
  36. .consume_context::<Rc<SuspenseContext>>()
  37. .unwrap()
  38. }
  39. pub(crate) fn handle_suspense_wakeup(&mut self, id: SuspenseId) {
  40. let leaves = self.scheduler.leaves.borrow_mut();
  41. let leaf = leaves.get(id.0).unwrap();
  42. let scope_id = leaf.scope_id;
  43. // todo: cache the waker
  44. let mut cx = Context::from_waker(&leaf.waker);
  45. // Safety: the future is always pinned to the bump arena
  46. let mut pinned = unsafe { std::pin::Pin::new_unchecked(&mut *leaf.task) };
  47. let as_pinned_mut = &mut pinned;
  48. // the component finished rendering and gave us nodes
  49. // we should attach them to that component and then render its children
  50. // continue rendering the tree until we hit yet another suspended component
  51. if let Poll::Ready(new_nodes) = as_pinned_mut.poll_unpin(&mut cx) {
  52. let fiber = self.acquire_suspense_boundary(leaf.scope_id);
  53. let scope = &self.scopes[scope_id];
  54. let arena = scope.current_frame();
  55. let ret = arena.bump().alloc(match new_nodes {
  56. Some(new) => RenderReturn::Ready(new),
  57. None => RenderReturn::default(),
  58. });
  59. arena.node.set(ret);
  60. fiber.waiting_on.borrow_mut().remove(&id);
  61. if let RenderReturn::Ready(template) = ret {
  62. let mutations_ref = &mut fiber.mutations.borrow_mut();
  63. let mutations = &mut **mutations_ref;
  64. let template: &VNode = unsafe { std::mem::transmute(template) };
  65. let mutations: &mut Mutations = unsafe { std::mem::transmute(mutations) };
  66. std::mem::swap(&mut self.mutations, mutations);
  67. let place_holder_id = scope.placeholder.get().unwrap();
  68. self.scope_stack.push(scope_id);
  69. drop(leaves);
  70. let created = self.create(template);
  71. self.scope_stack.pop();
  72. mutations.push(Mutation::ReplaceWith {
  73. id: place_holder_id,
  74. m: created,
  75. });
  76. for leaf in self.collected_leaves.drain(..) {
  77. fiber.waiting_on.borrow_mut().insert(leaf);
  78. }
  79. std::mem::swap(&mut self.mutations, mutations);
  80. if fiber.waiting_on.borrow().is_empty() {
  81. self.finished_fibers.push(fiber.id);
  82. }
  83. }
  84. }
  85. }
  86. }