Allow listening to runtime events in subscriptions

This commit is contained in:
Héctor Ramón Jiménez 2019-12-08 08:21:26 +01:00
parent 9b84b6e403
commit 98160406f7
7 changed files with 89 additions and 40 deletions

View File

@ -1,51 +1,59 @@
//! Generate events asynchronously for you application. //! Generate events asynchronously for you application.
/// An event subscription. /// An event subscription.
pub struct Subscription<T> { pub struct Subscription<I, O> {
handles: Vec<Box<dyn Handle<Output = T>>>, connections: Vec<Box<dyn Connection<Input = I, Output = O>>>,
} }
impl<T> Subscription<T> { impl<I, O> Subscription<I, O> {
pub fn none() -> Self { pub fn none() -> Self {
Self { Self {
handles: Vec::new(), connections: Vec::new(),
} }
} }
pub fn batch(subscriptions: impl Iterator<Item = Subscription<T>>) -> Self { pub fn batch(
subscriptions: impl Iterator<Item = Subscription<I, O>>,
) -> Self {
Self { Self {
handles: subscriptions connections: subscriptions
.flat_map(|subscription| subscription.handles) .flat_map(|subscription| subscription.connections)
.collect(), .collect(),
} }
} }
pub fn handles(self) -> Vec<Box<dyn Handle<Output = T>>> { pub fn connections(
self.handles self,
) -> Vec<Box<dyn Connection<Input = I, Output = O>>> {
self.connections
} }
} }
impl<T, A> From<A> for Subscription<T> impl<I, O, T> From<T> for Subscription<I, O>
where where
A: Handle<Output = T> + 'static, T: Connection<Input = I, Output = O> + 'static,
{ {
fn from(handle: A) -> Self { fn from(handle: T) -> Self {
Self { Self {
handles: vec![Box::new(handle)], connections: vec![Box::new(handle)],
} }
} }
} }
/// The handle of an event subscription. /// The connection of an event subscription.
pub trait Handle { pub trait Connection {
type Input;
type Output; type Output;
fn id(&self) -> u64; fn id(&self) -> u64;
fn stream(&self) -> futures::stream::BoxStream<'static, Self::Output>; fn stream(
&self,
input: Self::Input,
) -> futures::stream::BoxStream<'static, Self::Output>;
} }
impl<T> std::fmt::Debug for Subscription<T> { impl<I, O> std::fmt::Debug for Subscription<I, O> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Subscription").finish() f.debug_struct("Subscription").finish()
} }

View File

@ -4,6 +4,8 @@ use iced::{
}; };
pub fn main() { pub fn main() {
env_logger::init();
Clock::run(Settings::default()) Clock::run(Settings::default())
} }
@ -108,30 +110,26 @@ mod time {
>, >,
} }
impl<Message> iced::subscription::Handle for Tick<Message> impl<Message> iced_native::subscription::Connection for Tick<Message>
where where
Message: 'static, Message: 'static,
{ {
type Input = iced_native::subscription::Input;
type Output = Message; type Output = Message;
fn id(&self) -> u64 { fn id(&self) -> u64 {
0 0
} }
fn stream(&self) -> futures::stream::BoxStream<'static, Message> { fn stream(
&self,
input: iced_native::subscription::Input,
) -> futures::stream::BoxStream<'static, Message> {
use futures::StreamExt; use futures::StreamExt;
let duration = self.duration.clone();
let function = self.message.clone(); let function = self.message.clone();
let stream = input.map(move |_| function(chrono::Local::now())).boxed()
futures::stream::iter(std::iter::repeat(())).map(move |_| {
std::thread::sleep(duration);
function(chrono::Local::now())
});
stream.boxed()
} }
} }
} }

View File

@ -12,3 +12,4 @@ iced_core = { version = "0.1.0", path = "../core", features = ["command", "subsc
twox-hash = "1.5" twox-hash = "1.5"
raw-window-handle = "0.3" raw-window-handle = "0.3"
unicode-segmentation = "1.6" unicode-segmentation = "1.6"
futures = "0.3"

View File

@ -34,7 +34,7 @@
//! [`Windowed`]: renderer/trait.Windowed.html //! [`Windowed`]: renderer/trait.Windowed.html
//! [`UserInterface`]: struct.UserInterface.html //! [`UserInterface`]: struct.UserInterface.html
//! [renderer]: renderer/index.html //! [renderer]: renderer/index.html
#![deny(missing_docs)] //#![deny(missing_docs)]
#![deny(missing_debug_implementations)] #![deny(missing_debug_implementations)]
#![deny(unused_results)] #![deny(unused_results)]
#![deny(unsafe_code)] #![deny(unsafe_code)]
@ -42,6 +42,7 @@
pub mod input; pub mod input;
pub mod layout; pub mod layout;
pub mod renderer; pub mod renderer;
pub mod subscription;
pub mod widget; pub mod widget;
mod element; mod element;
@ -52,8 +53,8 @@ mod size;
mod user_interface; mod user_interface;
pub use iced_core::{ pub use iced_core::{
subscription, Align, Background, Color, Command, Font, HorizontalAlignment, Align, Background, Color, Command, Font, HorizontalAlignment, Length,
Length, Point, Rectangle, Subscription, Vector, VerticalAlignment, Point, Rectangle, Vector, VerticalAlignment,
}; };
pub use element::Element; pub use element::Element;
@ -63,5 +64,6 @@ pub use layout::Layout;
pub use mouse_cursor::MouseCursor; pub use mouse_cursor::MouseCursor;
pub use renderer::Renderer; pub use renderer::Renderer;
pub use size::Size; pub use size::Size;
pub use subscription::Subscription;
pub use user_interface::{Cache, UserInterface}; pub use user_interface::{Cache, UserInterface};
pub use widget::*; pub use widget::*;

View File

@ -0,0 +1,6 @@
use crate::Event;
pub type Subscription<T> = iced_core::Subscription<Input, T>;
pub type Input = futures::channel::mpsc::Receiver<Event>;
pub use iced_core::subscription::Connection;

View File

@ -1,6 +1,6 @@
pub use iced_winit::{ pub use iced_winit::{
subscription, Align, Background, Color, Command, Font, HorizontalAlignment, Align, Background, Color, Command, Font, HorizontalAlignment, Length,
Length, Subscription, VerticalAlignment, Subscription, VerticalAlignment,
}; };
pub mod widget { pub mod widget {

View File

@ -184,6 +184,10 @@ pub trait Application: Sized {
debug.layout_finished(); debug.layout_finished();
debug.event_processing_started(); debug.event_processing_started();
events
.iter()
.for_each(|event| alive_subscriptions.send_event(*event));
let mut messages = let mut messages =
user_interface.update(&renderer, events.drain(..)); user_interface.update(&renderer, events.drain(..));
messages.extend(external_messages.drain(..)); messages.extend(external_messages.drain(..));
@ -207,7 +211,6 @@ pub trait Application: Sized {
debug.update_started(); debug.update_started();
let command = application.update(message); let command = application.update(message);
spawn(command, &mut thread_pool, &proxy); spawn(command, &mut thread_pool, &proxy);
debug.update_finished(); debug.update_finished();
} }
@ -422,7 +425,12 @@ fn spawn<Message: Send>(
} }
pub struct Subscriptions { pub struct Subscriptions {
alive: HashMap<u64, futures::channel::oneshot::Sender<()>>, alive: HashMap<u64, Connection>,
}
pub struct Connection {
_cancel: futures::channel::oneshot::Sender<()>,
listener: Option<futures::channel::mpsc::Sender<Event>>,
} }
impl Subscriptions { impl Subscriptions {
@ -440,17 +448,19 @@ impl Subscriptions {
) { ) {
use futures::{future::FutureExt, stream::StreamExt}; use futures::{future::FutureExt, stream::StreamExt};
let handles = subscriptions.handles(); let connections = subscriptions.connections();
let mut alive = std::collections::HashSet::new(); let mut alive = std::collections::HashSet::new();
for handle in handles { for connection in connections {
let id = handle.id(); let id = connection.id();
let _ = alive.insert(id); let _ = alive.insert(id);
if !self.alive.contains_key(&id) { if !self.alive.contains_key(&id) {
let (cancel, cancelled) = futures::channel::oneshot::channel(); let (cancel, cancelled) = futures::channel::oneshot::channel();
let (event_sender, event_receiver) =
futures::channel::mpsc::channel(100);
let stream = handle.stream(); let stream = connection.stream(event_receiver);
let proxy = let proxy =
std::sync::Arc::new(std::sync::Mutex::new(proxy.clone())); std::sync::Arc::new(std::sync::Mutex::new(proxy.clone()));
@ -471,12 +481,36 @@ impl Subscriptions {
thread_pool.spawn_ok(future); thread_pool.spawn_ok(future);
let _ = self.alive.insert(id, cancel); let _ = self.alive.insert(
id,
Connection {
_cancel: cancel,
listener: if event_sender.is_closed() {
None
} else {
Some(event_sender)
},
},
);
} }
} }
self.alive.retain(|id, _| alive.contains(&id)); self.alive.retain(|id, _| alive.contains(&id));
} }
fn send_event(&mut self, event: Event) {
self.alive
.values_mut()
.filter_map(|connection| connection.listener.as_mut())
.for_each(|listener| {
if let Err(error) = listener.try_send(event) {
log::warn!(
"Error sending event to subscription: {:?}",
error
);
}
});
}
} }
// As defined in: http://www.unicode.org/faq/private_use.html // As defined in: http://www.unicode.org/faq/private_use.html