Converge Command types from iced_futures and iced_native
This commit is contained in:
parent
43414bbdfb
commit
8af69be47e
9 changed files with 161 additions and 266 deletions
|
|
@ -11,6 +11,7 @@ use crate::widget::Id;
|
||||||
|
|
||||||
use std::any::Any;
|
use std::any::Any;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
/// A piece of logic that can traverse the widget tree of an application in
|
/// A piece of logic that can traverse the widget tree of an application in
|
||||||
/// order to query or update some widget state.
|
/// order to query or update some widget state.
|
||||||
|
|
@ -68,9 +69,122 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Maps the output of an [`Operation`] using the given function.
|
||||||
|
pub fn map<A, B>(
|
||||||
|
operation: Box<dyn Operation<A>>,
|
||||||
|
f: impl Fn(A) -> B + 'static,
|
||||||
|
) -> impl Operation<B>
|
||||||
|
where
|
||||||
|
A: 'static,
|
||||||
|
B: 'static,
|
||||||
|
{
|
||||||
|
#[allow(missing_debug_implementations)]
|
||||||
|
struct Map<A, B> {
|
||||||
|
operation: Box<dyn Operation<A>>,
|
||||||
|
f: Rc<dyn Fn(A) -> B>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<A, B> Operation<B> for Map<A, B>
|
||||||
|
where
|
||||||
|
A: 'static,
|
||||||
|
B: 'static,
|
||||||
|
{
|
||||||
|
fn container(
|
||||||
|
&mut self,
|
||||||
|
id: Option<&Id>,
|
||||||
|
operate_on_children: &mut dyn FnMut(&mut dyn Operation<B>),
|
||||||
|
) {
|
||||||
|
struct MapRef<'a, A> {
|
||||||
|
operation: &'a mut dyn Operation<A>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, A, B> Operation<B> for MapRef<'a, A> {
|
||||||
|
fn container(
|
||||||
|
&mut self,
|
||||||
|
id: Option<&Id>,
|
||||||
|
operate_on_children: &mut dyn FnMut(&mut dyn Operation<B>),
|
||||||
|
) {
|
||||||
|
let Self { operation, .. } = self;
|
||||||
|
|
||||||
|
operation.container(id, &mut |operation| {
|
||||||
|
operate_on_children(&mut MapRef { operation });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn scrollable(
|
||||||
|
&mut self,
|
||||||
|
state: &mut dyn Scrollable,
|
||||||
|
id: Option<&Id>,
|
||||||
|
) {
|
||||||
|
self.operation.scrollable(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn focusable(
|
||||||
|
&mut self,
|
||||||
|
state: &mut dyn Focusable,
|
||||||
|
id: Option<&Id>,
|
||||||
|
) {
|
||||||
|
self.operation.focusable(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn text_input(
|
||||||
|
&mut self,
|
||||||
|
state: &mut dyn TextInput,
|
||||||
|
id: Option<&Id>,
|
||||||
|
) {
|
||||||
|
self.operation.text_input(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn custom(&mut self, state: &mut dyn Any, id: Option<&Id>) {
|
||||||
|
self.operation.custom(state, id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let Self { operation, .. } = self;
|
||||||
|
|
||||||
|
MapRef {
|
||||||
|
operation: operation.as_mut(),
|
||||||
|
}
|
||||||
|
.container(id, operate_on_children);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn focusable(&mut self, state: &mut dyn Focusable, id: Option<&Id>) {
|
||||||
|
self.operation.focusable(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn scrollable(&mut self, state: &mut dyn Scrollable, id: Option<&Id>) {
|
||||||
|
self.operation.scrollable(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn text_input(&mut self, state: &mut dyn TextInput, id: Option<&Id>) {
|
||||||
|
self.operation.text_input(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn custom(&mut self, state: &mut dyn Any, id: Option<&Id>) {
|
||||||
|
self.operation.custom(state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn finish(&self) -> Outcome<B> {
|
||||||
|
match self.operation.finish() {
|
||||||
|
Outcome::None => Outcome::None,
|
||||||
|
Outcome::Some(output) => Outcome::Some((self.f)(output)),
|
||||||
|
Outcome::Chain(next) => Outcome::Chain(Box::new(Map {
|
||||||
|
operation: next,
|
||||||
|
f: self.f.clone(),
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Map {
|
||||||
|
operation,
|
||||||
|
f: Rc::new(f),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Produces an [`Operation`] that applies the given [`Operation`] to the
|
/// Produces an [`Operation`] that applies the given [`Operation`] to the
|
||||||
/// children of a container with the given [`Id`].
|
/// children of a container with the given [`Id`].
|
||||||
pub fn scoped<T: 'static>(
|
pub fn scope<T: 'static>(
|
||||||
target: Id,
|
target: Id,
|
||||||
operation: impl Operation<T> + 'static,
|
operation: impl Operation<T> + 'static,
|
||||||
) -> impl Operation<T> {
|
) -> impl Operation<T> {
|
||||||
|
|
|
||||||
|
|
@ -1,70 +0,0 @@
|
||||||
/// A set of asynchronous actions to be performed by some runtime.
|
|
||||||
#[must_use = "`Command` must be returned to runtime to take effect"]
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct Command<T>(Internal<T>);
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
enum Internal<T> {
|
|
||||||
None,
|
|
||||||
Single(T),
|
|
||||||
Batch(Vec<T>),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T> Command<T> {
|
|
||||||
/// Creates an empty [`Command`].
|
|
||||||
///
|
|
||||||
/// In other words, a [`Command`] that does nothing.
|
|
||||||
pub const fn none() -> Self {
|
|
||||||
Self(Internal::None)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a [`Command`] that performs a single action.
|
|
||||||
pub const fn single(action: T) -> Self {
|
|
||||||
Self(Internal::Single(action))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a [`Command`] that performs the actions of all the given
|
|
||||||
/// commands.
|
|
||||||
///
|
|
||||||
/// Once this command is run, all the commands will be executed at once.
|
|
||||||
pub fn batch(commands: impl IntoIterator<Item = Command<T>>) -> Self {
|
|
||||||
let mut batch = Vec::new();
|
|
||||||
|
|
||||||
for Command(command) in commands {
|
|
||||||
match command {
|
|
||||||
Internal::None => {}
|
|
||||||
Internal::Single(command) => batch.push(command),
|
|
||||||
Internal::Batch(commands) => batch.extend(commands),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Self(Internal::Batch(batch))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Applies a transformation to the result of a [`Command`].
|
|
||||||
pub fn map<A>(self, f: impl Fn(T) -> A) -> Command<A>
|
|
||||||
where
|
|
||||||
T: 'static,
|
|
||||||
{
|
|
||||||
let Command(command) = self;
|
|
||||||
|
|
||||||
match command {
|
|
||||||
Internal::None => Command::none(),
|
|
||||||
Internal::Single(action) => Command::single(f(action)),
|
|
||||||
Internal::Batch(batch) => {
|
|
||||||
Command(Internal::Batch(batch.into_iter().map(f).collect()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns all of the actions of the [`Command`].
|
|
||||||
pub fn actions(self) -> Vec<T> {
|
|
||||||
let Command(command) = self;
|
|
||||||
|
|
||||||
match command {
|
|
||||||
Internal::None => Vec::new(),
|
|
||||||
Internal::Single(action) => vec![action],
|
|
||||||
Internal::Batch(batch) => batch,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -20,7 +20,6 @@
|
||||||
pub use futures;
|
pub use futures;
|
||||||
pub use iced_core as core;
|
pub use iced_core as core;
|
||||||
|
|
||||||
mod command;
|
|
||||||
mod maybe_send;
|
mod maybe_send;
|
||||||
mod runtime;
|
mod runtime;
|
||||||
|
|
||||||
|
|
@ -28,7 +27,6 @@ pub mod backend;
|
||||||
pub mod executor;
|
pub mod executor;
|
||||||
pub mod subscription;
|
pub mod subscription;
|
||||||
|
|
||||||
pub use command::Command;
|
|
||||||
pub use executor::Executor;
|
pub use executor::Executor;
|
||||||
pub use maybe_send::MaybeSend;
|
pub use maybe_send::MaybeSend;
|
||||||
pub use platform::*;
|
pub use platform::*;
|
||||||
|
|
|
||||||
|
|
@ -3,37 +3,39 @@ mod action;
|
||||||
|
|
||||||
pub use action::Action;
|
pub use action::Action;
|
||||||
|
|
||||||
use crate::widget;
|
use crate::core::widget;
|
||||||
|
use crate::futures::MaybeSend;
|
||||||
use iced_futures::MaybeSend;
|
|
||||||
|
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::future::Future;
|
use std::future::Future;
|
||||||
|
|
||||||
/// A set of asynchronous actions to be performed by some runtime.
|
/// A set of asynchronous actions to be performed by some runtime.
|
||||||
#[must_use = "`Command` must be returned to runtime to take effect"]
|
#[must_use = "`Command` must be returned to runtime to take effect"]
|
||||||
pub struct Command<T>(iced_futures::Command<Action<T>>);
|
pub struct Command<T>(Internal<Action<T>>);
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
enum Internal<T> {
|
||||||
|
None,
|
||||||
|
Single(T),
|
||||||
|
Batch(Vec<T>),
|
||||||
|
}
|
||||||
|
|
||||||
impl<T> Command<T> {
|
impl<T> Command<T> {
|
||||||
/// Creates an empty [`Command`].
|
/// Creates an empty [`Command`].
|
||||||
///
|
///
|
||||||
/// In other words, a [`Command`] that does nothing.
|
/// In other words, a [`Command`] that does nothing.
|
||||||
pub const fn none() -> Self {
|
pub const fn none() -> Self {
|
||||||
Self(iced_futures::Command::none())
|
Self(Internal::None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a [`Command`] that performs a single [`Action`].
|
/// Creates a [`Command`] that performs a single [`Action`].
|
||||||
pub const fn single(action: Action<T>) -> Self {
|
pub const fn single(action: Action<T>) -> Self {
|
||||||
Self(iced_futures::Command::single(action))
|
Self(Internal::Single(action))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a [`Command`] that performs a [`widget::Operation`].
|
/// Creates a [`Command`] that performs a [`widget::Operation`].
|
||||||
pub fn widget(
|
pub fn widget(operation: impl widget::Operation<T> + 'static) -> Self {
|
||||||
operation: impl iced_core::widget::Operation<T> + 'static,
|
Self::single(Action::Widget(Box::new(operation)))
|
||||||
) -> Self {
|
|
||||||
Self(iced_futures::Command::single(Action::Widget(
|
|
||||||
widget::Action::new(operation),
|
|
||||||
)))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a [`Command`] that performs the action of the given future.
|
/// Creates a [`Command`] that performs the action of the given future.
|
||||||
|
|
@ -51,9 +53,17 @@ impl<T> Command<T> {
|
||||||
///
|
///
|
||||||
/// Once this command is run, all the commands will be executed at once.
|
/// Once this command is run, all the commands will be executed at once.
|
||||||
pub fn batch(commands: impl IntoIterator<Item = Command<T>>) -> Self {
|
pub fn batch(commands: impl IntoIterator<Item = Command<T>>) -> Self {
|
||||||
Self(iced_futures::Command::batch(
|
let mut batch = Vec::new();
|
||||||
commands.into_iter().map(|Command(command)| command),
|
|
||||||
))
|
for Command(command) in commands {
|
||||||
|
match command {
|
||||||
|
Internal::None => {}
|
||||||
|
Internal::Single(command) => batch.push(command),
|
||||||
|
Internal::Batch(commands) => batch.extend(commands),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Self(Internal::Batch(batch))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Applies a transformation to the result of a [`Command`].
|
/// Applies a transformation to the result of a [`Command`].
|
||||||
|
|
@ -65,16 +75,27 @@ impl<T> Command<T> {
|
||||||
T: 'static,
|
T: 'static,
|
||||||
A: 'static,
|
A: 'static,
|
||||||
{
|
{
|
||||||
let Command(command) = self;
|
match self.0 {
|
||||||
|
Internal::None => Command::none(),
|
||||||
Command(command.map(move |action| action.map(f.clone())))
|
Internal::Single(action) => Command::single(action.map(f)),
|
||||||
|
Internal::Batch(batch) => Command(Internal::Batch(
|
||||||
|
batch
|
||||||
|
.into_iter()
|
||||||
|
.map(|action| action.map(f.clone()))
|
||||||
|
.collect(),
|
||||||
|
)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns all of the actions of the [`Command`].
|
/// Returns all of the actions of the [`Command`].
|
||||||
pub fn actions(self) -> Vec<Action<T>> {
|
pub fn actions(self) -> Vec<Action<T>> {
|
||||||
let Command(command) = self;
|
let Command(command) = self;
|
||||||
|
|
||||||
command.actions()
|
match command {
|
||||||
|
Internal::None => Vec::new(),
|
||||||
|
Internal::Single(action) => vec![action],
|
||||||
|
Internal::Batch(batch) => batch,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::clipboard;
|
use crate::clipboard;
|
||||||
|
use crate::core::widget;
|
||||||
use crate::font;
|
use crate::font;
|
||||||
use crate::system;
|
use crate::system;
|
||||||
use crate::widget;
|
|
||||||
use crate::window;
|
use crate::window;
|
||||||
|
|
||||||
use iced_futures::MaybeSend;
|
use iced_futures::MaybeSend;
|
||||||
|
|
@ -28,7 +28,7 @@ pub enum Action<T> {
|
||||||
System(system::Action<T>),
|
System(system::Action<T>),
|
||||||
|
|
||||||
/// Run a widget action.
|
/// Run a widget action.
|
||||||
Widget(widget::Action<T>),
|
Widget(Box<dyn widget::Operation<T>>),
|
||||||
|
|
||||||
/// Load a font from its bytes.
|
/// Load a font from its bytes.
|
||||||
LoadFont {
|
LoadFont {
|
||||||
|
|
@ -59,7 +59,9 @@ impl<T> Action<T> {
|
||||||
Self::Clipboard(action) => Action::Clipboard(action.map(f)),
|
Self::Clipboard(action) => Action::Clipboard(action.map(f)),
|
||||||
Self::Window(window) => Action::Window(window.map(f)),
|
Self::Window(window) => Action::Window(window.map(f)),
|
||||||
Self::System(system) => Action::System(system.map(f)),
|
Self::System(system) => Action::System(system.map(f)),
|
||||||
Self::Widget(widget) => Action::Widget(widget.map(f)),
|
Self::Widget(operation) => {
|
||||||
|
Action::Widget(Box::new(widget::operation::map(operation, f)))
|
||||||
|
}
|
||||||
Self::LoadFont { bytes, tagger } => Action::LoadFont {
|
Self::LoadFont { bytes, tagger } => Action::LoadFont {
|
||||||
bytes,
|
bytes,
|
||||||
tagger: Box::new(move |result| f(tagger(result))),
|
tagger: Box::new(move |result| f(tagger(result))),
|
||||||
|
|
|
||||||
|
|
@ -50,7 +50,6 @@ pub mod keyboard;
|
||||||
pub mod program;
|
pub mod program;
|
||||||
pub mod system;
|
pub mod system;
|
||||||
pub mod user_interface;
|
pub mod user_interface;
|
||||||
pub mod widget;
|
|
||||||
pub mod window;
|
pub mod window;
|
||||||
|
|
||||||
// We disable debug capabilities on release builds unless the `debug` feature
|
// We disable debug capabilities on release builds unless the `debug` feature
|
||||||
|
|
|
||||||
|
|
@ -1,16 +0,0 @@
|
||||||
//! Use the built-in widgets or create your own.
|
|
||||||
//!
|
|
||||||
//! # Built-in widgets
|
|
||||||
//! Every built-in drawable widget has its own module with a `Renderer` trait
|
|
||||||
//! that must be implemented by a [renderer] before being able to use it as
|
|
||||||
//! a [`Widget`].
|
|
||||||
//!
|
|
||||||
//! # Custom widgets
|
|
||||||
//! If you want to implement a custom widget, you simply need to implement the
|
|
||||||
//! [`Widget`] trait. You can use the API of the built-in widgets as a guide or
|
|
||||||
//! source of inspiration.
|
|
||||||
//!
|
|
||||||
//! [renderer]: crate::renderer
|
|
||||||
mod action;
|
|
||||||
|
|
||||||
pub use action::Action;
|
|
||||||
|
|
@ -1,153 +0,0 @@
|
||||||
use iced_core::widget::operation::{
|
|
||||||
self, Focusable, Operation, Scrollable, TextInput,
|
|
||||||
};
|
|
||||||
use iced_core::widget::Id;
|
|
||||||
use iced_futures::MaybeSend;
|
|
||||||
|
|
||||||
use std::any::Any;
|
|
||||||
use std::rc::Rc;
|
|
||||||
|
|
||||||
/// An operation to be performed on the widget tree.
|
|
||||||
#[allow(missing_debug_implementations)]
|
|
||||||
pub struct Action<T>(Box<dyn Operation<T>>);
|
|
||||||
|
|
||||||
impl<T> Action<T> {
|
|
||||||
/// Creates a new [`Action`] with the given [`Operation`].
|
|
||||||
pub fn new(operation: impl Operation<T> + 'static) -> Self {
|
|
||||||
Self(Box::new(operation))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Maps the output of an [`Action`] using the given function.
|
|
||||||
pub fn map<A>(
|
|
||||||
self,
|
|
||||||
f: impl Fn(T) -> A + 'static + MaybeSend + Sync,
|
|
||||||
) -> Action<A>
|
|
||||||
where
|
|
||||||
T: 'static,
|
|
||||||
A: 'static,
|
|
||||||
{
|
|
||||||
Action(Box::new(Map {
|
|
||||||
operation: self.0,
|
|
||||||
f: Rc::new(f),
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Consumes the [`Action`] and returns the internal [`Operation`].
|
|
||||||
pub fn into_operation(self) -> Box<dyn Operation<T>> {
|
|
||||||
self.0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(missing_debug_implementations)]
|
|
||||||
struct Map<A, B> {
|
|
||||||
operation: Box<dyn Operation<A>>,
|
|
||||||
f: Rc<dyn Fn(A) -> B>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<A, B> Operation<B> for Map<A, B>
|
|
||||||
where
|
|
||||||
A: 'static,
|
|
||||||
B: 'static,
|
|
||||||
{
|
|
||||||
fn container(
|
|
||||||
&mut self,
|
|
||||||
id: Option<&Id>,
|
|
||||||
operate_on_children: &mut dyn FnMut(&mut dyn Operation<B>),
|
|
||||||
) {
|
|
||||||
struct MapRef<'a, A> {
|
|
||||||
operation: &'a mut dyn Operation<A>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, A, B> Operation<B> for MapRef<'a, A> {
|
|
||||||
fn container(
|
|
||||||
&mut self,
|
|
||||||
id: Option<&Id>,
|
|
||||||
operate_on_children: &mut dyn FnMut(&mut dyn Operation<B>),
|
|
||||||
) {
|
|
||||||
let Self { operation, .. } = self;
|
|
||||||
|
|
||||||
operation.container(id, &mut |operation| {
|
|
||||||
operate_on_children(&mut MapRef { operation });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn scrollable(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn Scrollable,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.scrollable(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn focusable(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn Focusable,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.focusable(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn text_input(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn TextInput,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.text_input(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn custom(&mut self, state: &mut dyn Any, id: Option<&Id>) {
|
|
||||||
self.operation.custom(state, id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let Self { operation, .. } = self;
|
|
||||||
|
|
||||||
MapRef {
|
|
||||||
operation: operation.as_mut(),
|
|
||||||
}
|
|
||||||
.container(id, operate_on_children);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn focusable(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn operation::Focusable,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.focusable(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn scrollable(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn operation::Scrollable,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.scrollable(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn text_input(
|
|
||||||
&mut self,
|
|
||||||
state: &mut dyn operation::TextInput,
|
|
||||||
id: Option<&Id>,
|
|
||||||
) {
|
|
||||||
self.operation.text_input(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn custom(&mut self, state: &mut dyn Any, id: Option<&Id>) {
|
|
||||||
self.operation.custom(state, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn finish(&self) -> operation::Outcome<B> {
|
|
||||||
match self.operation.finish() {
|
|
||||||
operation::Outcome::None => operation::Outcome::None,
|
|
||||||
operation::Outcome::Some(output) => {
|
|
||||||
operation::Outcome::Some((self.f)(output))
|
|
||||||
}
|
|
||||||
operation::Outcome::Chain(next) => {
|
|
||||||
operation::Outcome::Chain(Box::new(Map {
|
|
||||||
operation: next,
|
|
||||||
f: self.f.clone(),
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -818,7 +818,7 @@ pub fn run_command<A, E>(
|
||||||
},
|
},
|
||||||
command::Action::Widget(action) => {
|
command::Action::Widget(action) => {
|
||||||
let mut current_cache = std::mem::take(cache);
|
let mut current_cache = std::mem::take(cache);
|
||||||
let mut current_operation = Some(action.into_operation());
|
let mut current_operation = Some(action);
|
||||||
|
|
||||||
let mut user_interface = build_user_interface(
|
let mut user_interface = build_user_interface(
|
||||||
application,
|
application,
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue