123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384 |
- use futures_util::task::ArcWake;
- use super::{Scheduler, SchedulerMsg};
- use crate::ScopeId;
- use std::cell::RefCell;
- use std::future::Future;
- use std::pin::Pin;
- use std::sync::Arc;
- use std::task::Waker;
- /// A task's unique identifier.
- ///
- /// `TaskId` is a `usize` that is unique across the entire VirtualDOM and across time. TaskIDs will never be reused
- /// once a Task has been completed.
- #[cfg_attr(feature = "serialize", derive(serde::Serialize, serde::Deserialize))]
- #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
- pub struct TaskId(pub usize);
- /// the task itself is the waker
- pub(crate) struct LocalTask {
- pub scope: ScopeId,
- pub task: RefCell<Pin<Box<dyn Future<Output = ()> + 'static>>>,
- pub waker: Waker,
- }
- impl Scheduler {
- /// Start a new future on the same thread as the rest of the VirtualDom.
- ///
- /// This future will not contribute to suspense resolving, so you should primarily use this for reacting to changes
- /// and long running tasks.
- ///
- /// Whenever the component that owns this future is dropped, the future will be dropped as well.
- ///
- /// Spawning a future onto the root scope will cause it to be dropped when the root component is dropped - which
- /// will only occur when the VirtuaalDom itself has been dropped.
- pub fn spawn(&self, scope: ScopeId, task: impl Future<Output = ()> + 'static) -> TaskId {
- let mut tasks = self.tasks.borrow_mut();
- let entry = tasks.vacant_entry();
- let task_id = TaskId(entry.key());
- let task = LocalTask {
- task: RefCell::new(Box::pin(task)),
- scope,
- waker: futures_util::task::waker(Arc::new(LocalTaskHandle {
- id: task_id,
- tx: self.sender.clone(),
- })),
- };
- let mut cx = std::task::Context::from_waker(&task.waker);
- if !task.task.borrow_mut().as_mut().poll(&mut cx).is_ready() {
- self.sender
- .unbounded_send(SchedulerMsg::TaskNotified(task_id))
- .expect("Scheduler should exist");
- }
- entry.insert(task);
- task_id
- }
- /// Drop the future with the given TaskId
- ///
- /// This does not abort the task, so you'll want to wrap it in an aborthandle if that's important to you
- pub fn remove(&self, id: TaskId) -> Option<LocalTask> {
- self.tasks.borrow_mut().try_remove(id.0)
- }
- }
- pub struct LocalTaskHandle {
- id: TaskId,
- tx: futures_channel::mpsc::UnboundedSender<SchedulerMsg>,
- }
- impl ArcWake for LocalTaskHandle {
- fn wake_by_ref(arc_self: &Arc<Self>) {
- // This can fail if the scheduler has been dropped while the application is shutting down
- let _ = arc_self
- .tx
- .unbounded_send(SchedulerMsg::TaskNotified(arc_self.id));
- }
- }
|