diff options
Diffstat (limited to 'core')
| -rw-r--r-- | core/src/element.rs | 5 | ||||
| -rw-r--r-- | core/src/lib.rs | 57 | 
2 files changed, 60 insertions, 2 deletions
| diff --git a/core/src/element.rs b/core/src/element.rs index ede9e16c..b7d51aeb 100644 --- a/core/src/element.rs +++ b/core/src/element.rs @@ -93,6 +93,7 @@ impl<'a, Message, Theme, Renderer> Element<'a, Message, Theme, Renderer> {      ///      /// ```no_run      /// # mod iced { +    /// #     pub use iced_core::Function;      /// #     pub type Element<'a, Message> = iced_core::Element<'a, Message, iced_core::Theme, ()>;      /// #      /// #     pub mod widget { @@ -119,7 +120,7 @@ impl<'a, Message, Theme, Renderer> Element<'a, Message, Theme, Renderer> {      /// use counter::Counter;      ///      /// use iced::widget::row; -    /// use iced::Element; +    /// use iced::{Element, Function};      ///      /// struct ManyCounters {      ///     counters: Vec<Counter>, @@ -142,7 +143,7 @@ impl<'a, Message, Theme, Renderer> Element<'a, Message, Theme, Renderer> {      ///                     // Here we turn our `Element<counter::Message>` into      ///                     // an `Element<Message>` by combining the `index` and the      ///                     // message of the `element`. -    ///                     counter.map(move |message| Message::Counter(index, message)) +    ///                     counter.map(Message::Counter.with(index))      ///                 }),      ///         )      ///         .into() diff --git a/core/src/lib.rs b/core/src/lib.rs index d5c221ac..03cc0632 100644 --- a/core/src/lib.rs +++ b/core/src/lib.rs @@ -93,3 +93,60 @@ pub use smol_str::SmolStr;  pub fn never<T>(never: std::convert::Infallible) -> T {      match never {}  } + +/// A trait extension for binary functions (`Fn(A, B) -> O`). +/// +/// It enables you to use a bunch of nifty functional programming paradigms +/// that work well with iced. +pub trait Function<A, B, O> { +    /// Applies the given first argument to a binary function and returns +    /// a new function that takes the other argument. +    /// +    /// This lets you partially "apply" a function—equivalent to currying, +    /// but it only works with binary functions. If you want to apply an +    /// arbitrary number of arguments, create a little struct for them. +    /// +    /// # When is this useful? +    /// Sometimes you will want to identify the source or target +    /// of some message in your user interface. This can be achieved through +    /// normal means by defining a closure and moving the identifier +    /// inside: +    /// +    /// ```rust +    /// # let element: Option<()> = Some(()); +    /// # enum Message { ButtonPressed(u32, ()) } +    /// let id = 123; +    /// +    /// # let _ = { +    /// element.map(move |result| Message::ButtonPressed(id, result)) +    /// # }; +    /// ``` +    /// +    /// That's quite a mouthful. [`with`](Self::with) lets you write: +    /// +    /// ```rust +    /// # use iced_core::Function; +    /// # let element: Option<()> = Some(()); +    /// # enum Message { ButtonPressed(u32, ()) } +    /// let id = 123; +    /// +    /// # let _ = { +    /// element.map(Message::ButtonPressed.with(id)) +    /// # }; +    /// ``` +    /// +    /// Effectively creating the same closure that partially applies +    /// the `id` to the message—but much more concise! +    fn with(self, prefix: A) -> impl Fn(B) -> O; +} + +impl<F, A, B, O> Function<A, B, O> for F +where +    F: Fn(A, B) -> O, +    Self: Sized, +    A: Copy, +{ +    fn with(self, prefix: A) -> impl Fn(B) -> O { +        move |result| self(prefix, result) +    } +} | 
