diff options
Diffstat (limited to 'native')
| -rw-r--r-- | native/src/element.rs | 122 | ||||
| -rw-r--r-- | native/src/input/mouse/event.rs | 8 | ||||
| -rw-r--r-- | native/src/layout.rs | 16 | ||||
| -rw-r--r-- | native/src/layout/flex.rs | 24 | ||||
| -rw-r--r-- | native/src/layout/limits.rs | 53 | ||||
| -rw-r--r-- | native/src/layout/node.rs | 23 | ||||
| -rw-r--r-- | native/src/lib.rs | 227 | ||||
| -rw-r--r-- | native/src/renderer.rs | 16 | ||||
| -rw-r--r-- | native/src/renderer/null.rs | 145 | ||||
| -rw-r--r-- | native/src/renderer/windowed.rs | 19 | ||||
| -rw-r--r-- | native/src/size.rs | 14 | ||||
| -rw-r--r-- | native/src/user_interface.rs | 12 | ||||
| -rw-r--r-- | native/src/widget.rs | 20 | ||||
| -rw-r--r-- | native/src/widget/button.rs | 183 | ||||
| -rw-r--r-- | native/src/widget/checkbox.rs | 134 | ||||
| -rw-r--r-- | native/src/widget/column.rs | 131 | ||||
| -rw-r--r-- | native/src/widget/container.rs | 93 | ||||
| -rw-r--r-- | native/src/widget/image.rs | 81 | ||||
| -rw-r--r-- | native/src/widget/radio.rs | 149 | ||||
| -rw-r--r-- | native/src/widget/row.rs | 132 | ||||
| -rw-r--r-- | native/src/widget/scrollable.rs | 242 | ||||
| -rw-r--r-- | native/src/widget/slider.rs | 132 | ||||
| -rw-r--r-- | native/src/widget/text.rs | 166 | ||||
| -rw-r--r-- | native/src/widget/text_input.rs | 285 | 
24 files changed, 1981 insertions, 446 deletions
| diff --git a/native/src/element.rs b/native/src/element.rs index 791bf9cf..d4237fd0 100644 --- a/native/src/element.rs +++ b/native/src/element.rs @@ -13,18 +13,11 @@ use crate::{  /// [built-in widget]: widget/index.html#built-in-widgets  /// [`Widget`]: widget/trait.Widget.html  /// [`Element`]: struct.Element.html +#[allow(missing_debug_implementations)]  pub struct Element<'a, Message, Renderer> {      pub(crate) widget: Box<dyn Widget<Message, Renderer> + 'a>,  } -impl<'a, Message, Renderer> std::fmt::Debug for Element<'a, Message, Renderer> { -    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { -        f.debug_struct("Element") -            .field("widget", &self.widget) -            .finish() -    } -} -  impl<'a, Message, Renderer> Element<'a, Message, Renderer>  where      Renderer: crate::Renderer, @@ -41,31 +34,6 @@ where          }      } -    pub fn width(&self) -> Length { -        self.widget.width() -    } - -    pub fn height(&self) -> Length { -        self.widget.height() -    } - -    pub fn layout( -        &self, -        renderer: &Renderer, -        limits: &layout::Limits, -    ) -> layout::Node { -        self.widget.layout(renderer, limits) -    } - -    pub fn draw( -        &self, -        renderer: &mut Renderer, -        layout: Layout<'_>, -        cursor_position: Point, -    ) -> Renderer::Output { -        self.widget.draw(renderer, layout, cursor_position) -    } -      /// Applies a transformation to the produced message of the [`Element`].      ///      /// This method is useful when you want to decouple different parts of your @@ -127,37 +95,7 @@ where      /// # }      /// #      /// # mod iced_wgpu { -    /// #     use iced_native::{ -    /// #         text, row, layout, Text, Size, Point, Rectangle, Layout, Row -    /// #     }; -    /// #     pub struct Renderer; -    /// # -    /// #     impl iced_native::Renderer for Renderer { type Output = (); } -    /// # -    /// #     impl iced_native::row::Renderer for Renderer { -    /// #         fn draw<Message>( -    /// #             &mut self, -    /// #             _column: &Row<'_, Message, Self>, -    /// #             _layout: Layout<'_>, -    /// #             _cursor_position: Point, -    /// #         ) {} -    /// #     } -    /// # -    /// #     impl text::Renderer for Renderer { -    /// #         fn layout( -    /// #             &self, -    /// #             _text: &Text, -    /// #             _limits: &layout::Limits, -    /// #         ) -> layout::Node { -    /// #             layout::Node::new(Size::ZERO) -    /// #         } -    /// # -    /// #         fn draw( -    /// #             &mut self, -    /// #             _text: &Text, -    /// #             _layout: Layout<'_>, -    /// #         ) {} -    /// #     } +    /// #     pub use iced_native::renderer::Null as Renderer;      /// # }      /// #      /// # use counter::Counter; @@ -263,6 +201,45 @@ where          }      } +    /// Returns the width of the [`Element`]. +    /// +    /// [`Element`]: struct.Element.html +    pub fn width(&self) -> Length { +        self.widget.width() +    } + +    /// Returns the height of the [`Element`]. +    /// +    /// [`Element`]: struct.Element.html +    pub fn height(&self) -> Length { +        self.widget.height() +    } + +    /// Computes the layout of the [`Element`] in the given [`Limits`]. +    /// +    /// [`Element`]: struct.Element.html +    /// [`Limits`]: layout/struct.Limits.html +    pub fn layout( +        &self, +        renderer: &Renderer, +        limits: &layout::Limits, +    ) -> layout::Node { +        self.widget.layout(renderer, limits) +    } + +    /// Draws the [`Element`] and its children using the given [`Layout`]. +    /// +    /// [`Element`]: struct.Element.html +    /// [`Layout`]: layout/struct.Layout.html +    pub fn draw( +        &self, +        renderer: &mut Renderer, +        layout: Layout<'_>, +        cursor_position: Point, +    ) -> Renderer::Output { +        self.widget.draw(renderer, layout, cursor_position) +    } +      pub(crate) fn hash_layout(&self, state: &mut Hasher) {          self.widget.hash_layout(state);      } @@ -273,12 +250,6 @@ struct Map<'a, A, B, Renderer> {      mapper: Box<dyn Fn(A) -> B>,  } -impl<'a, A, B, Renderer> std::fmt::Debug for Map<'a, A, B, Renderer> { -    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { -        f.debug_struct("Map").field("widget", &self.widget).finish() -    } -} -  impl<'a, A, B, Renderer> Map<'a, A, B, Renderer> {      pub fn new<F>(          widget: Box<dyn Widget<A, Renderer> + 'a>, @@ -358,17 +329,6 @@ struct Explain<'a, Message, Renderer: crate::Renderer> {      color: Color,  } -impl<'a, Message, Renderer> std::fmt::Debug for Explain<'a, Message, Renderer> -where -    Renderer: crate::Renderer, -{ -    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { -        f.debug_struct("Explain") -            .field("element", &self.element) -            .finish() -    } -} -  impl<'a, Message, Renderer> Explain<'a, Message, Renderer>  where      Renderer: crate::Renderer, diff --git a/native/src/input/mouse/event.rs b/native/src/input/mouse/event.rs index 478f9b4d..aafc4fe3 100644 --- a/native/src/input/mouse/event.rs +++ b/native/src/input/mouse/event.rs @@ -34,11 +34,16 @@ pub enum Event {      },      /// The mouse wheel was scrolled. -    WheelScrolled { delta: ScrollDelta }, +    WheelScrolled { +        /// The scroll movement. +        delta: ScrollDelta, +    },  } +/// A scroll movement.  #[derive(Debug, Clone, Copy, PartialEq)]  pub enum ScrollDelta { +    /// A line-based scroll movement      Lines {          /// The number of horizontal lines scrolled          x: f32, @@ -46,6 +51,7 @@ pub enum ScrollDelta {          /// The number of vertical lines scrolled          y: f32,      }, +    /// A pixel-based scroll movement      Pixels {          /// The number of horizontal pixels scrolled          x: f32, diff --git a/native/src/layout.rs b/native/src/layout.rs index 0a744346..e945706b 100644 --- a/native/src/layout.rs +++ b/native/src/layout.rs @@ -1,3 +1,4 @@ +//! Position your widgets properly.  mod limits;  mod node; @@ -8,6 +9,9 @@ pub use node::Node;  use crate::{Point, Rectangle, Vector}; +/// The bounds of a [`Node`] and its children, using absolute coordinates. +/// +/// [`Node`]: struct.Node.html  #[derive(Debug, Clone, Copy)]  pub struct Layout<'a> {      position: Point, @@ -28,6 +32,14 @@ impl<'a> Layout<'a> {          }      } +    /// Gets the bounds of the [`Layout`]. +    /// +    /// The returned [`Rectangle`] describes the position and size of a +    /// [`Node`]. +    /// +    /// [`Layout`]: struct.Layout.html +    /// [`Rectangle`]: struct.Rectangle.html +    /// [`Node`]: struct.Node.html      pub fn bounds(&self) -> Rectangle {          let bounds = self.node.bounds(); @@ -39,6 +51,10 @@ impl<'a> Layout<'a> {          }      } +    /// Returns an iterator over the [`Layout`] of the children of a [`Node`]. +    /// +    /// [`Layout`]: struct.Layout.html +    /// [`Node`]: struct.Node.html      pub fn children(&'a self) -> impl Iterator<Item = Layout<'a>> {          self.node.children().iter().map(move |node| {              Layout::with_offset( diff --git a/native/src/layout/flex.rs b/native/src/layout/flex.rs index 7a2b0d70..bc90553e 100644 --- a/native/src/layout/flex.rs +++ b/native/src/layout/flex.rs @@ -1,3 +1,4 @@ +//! Distribute elements using a flex-based layout.  // This code is heavily inspired by the [`druid`] codebase.  //  // [`druid`]: https://github.com/xi-editor/druid @@ -20,9 +21,13 @@ use crate::{      Align, Element, Size,  }; +/// The main axis of a flex layout.  #[derive(Debug)]  pub enum Axis { +    /// The horizontal axis      Horizontal, + +    /// The vertical axis      Vertical,  } @@ -49,7 +54,12 @@ impl Axis {      }  } -// TODO: Remove `Message` type parameter +/// Computes the flex layout with the given axis and limits, applying spacing, +/// padding and alignment to the items as needed. +/// +/// It returns a new layout [`Node`]. +/// +/// [`Node`]: ../struct.Node.html  pub fn resolve<Message, Renderer>(      axis: Axis,      renderer: &Renderer, @@ -57,7 +67,7 @@ pub fn resolve<Message, Renderer>(      padding: f32,      spacing: f32,      align_items: Align, -    children: &[Element<'_, Message, Renderer>], +    items: &[Element<'_, Message, Renderer>],  ) -> Node  where      Renderer: crate::Renderer, @@ -65,14 +75,14 @@ where      let limits = limits.pad(padding);      let mut total_non_fill = -        spacing as f32 * (children.len() as i32 - 1).max(0) as f32; +        spacing as f32 * (items.len() as i32 - 1).max(0) as f32;      let mut fill_sum = 0;      let mut cross = axis.cross(limits.min()); -    let mut nodes: Vec<Node> = Vec::with_capacity(children.len()); -    nodes.resize(children.len(), Node::default()); +    let mut nodes: Vec<Node> = Vec::with_capacity(items.len()); +    nodes.resize(items.len(), Node::default()); -    for (i, child) in children.iter().enumerate() { +    for (i, child) in items.iter().enumerate() {          let fill_factor = match axis {              Axis::Horizontal => child.width(),              Axis::Vertical => child.height(), @@ -97,7 +107,7 @@ where      let available = axis.main(limits.max());      let remaining = (available - total_non_fill).max(0.0); -    for (i, child) in children.iter().enumerate() { +    for (i, child) in items.iter().enumerate() {          let fill_factor = match axis {              Axis::Horizontal => child.width(),              Axis::Vertical => child.height(), diff --git a/native/src/layout/limits.rs b/native/src/layout/limits.rs index af269acd..2705a47d 100644 --- a/native/src/layout/limits.rs +++ b/native/src/layout/limits.rs @@ -1,5 +1,6 @@  use crate::{Length, Size}; +/// A set of size constraints for layouting.  #[derive(Debug, Clone, Copy)]  pub struct Limits {      min: Size, @@ -8,12 +9,17 @@ pub struct Limits {  }  impl Limits { +    /// No limits      pub const NONE: Limits = Limits {          min: Size::ZERO,          max: Size::INFINITY,          fill: Size::INFINITY,      }; +    /// Creates new [`Limits`] with the given minimum and maximum [`Size`]. +    /// +    /// [`Limits`]: struct.Limits.html +    /// [`Size`]: ../struct.Size.html      pub fn new(min: Size, max: Size) -> Limits {          Limits {              min, @@ -22,14 +28,25 @@ impl Limits {          }      } +    /// Returns the minimum [`Size`] of the [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html +    /// [`Size`]: ../struct.Size.html      pub fn min(&self) -> Size {          self.min      } +    /// Returns the maximum [`Size`] of the [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html +    /// [`Size`]: ../struct.Size.html      pub fn max(&self) -> Size {          self.max      } +    /// Applies a width constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn width(mut self, width: Length) -> Limits {          match width {              Length::Shrink => { @@ -51,6 +68,9 @@ impl Limits {          self      } +    /// Applies a height constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn height(mut self, height: Length) -> Limits {          match height {              Length::Shrink => { @@ -72,6 +92,9 @@ impl Limits {          self      } +    /// Applies a minimum width constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn min_width(mut self, min_width: u32) -> Limits {          self.min.width =              self.min.width.max(min_width as f32).min(self.max.width); @@ -79,6 +102,9 @@ impl Limits {          self      } +    /// Applies a maximum width constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn max_width(mut self, max_width: u32) -> Limits {          self.max.width =              self.max.width.min(max_width as f32).max(self.min.width); @@ -86,6 +112,19 @@ impl Limits {          self      } +    /// Applies a minimum height constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html +    pub fn min_height(mut self, min_height: u32) -> Limits { +        self.min.height = +            self.min.height.max(min_height as f32).min(self.max.height); + +        self +    } + +    /// Applies a maximum height constraint to the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn max_height(mut self, max_height: u32) -> Limits {          self.max.height =              self.max.height.min(max_height as f32).max(self.min.height); @@ -93,10 +132,17 @@ impl Limits {          self      } +    /// Shrinks the current [`Limits`] to account for the given padding. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn pad(&self, padding: f32) -> Limits {          self.shrink(Size::new(padding * 2.0, padding * 2.0))      } +    /// Shrinks the current [`Limits`] by the given [`Size`]. +    /// +    /// [`Limits`]: struct.Limits.html +    /// [`Size`]: ../struct.Size.html      pub fn shrink(&self, size: Size) -> Limits {          let min = Size::new(              (self.min().width - size.width).max(0.0), @@ -116,6 +162,9 @@ impl Limits {          Limits { min, max, fill }      } +    /// Removes the minimum width constraint for the current [`Limits`]. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn loose(&self) -> Limits {          Limits {              min: Size::ZERO, @@ -124,6 +173,10 @@ impl Limits {          }      } +    /// Computes the resulting [`Size`] that fits the [`Limits`] given the +    /// intrinsic size of some content. +    /// +    /// [`Limits`]: struct.Limits.html      pub fn resolve(&self, intrinsic_size: Size) -> Size {          Size::new(              intrinsic_size diff --git a/native/src/layout/node.rs b/native/src/layout/node.rs index 64ebf2d0..ed1cd3da 100644 --- a/native/src/layout/node.rs +++ b/native/src/layout/node.rs @@ -1,16 +1,25 @@  use crate::{Align, Rectangle, Size}; +/// The bounds of an element and its children.  #[derive(Debug, Clone, Default)]  pub struct Node { -    pub bounds: Rectangle, +    pub(crate) bounds: Rectangle,      children: Vec<Node>,  }  impl Node { +    /// Creates a new [`Node`] with the given [`Size`]. +    /// +    /// [`Node`]: struct.Node.html +    /// [`Size`]: ../struct.Size.html      pub fn new(size: Size) -> Self {          Self::with_children(size, Vec::new())      } +    /// Creates a new [`Node`] with the given [`Size`] and children. +    /// +    /// [`Node`]: struct.Node.html +    /// [`Size`]: ../struct.Size.html      pub fn with_children(size: Size, children: Vec<Node>) -> Self {          Node {              bounds: Rectangle { @@ -23,19 +32,29 @@ impl Node {          }      } +    /// Returns the [`Size`] of the [`Node`]. +    /// +    /// [`Node`]: struct.Node.html +    /// [`Size`]: ../struct.Size.html      pub fn size(&self) -> Size {          Size::new(self.bounds.width, self.bounds.height)      } +    /// Returns the bounds of the [`Node`]. +    /// +    /// [`Node`]: struct.Node.html      pub fn bounds(&self) -> Rectangle {          self.bounds      } +    /// Returns the children of the [`Node`]. +    /// +    /// [`Node`]: struct.Node.html      pub fn children(&self) -> &[Node] {          &self.children      } -    pub fn align( +    pub(crate) fn align(          &mut self,          horizontal_alignment: Align,          vertical_alignment: Align, diff --git a/native/src/lib.rs b/native/src/lib.rs index bd03ddcd..45c3c699 100644 --- a/native/src/lib.rs +++ b/native/src/lib.rs @@ -1,204 +1,40 @@ -//! Iced is a renderer-agnostic GUI library focused on simplicity and -//! type-safety. Inspired by [Elm]. +//! A renderer-agnostic native GUI runtime.  //! -//! # Features -//!   * Simple, easy-to-use, renderer-agnostic API -//!   * Responsive, flexbox-based layouting -//!   * Type-safe, reactive programming model -//!   * Built-in widgets -//!   * Custom widget support +//!   //! -//! Check out the [repository] and the [examples] for more details! +//! `iced_native` takes [`iced_core`] and builds a native runtime on top of it, +//! featuring:  //! -//! [examples]: https://github.com/hecrj/iced/tree/0.1.0/examples -//! [repository]: https://github.com/hecrj/iced +//! - A custom layout engine, greatly inspired by [`druid`] +//! - Event handling for all the built-in widgets +//! - A renderer-agnostic API  //! -//! # Usage -//! Inspired by [The Elm Architecture], Iced expects you to split user interfaces -//! into four different concepts: -//! -//!   * __State__ — the state of your application -//!   * __Messages__ — user interactions or meaningful events that you care -//!   about -//!   * __View logic__ — a way to display your __state__ as widgets that -//!   may produce __messages__ on user interaction -//!   * __Update logic__ — a way to react to __messages__ and update your -//!   __state__ -//! -//! We can build something to see how this works! Let's say we want a simple counter -//! that can be incremented and decremented using two buttons. -//! -//! We start by modelling the __state__ of our application: -//! -//! ``` -//! use iced_native::button; -//! -//! struct Counter { -//!     // The counter value -//!     value: i32, -//! -//!     // The local state of the two buttons -//!     increment_button: button::State, -//!     decrement_button: button::State, -//! } -//! ``` -//! -//! Next, we need to define the possible user interactions of our counter: -//! the button presses. These interactions are our __messages__: -//! -//! ``` -//! #[derive(Debug, Clone, Copy)] -//! pub enum Message { -//!     IncrementPressed, -//!     DecrementPressed, -//! } -//! ``` +//! To achieve this, it introduces a bunch of reusable interfaces:  //! -//! Now, let's show the actual counter by putting it all together in our -//! __view logic__: +//! - A [`Widget`] trait, which is used to implement new widgets: from layout +//!   requirements to event and drawing logic. +//! - A bunch of `Renderer` traits, meant to keep the crate renderer-agnostic. +//! - A [`Windowed`] trait, leveraging [`raw-window-handle`], which can be +//!   implemented by graphical renderers that target _windows_. Window-based +//!   shells (like [`iced_winit`]) can use this trait to stay renderer-agnostic.  //! -//! ``` -//! # use iced_native::button; -//! # -//! # struct Counter { -//! #     // The counter value -//! #     value: i32, -//! # -//! #     // The local state of the two buttons -//! #     increment_button: button::State, -//! #     decrement_button: button::State, -//! # } -//! # -//! # #[derive(Debug, Clone, Copy)] -//! # pub enum Message { -//! #     IncrementPressed, -//! #     DecrementPressed, -//! # } -//! # -//! # mod iced_wgpu { -//! #     use iced_native::{ -//! #         button, text, layout, Button, Text, Point, Rectangle, Color, Layout, Size -//! #     }; -//! # -//! #     pub struct Renderer {} -//! # -//! #     impl iced_native::Renderer for Renderer { -//! #         type Output = (); -//! #     } -//! # -//! #     impl button::Renderer for Renderer { -//! #         fn layout<Message>( -//! #             &self, -//! #             _button: &Button<'_, Message, Self>, -//! #             _limits: &layout::Limits, -//! #         ) -> layout::Node { -//! #             layout::Node::new(Size::ZERO) -//! #         } -//! # -//! #         fn draw<Message>( -//! #             &mut self, -//! #             _button: &Button<'_, Message, Self>, -//! #             _layout: Layout<'_>, -//! #             _cursor_position: Point, -//! #         ) {} -//! #     } -//! # -//! #     impl text::Renderer for Renderer { -//! #         fn layout( -//! #             &self, -//! #             _text: &Text, -//! #             _limits: &layout::Limits, -//! #         ) -> layout::Node { -//! #             layout::Node::new(Size::ZERO) -//! #         } -//! # -//! #         fn draw( -//! #             &mut self, -//! #             _text: &Text, -//! #             _layout: Layout<'_>, -//! #         ) { -//! #         } -//! #     } -//! # } -//! use iced_native::{Button, Column, Text}; -//! use iced_wgpu::Renderer; // Iced does not include a renderer! We need to bring our own! -//! -//! impl Counter { -//!     pub fn view(&mut self) -> Column<Message, Renderer> { -//!         // We use a column: a simple vertical layout -//!         Column::new() -//!             .push( -//!                 // The increment button. We tell it to produce an -//!                 // `IncrementPressed` message when pressed -//!                 Button::new(&mut self.increment_button, Text::new("+")) -//!                     .on_press(Message::IncrementPressed), -//!             ) -//!             .push( -//!                 // We show the value of the counter here -//!                 Text::new(&self.value.to_string()).size(50), -//!             ) -//!             .push( -//!                 // The decrement button. We tell it to produce a -//!                 // `DecrementPressed` message when pressed -//!                 Button::new(&mut self.decrement_button, Text::new("-")) -//!                     .on_press(Message::DecrementPressed), -//!             ) -//!     } -//! } -//! ``` -//! -//! Finally, we need to be able to react to any produced __messages__ and change -//! our __state__ accordingly in our __update logic__: -//! -//! ``` -//! # use iced_native::button; -//! # -//! # struct Counter { -//! #     // The counter value -//! #     value: i32, -//! # -//! #     // The local state of the two buttons -//! #     increment_button: button::State, -//! #     decrement_button: button::State, -//! # } -//! # -//! # #[derive(Debug, Clone, Copy)] -//! # pub enum Message { -//! #     IncrementPressed, -//! #     DecrementPressed, -//! # } -//! impl Counter { -//!     // ... -//! -//!     pub fn update(&mut self, message: Message) { -//!         match message { -//!             Message::IncrementPressed => { -//!                 self.value += 1; -//!             } -//!             Message::DecrementPressed => { -//!                 self.value -= 1; -//!             } -//!         } -//!     } -//! } -//! ``` -//! -//! And that's everything! We just wrote a whole user interface. Iced is now able -//! to: -//! -//!   1. Take the result of our __view logic__ and layout its widgets. -//!   1. Process events from our system and produce __messages__ for our -//!      __update logic__. -//!   1. Draw the resulting user interface using our chosen __renderer__. -//! -//! Check out the [`UserInterface`] type to learn how to wire everything up! -//! -//! [Elm]: https://elm-lang.org/ -//! [The Elm Architecture]: https://guide.elm-lang.org/architecture/ -//! [documentation]: https://docs.rs/iced -//! [examples]: https://github.com/hecrj/iced/tree/master/examples +//! # Usage +//! The strategy to use this crate depends on your particular use case. If you +//! want to: +//! - Implement a custom shell or integrate it in your own system, you should +//!   check out the [`UserInterface`] type. +//! - Build a new renderer, see the [renderer] module. +//! - Build a custom widget, start at the [`Widget`] trait. +//! +//! [`iced_core`]: https://github.com/hecrj/iced/tree/master/core +//! [`iced_winit`]: https://github.com/hecrj/iced/tree/master/winit +//! [`druid`]: https://github.com/xi-editor/druid +//! [`raw-window-handle`]: https://github.com/rust-windowing/raw-window-handle +//! [`Widget`]: widget/trait.Widget.html +//! [`Windowed`]: renderer/trait.Windowed.html  //! [`UserInterface`]: struct.UserInterface.html -//#![deny(missing_docs)] +//! [renderer]: renderer/index.html +#![deny(missing_docs)]  #![deny(missing_debug_implementations)]  #![deny(unused_results)]  #![deny(unsafe_code)] @@ -216,7 +52,8 @@ mod size;  mod user_interface;  pub use iced_core::{ -    Align, Background, Color, Command, Font, Length, Point, Rectangle, Vector, +    Align, Background, Color, Command, Font, HorizontalAlignment, Length, +    Point, Rectangle, Vector, VerticalAlignment,  };  pub use element::Element; diff --git a/native/src/renderer.rs b/native/src/renderer.rs index 833de571..7a68ada4 100644 --- a/native/src/renderer.rs +++ b/native/src/renderer.rs @@ -21,16 +21,32 @@  //! [`checkbox::Renderer`]: ../widget/checkbox/trait.Renderer.html  mod debugger; +#[cfg(debug_assertions)] +mod null;  mod windowed;  pub use debugger::Debugger; +#[cfg(debug_assertions)] +pub use null::Null;  pub use windowed::{Target, Windowed};  use crate::{layout, Element}; +/// A component that can take the state of a user interface and produce an +/// output for its users.  pub trait Renderer: Sized { +    /// The type of output of the [`Renderer`]. +    /// +    /// If you are implementing a graphical renderer, your output will most +    /// likely be a tree of visual primitives. +    /// +    /// [`Renderer`]: trait.Renderer.html      type Output; +    /// Lays out the elements of a user interface. +    /// +    /// You should override this if you need to perform any operations before or +    /// after layouting. For instance, trimming the measurements cache.      fn layout<'a, Message>(          &mut self,          element: &Element<'a, Message, Self>, diff --git a/native/src/renderer/null.rs b/native/src/renderer/null.rs new file mode 100644 index 00000000..182f033a --- /dev/null +++ b/native/src/renderer/null.rs @@ -0,0 +1,145 @@ +use crate::{ +    button, checkbox, column, radio, row, scrollable, text, text_input, +    Background, Color, Element, Font, HorizontalAlignment, Layout, Point, +    Rectangle, Renderer, Size, VerticalAlignment, +}; + +/// A renderer that does nothing. +#[derive(Debug, Clone, Copy)] +pub struct Null; + +impl Renderer for Null { +    type Output = (); +} + +impl column::Renderer for Null { +    fn draw<Message>( +        &mut self, +        _content: &[Element<'_, Message, Self>], +        _layout: Layout<'_>, +        _cursor_position: Point, +    ) { +    } +} + +impl row::Renderer for Null { +    fn draw<Message>( +        &mut self, +        _content: &[Element<'_, Message, Self>], +        _layout: Layout<'_>, +        _cursor_position: Point, +    ) { +    } +} + +impl text::Renderer for Null { +    fn default_size(&self) -> u16 { +        20 +    } + +    fn measure( +        &self, +        _content: &str, +        _size: u16, +        _font: Font, +        _bounds: Size, +    ) -> (f32, f32) { +        (0.0, 20.0) +    } + +    fn draw( +        &mut self, +        _bounds: Rectangle, +        _content: &str, +        _size: u16, +        _font: Font, +        _color: Option<Color>, +        _horizontal_alignment: HorizontalAlignment, +        _vertical_alignment: VerticalAlignment, +    ) { +    } +} + +impl scrollable::Renderer for Null { +    fn is_mouse_over_scrollbar( +        &self, +        _bounds: Rectangle, +        _content_bounds: Rectangle, +        _cursor_position: Point, +    ) -> bool { +        false +    } + +    fn draw( +        &mut self, +        _scrollable: &scrollable::State, +        _bounds: Rectangle, +        _content_bounds: Rectangle, +        _is_mouse_over: bool, +        _is_mouse_over_scrollbar: bool, +        _offset: u32, +        _content: Self::Output, +    ) { +    } +} + +impl text_input::Renderer for Null { +    fn default_size(&self) -> u16 { +        20 +    } + +    fn draw( +        &mut self, +        _bounds: Rectangle, +        _text_bounds: Rectangle, +        _cursor_position: Point, +        _size: u16, +        _placeholder: &str, +        _value: &text_input::Value, +        _state: &text_input::State, +    ) -> Self::Output { +    } +} + +impl button::Renderer for Null { +    fn draw( +        &mut self, +        _bounds: Rectangle, +        _cursor_position: Point, +        _is_pressed: bool, +        _background: Option<Background>, +        _border_radius: u16, +        _content: Self::Output, +    ) -> Self::Output { +    } +} + +impl radio::Renderer for Null { +    fn default_size(&self) -> u32 { +        20 +    } + +    fn draw( +        &mut self, +        _bounds: Rectangle, +        _is_selected: bool, +        _is_mouse_over: bool, +        _label: Self::Output, +    ) { +    } +} + +impl checkbox::Renderer for Null { +    fn default_size(&self) -> u32 { +        20 +    } + +    fn draw( +        &mut self, +        _bounds: Rectangle, +        _is_checked: bool, +        _is_mouse_over: bool, +        _label: Self::Output, +    ) { +    } +} diff --git a/native/src/renderer/windowed.rs b/native/src/renderer/windowed.rs index 6d0419d2..813a03f2 100644 --- a/native/src/renderer/windowed.rs +++ b/native/src/renderer/windowed.rs @@ -2,11 +2,21 @@ use crate::MouseCursor;  use raw_window_handle::HasRawWindowHandle; +/// A renderer that can target windows.  pub trait Windowed: super::Renderer + Sized { +    /// The type of target.      type Target: Target<Renderer = Self>; +    /// Creates a new [`Windowed`] renderer. +    /// +    /// [`Windowed`]: trait.Windowed.html      fn new() -> Self; +    /// Performs the drawing operations described in the output on the given +    /// target. +    /// +    /// The overlay can be a bunch of debug text logs. It should be rendered on +    /// top of the GUI on most scenarios.      fn draw<T: AsRef<str>>(          &mut self,          output: &Self::Output, @@ -15,9 +25,15 @@ pub trait Windowed: super::Renderer + Sized {      ) -> MouseCursor;  } +/// A rendering target.  pub trait Target { +    /// The renderer of this target.      type Renderer; +    /// Creates a new rendering [`Target`] from the given window handle, width, +    /// height and dpi factor. +    /// +    /// [`Target`]: trait.Target.html      fn new<W: HasRawWindowHandle>(          window: &W,          width: u16, @@ -26,6 +42,9 @@ pub trait Target {          renderer: &Self::Renderer,      ) -> Self; +    /// Resizes the current [`Target`]. +    /// +    /// [`Target`]: trait.Target.html      fn resize(          &mut self,          width: u16, diff --git a/native/src/size.rs b/native/src/size.rs index bd909292..30e2a57e 100644 --- a/native/src/size.rs +++ b/native/src/size.rs @@ -1,5 +1,6 @@  use std::f32; +/// An amount of space in 2 dimensions.  #[derive(Debug, Clone, Copy, PartialEq)]  pub struct Size {      /// The width. @@ -9,13 +10,26 @@ pub struct Size {  }  impl Size { +    /// A [`Size`] with zero width and height. +    /// +    /// [`Size`]: struct.Size.html      pub const ZERO: Size = Size::new(0., 0.); + +    /// A [`Size`] with infinite width and height. +    /// +    /// [`Size`]: struct.Size.html      pub const INFINITY: Size = Size::new(f32::INFINITY, f32::INFINITY); +    /// A [`Size`] of infinite width and height. +    /// +    /// [`Size`]: struct.Size.html      pub const fn new(width: f32, height: f32) -> Self {          Size { width, height }      } +    /// Increments the [`Size`] to account for the given padding. +    /// +    /// [`Size`]: struct.Size.html      pub fn pad(&self, padding: f32) -> Self {          Size {              width: self.width + padding * 2.0, diff --git a/native/src/user_interface.rs b/native/src/user_interface.rs index 9324fe0b..9833c815 100644 --- a/native/src/user_interface.rs +++ b/native/src/user_interface.rs @@ -10,7 +10,7 @@ use std::hash::Hasher;  /// charge of using this type in your system in any way you want.  ///  /// [`Layout`]: struct.Layout.html -#[derive(Debug)] +#[allow(missing_debug_implementations)]  pub struct UserInterface<'a, Message, Renderer> {      hash: u64,      root: Element<'a, Message, Renderer>, @@ -52,7 +52,7 @@ where      /// #     impl iced_native::column::Renderer for Renderer {      /// #         fn draw<Message>(      /// #             &mut self, -    /// #             _column: &iced_native::Column<'_, Message, Self>, +    /// #             _children: &[iced_native::Element<'_, Message, Self>],      /// #             _layout: iced_native::Layout<'_>,      /// #             _cursor_position: iced_native::Point,      /// #         ) -> Self::Output { @@ -129,8 +129,8 @@ where      /// [`Event`]: enum.Event.html      ///      /// # Example -    /// Let's allow our [counter](index.html#usage) to change state by completing -    /// [the previous example](#example): +    /// Let's allow our [counter](index.html#usage) to change state by +    /// completing [the previous example](#example):      ///      /// ```no_run      /// use iced_native::{UserInterface, Cache}; @@ -148,7 +148,7 @@ where      /// #     impl iced_native::column::Renderer for Renderer {      /// #         fn draw<Message>(      /// #             &mut self, -    /// #             _column: &iced_native::Column<'_, Message, Self>, +    /// #             _children: &[iced_native::Element<'_, Message, Self>],      /// #             _layout: iced_native::Layout<'_>,      /// #             _cursor_position: iced_native::Point,      /// #         ) -> Self::Output { @@ -248,7 +248,7 @@ where      /// #     impl iced_native::column::Renderer for Renderer {      /// #         fn draw<Message>(      /// #             &mut self, -    /// #             _column: &iced_native::Column<'_, Message, Self>, +    /// #             _children: &[iced_native::Element<'_, Message, Self>],      /// #             _layout: iced_native::Layout<'_>,      /// #             _cursor_position: iced_native::Point,      /// #         ) -> Self::Output { diff --git a/native/src/widget.rs b/native/src/widget.rs index ff765ee6..71dcdc0d 100644 --- a/native/src/widget.rs +++ b/native/src/widget.rs @@ -23,6 +23,7 @@  pub mod button;  pub mod checkbox;  pub mod column; +pub mod container;  pub mod image;  pub mod radio;  pub mod row; @@ -31,8 +32,6 @@ pub mod slider;  pub mod text;  pub mod text_input; -mod container; -  #[doc(no_inline)]  pub use button::Button;  #[doc(no_inline)] @@ -65,12 +64,18 @@ use crate::{layout, Event, Hasher, Layout, Length, Point};  ///  /// [`Widget`]: trait.Widget.html  /// [`Element`]: ../struct.Element.html -pub trait Widget<Message, Renderer>: std::fmt::Debug +pub trait Widget<Message, Renderer>  where      Renderer: crate::Renderer,  { +    /// Returns the width of the [`Widget`]. +    /// +    /// [`Widget`]: trait.Widget.html      fn width(&self) -> Length; +    /// Returns the height of the [`Widget`]. +    /// +    /// [`Widget`]: trait.Widget.html      fn height(&self) -> Length;      /// Returns the [`Node`] of the [`Widget`]. @@ -78,9 +83,9 @@ where      /// This [`Node`] is used by the runtime to compute the [`Layout`] of the      /// user interface.      /// -    /// [`Node`]: ../struct.Node.html +    /// [`Node`]: ../layout/struct.Node.html      /// [`Widget`]: trait.Widget.html -    /// [`Layout`]: ../struct.Layout.html +    /// [`Layout`]: ../layout/struct.Layout.html      fn layout(          &self,          renderer: &Renderer, @@ -108,7 +113,7 @@ where      /// its value cannot affect the overall [`Layout`] of the user interface.      ///      /// [`Widget`]: trait.Widget.html -    /// [`Layout`]: ../struct.Layout.html +    /// [`Layout`]: ../layout/struct.Layout.html      /// [`Text`]: text/struct.Text.html      fn hash_layout(&self, state: &mut Hasher); @@ -120,12 +125,13 @@ where      ///   * the current cursor position      ///   * a mutable `Message` list, allowing the [`Widget`] to produce      ///   new messages based on user interaction. +    ///   * the `Renderer`      ///      /// By default, it does nothing.      ///      /// [`Event`]: ../enum.Event.html      /// [`Widget`]: trait.Widget.html -    /// [`Layout`]: ../struct.Layout.html +    /// [`Layout`]: ../layout/struct.Layout.html      fn on_event(          &mut self,          _event: Event, diff --git a/native/src/widget/button.rs b/native/src/widget/button.rs index 248aaaf9..023c4ee8 100644 --- a/native/src/widget/button.rs +++ b/native/src/widget/button.rs @@ -1,25 +1,138 @@  //! Allow your users to perform actions by pressing a button.  //! -//! A [`Button`] has some local [`State`] and a [`Class`]. +//! A [`Button`] has some local [`State`].  //!  //! [`Button`]: struct.Button.html  //! [`State`]: struct.State.html -//! [`Class`]: enum.Class.html - -use crate::input::{mouse, ButtonState}; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    input::{mouse, ButtonState}, +    layout, Background, Element, Event, Hasher, Layout, Length, Point, +    Rectangle, Widget, +};  use std::hash::Hash; -pub use iced_core::button::State; +/// A generic widget that produces a message when pressed. +/// +/// ``` +/// # use iced_native::{button, Text}; +/// # +/// # type Button<'a, Message> = +/// #     iced_native::Button<'a, Message, iced_native::renderer::Null>; +/// # +/// enum Message { +///     ButtonPressed, +/// } +/// +/// let mut state = button::State::new(); +/// let button = Button::new(&mut state, Text::new("Press me!")) +///     .on_press(Message::ButtonPressed); +/// ``` +#[allow(missing_debug_implementations)] +pub struct Button<'a, Message, Renderer> { +    state: &'a mut State, +    content: Element<'a, Message, Renderer>, +    on_press: Option<Message>, +    width: Length, +    min_width: u32, +    padding: u16, +    background: Option<Background>, +    border_radius: u16, +} + +impl<'a, Message, Renderer> Button<'a, Message, Renderer> { +    /// Creates a new [`Button`] with some local [`State`] and the given +    /// content. +    /// +    /// [`Button`]: struct.Button.html +    /// [`State`]: struct.State.html +    pub fn new<E>(state: &'a mut State, content: E) -> Self +    where +        E: Into<Element<'a, Message, Renderer>>, +    { +        Button { +            state, +            content: content.into(), +            on_press: None, +            width: Length::Shrink, +            min_width: 0, +            padding: 0, +            background: None, +            border_radius: 0, +        } +    } -pub type Button<'a, Message, Renderer> = -    iced_core::Button<'a, Message, Element<'a, Message, Renderer>>; +    /// Sets the width of the [`Button`]. +    /// +    /// [`Button`]: struct.Button.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the minimum width of the [`Button`]. +    /// +    /// [`Button`]: struct.Button.html +    pub fn min_width(mut self, min_width: u32) -> Self { +        self.min_width = min_width; +        self +    } + +    /// Sets the padding of the [`Button`]. +    /// +    /// [`Button`]: struct.Button.html +    pub fn padding(mut self, padding: u16) -> Self { +        self.padding = padding; +        self +    } + +    /// Sets the [`Background`] of the [`Button`]. +    /// +    /// [`Button`]: struct.Button.html +    /// [`Background`]: ../../struct.Background.html +    pub fn background(mut self, background: Background) -> Self { +        self.background = Some(background); +        self +    } + +    /// Sets the border radius of the [`Button`]. +    /// +    /// [`Button`]: struct.Button.html +    pub fn border_radius(mut self, border_radius: u16) -> Self { +        self.border_radius = border_radius; +        self +    } + +    /// Sets the message that will be produced when the [`Button`] is pressed. +    /// +    /// [`Button`]: struct.Button.html +    pub fn on_press(mut self, msg: Message) -> Self { +        self.on_press = Some(msg); +        self +    } +} + +/// The local state of a [`Button`]. +/// +/// [`Button`]: struct.Button.html +#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)] +pub struct State { +    is_pressed: bool, +} + +impl State { +    /// Creates a new [`State`]. +    /// +    /// [`State`]: struct.State.html +    pub fn new() -> State { +        State::default() +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer>      for Button<'a, Message, Renderer>  where      Renderer: self::Renderer, -    Message: Clone + std::fmt::Debug, +    Message: Clone,  {      fn width(&self) -> Length {          self.width @@ -34,7 +147,21 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let padding = f32::from(self.padding); +        let limits = limits +            .min_width(self.min_width) +            .width(self.width) +            .height(Length::Shrink) +            .pad(padding); + +        let mut content = self.content.layout(renderer, &limits); + +        content.bounds.x = padding; +        content.bounds.y = padding; + +        let size = limits.resolve(content.size()).pad(padding); + +        layout::Node::with_children(size, vec![content])      }      fn on_event( @@ -81,7 +208,20 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        let content = self.content.draw( +            renderer, +            layout.children().next().unwrap(), +            cursor_position, +        ); + +        renderer.draw( +            layout.bounds(), +            cursor_position, +            self.state.is_pressed, +            self.background, +            self.border_radius, +            content, +        )      }      fn hash_layout(&self, state: &mut Hasher) { @@ -98,24 +238,17 @@ where  /// [`Button`]: struct.Button.html  /// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer + Sized { -    /// Creates a [`Node`] for the provided [`Button`]. -    /// -    /// [`Node`]: ../../struct.Node.html -    /// [`Button`]: struct.Button.html -    fn layout<Message>( -        &self, -        button: &Button<'_, Message, Self>, -        limits: &layout::Limits, -    ) -> layout::Node; -      /// Draws a [`Button`].      ///      /// [`Button`]: struct.Button.html -    fn draw<Message>( +    fn draw(          &mut self, -        button: &Button<'_, Message, Self>, -        layout: Layout<'_>, +        bounds: Rectangle,          cursor_position: Point, +        is_pressed: bool, +        background: Option<Background>, +        border_radius: u16, +        content: Self::Output,      ) -> Self::Output;  } @@ -123,7 +256,7 @@ impl<'a, Message, Renderer> From<Button<'a, Message, Renderer>>      for Element<'a, Message, Renderer>  where      Renderer: 'static + self::Renderer, -    Message: 'static + Clone + std::fmt::Debug, +    Message: 'static + Clone,  {      fn from(          button: Button<'a, Message, Renderer>, diff --git a/native/src/widget/checkbox.rs b/native/src/widget/checkbox.rs index 4ca44e02..9563291c 100644 --- a/native/src/widget/checkbox.rs +++ b/native/src/widget/checkbox.rs @@ -1,14 +1,73 @@  //! Show toggle controls using checkboxes.  use std::hash::Hash; -use crate::input::{mouse, ButtonState}; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    input::{mouse, ButtonState}, +    layout, row, text, Align, Color, Element, Event, Font, Hasher, +    HorizontalAlignment, Layout, Length, Point, Rectangle, Row, Text, +    VerticalAlignment, Widget, +}; -pub use iced_core::Checkbox; +/// A box that can be checked. +/// +/// # Example +/// +/// ``` +/// # use iced_native::Checkbox; +/// # +/// pub enum Message { +///     CheckboxToggled(bool), +/// } +/// +/// let is_checked = true; +/// +/// Checkbox::new(is_checked, "Toggle me!", Message::CheckboxToggled); +/// ``` +/// +///  +#[allow(missing_debug_implementations)] +pub struct Checkbox<Message> { +    is_checked: bool, +    on_toggle: Box<dyn Fn(bool) -> Message>, +    label: String, +    label_color: Option<Color>, +} + +impl<Message> Checkbox<Message> { +    /// Creates a new [`Checkbox`]. +    /// +    /// It expects: +    ///   * a boolean describing whether the [`Checkbox`] is checked or not +    ///   * the label of the [`Checkbox`] +    ///   * a function that will be called when the [`Checkbox`] is toggled. It +    ///     will receive the new state of the [`Checkbox`] and must produce a +    ///     `Message`. +    /// +    /// [`Checkbox`]: struct.Checkbox.html +    pub fn new<F>(is_checked: bool, label: &str, f: F) -> Self +    where +        F: 'static + Fn(bool) -> Message, +    { +        Checkbox { +            is_checked, +            on_toggle: Box::new(f), +            label: String::from(label), +            label_color: None, +        } +    } + +    /// Sets the color of the label of the [`Checkbox`]. +    /// +    /// [`Checkbox`]: struct.Checkbox.html +    pub fn label_color<C: Into<Color>>(mut self, color: C) -> Self { +        self.label_color = Some(color.into()); +        self +    } +}  impl<Message, Renderer> Widget<Message, Renderer> for Checkbox<Message>  where -    Renderer: self::Renderer, +    Renderer: self::Renderer + text::Renderer + row::Renderer,  {      fn width(&self) -> Length {          Length::Fill @@ -23,7 +82,18 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let size = self::Renderer::default_size(renderer); + +        Row::<(), Renderer>::new() +            .spacing(15) +            .align_items(Align::Center) +            .push( +                Row::new() +                    .width(Length::Units(size as u16)) +                    .height(Length::Units(size as u16)), +            ) +            .push(Text::new(&self.label)) +            .layout(renderer, limits)      }      fn on_event( @@ -55,7 +125,33 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        let bounds = layout.bounds(); +        let mut children = layout.children(); + +        let checkbox_layout = children.next().unwrap(); +        let label_layout = children.next().unwrap(); +        let checkbox_bounds = checkbox_layout.bounds(); + +        let label = text::Renderer::draw( +            renderer, +            label_layout.bounds(), +            &self.label, +            text::Renderer::default_size(renderer), +            Font::Default, +            self.label_color, +            HorizontalAlignment::Left, +            VerticalAlignment::Center, +        ); + +        let is_mouse_over = bounds.contains(cursor_position); + +        self::Renderer::draw( +            renderer, +            checkbox_bounds, +            self.is_checked, +            is_mouse_over, +            label, +        )      }      fn hash_layout(&self, state: &mut Hasher) { @@ -71,37 +167,33 @@ where  /// [`Checkbox`]: struct.Checkbox.html  /// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer { -    /// Creates a [`Node`] for the provided [`Checkbox`]. +    /// Returns the default size of a [`Checkbox`].      /// -    /// [`Node`]: ../../struct.Node.html      /// [`Checkbox`]: struct.Checkbox.html -    fn layout<Message>( -        &self, -        checkbox: &Checkbox<Message>, -        limits: &layout::Limits, -    ) -> layout::Node; +    fn default_size(&self) -> u32;      /// Draws a [`Checkbox`].      ///      /// It receives: -    ///   * the current cursor position      ///   * the bounds of the [`Checkbox`] -    ///   * the bounds of the label of the [`Checkbox`] -    ///   * whether the [`Checkbox`] is checked or not +    ///   * whether the [`Checkbox`] is selected or not +    ///   * whether the mouse is over the [`Checkbox`] or not +    ///   * the drawn label of the [`Checkbox`]      ///      /// [`Checkbox`]: struct.Checkbox.html -    fn draw<Message>( +    fn draw(          &mut self, -        checkbox: &Checkbox<Message>, -        layout: Layout<'_>, -        cursor_position: Point, +        bounds: Rectangle, +        is_checked: bool, +        is_mouse_over: bool, +        label: Self::Output,      ) -> Self::Output;  }  impl<'a, Message, Renderer> From<Checkbox<Message>>      for Element<'a, Message, Renderer>  where -    Renderer: self::Renderer, +    Renderer: self::Renderer + text::Renderer + row::Renderer,      Message: 'static,  {      fn from(checkbox: Checkbox<Message>) -> Element<'a, Message, Renderer> { diff --git a/native/src/widget/column.rs b/native/src/widget/column.rs index 38dbcdc5..cdcf25af 100644 --- a/native/src/widget/column.rs +++ b/native/src/widget/column.rs @@ -1,10 +1,115 @@ +//! Distribute content vertically.  use std::hash::Hash; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    layout, Align, Element, Event, Hasher, Layout, Length, Point, Widget, +}; + +use std::u32;  /// A container that distributes its contents vertically. -pub type Column<'a, Message, Renderer> = -    iced_core::Column<Element<'a, Message, Renderer>>; +/// +/// A [`Column`] will try to fill the horizontal space of its container. +/// +/// [`Column`]: struct.Column.html +#[allow(missing_debug_implementations)] +pub struct Column<'a, Message, Renderer> { +    spacing: u16, +    padding: u16, +    width: Length, +    height: Length, +    max_width: u32, +    max_height: u32, +    align_items: Align, +    children: Vec<Element<'a, Message, Renderer>>, +} + +impl<'a, Message, Renderer> Column<'a, Message, Renderer> { +    /// Creates an empty [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn new() -> Self { +        Column { +            spacing: 0, +            padding: 0, +            width: Length::Fill, +            height: Length::Shrink, +            max_width: u32::MAX, +            max_height: u32::MAX, +            align_items: Align::Start, +            children: Vec::new(), +        } +    } + +    /// Sets the vertical spacing _between_ elements. +    /// +    /// Custom margins per element do not exist in Iced. You should use this +    /// method instead! While less flexible, it helps you keep spacing between +    /// elements consistent. +    pub fn spacing(mut self, units: u16) -> Self { +        self.spacing = units; +        self +    } + +    /// Sets the padding of the [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn padding(mut self, units: u16) -> Self { +        self.padding = units; +        self +    } + +    /// Sets the width of the [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the height of the [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } + +    /// Sets the maximum width of the [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn max_width(mut self, max_width: u32) -> Self { +        self.max_width = max_width; +        self +    } + +    /// Sets the maximum height of the [`Column`] in pixels. +    /// +    /// [`Column`]: struct.Column.html +    pub fn max_height(mut self, max_height: u32) -> Self { +        self.max_height = max_height; +        self +    } + +    /// Sets the horizontal alignment of the contents of the [`Column`] . +    /// +    /// [`Column`]: struct.Column.html +    pub fn align_items(mut self, align: Align) -> Self { +        self.align_items = align; +        self +    } + +    /// Adds an element to the [`Column`]. +    /// +    /// [`Column`]: struct.Column.html +    pub fn push<E>(mut self, child: E) -> Self +    where +        E: Into<Element<'a, Message, Renderer>>, +    { +        self.children.push(child.into()); +        self +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer>      for Column<'a, Message, Renderer> @@ -68,7 +173,7 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        renderer.draw(&self.children, layout, cursor_position)      }      fn hash_layout(&self, state: &mut Hasher) { @@ -86,10 +191,26 @@ where      }  } +/// The renderer of a [`Column`]. +/// +/// Your [renderer] will need to implement this trait before being +/// able to use a [`Column`] in your user interface. +/// +/// [`Column`]: struct.Column.html +/// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer + Sized { +    /// Draws a [`Column`]. +    /// +    /// It receives: +    /// - the children of the [`Column`] +    /// - the [`Layout`] of the [`Column`] and its children +    /// - the cursor position +    /// +    /// [`Column`]: struct.Row.html +    /// [`Layout`]: ../layout/struct.Layout.html      fn draw<Message>(          &mut self, -        row: &Column<'_, Message, Self>, +        content: &[Element<'_, Message, Self>],          layout: Layout<'_>,          cursor_position: Point,      ) -> Self::Output; diff --git a/native/src/widget/container.rs b/native/src/widget/container.rs index 5aed3121..7852eecf 100644 --- a/native/src/widget/container.rs +++ b/native/src/widget/container.rs @@ -1,10 +1,95 @@ +//! Decorate content and apply alignment.  use std::hash::Hash; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    layout, Align, Element, Event, Hasher, Layout, Length, Point, Widget, +}; -/// A container that distributes its contents vertically. -pub type Container<'a, Message, Renderer> = -    iced_core::Container<Element<'a, Message, Renderer>>; +use std::u32; + +/// An element decorating some content. +/// +/// It is normally used for alignment purposes. +#[allow(missing_debug_implementations)] +pub struct Container<'a, Message, Renderer> { +    width: Length, +    height: Length, +    max_width: u32, +    max_height: u32, +    horizontal_alignment: Align, +    vertical_alignment: Align, +    content: Element<'a, Message, Renderer>, +} + +impl<'a, Message, Renderer> Container<'a, Message, Renderer> { +    /// Creates an empty [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn new<T>(content: T) -> Self +    where +        T: Into<Element<'a, Message, Renderer>>, +    { +        Container { +            width: Length::Shrink, +            height: Length::Shrink, +            max_width: u32::MAX, +            max_height: u32::MAX, +            horizontal_alignment: Align::Start, +            vertical_alignment: Align::Start, +            content: content.into(), +        } +    } + +    /// Sets the width of the [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the height of the [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } + +    /// Sets the maximum width of the [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn max_width(mut self, max_width: u32) -> Self { +        self.max_width = max_width; +        self +    } + +    /// Sets the maximum height of the [`Container`] in pixels. +    /// +    /// [`Container`]: struct.Container.html +    pub fn max_height(mut self, max_height: u32) -> Self { +        self.max_height = max_height; +        self +    } + +    /// Centers the contents in the horizontal axis of the [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn center_x(mut self) -> Self { +        self.horizontal_alignment = Align::Center; + +        self +    } + +    /// Centers the contents in the vertical axis of the [`Container`]. +    /// +    /// [`Container`]: struct.Container.html +    pub fn center_y(mut self) -> Self { +        self.vertical_alignment = Align::Center; + +        self +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer>      for Container<'a, Message, Renderer> diff --git a/native/src/widget/image.rs b/native/src/widget/image.rs index 1e203077..4c588c9d 100644 --- a/native/src/widget/image.rs +++ b/native/src/widget/image.rs @@ -1,10 +1,55 @@  //! Display images in your user interface. -use crate::{layout, Element, Hasher, Layout, Length, Point, Widget}; +use crate::{layout, Element, Hasher, Layout, Length, Point, Size, Widget};  use std::hash::Hash; -pub use iced_core::Image; +/// A frame that displays an image while keeping aspect ratio. +/// +/// # Example +/// +/// ``` +/// # use iced_native::Image; +/// # +/// let image = Image::new("resources/ferris.png"); +/// ``` +/// +/// <img src="https://github.com/hecrj/iced/blob/9712b319bb7a32848001b96bd84977430f14b623/examples/resources/ferris.png?raw=true" width="300"> +#[derive(Debug)] +pub struct Image { +    path: String, +    width: Length, +    height: Length, +} + +impl Image { +    /// Creates a new [`Image`] with the given path. +    /// +    /// [`Image`]: struct.Image.html +    pub fn new<T: Into<String>>(path: T) -> Self { +        Image { +            path: path.into(), +            width: Length::Shrink, +            height: Length::Shrink, +        } +    } + +    /// Sets the width of the [`Image`] boundaries. +    /// +    /// [`Image`]: struct.Image.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the height of the [`Image`] boundaries. +    /// +    /// [`Image`]: struct.Image.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } +}  impl<Message, Renderer> Widget<Message, Renderer> for Image  where @@ -23,7 +68,26 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let (width, height) = renderer.dimensions(&self.path); + +        let aspect_ratio = width as f32 / height as f32; + +        // TODO: Deal with additional cases +        let (width, height) = match (self.width, self.height) { +            (Length::Units(width), _) => ( +                self.width, +                Length::Units((width as f32 / aspect_ratio).round() as u16), +            ), +            (_, _) => { +                (Length::Units(width as u16), Length::Units(height as u16)) +            } +        }; + +        let mut size = limits.width(width).height(height).resolve(Size::ZERO); + +        size.height = size.width / aspect_ratio; + +        layout::Node::new(size)      }      fn draw( @@ -32,7 +96,7 @@ where          layout: Layout<'_>,          _cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout) +        renderer.draw(&self.path, layout)      }      fn hash_layout(&self, state: &mut Hasher) { @@ -49,18 +113,15 @@ where  /// [`Image`]: struct.Image.html  /// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer { -    /// Creates a [`Node`] for the provided [`Image`]. -    /// -    /// You should probably keep the original aspect ratio, if possible. +    /// Returns the dimensions of an [`Image`] located on the given path.      /// -    /// [`Node`]: ../../struct.Node.html      /// [`Image`]: struct.Image.html -    fn layout(&self, image: &Image, limits: &layout::Limits) -> layout::Node; +    fn dimensions(&self, path: &str) -> (u32, u32);      /// Draws an [`Image`].      ///      /// [`Image`]: struct.Image.html -    fn draw(&mut self, image: &Image, layout: Layout<'_>) -> Self::Output; +    fn draw(&mut self, path: &str, layout: Layout<'_>) -> Self::Output;  }  impl<'a, Message, Renderer> From<Image> for Element<'a, Message, Renderer> diff --git a/native/src/widget/radio.rs b/native/src/widget/radio.rs index c3405d1f..a9d145db 100644 --- a/native/src/widget/radio.rs +++ b/native/src/widget/radio.rs @@ -1,15 +1,83 @@  //! Create choices using radio buttons. -use crate::input::{mouse, ButtonState}; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    input::{mouse, ButtonState}, +    layout, row, text, Align, Color, Element, Event, Font, Hasher, +    HorizontalAlignment, Layout, Length, Point, Rectangle, Row, Text, +    VerticalAlignment, Widget, +};  use std::hash::Hash; -pub use iced_core::Radio; +/// A circular button representing a choice. +/// +/// # Example +/// ``` +/// # use iced_native::Radio; +/// # +/// #[derive(Debug, Clone, Copy, PartialEq, Eq)] +/// pub enum Choice { +///     A, +///     B, +/// } +/// +/// #[derive(Debug, Clone, Copy)] +/// pub enum Message { +///     RadioSelected(Choice), +/// } +/// +/// let selected_choice = Some(Choice::A); +/// +/// Radio::new(Choice::A, "This is A", selected_choice, Message::RadioSelected); +/// +/// Radio::new(Choice::B, "This is B", selected_choice, Message::RadioSelected); +/// ``` +/// +///  +#[allow(missing_debug_implementations)] +pub struct Radio<Message> { +    is_selected: bool, +    on_click: Message, +    label: String, +    label_color: Option<Color>, +} + +impl<Message> Radio<Message> { +    /// Creates a new [`Radio`] button. +    /// +    /// It expects: +    ///   * the value related to the [`Radio`] button +    ///   * the label of the [`Radio`] button +    ///   * the current selected value +    ///   * a function that will be called when the [`Radio`] is selected. It +    ///   receives the value of the radio and must produce a `Message`. +    /// +    /// [`Radio`]: struct.Radio.html +    pub fn new<F, V>(value: V, label: &str, selected: Option<V>, f: F) -> Self +    where +        V: Eq + Copy, +        F: 'static + Fn(V) -> Message, +    { +        Radio { +            is_selected: Some(value) == selected, +            on_click: f(value), +            label: String::from(label), +            label_color: None, +        } +    } + +    /// Sets the `Color` of the label of the [`Radio`]. +    /// +    /// [`Radio`]: struct.Radio.html +    pub fn label_color<C: Into<Color>>(mut self, color: C) -> Self { +        self.label_color = Some(color.into()); +        self +    } +}  impl<Message, Renderer> Widget<Message, Renderer> for Radio<Message>  where -    Renderer: self::Renderer, -    Message: Clone + std::fmt::Debug, +    Renderer: self::Renderer + text::Renderer + row::Renderer, +    Message: Clone,  {      fn width(&self) -> Length {          Length::Fill @@ -24,7 +92,18 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let size = self::Renderer::default_size(renderer); + +        Row::<(), Renderer>::new() +            .spacing(15) +            .align_items(Align::Center) +            .push( +                Row::new() +                    .width(Length::Units(size as u16)) +                    .height(Length::Units(size as u16)), +            ) +            .push(Text::new(&self.label)) +            .layout(renderer, limits)      }      fn on_event( @@ -54,7 +133,33 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        let bounds = layout.bounds(); +        let mut children = layout.children(); + +        let radio_layout = children.next().unwrap(); +        let label_layout = children.next().unwrap(); +        let radio_bounds = radio_layout.bounds(); + +        let label = text::Renderer::draw( +            renderer, +            label_layout.bounds(), +            &self.label, +            text::Renderer::default_size(renderer), +            Font::Default, +            self.label_color, +            HorizontalAlignment::Left, +            VerticalAlignment::Center, +        ); + +        let is_mouse_over = bounds.contains(cursor_position); + +        self::Renderer::draw( +            renderer, +            radio_bounds, +            self.is_selected, +            is_mouse_over, +            label, +        )      }      fn hash_layout(&self, state: &mut Hasher) { @@ -70,40 +175,36 @@ where  /// [`Radio`]: struct.Radio.html  /// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer { -    /// Creates a [`Node`] for the provided [`Radio`]. +    /// Returns the default size of a [`Radio`] button.      /// -    /// [`Node`]: ../../struct.Node.html      /// [`Radio`]: struct.Radio.html -    fn layout<Message>( -        &self, -        radio: &Radio<Message>, -        limits: &layout::Limits, -    ) -> layout::Node; +    fn default_size(&self) -> u32;      /// Draws a [`Radio`] button.      ///      /// It receives: -    ///   * the current cursor position      ///   * the bounds of the [`Radio`] -    ///   * the bounds of the label of the [`Radio`]      ///   * whether the [`Radio`] is selected or not +    ///   * whether the mouse is over the [`Radio`] or not +    ///   * the drawn label of the [`Radio`]      ///      /// [`Radio`]: struct.Radio.html -    fn draw<Message>( +    fn draw(          &mut self, -        radio: &Radio<Message>, -        layout: Layout<'_>, -        cursor_position: Point, +        bounds: Rectangle, +        is_selected: bool, +        is_mouse_over: bool, +        label: Self::Output,      ) -> Self::Output;  }  impl<'a, Message, Renderer> From<Radio<Message>>      for Element<'a, Message, Renderer>  where -    Renderer: self::Renderer, -    Message: 'static + Clone + std::fmt::Debug, +    Renderer: self::Renderer + row::Renderer + text::Renderer, +    Message: 'static + Clone,  { -    fn from(checkbox: Radio<Message>) -> Element<'a, Message, Renderer> { -        Element::new(checkbox) +    fn from(radio: Radio<Message>) -> Element<'a, Message, Renderer> { +        Element::new(radio)      }  } diff --git a/native/src/widget/row.rs b/native/src/widget/row.rs index 3827fd9a..c854aff7 100644 --- a/native/src/widget/row.rs +++ b/native/src/widget/row.rs @@ -1,10 +1,116 @@ +//! Distribute content horizontally.  use std::hash::Hash; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    layout, Align, Element, Event, Hasher, Layout, Length, Point, Widget, +}; + +use std::u32;  /// A container that distributes its contents horizontally. -pub type Row<'a, Message, Renderer> = -    iced_core::Row<Element<'a, Message, Renderer>>; +/// +/// A [`Row`] will try to fill the horizontal space of its container. +/// +/// [`Row`]: struct.Row.html +#[allow(missing_debug_implementations)] +pub struct Row<'a, Message, Renderer> { +    spacing: u16, +    padding: u16, +    width: Length, +    height: Length, +    max_width: u32, +    max_height: u32, +    align_items: Align, +    children: Vec<Element<'a, Message, Renderer>>, +} + +impl<'a, Message, Renderer> Row<'a, Message, Renderer> { +    /// Creates an empty [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn new() -> Self { +        Row { +            spacing: 0, +            padding: 0, +            width: Length::Fill, +            height: Length::Shrink, +            max_width: u32::MAX, +            max_height: u32::MAX, +            align_items: Align::Start, +            children: Vec::new(), +        } +    } + +    /// Sets the horizontal spacing _between_ elements. +    /// +    /// Custom margins per element do not exist in Iced. You should use this +    /// method instead! While less flexible, it helps you keep spacing between +    /// elements consistent. +    pub fn spacing(mut self, units: u16) -> Self { +        self.spacing = units; +        self +    } + +    /// Sets the padding of the [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn padding(mut self, units: u16) -> Self { +        self.padding = units; +        self +    } + +    /// Sets the width of the [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the height of the [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } + +    /// Sets the maximum width of the [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn max_width(mut self, max_width: u32) -> Self { +        self.max_width = max_width; +        self +    } + +    /// Sets the maximum height of the [`Row`]. +    /// +    /// [`Row`]: struct.Row.html +    pub fn max_height(mut self, max_height: u32) -> Self { +        self.max_height = max_height; +        self +    } + +    /// Sets the vertical alignment of the contents of the [`Row`] . +    /// +    /// [`Row`]: struct.Row.html +    pub fn align_items(mut self, align: Align) -> Self { +        self.align_items = align; +        self +    } + +    /// Adds an [`Element`] to the [`Row`]. +    /// +    /// [`Element`]: ../struct.Element.html +    /// [`Row`]: struct.Row.html +    pub fn push<E>(mut self, child: E) -> Self +    where +        E: Into<Element<'a, Message, Renderer>>, +    { +        self.children.push(child.into()); +        self +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer>      for Row<'a, Message, Renderer> @@ -68,7 +174,7 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        renderer.draw(&self.children, layout, cursor_position)      }      fn hash_layout(&self, state: &mut Hasher) { @@ -87,10 +193,26 @@ where      }  } +/// The renderer of a [`Row`]. +/// +/// Your [renderer] will need to implement this trait before being +/// able to use a [`Row`] in your user interface. +/// +/// [`Row`]: struct.Row.html +/// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer + Sized { +    /// Draws a [`Row`]. +    /// +    /// It receives: +    /// - the children of the [`Row`] +    /// - the [`Layout`] of the [`Row`] and its children +    /// - the cursor position +    /// +    /// [`Row`]: struct.Row.html +    /// [`Layout`]: ../layout/struct.Layout.html      fn draw<Message>(          &mut self, -        row: &Row<'_, Message, Self>, +        children: &[Element<'_, Message, Self>],          layout: Layout<'_>,          cursor_position: Point,      ) -> Self::Output; diff --git a/native/src/widget/scrollable.rs b/native/src/widget/scrollable.rs index 3877f6fe..0d745756 100644 --- a/native/src/widget/scrollable.rs +++ b/native/src/widget/scrollable.rs @@ -1,20 +1,106 @@ +//! Navigate an endless amount of content with a scrollbar.  use crate::{      column,      input::{mouse, ButtonState}, -    layout, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, -    Widget, +    layout, Align, Column, Element, Event, Hasher, Layout, Length, Point, +    Rectangle, Size, Widget,  }; -pub use iced_core::scrollable::State; +use std::{f32, hash::Hash, u32}; -use std::f32; -use std::hash::Hash; +/// A widget that can vertically display an infinite amount of content with a +/// scrollbar. +#[allow(missing_debug_implementations)] +pub struct Scrollable<'a, Message, Renderer> { +    state: &'a mut State, +    height: Length, +    max_height: u32, +    content: Column<'a, Message, Renderer>, +} -/// A scrollable [`Column`]. -/// -/// [`Column`]: ../column/struct.Column.html -pub type Scrollable<'a, Message, Renderer> = -    iced_core::Scrollable<'a, Element<'a, Message, Renderer>>; +impl<'a, Message, Renderer> Scrollable<'a, Message, Renderer> { +    /// Creates a new [`Scrollable`] with the given [`State`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    /// [`State`]: struct.State.html +    pub fn new(state: &'a mut State) -> Self { +        Scrollable { +            state, +            height: Length::Shrink, +            max_height: u32::MAX, +            content: Column::new(), +        } +    } + +    /// Sets the vertical spacing _between_ elements. +    /// +    /// Custom margins per element do not exist in Iced. You should use this +    /// method instead! While less flexible, it helps you keep spacing between +    /// elements consistent. +    pub fn spacing(mut self, units: u16) -> Self { +        self.content = self.content.spacing(units); +        self +    } + +    /// Sets the padding of the [`Scrollable`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn padding(mut self, units: u16) -> Self { +        self.content = self.content.padding(units); +        self +    } + +    /// Sets the width of the [`Scrollable`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn width(mut self, width: Length) -> Self { +        self.content = self.content.width(width); +        self +    } + +    /// Sets the height of the [`Scrollable`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } + +    /// Sets the maximum width of the [`Scrollable`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn max_width(mut self, max_width: u32) -> Self { +        self.content = self.content.max_width(max_width); +        self +    } + +    /// Sets the maximum height of the [`Scrollable`] in pixels. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn max_height(mut self, max_height: u32) -> Self { +        self.max_height = max_height; +        self +    } + +    /// Sets the horizontal alignment of the contents of the [`Scrollable`] . +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn align_items(mut self, align_items: Align) -> Self { +        self.content = self.content.align_items(align_items); +        self +    } + +    /// Adds an element to the [`Scrollable`]. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    pub fn push<E>(mut self, child: E) -> Self +    where +        E: Into<Element<'a, Message, Renderer>>, +    { +        self.content = self.content.push(child); +        self +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer>      for Scrollable<'a, Message, Renderer> @@ -161,13 +247,35 @@ where      ) -> Renderer::Output {          let bounds = layout.bounds();          let content_layout = layout.children().next().unwrap(); +        let content_bounds = content_layout.bounds(); +        let offset = self.state.offset(bounds, content_bounds); + +        let is_mouse_over = bounds.contains(cursor_position); +        let is_mouse_over_scrollbar = renderer.is_mouse_over_scrollbar( +            bounds, +            content_bounds, +            cursor_position, +        ); + +        let content = { +            let cursor_position = if is_mouse_over && !is_mouse_over_scrollbar { +                Point::new(cursor_position.x, cursor_position.y + offset as f32) +            } else { +                Point::new(cursor_position.x, -1.0) +            }; + +            self.content.draw(renderer, content_layout, cursor_position) +        };          self::Renderer::draw(              renderer, -            &self, +            &self.state,              bounds, -            content_layout, -            cursor_position, +            content_layout.bounds(), +            is_mouse_over, +            is_mouse_over_scrollbar, +            offset, +            content,          )      } @@ -181,7 +289,92 @@ where      }  } +/// The local state of a [`Scrollable`]. +/// +/// [`Scrollable`]: struct.Scrollable.html +#[derive(Debug, Clone, Copy, Default)] +pub struct State { +    scrollbar_grabbed_at: Option<Point>, +    offset: u32, +} + +impl State { +    /// Creates a new [`State`] with the scrollbar located at the top. +    /// +    /// [`State`]: struct.State.html +    pub fn new() -> Self { +        State::default() +    } + +    /// Apply a scrolling offset to the current [`State`], given the bounds of +    /// the [`Scrollable`] and its contents. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    /// [`State`]: struct.State.html +    pub fn scroll( +        &mut self, +        delta_y: f32, +        bounds: Rectangle, +        content_bounds: Rectangle, +    ) { +        if bounds.height >= content_bounds.height { +            return; +        } + +        self.offset = (self.offset as i32 - delta_y.round() as i32) +            .max(0) +            .min((content_bounds.height - bounds.height) as i32) +            as u32; +    } + +    /// Moves the scroll position to a relative amount, given the bounds of +    /// the [`Scrollable`] and its contents. +    /// +    /// `0` represents scrollbar at the top, while `1` represents scrollbar at +    /// the bottom. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    /// [`State`]: struct.State.html +    pub fn scroll_to( +        &mut self, +        percentage: f32, +        bounds: Rectangle, +        content_bounds: Rectangle, +    ) { +        self.offset = ((content_bounds.height - bounds.height) * percentage) +            .max(0.0) as u32; +    } + +    /// Returns the current scrolling offset of the [`State`], given the bounds +    /// of the [`Scrollable`] and its contents. +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    /// [`State`]: struct.State.html +    pub fn offset(&self, bounds: Rectangle, content_bounds: Rectangle) -> u32 { +        let hidden_content = +            (content_bounds.height - bounds.height).max(0.0).round() as u32; + +        self.offset.min(hidden_content) +    } + +    /// Returns whether the scrollbar is currently grabbed or not. +    pub fn is_scrollbar_grabbed(&self) -> bool { +        self.scrollbar_grabbed_at.is_some() +    } +} + +/// The renderer of a [`Scrollable`]. +/// +/// Your [renderer] will need to implement this trait before being +/// able to use a [`Scrollable`] in your user interface. +/// +/// [`Scrollable`]: struct.Scrollable.html +/// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer + Sized { +    /// Returns whether the mouse is over the scrollbar given the bounds of +    /// the [`Scrollable`] and its contents. +    /// +    /// [`Scrollable`]: struct.Scrollable.html      fn is_mouse_over_scrollbar(          &self,          bounds: Rectangle, @@ -189,12 +382,27 @@ pub trait Renderer: crate::Renderer + Sized {          cursor_position: Point,      ) -> bool; -    fn draw<Message>( +    /// Draws the [`Scrollable`]. +    /// +    /// It receives: +    /// - the [`State`] of the [`Scrollable`] +    /// - the bounds of the [`Scrollable`] +    /// - whether the mouse is over the [`Scrollable`] or not +    /// - whether the mouse is over the scrollbar or not +    /// - the scrolling offset +    /// - the drawn content +    /// +    /// [`Scrollable`]: struct.Scrollable.html +    /// [`State`]: struct.State.html +    fn draw(          &mut self, -        scrollable: &Scrollable<'_, Message, Self>, +        scrollable: &State,          bounds: Rectangle, -        content_layout: Layout<'_>, -        cursor_position: Point, +        content_bounds: Rectangle, +        is_mouse_over: bool, +        is_mouse_over_scrollbar: bool, +        offset: u32, +        content: Self::Output,      ) -> Self::Output;  } diff --git a/native/src/widget/slider.rs b/native/src/widget/slider.rs index e4f5c7a6..f07ea7cd 100644 --- a/native/src/widget/slider.rs +++ b/native/src/widget/slider.rs @@ -4,12 +4,101 @@  //!  //! [`Slider`]: struct.Slider.html  //! [`State`]: struct.State.html -use std::hash::Hash; +use crate::{ +    input::{mouse, ButtonState}, +    layout, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size, +    Widget, +}; -use crate::input::{mouse, ButtonState}; -use crate::{layout, Element, Event, Hasher, Layout, Length, Point, Widget}; +use std::{hash::Hash, ops::RangeInclusive}; -pub use iced_core::slider::*; +/// An horizontal bar and a handle that selects a single value from a range of +/// values. +/// +/// A [`Slider`] will try to fill the horizontal space of its container. +/// +/// [`Slider`]: struct.Slider.html +/// +/// # Example +/// ``` +/// # use iced_native::{slider, Slider}; +/// # +/// pub enum Message { +///     SliderChanged(f32), +/// } +/// +/// let state = &mut slider::State::new(); +/// let value = 50.0; +/// +/// Slider::new(state, 0.0..=100.0, value, Message::SliderChanged); +/// ``` +/// +///  +#[allow(missing_debug_implementations)] +pub struct Slider<'a, Message> { +    state: &'a mut State, +    range: RangeInclusive<f32>, +    value: f32, +    on_change: Box<dyn Fn(f32) -> Message>, +    width: Length, +} + +impl<'a, Message> Slider<'a, Message> { +    /// Creates a new [`Slider`]. +    /// +    /// It expects: +    ///   * the local [`State`] of the [`Slider`] +    ///   * an inclusive range of possible values +    ///   * the current value of the [`Slider`] +    ///   * a function that will be called when the [`Slider`] is dragged. +    ///   It receives the new value of the [`Slider`] and must produce a +    ///   `Message`. +    /// +    /// [`Slider`]: struct.Slider.html +    /// [`State`]: struct.State.html +    pub fn new<F>( +        state: &'a mut State, +        range: RangeInclusive<f32>, +        value: f32, +        on_change: F, +    ) -> Self +    where +        F: 'static + Fn(f32) -> Message, +    { +        Slider { +            state, +            value: value.max(*range.start()).min(*range.end()), +            range, +            on_change: Box::new(on_change), +            width: Length::Fill, +        } +    } + +    /// Sets the width of the [`Slider`]. +    /// +    /// [`Slider`]: struct.Slider.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } +} + +/// The local state of a [`Slider`]. +/// +/// [`Slider`]: struct.Slider.html +#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)] +pub struct State { +    is_dragging: bool, +} + +impl State { +    /// Creates a new [`State`]. +    /// +    /// [`State`]: struct.State.html +    pub fn new() -> State { +        State::default() +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer> for Slider<'a, Message>  where @@ -28,7 +117,13 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let limits = limits +            .width(self.width) +            .height(Length::Units(renderer.height() as u16)); + +        let size = limits.resolve(Size::ZERO); + +        layout::Node::new(size)      }      fn on_event( @@ -85,7 +180,13 @@ where          layout: Layout<'_>,          cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout, cursor_position) +        renderer.draw( +            layout.bounds(), +            cursor_position, +            self.range.clone(), +            self.value, +            self.state.is_dragging, +        )      }      fn hash_layout(&self, state: &mut Hasher) { @@ -101,15 +202,10 @@ where  /// [`Slider`]: struct.Slider.html  /// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer { -    /// Creates a [`Node`] for the provided [`Radio`]. +    /// Returns the height of the [`Slider`].      /// -    /// [`Node`]: ../../struct.Node.html -    /// [`Radio`]: struct.Radio.html -    fn layout<Message>( -        &self, -        slider: &Slider<'_, Message>, -        limits: &layout::Limits, -    ) -> layout::Node; +    /// [`Slider`]: struct.Slider.html +    fn height(&self) -> u32;      /// Draws a [`Slider`].      /// @@ -123,11 +219,13 @@ pub trait Renderer: crate::Renderer {      /// [`Slider`]: struct.Slider.html      /// [`State`]: struct.State.html      /// [`Class`]: enum.Class.html -    fn draw<Message>( +    fn draw(          &mut self, -        slider: &Slider<'_, Message>, -        layout: Layout<'_>, +        bounds: Rectangle,          cursor_position: Point, +        range: RangeInclusive<f32>, +        value: f32, +        is_dragging: bool,      ) -> Self::Output;  } diff --git a/native/src/widget/text.rs b/native/src/widget/text.rs index 3097b8f3..cf9c9565 100644 --- a/native/src/widget/text.rs +++ b/native/src/widget/text.rs @@ -1,9 +1,116 @@  //! Write some text for your users to read. -use crate::{layout, Element, Hasher, Layout, Length, Point, Widget}; +use crate::{ +    layout, Color, Element, Font, Hasher, HorizontalAlignment, Layout, Length, +    Point, Rectangle, Size, VerticalAlignment, Widget, +};  use std::hash::Hash; -pub use iced_core::text::*; +/// A paragraph of text. +/// +/// # Example +/// +/// ``` +/// # use iced_native::Text; +/// # +/// Text::new("I <3 iced!") +///     .color([0.0, 0.0, 1.0]) +///     .size(40); +/// ``` +/// +///  +#[derive(Debug, Clone)] +pub struct Text { +    content: String, +    size: Option<u16>, +    color: Option<Color>, +    font: Font, +    width: Length, +    height: Length, +    horizontal_alignment: HorizontalAlignment, +    vertical_alignment: VerticalAlignment, +} + +impl Text { +    /// Create a new fragment of [`Text`] with the given contents. +    /// +    /// [`Text`]: struct.Text.html +    pub fn new<T: Into<String>>(label: T) -> Self { +        Text { +            content: label.into(), +            size: None, +            color: None, +            font: Font::Default, +            width: Length::Fill, +            height: Length::Shrink, +            horizontal_alignment: HorizontalAlignment::Left, +            vertical_alignment: VerticalAlignment::Top, +        } +    } + +    /// Sets the size of the [`Text`]. +    /// +    /// [`Text`]: struct.Text.html +    pub fn size(mut self, size: u16) -> Self { +        self.size = Some(size); +        self +    } + +    /// Sets the [`Color`] of the [`Text`]. +    /// +    /// [`Text`]: struct.Text.html +    /// [`Color`]: ../../struct.Color.html +    pub fn color<C: Into<Color>>(mut self, color: C) -> Self { +        self.color = Some(color.into()); +        self +    } + +    /// Sets the [`Font`] of the [`Text`]. +    /// +    /// [`Text`]: struct.Text.html +    /// [`Font`]: ../../struct.Font.html +    pub fn font(mut self, font: Font) -> Self { +        self.font = font; +        self +    } + +    /// Sets the width of the [`Text`] boundaries. +    /// +    /// [`Text`]: struct.Text.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the height of the [`Text`] boundaries. +    /// +    /// [`Text`]: struct.Text.html +    pub fn height(mut self, height: Length) -> Self { +        self.height = height; +        self +    } + +    /// Sets the [`HorizontalAlignment`] of the [`Text`]. +    /// +    /// [`Text`]: struct.Text.html +    /// [`HorizontalAlignment`]: enum.HorizontalAlignment.html +    pub fn horizontal_alignment( +        mut self, +        alignment: HorizontalAlignment, +    ) -> Self { +        self.horizontal_alignment = alignment; +        self +    } + +    /// Sets the [`VerticalAlignment`] of the [`Text`]. +    /// +    /// [`Text`]: struct.Text.html +    /// [`VerticalAlignment`]: enum.VerticalAlignment.html +    pub fn vertical_alignment(mut self, alignment: VerticalAlignment) -> Self { +        self.vertical_alignment = alignment; +        self +    } +}  impl<Message, Renderer> Widget<Message, Renderer> for Text  where @@ -22,7 +129,18 @@ where          renderer: &Renderer,          limits: &layout::Limits,      ) -> layout::Node { -        renderer.layout(&self, limits) +        let limits = limits.width(self.width).height(self.height); + +        let size = self.size.unwrap_or(renderer.default_size()); + +        let bounds = limits.max(); + +        let (width, height) = +            renderer.measure(&self.content, size, self.font, bounds); + +        let size = limits.resolve(Size::new(width, height)); + +        layout::Node::new(size)      }      fn draw( @@ -31,7 +149,15 @@ where          layout: Layout<'_>,          _cursor_position: Point,      ) -> Renderer::Output { -        renderer.draw(&self, layout) +        renderer.draw( +            layout.bounds(), +            &self.content, +            self.size.unwrap_or(renderer.default_size()), +            self.font, +            self.color, +            self.horizontal_alignment, +            self.vertical_alignment, +        )      }      fn hash_layout(&self, state: &mut Hasher) { @@ -51,17 +177,22 @@ where  /// [renderer]: ../../renderer/index.html  /// [`UserInterface`]: ../../struct.UserInterface.html  pub trait Renderer: crate::Renderer { -    /// Creates a [`Node`] with the given [`Style`] for the provided [`Text`] -    /// contents and size. +    /// Returns the default size of the [`Text`].      /// -    /// You should probably use [`Node::with_measure`] to allow [`Text`] to -    /// adapt to the dimensions of its container. +    /// [`Text`]: struct.Text.html +    fn default_size(&self) -> u16; + +    /// Measures the [`Text`] in the given bounds and returns the minimum +    /// boundaries that can fit the contents.      /// -    /// [`Node`]: ../../struct.Node.html -    /// [`Style`]: ../../struct.Style.html      /// [`Text`]: struct.Text.html -    /// [`Node::with_measure`]: ../../struct.Node.html#method.with_measure -    fn layout(&self, text: &Text, limits: &layout::Limits) -> layout::Node; +    fn measure( +        &self, +        content: &str, +        size: u16, +        font: Font, +        bounds: Size, +    ) -> (f32, f32);      /// Draws a [`Text`] fragment.      /// @@ -76,7 +207,16 @@ pub trait Renderer: crate::Renderer {      /// [`Text`]: struct.Text.html      /// [`HorizontalAlignment`]: enum.HorizontalAlignment.html      /// [`VerticalAlignment`]: enum.VerticalAlignment.html -    fn draw(&mut self, text: &Text, layout: Layout<'_>) -> Self::Output; +    fn draw( +        &mut self, +        bounds: Rectangle, +        content: &str, +        size: u16, +        font: Font, +        color: Option<Color>, +        horizontal_alignment: HorizontalAlignment, +        vertical_alignment: VerticalAlignment, +    ) -> Self::Output;  }  impl<'a, Message, Renderer> From<Text> for Element<'a, Message, Renderer> diff --git a/native/src/widget/text_input.rs b/native/src/widget/text_input.rs index d54cf82c..f97ed424 100644 --- a/native/src/widget/text_input.rs +++ b/native/src/widget/text_input.rs @@ -1,10 +1,125 @@ +//! Display fields that can be filled with text. +//! +//! A [`TextInput`] has some local [`State`]. +//! +//! [`TextInput`]: struct.TextInput.html +//! [`State`]: struct.State.html  use crate::{      input::{keyboard, mouse, ButtonState},      layout, Element, Event, Hasher, Layout, Length, Point, Rectangle, Size,      Widget,  }; -pub use iced_core::{text_input::State, TextInput}; +/// A field that can be filled with text. +/// +/// # Example +/// ``` +/// # use iced_native::{text_input, TextInput}; +/// # +/// #[derive(Debug, Clone)] +/// enum Message { +///     TextInputChanged(String), +/// } +/// +/// let mut state = text_input::State::new(); +/// let value = "Some text"; +/// +/// let input = TextInput::new( +///     &mut state, +///     "This is the placeholder...", +///     value, +///     Message::TextInputChanged, +/// ) +/// .padding(10); +/// ``` +///  +#[allow(missing_debug_implementations)] +pub struct TextInput<'a, Message> { +    state: &'a mut State, +    placeholder: String, +    value: Value, +    width: Length, +    max_width: Length, +    padding: u16, +    size: Option<u16>, +    on_change: Box<dyn Fn(String) -> Message>, +    on_submit: Option<Message>, +} + +impl<'a, Message> TextInput<'a, Message> { +    /// Creates a new [`TextInput`]. +    /// +    /// It expects: +    /// - some [`State`] +    /// - a placeholder +    /// - the current value +    /// - a function that produces a message when the [`TextInput`] changes +    /// +    /// [`TextInput`]: struct.TextInput.html +    /// [`State`]: struct.State.html +    pub fn new<F>( +        state: &'a mut State, +        placeholder: &str, +        value: &str, +        on_change: F, +    ) -> Self +    where +        F: 'static + Fn(String) -> Message, +    { +        Self { +            state, +            placeholder: String::from(placeholder), +            value: Value::new(value), +            width: Length::Fill, +            max_width: Length::Shrink, +            padding: 0, +            size: None, +            on_change: Box::new(on_change), +            on_submit: None, +        } +    } + +    /// Sets the width of the [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn width(mut self, width: Length) -> Self { +        self.width = width; +        self +    } + +    /// Sets the maximum width of the [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn max_width(mut self, max_width: Length) -> Self { +        self.max_width = max_width; +        self +    } + +    /// Sets the padding of the [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn padding(mut self, units: u16) -> Self { +        self.padding = units; +        self +    } + +    /// Sets the text size of the [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn size(mut self, size: u16) -> Self { +        self.size = Some(size); +        self +    } + +    /// Sets the message that should be produced when the [`TextInput`] is +    /// focused and the enter key is pressed. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn on_submit(mut self, message: Message) -> Self { +        self.on_submit = Some(message); +        self +    } +}  impl<'a, Message, Renderer> Widget<Message, Renderer> for TextInput<'a, Message>  where @@ -124,12 +239,19 @@ where          let bounds = layout.bounds();          let text_bounds = layout.children().next().unwrap().bounds(); -        renderer.draw(&self, bounds, text_bounds, cursor_position) +        renderer.draw( +            bounds, +            text_bounds, +            cursor_position, +            self.size.unwrap_or(renderer.default_size()), +            &self.placeholder, +            &self.value, +            &self.state, +        )      }      fn hash_layout(&self, state: &mut Hasher) { -        use std::any::TypeId; -        use std::hash::Hash; +        use std::{any::TypeId, hash::Hash};          TypeId::of::<TextInput<'static, ()>>().hash(state); @@ -140,15 +262,41 @@ where      }  } +/// The renderer of a [`TextInput`]. +/// +/// Your [renderer] will need to implement this trait before being +/// able to use a [`TextInput`] in your user interface. +/// +/// [`TextInput`]: struct.TextInput.html +/// [renderer]: ../../renderer/index.html  pub trait Renderer: crate::Renderer + Sized { +    /// Returns the default size of the text of the [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html      fn default_size(&self) -> u16; -    fn draw<Message>( +    /// Draws a [`TextInput`]. +    /// +    /// It receives: +    /// - its bounds of the [`TextInput`] +    /// - the bounds of the text (i.e. the current value) +    /// - the cursor position +    /// - the placeholder to show when the value is empty +    /// - the current [`Value`] +    /// - the current [`State`] +    /// +    /// [`TextInput`]: struct.TextInput.html +    /// [`Value`]: struct.Value.html +    /// [`State`]: struct.State.html +    fn draw(          &mut self, -        text_input: &TextInput<'_, Message>,          bounds: Rectangle,          text_bounds: Rectangle,          cursor_position: Point, +        size: u16, +        placeholder: &str, +        value: &Value, +        state: &State,      ) -> Self::Output;  } @@ -164,3 +312,128 @@ where          Element::new(text_input)      }  } + +/// The state of a [`TextInput`]. +/// +/// [`TextInput`]: struct.TextInput.html +#[derive(Debug, Default, Clone)] +pub struct State { +    is_focused: bool, +    cursor_position: usize, +} + +impl State { +    /// Creates a new [`State`], representing an unfocused [`TextInput`]. +    /// +    /// [`State`]: struct.State.html +    pub fn new() -> Self { +        Self::default() +    } + +    /// Creates a new [`State`], representing a focused [`TextInput`]. +    /// +    /// [`State`]: struct.State.html +    pub fn focused() -> Self { +        use std::usize; + +        Self { +            is_focused: true, +            cursor_position: usize::MAX, +        } +    } + +    /// Returns whether the [`TextInput`] is currently focused or not. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn is_focused(&self) -> bool { +        self.is_focused +    } + +    /// Returns the cursor position of a [`TextInput`]. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub fn cursor_position(&self, value: &Value) -> usize { +        self.cursor_position.min(value.len()) +    } + +    /// Moves the cursor of a [`TextInput`] to the right. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub(crate) fn move_cursor_right(&mut self, value: &Value) { +        let current = self.cursor_position(value); + +        if current < value.len() { +            self.cursor_position = current + 1; +        } +    } + +    /// Moves the cursor of a [`TextInput`] to the left. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub(crate) fn move_cursor_left(&mut self, value: &Value) { +        let current = self.cursor_position(value); + +        if current > 0 { +            self.cursor_position = current - 1; +        } +    } + +    /// Moves the cursor of a [`TextInput`] to the end. +    /// +    /// [`TextInput`]: struct.TextInput.html +    pub(crate) fn move_cursor_to_end(&mut self, value: &Value) { +        self.cursor_position = value.len(); +    } +} + +/// The value of a [`TextInput`]. +/// +/// [`TextInput`]: struct.TextInput.html +// TODO: Use `unicode-segmentation` +#[derive(Debug)] +pub struct Value(Vec<char>); + +impl Value { +    /// Creates a new [`Value`] from a string slice. +    /// +    /// [`Value`]: struct.Value.html +    pub fn new(string: &str) -> Self { +        Self(string.chars().collect()) +    } + +    /// Returns the total amount of `char` in the [`Value`]. +    /// +    /// [`Value`]: struct.Value.html +    pub fn len(&self) -> usize { +        self.0.len() +    } + +    /// Returns a new [`Value`] containing the `char` until the given `index`. +    /// +    /// [`Value`]: struct.Value.html +    pub fn until(&self, index: usize) -> Self { +        Self(self.0[..index.min(self.len())].to_vec()) +    } + +    /// Converts the [`Value`] into a `String`. +    /// +    /// [`Value`]: struct.Value.html +    pub fn to_string(&self) -> String { +        use std::iter::FromIterator; +        String::from_iter(self.0.iter()) +    } + +    /// Inserts a new `char` at the given `index`. +    /// +    /// [`Value`]: struct.Value.html +    pub fn insert(&mut self, index: usize, c: char) { +        self.0.insert(index, c); +    } + +    /// Removes the `char` at the given `index`. +    /// +    /// [`Value`]: struct.Value.html +    pub fn remove(&mut self, index: usize) { +        let _ = self.0.remove(index); +    } +} | 
