summaryrefslogtreecommitdiffstats
path: root/runtime
diff options
context:
space:
mode:
authorLibravatar Héctor Ramón Jiménez <hector@hecrj.dev>2024-11-22 01:51:00 +0100
committerLibravatar Héctor Ramón Jiménez <hector@hecrj.dev>2024-11-22 01:51:00 +0100
commit6d50c62bc79644ee260eb9ced497ddfd688667c1 (patch)
tree4fcf233ba17ca2c9dcb2f8f7bc2d2efa6caf357e /runtime
parent2a2e20b0a35bcf4da9ac34ab2f664485c4e1dbe3 (diff)
downloadiced-6d50c62bc79644ee260eb9ced497ddfd688667c1.tar.gz
iced-6d50c62bc79644ee260eb9ced497ddfd688667c1.tar.bz2
iced-6d50c62bc79644ee260eb9ced497ddfd688667c1.zip
Honor clones of `task::Handle` with `abort_on_drop`
Diffstat (limited to 'runtime')
-rw-r--r--runtime/src/task.rs63
1 files changed, 42 insertions, 21 deletions
diff --git a/runtime/src/task.rs b/runtime/src/task.rs
index 4554c74b..22cfb63e 100644
--- a/runtime/src/task.rs
+++ b/runtime/src/task.rs
@@ -9,6 +9,7 @@ use crate::futures::{boxed_stream, BoxStream, MaybeSend};
use crate::Action;
use std::future::Future;
+use std::sync::Arc;
/// A set of concurrent actions to be performed by the iced runtime.
///
@@ -183,16 +184,16 @@ impl<T> Task<T> {
(
Self(Some(boxed_stream(stream))),
Handle {
- raw: Some(handle),
- abort_on_drop: false,
+ internal: InternalHandle::Manual(handle),
},
)
}
None => (
Self(None),
Handle {
- raw: None,
- abort_on_drop: false,
+ internal: InternalHandle::Manual(
+ stream::AbortHandle::new_pair().0,
+ ),
},
),
}
@@ -220,44 +221,64 @@ impl<T> Task<T> {
/// A handle to a [`Task`] that can be used for aborting it.
#[derive(Debug, Clone)]
pub struct Handle {
- raw: Option<stream::AbortHandle>,
- abort_on_drop: bool,
+ internal: InternalHandle,
+}
+
+#[derive(Debug, Clone)]
+enum InternalHandle {
+ Manual(stream::AbortHandle),
+ AbortOnDrop(Arc<stream::AbortHandle>),
+}
+
+impl InternalHandle {
+ pub fn as_ref(&self) -> &stream::AbortHandle {
+ match self {
+ InternalHandle::Manual(handle) => handle,
+ InternalHandle::AbortOnDrop(handle) => handle.as_ref(),
+ }
+ }
}
impl Handle {
/// Aborts the [`Task`] of this [`Handle`].
pub fn abort(&self) {
- if let Some(handle) = &self.raw {
- handle.abort();
- }
+ self.internal.as_ref().abort();
}
/// Returns a new [`Handle`] that will call [`Handle::abort`] whenever
- /// it is dropped.
+ /// all of its instances are dropped.
+ ///
+ /// If a [`Handle`] is cloned, [`Handle::abort`] will only be called
+ /// once all of the clones are dropped.
///
/// This can be really useful if you do not want to worry about calling
/// [`Handle::abort`] yourself.
- pub fn abort_on_drop(mut self) -> Self {
- Self {
- raw: self.raw.take(),
- abort_on_drop: true,
+ pub fn abort_on_drop(self) -> Self {
+ match &self.internal {
+ InternalHandle::Manual(handle) => Self {
+ internal: InternalHandle::AbortOnDrop(Arc::new(handle.clone())),
+ },
+ InternalHandle::AbortOnDrop(_) => self,
}
}
/// Returns `true` if the [`Task`] of this [`Handle`] has been aborted.
pub fn is_aborted(&self) -> bool {
- if let Some(handle) = &self.raw {
- handle.is_aborted()
- } else {
- true
- }
+ self.internal.as_ref().is_aborted()
}
}
impl Drop for Handle {
fn drop(&mut self) {
- if self.abort_on_drop {
- self.abort();
+ if let InternalHandle::AbortOnDrop(handle) = &mut self.internal {
+ let handle = std::mem::replace(
+ handle,
+ Arc::new(stream::AbortHandle::new_pair().0),
+ );
+
+ if let Some(handle) = Arc::into_inner(handle) {
+ handle.abort();
+ }
}
}
}