diff options
| author | 2019-11-05 03:43:15 +0100 | |
|---|---|---|
| committer | 2019-11-05 03:43:15 +0100 | |
| commit | da2717c74dbe3e1123ff41de345a409c1afc2f18 (patch) | |
| tree | f8e5615166a5d5fa820a4d2acd9162e3a542b199 /wgpu/src/renderer | |
| parent | 0ea911ae36bbde8c288f7ae1ba8a0049b696d7c4 (diff) | |
| parent | a2161586dab6837d8c641b6f93ad476f861d8580 (diff) | |
| download | iced-da2717c74dbe3e1123ff41de345a409c1afc2f18.tar.gz iced-da2717c74dbe3e1123ff41de345a409c1afc2f18.tar.bz2 iced-da2717c74dbe3e1123ff41de345a409c1afc2f18.zip | |
Merge pull request #37 from hecrj/feature/text-input
Text input widget
Diffstat (limited to 'wgpu/src/renderer')
| -rw-r--r-- | wgpu/src/renderer/checkbox.rs | 1 | ||||
| -rw-r--r-- | wgpu/src/renderer/scrollable.rs | 16 | ||||
| -rw-r--r-- | wgpu/src/renderer/text.rs | 2 | ||||
| -rw-r--r-- | wgpu/src/renderer/text_input.rs | 171 | 
4 files changed, 179 insertions, 11 deletions
| diff --git a/wgpu/src/renderer/checkbox.rs b/wgpu/src/renderer/checkbox.rs index fd3f08b1..ea7a4c0b 100644 --- a/wgpu/src/renderer/checkbox.rs +++ b/wgpu/src/renderer/checkbox.rs @@ -10,6 +10,7 @@ const SIZE: f32 = 28.0;  impl checkbox::Renderer for Renderer {      fn node<Message>(&self, checkbox: &Checkbox<Message>) -> Node {          Row::<(), Self>::new() +            .width(Length::Fill)              .spacing(15)              .align_items(Align::Center)              .push( diff --git a/wgpu/src/renderer/scrollable.rs b/wgpu/src/renderer/scrollable.rs index e812a7e1..360759a5 100644 --- a/wgpu/src/renderer/scrollable.rs +++ b/wgpu/src/renderer/scrollable.rs @@ -1,7 +1,7 @@  use crate::{Primitive, Renderer};  use iced_native::{      scrollable, Background, Color, Layout, MouseCursor, Point, Rectangle, -    Scrollable, Widget, +    Scrollable, Vector, Widget,  };  const SCROLLBAR_WIDTH: u16 = 10; @@ -56,9 +56,9 @@ impl scrollable::Renderer for Renderer {          let (content, mouse_cursor) =              scrollable.content.draw(self, content, cursor_position); -        let primitive = Primitive::Clip { +        let clip = Primitive::Clip {              bounds, -            offset, +            offset: Vector::new(0, offset),              content: Box::new(content),          }; @@ -107,19 +107,15 @@ impl scrollable::Renderer for Renderer {                      };                      Primitive::Group { -                        primitives: vec![ -                            primitive, -                            scrollbar_background, -                            scrollbar, -                        ], +                        primitives: vec![clip, scrollbar_background, scrollbar],                      }                  } else {                      Primitive::Group { -                        primitives: vec![primitive, scrollbar], +                        primitives: vec![clip, scrollbar],                      }                  }              } else { -                primitive +                clip              },              if is_mouse_over_scrollbar                  || scrollable.state.is_scrollbar_grabbed() diff --git a/wgpu/src/renderer/text.rs b/wgpu/src/renderer/text.rs index 8fbade4e..82f75f09 100644 --- a/wgpu/src/renderer/text.rs +++ b/wgpu/src/renderer/text.rs @@ -47,7 +47,7 @@ impl text::Renderer for Renderer {                  let (width, height) = if let Some(bounds) =                      glyph_brush.borrow_mut().glyph_bounds(&text)                  { -                    (bounds.width(), bounds.height()) +                    (bounds.width().ceil(), bounds.height().ceil())                  } else {                      (0.0, 0.0)                  }; diff --git a/wgpu/src/renderer/text_input.rs b/wgpu/src/renderer/text_input.rs new file mode 100644 index 00000000..cff8bf23 --- /dev/null +++ b/wgpu/src/renderer/text_input.rs @@ -0,0 +1,171 @@ +use crate::{Primitive, Renderer}; + +use iced_native::{ +    text::HorizontalAlignment, text::VerticalAlignment, text_input, Background, +    Color, MouseCursor, Point, Rectangle, TextInput, Vector, +}; +use std::f32; + +impl text_input::Renderer for Renderer { +    fn default_size(&self) -> u16 { +        // TODO: Make this configurable +        20 +    } + +    fn draw<Message>( +        &mut self, +        text_input: &TextInput<Message>, +        bounds: Rectangle, +        text_bounds: Rectangle, +        cursor_position: Point, +    ) -> Self::Output { +        let is_mouse_over = bounds.contains(cursor_position); + +        let border = Primitive::Quad { +            bounds, +            background: Background::Color( +                if is_mouse_over || text_input.state.is_focused { +                    Color { +                        r: 0.5, +                        g: 0.5, +                        b: 0.5, +                        a: 1.0, +                    } +                } else { +                    Color { +                        r: 0.7, +                        g: 0.7, +                        b: 0.7, +                        a: 1.0, +                    } +                }, +            ), +            border_radius: 5, +        }; + +        let input = Primitive::Quad { +            bounds: Rectangle { +                x: bounds.x + 1.0, +                y: bounds.y + 1.0, +                width: bounds.width - 2.0, +                height: bounds.height - 2.0, +            }, +            background: Background::Color(Color::WHITE), +            border_radius: 5, +        }; + +        let size = f32::from(text_input.size.unwrap_or(self.default_size())); +        let text = text_input.value.to_string(); + +        let value = Primitive::Text { +            content: if text.is_empty() { +                text_input.placeholder.clone() +            } else { +                text.clone() +            }, +            color: if text.is_empty() { +                Color { +                    r: 0.7, +                    g: 0.7, +                    b: 0.7, +                    a: 1.0, +                } +            } else { +                Color { +                    r: 0.3, +                    g: 0.3, +                    b: 0.3, +                    a: 1.0, +                } +            }, +            bounds: Rectangle { +                width: f32::INFINITY, +                ..text_bounds +            }, +            size, +            horizontal_alignment: HorizontalAlignment::Left, +            vertical_alignment: VerticalAlignment::Center, +        }; + +        let (contents_primitive, offset) = if text_input.state.is_focused { +            use wgpu_glyph::{GlyphCruncher, Scale, Section}; + +            let text_before_cursor = &text_input +                .value +                .until(text_input.state.cursor_position(&text_input.value)) +                .to_string(); + +            let mut text_value_width = self +                .glyph_brush +                .borrow_mut() +                .glyph_bounds(Section { +                    text: text_before_cursor, +                    bounds: (f32::INFINITY, text_bounds.height), +                    scale: Scale { x: size, y: size }, +                    ..Default::default() +                }) +                .map(|bounds| bounds.width().round()) +                .unwrap_or(0.0); + +            let spaces_at_the_end = +                text_before_cursor.len() - text_before_cursor.trim_end().len(); + +            if spaces_at_the_end > 0 { +                let space_width = { +                    let glyph_brush = self.glyph_brush.borrow(); + +                    // TODO: Select appropriate font +                    let font = &glyph_brush.fonts()[0]; + +                    font.glyph(' ') +                        .scaled(Scale { x: size, y: size }) +                        .h_metrics() +                        .advance_width +                }; + +                text_value_width += spaces_at_the_end as f32 * space_width; +            } + +            let cursor = Primitive::Quad { +                bounds: Rectangle { +                    x: text_bounds.x + text_value_width, +                    y: text_bounds.y, +                    width: 1.0, +                    height: text_bounds.height, +                }, +                background: Background::Color(Color::BLACK), +                border_radius: 0, +            }; + +            ( +                Primitive::Group { +                    primitives: vec![value, cursor], +                }, +                Vector::new( +                    ((text_value_width + 5.0) - text_bounds.width).max(0.0) +                        as u32, +                    0, +                ), +            ) +        } else { +            (value, Vector::new(0, 0)) +        }; + +        let contents = Primitive::Clip { +            bounds: text_bounds, +            offset, +            content: Box::new(contents_primitive), +        }; + +        ( +            Primitive::Group { +                primitives: vec![border, input, contents], +            }, +            if is_mouse_over { +                MouseCursor::Text +            } else { +                MouseCursor::OutOfBounds +            }, +        ) +    } +} | 
