summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLibravatar Héctor Ramón <hector@hecrj.dev>2023-11-29 14:26:44 +0100
committerLibravatar GitHub <noreply@github.com>2023-11-29 14:26:44 +0100
commit7f8b17604a31e00becc43130ec516c1a53552c88 (patch)
treeb7cbc5ba003d61416d7959a6a704839b53822660
parent7e7d65a23d864e4662c43028a41244ca30cac540 (diff)
parenta761448858521d11dc646e2ef5217e9e06628932 (diff)
downloadiced-7f8b17604a31e00becc43130ec516c1a53552c88.tar.gz
iced-7f8b17604a31e00becc43130ec516c1a53552c88.tar.bz2
iced-7f8b17604a31e00becc43130ec516c1a53552c88.zip
Merge pull request #2150 from iced-rs/feature/command-run
`Stream` support for `Command`
-rw-r--r--futures/src/runtime.rs25
-rw-r--r--runtime/src/command.rs35
-rw-r--r--runtime/src/command/action.rs9
-rw-r--r--src/lib.rs7
-rw-r--r--winit/src/application.rs3
5 files changed, 75 insertions, 4 deletions
diff --git a/futures/src/runtime.rs b/futures/src/runtime.rs
index 16111b36..cac7b7e1 100644
--- a/futures/src/runtime.rs
+++ b/futures/src/runtime.rs
@@ -1,7 +1,7 @@
//! Run commands and keep track of subscriptions.
use crate::core::event::{self, Event};
use crate::subscription;
-use crate::{BoxFuture, Executor, MaybeSend};
+use crate::{BoxFuture, BoxStream, Executor, MaybeSend};
use futures::{channel::mpsc, Sink};
use std::marker::PhantomData;
@@ -69,6 +69,29 @@ where
self.executor.spawn(future);
}
+ /// Runs a [`Stream`] in the [`Runtime`] until completion.
+ ///
+ /// The resulting `Message`s will be forwarded to the `Sender` of the
+ /// [`Runtime`].
+ ///
+ /// [`Stream`]: BoxStream
+ pub fn run(&mut self, stream: BoxStream<Message>) {
+ use futures::{FutureExt, StreamExt};
+
+ let sender = self.sender.clone();
+ let future =
+ stream.map(Ok).forward(sender).map(|result| match result {
+ Ok(()) => (),
+ Err(error) => {
+ log::warn!(
+ "Stream could not run until completion: {error}"
+ );
+ }
+ });
+
+ self.executor.spawn(future);
+ }
+
/// Tracks a [`Subscription`] in the [`Runtime`].
///
/// It will spawn new streams or close old ones as necessary! See
diff --git a/runtime/src/command.rs b/runtime/src/command.rs
index b74097bd..f70da915 100644
--- a/runtime/src/command.rs
+++ b/runtime/src/command.rs
@@ -4,8 +4,11 @@ mod action;
pub use action::Action;
use crate::core::widget;
+use crate::futures::futures;
use crate::futures::MaybeSend;
+use futures::channel::mpsc;
+use futures::Stream;
use std::fmt;
use std::future::Future;
@@ -43,11 +46,21 @@ impl<T> Command<T> {
future: impl Future<Output = A> + 'static + MaybeSend,
f: impl FnOnce(A) -> T + 'static + MaybeSend,
) -> Command<T> {
- use iced_futures::futures::FutureExt;
+ use futures::FutureExt;
Command::single(Action::Future(Box::pin(future.map(f))))
}
+ /// Creates a [`Command`] that runs the given stream to completion.
+ pub fn run<A>(
+ stream: impl Stream<Item = A> + 'static + MaybeSend,
+ f: impl Fn(A) -> T + 'static + MaybeSend,
+ ) -> Command<T> {
+ use futures::StreamExt;
+
+ Command::single(Action::Stream(Box::pin(stream.map(f))))
+ }
+
/// Creates a [`Command`] that performs the actions of all the given
/// commands.
///
@@ -106,3 +119,23 @@ impl<T> fmt::Debug for Command<T> {
command.fmt(f)
}
}
+
+/// Creates a [`Command`] that produces the `Message`s published from a [`Future`]
+/// to an [`mpsc::Sender`] with the given bounds.
+pub fn channel<Fut, Message>(
+ size: usize,
+ f: impl FnOnce(mpsc::Sender<Message>) -> Fut + MaybeSend + 'static,
+) -> Command<Message>
+where
+ Fut: Future<Output = ()> + MaybeSend + 'static,
+ Message: 'static + MaybeSend,
+{
+ use futures::future;
+ use futures::stream::{self, StreamExt};
+
+ let (sender, receiver) = mpsc::channel(size);
+
+ let runner = stream::once(f(sender)).filter_map(|_| future::ready(None));
+
+ Command::single(Action::Stream(Box::pin(stream::select(receiver, runner))))
+}
diff --git a/runtime/src/command/action.rs b/runtime/src/command/action.rs
index 6c74f0ef..6551e233 100644
--- a/runtime/src/command/action.rs
+++ b/runtime/src/command/action.rs
@@ -18,6 +18,11 @@ pub enum Action<T> {
/// [`Future`]: iced_futures::BoxFuture
Future(iced_futures::BoxFuture<T>),
+ /// Run a [`Stream`] to completion.
+ ///
+ /// [`Stream`]: iced_futures::BoxStream
+ Stream(iced_futures::BoxStream<T>),
+
/// Run a clipboard action.
Clipboard(clipboard::Action<T>),
@@ -52,10 +57,11 @@ impl<T> Action<T> {
A: 'static,
T: 'static,
{
- use iced_futures::futures::FutureExt;
+ use iced_futures::futures::{FutureExt, StreamExt};
match self {
Self::Future(future) => Action::Future(Box::pin(future.map(f))),
+ Self::Stream(stream) => Action::Stream(Box::pin(stream.map(f))),
Self::Clipboard(action) => Action::Clipboard(action.map(f)),
Self::Window(window) => Action::Window(window.map(f)),
Self::System(system) => Action::System(system.map(f)),
@@ -74,6 +80,7 @@ impl<T> fmt::Debug for Action<T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Future(_) => write!(f, "Action::Future"),
+ Self::Stream(_) => write!(f, "Action::Stream"),
Self::Clipboard(action) => {
write!(f, "Action::Clipboard({action:?})")
}
diff --git a/src/lib.rs b/src/lib.rs
index f9f3952c..47766e6f 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -190,7 +190,6 @@ pub use crate::core::{
color, Alignment, Background, BorderRadius, Color, ContentFit, Degrees,
Gradient, Length, Padding, Pixels, Point, Radians, Rectangle, Size, Vector,
};
-pub use crate::runtime::Command;
pub mod clipboard {
//! Access the clipboard.
@@ -239,6 +238,11 @@ pub mod mouse {
};
}
+pub mod command {
+ //! Run asynchronous actions.
+ pub use crate::runtime::command::{channel, Command};
+}
+
pub mod subscription {
//! Listen to external events in your application.
pub use iced_futures::subscription::{
@@ -287,6 +291,7 @@ pub mod widget {
}
pub use application::Application;
+pub use command::Command;
pub use error::Error;
pub use event::Event;
pub use executor::Executor;
diff --git a/winit/src/application.rs b/winit/src/application.rs
index 315e34d9..2c5c864a 100644
--- a/winit/src/application.rs
+++ b/winit/src/application.rs
@@ -736,6 +736,9 @@ pub fn run_command<A, C, E>(
command::Action::Future(future) => {
runtime.spawn(future);
}
+ command::Action::Stream(stream) => {
+ runtime.run(stream);
+ }
command::Action::Clipboard(action) => match action {
clipboard::Action::Read(tag) => {
let message = tag(clipboard.read());