diff options
| author | 2023-11-29 00:12:48 +0100 | |
|---|---|---|
| committer | 2023-11-29 00:12:48 +0100 | |
| commit | 3b7d479534d9114ed12bb5d9ccd910e85d5c13c7 (patch) | |
| tree | ab5c3e2a8f074d9a02dd470bfd659fb1740bf82a | |
| parent | 133f4da9014fcdc331ac44269209ee61ca56d007 (diff) | |
| download | iced-3b7d479534d9114ed12bb5d9ccd910e85d5c13c7.tar.gz iced-3b7d479534d9114ed12bb5d9ccd910e85d5c13c7.tar.bz2 iced-3b7d479534d9114ed12bb5d9ccd910e85d5c13c7.zip  | |
Implement `Command::run` for executing a `Stream` to completion
| -rw-r--r-- | futures/src/runtime.rs | 25 | ||||
| -rw-r--r-- | runtime/src/command.rs | 14 | ||||
| -rw-r--r-- | runtime/src/command/action.rs | 9 | ||||
| -rw-r--r-- | winit/src/application.rs | 3 | 
4 files changed, 48 insertions, 3 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..b942f3ce 100644 --- a/runtime/src/command.rs +++ b/runtime/src/command.rs @@ -4,8 +4,10 @@ mod action;  pub use action::Action;  use crate::core::widget; +use crate::futures::futures;  use crate::futures::MaybeSend; +use futures::Stream;  use std::fmt;  use std::future::Future; @@ -43,11 +45,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.      /// 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/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());  | 
