mirror of
https://github.com/DioxusLabs/dioxus
synced 2024-11-10 06:34:20 +00:00
Make cx.spawn() return an Option<TaskId>, returns None if finished immediately
This commit is contained in:
parent
695bf009d1
commit
69ca1fd8a7
8 changed files with 38 additions and 37 deletions
|
@ -19,7 +19,8 @@ fn app(cx: Scope) -> Element {
|
|||
emails_sent.write().push(message);
|
||||
}
|
||||
}
|
||||
});
|
||||
})
|
||||
.unwrap();
|
||||
|
||||
cx.render(rsx! {
|
||||
div {
|
||||
|
@ -27,13 +28,7 @@ fn app(cx: Scope) -> Element {
|
|||
|
||||
button {
|
||||
onclick: move |_| {
|
||||
let dom = VirtualDom::new_with_props(compose, ComposeProps {
|
||||
app_tx: tx.clone()
|
||||
});
|
||||
|
||||
// this returns a weak reference to the other window
|
||||
// Be careful not to keep a strong reference to the other window or it will never be dropped
|
||||
// and the window will never close.
|
||||
let dom = VirtualDom::new_with_props(compose, ComposeProps { app_tx: tx.clone() });
|
||||
window.new_window(dom, Default::default());
|
||||
},
|
||||
"Click to compose a new email"
|
||||
|
|
|
@ -19,7 +19,7 @@ pub struct TaskId(pub usize);
|
|||
/// the task itself is the waker
|
||||
pub(crate) struct LocalTask {
|
||||
pub scope: ScopeId,
|
||||
pub(super) task: RefCell<Pin<Box<dyn Future<Output = ()> + 'static>>>,
|
||||
pub task: RefCell<Pin<Box<dyn Future<Output = ()> + 'static>>>,
|
||||
pub waker: Waker,
|
||||
}
|
||||
|
||||
|
@ -33,7 +33,11 @@ impl Scheduler {
|
|||
///
|
||||
/// 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 {
|
||||
pub fn spawn(
|
||||
&self,
|
||||
scope: ScopeId,
|
||||
task: impl Future<Output = ()> + 'static,
|
||||
) -> Option<TaskId> {
|
||||
let mut tasks = self.tasks.borrow_mut();
|
||||
|
||||
let entry = tasks.vacant_entry();
|
||||
|
@ -48,20 +52,25 @@ impl Scheduler {
|
|||
})),
|
||||
};
|
||||
|
||||
let mut cx = std::task::Context::from_waker(&task.waker);
|
||||
if task.task.borrow_mut().as_mut().poll(&mut cx).is_ready() {
|
||||
return None;
|
||||
}
|
||||
|
||||
entry.insert(task);
|
||||
|
||||
self.sender
|
||||
.unbounded_send(SchedulerMsg::TaskNotified(task_id))
|
||||
.expect("Scheduler should exist");
|
||||
|
||||
task_id
|
||||
Some(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) {
|
||||
self.tasks.borrow_mut().try_remove(id.0);
|
||||
pub fn remove(&self, id: TaskId) -> Option<LocalTask> {
|
||||
self.tasks.borrow_mut().try_remove(id.0)
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -214,10 +214,10 @@ impl ScopeContext {
|
|||
}
|
||||
|
||||
/// Pushes the future onto the poll queue to be polled after the component renders.
|
||||
pub fn push_future(&self, fut: impl Future<Output = ()> + 'static) -> TaskId {
|
||||
let id = self.tasks.spawn(self.id, fut);
|
||||
pub fn push_future(&self, fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
let id = self.tasks.spawn(self.id, fut)?;
|
||||
self.spawned_tasks.borrow_mut().insert(id);
|
||||
id
|
||||
Some(id)
|
||||
}
|
||||
|
||||
/// Spawns the future but does not return the [`TaskId`]
|
||||
|
@ -228,9 +228,9 @@ impl ScopeContext {
|
|||
/// Spawn a future that Dioxus won't clean up when this component is unmounted
|
||||
///
|
||||
/// This is good for tasks that need to be run after the component has been dropped.
|
||||
pub fn spawn_forever(&self, fut: impl Future<Output = ()> + 'static) -> TaskId {
|
||||
pub fn spawn_forever(&self, fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
// The root scope will never be unmounted so we can just add the task at the top of the app
|
||||
let id = self.tasks.spawn(ScopeId::ROOT, fut);
|
||||
let id = self.tasks.spawn(ScopeId::ROOT, fut)?;
|
||||
|
||||
// wake up the scheduler if it is sleeping
|
||||
self.tasks
|
||||
|
@ -240,7 +240,7 @@ impl ScopeContext {
|
|||
|
||||
self.spawned_tasks.borrow_mut().insert(id);
|
||||
|
||||
id
|
||||
Some(id)
|
||||
}
|
||||
|
||||
/// Informs the scheduler that this task is no longer needed and should be removed.
|
||||
|
@ -339,7 +339,7 @@ pub fn throw(error: impl Debug + 'static) -> Option<()> {
|
|||
|
||||
/// Pushes the future onto the poll queue to be polled after the component renders.
|
||||
pub fn push_future(fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
with_current_scope(|cx| cx.push_future(fut))
|
||||
with_current_scope(|cx| cx.push_future(fut)).flatten()
|
||||
}
|
||||
|
||||
/// Spawns the future but does not return the [`TaskId`]
|
||||
|
@ -351,7 +351,7 @@ pub fn spawn(fut: impl Future<Output = ()> + 'static) {
|
|||
///
|
||||
/// This is good for tasks that need to be run after the component has been dropped.
|
||||
pub fn spawn_forever(fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
with_current_scope(|cx| cx.spawn_forever(fut))
|
||||
with_current_scope(|cx| cx.spawn_forever(fut)).flatten()
|
||||
}
|
||||
|
||||
/// Informs the scheduler that this task is no longer needed and should be removed.
|
||||
|
|
|
@ -309,7 +309,7 @@ impl<'src> ScopeState {
|
|||
}
|
||||
|
||||
/// Pushes the future onto the poll queue to be polled after the component renders.
|
||||
pub fn push_future(&self, fut: impl Future<Output = ()> + 'static) -> TaskId {
|
||||
pub fn push_future(&self, fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
self.context().push_future(fut)
|
||||
}
|
||||
|
||||
|
@ -321,7 +321,7 @@ impl<'src> ScopeState {
|
|||
/// Spawn a future that Dioxus won't clean up when this component is unmounted
|
||||
///
|
||||
/// This is good for tasks that need to be run after the component has been dropped.
|
||||
pub fn spawn_forever(&self, fut: impl Future<Output = ()> + 'static) -> TaskId {
|
||||
pub fn spawn_forever(&self, fut: impl Future<Output = ()> + 'static) -> Option<TaskId> {
|
||||
self.context().spawn_forever(fut)
|
||||
}
|
||||
|
||||
|
|
|
@ -76,7 +76,7 @@ where
|
|||
cx.remove_future(current);
|
||||
}
|
||||
|
||||
state.task.set(Some(cx.push_future(async move {
|
||||
state.task.set(cx.push_future(async move {
|
||||
let data;
|
||||
#[cfg(feature = "ssr")]
|
||||
{
|
||||
|
@ -94,7 +94,7 @@ where
|
|||
*value.borrow_mut() = Some(Box::new(data));
|
||||
|
||||
schedule_update();
|
||||
})));
|
||||
}));
|
||||
}
|
||||
|
||||
if first_run {
|
||||
|
|
|
@ -63,17 +63,15 @@ use std::future::Future;
|
|||
/// }
|
||||
/// })
|
||||
/// ```
|
||||
pub fn use_coroutine<M, G, F>(cx: &ScopeState, init: G) -> &Coroutine<M>
|
||||
pub fn use_coroutine<M, G, F>(cx: &ScopeState, init: G) -> Option<&Coroutine<M>>
|
||||
where
|
||||
M: 'static,
|
||||
G: FnOnce(UnboundedReceiver<M>) -> F,
|
||||
F: Future<Output = ()> + 'static,
|
||||
{
|
||||
cx.use_hook(|| {
|
||||
let (tx, rx) = futures_channel::mpsc::unbounded();
|
||||
let task = cx.push_future(init(rx));
|
||||
cx.provide_context(Coroutine { tx, task })
|
||||
})
|
||||
let (tx, rx) = futures_channel::mpsc::unbounded();
|
||||
let task = cx.push_future(init(rx))?;
|
||||
Some(cx.use_hook(|| cx.provide_context(Coroutine { tx, task })))
|
||||
}
|
||||
|
||||
/// Get a handle to a coroutine higher in the tree
|
||||
|
|
|
@ -117,7 +117,7 @@ where
|
|||
T: Future<Output = ()> + 'static,
|
||||
{
|
||||
fn apply(self, _: UseEffectCleanup, cx: &ScopeState) -> Option<TaskId> {
|
||||
Some(cx.push_future(self))
|
||||
cx.push_future(self)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -129,11 +129,10 @@ where
|
|||
F: FnOnce() + 'static,
|
||||
{
|
||||
fn apply(self, oncleanup: UseEffectCleanup, cx: &ScopeState) -> Option<TaskId> {
|
||||
let task = cx.push_future(async move {
|
||||
cx.push_future(async move {
|
||||
let cleanup = self.await;
|
||||
*oncleanup.borrow_mut() = Some(Box::new(cleanup) as Box<dyn FnOnce()>);
|
||||
});
|
||||
Some(task)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -49,10 +49,10 @@ where
|
|||
let val = val.clone();
|
||||
let task = state.task.clone();
|
||||
|
||||
state.task.set(Some(cx.push_future(async move {
|
||||
state.task.set(cx.push_future(async move {
|
||||
val.set(Some(fut.await));
|
||||
task.take();
|
||||
})));
|
||||
}));
|
||||
|
||||
// Mark that we don't need to regenerate
|
||||
state.needs_regen.set(false);
|
||||
|
|
Loading…
Reference in a new issue