diff options
| author | 2020-05-05 00:05:47 +0200 | |
|---|---|---|
| committer | 2020-05-05 00:05:47 +0200 | |
| commit | 7dc02a5e16a3143b7c3ba9270207e3ebda71d567 (patch) | |
| tree | dd727f138641fbda008af8e7827369cc99420749 /examples/bezier_tool | |
| parent | 27aad74a32fd8ac2b12f9d32df8a3b61a3175457 (diff) | |
| parent | 93c6be5eef577f0778b5787dac37351c035ed471 (diff) | |
| download | iced-7dc02a5e16a3143b7c3ba9270207e3ebda71d567.tar.gz iced-7dc02a5e16a3143b7c3ba9270207e3ebda71d567.tar.bz2 iced-7dc02a5e16a3143b7c3ba9270207e3ebda71d567.zip | |
Merge pull request #325 from hecrj/feature/canvas-interaction
Canvas interactivity and Game of Life example
Diffstat (limited to 'examples/bezier_tool')
| -rw-r--r-- | examples/bezier_tool/Cargo.toml | 5 | ||||
| -rw-r--r-- | examples/bezier_tool/README.md | 3 | ||||
| -rw-r--r-- | examples/bezier_tool/src/main.rs | 465 | 
3 files changed, 170 insertions, 303 deletions
| diff --git a/examples/bezier_tool/Cargo.toml b/examples/bezier_tool/Cargo.toml index b13a0aa5..a88975a7 100644 --- a/examples/bezier_tool/Cargo.toml +++ b/examples/bezier_tool/Cargo.toml @@ -6,7 +6,4 @@ edition = "2018"  publish = false  [dependencies] -iced = { path = "../.." } -iced_native = { path = "../../native" } -iced_wgpu = { path = "../../wgpu" } -lyon = "0.15" +iced = { path = "../..", features = ["canvas"] } diff --git a/examples/bezier_tool/README.md b/examples/bezier_tool/README.md index 933f2120..ebbb12cc 100644 --- a/examples/bezier_tool/README.md +++ b/examples/bezier_tool/README.md @@ -1,6 +1,6 @@  ## Bézier tool -A Paint-like tool for drawing Bézier curves using [`lyon`]. +A Paint-like tool for drawing Bézier curves using the `Canvas` widget.  The __[`main`]__ file contains all the code of the example. @@ -16,4 +16,3 @@ cargo run --package bezier_tool  ```  [`main`]: src/main.rs -[`lyon`]: https://github.com/nical/lyon diff --git a/examples/bezier_tool/src/main.rs b/examples/bezier_tool/src/main.rs index fcb7733c..fe41e1b2 100644 --- a/examples/bezier_tool/src/main.rs +++ b/examples/bezier_tool/src/main.rs @@ -1,291 +1,6 @@ -//! This example showcases a simple native custom widget that renders arbitrary -//! path with `lyon`. -mod bezier { -    // For now, to implement a custom native widget you will need to add -    // `iced_native` and `iced_wgpu` to your dependencies. -    // -    // Then, you simply need to define your widget type and implement the -    // `iced_native::Widget` trait with the `iced_wgpu::Renderer`. -    // -    // Of course, you can choose to make the implementation renderer-agnostic, -    // if you wish to, by creating your own `Renderer` trait, which could be -    // implemented by `iced_wgpu` and other renderers. -    use iced_native::{ -        input, layout, Clipboard, Color, Element, Event, Font, Hasher, -        HorizontalAlignment, Layout, Length, MouseCursor, Point, Rectangle, -        Size, Vector, VerticalAlignment, Widget, -    }; -    use iced_wgpu::{ -        triangle::{Mesh2D, Vertex2D}, -        Defaults, Primitive, Renderer, -    }; -    use lyon::tessellation::{ -        basic_shapes, BuffersBuilder, StrokeAttributes, StrokeOptions, -        StrokeTessellator, VertexBuffers, -    }; - -    pub struct Bezier<'a, Message> { -        state: &'a mut State, -        curves: &'a [Curve], -        // [from, to, ctrl] -        on_click: Box<dyn Fn(Curve) -> Message>, -    } - -    #[derive(Debug, Clone, Copy)] -    pub struct Curve { -        from: Point, -        to: Point, -        control: Point, -    } - -    #[derive(Default)] -    pub struct State { -        pending: Option<Pending>, -    } - -    enum Pending { -        One { from: Point }, -        Two { from: Point, to: Point }, -    } - -    impl<'a, Message> Bezier<'a, Message> { -        pub fn new<F>( -            state: &'a mut State, -            curves: &'a [Curve], -            on_click: F, -        ) -> Self -        where -            F: 'static + Fn(Curve) -> Message, -        { -            Self { -                state, -                curves, -                on_click: Box::new(on_click), -            } -        } -    } - -    impl<'a, Message> Widget<Message, Renderer> for Bezier<'a, Message> { -        fn width(&self) -> Length { -            Length::Fill -        } - -        fn height(&self) -> Length { -            Length::Fill -        } - -        fn layout( -            &self, -            _renderer: &Renderer, -            limits: &layout::Limits, -        ) -> layout::Node { -            let size = limits -                .height(Length::Fill) -                .width(Length::Fill) -                .resolve(Size::ZERO); -            layout::Node::new(size) -        } - -        fn draw( -            &self, -            _renderer: &mut Renderer, -            defaults: &Defaults, -            layout: Layout<'_>, -            cursor_position: Point, -        ) -> (Primitive, MouseCursor) { -            let mut buffer: VertexBuffers<Vertex2D, u32> = VertexBuffers::new(); -            let mut path_builder = lyon::path::Path::builder(); - -            let bounds = layout.bounds(); - -            // Draw rectangle border with lyon. -            basic_shapes::stroke_rectangle( -                &lyon::math::Rect::new( -                    lyon::math::Point::new(0.5, 0.5), -                    lyon::math::Size::new( -                        bounds.width - 1.0, -                        bounds.height - 1.0, -                    ), -                ), -                &StrokeOptions::default().with_line_width(1.0), -                &mut BuffersBuilder::new( -                    &mut buffer, -                    |pos: lyon::math::Point, _: StrokeAttributes| Vertex2D { -                        position: pos.to_array(), -                        color: [0.0, 0.0, 0.0, 1.0], -                    }, -                ), -            ) -            .unwrap(); - -            for curve in self.curves { -                path_builder.move_to(lyon::math::Point::new( -                    curve.from.x, -                    curve.from.y, -                )); - -                path_builder.quadratic_bezier_to( -                    lyon::math::Point::new(curve.control.x, curve.control.y), -                    lyon::math::Point::new(curve.to.x, curve.to.y), -                ); -            } - -            match self.state.pending { -                None => {} -                Some(Pending::One { from }) => { -                    path_builder -                        .move_to(lyon::math::Point::new(from.x, from.y)); -                    path_builder.line_to(lyon::math::Point::new( -                        cursor_position.x - bounds.x, -                        cursor_position.y - bounds.y, -                    )); -                } -                Some(Pending::Two { from, to }) => { -                    path_builder -                        .move_to(lyon::math::Point::new(from.x, from.y)); -                    path_builder.quadratic_bezier_to( -                        lyon::math::Point::new( -                            cursor_position.x - bounds.x, -                            cursor_position.y - bounds.y, -                        ), -                        lyon::math::Point::new(to.x, to.y), -                    ); -                } -            } - -            let mut tessellator = StrokeTessellator::new(); - -            // Draw strokes with lyon. -            tessellator -                .tessellate( -                    &path_builder.build(), -                    &StrokeOptions::default().with_line_width(3.0), -                    &mut BuffersBuilder::new( -                        &mut buffer, -                        |pos: lyon::math::Point, _: StrokeAttributes| { -                            Vertex2D { -                                position: pos.to_array(), -                                color: [0.0, 0.0, 0.0, 1.0], -                            } -                        }, -                    ), -                ) -                .unwrap(); - -            let mesh = Primitive::Mesh2D { -                origin: Point::new(bounds.x, bounds.y), -                buffers: Mesh2D { -                    vertices: buffer.vertices, -                    indices: buffer.indices, -                }, -            }; - -            ( -                Primitive::Clip { -                    bounds, -                    offset: Vector::new(0, 0), -                    content: Box::new( -                        if self.curves.is_empty() -                            && self.state.pending.is_none() -                        { -                            let instructions = Primitive::Text { -                                bounds: Rectangle { -                                    x: bounds.center_x(), -                                    y: bounds.center_y(), -                                    ..bounds -                                }, -                                color: Color { -                                    a: defaults.text.color.a * 0.7, -                                    ..defaults.text.color -                                }, -                                content: String::from( -                                    "Click to create bezier curves!", -                                ), -                                font: Font::Default, -                                size: 30.0, -                                horizontal_alignment: -                                    HorizontalAlignment::Center, -                                vertical_alignment: VerticalAlignment::Center, -                            }; - -                            Primitive::Group { -                                primitives: vec![mesh, instructions], -                            } -                        } else { -                            mesh -                        }, -                    ), -                }, -                MouseCursor::OutOfBounds, -            ) -        } - -        fn hash_layout(&self, _state: &mut Hasher) {} - -        fn on_event( -            &mut self, -            event: Event, -            layout: Layout<'_>, -            cursor_position: Point, -            messages: &mut Vec<Message>, -            _renderer: &Renderer, -            _clipboard: Option<&dyn Clipboard>, -        ) { -            let bounds = layout.bounds(); - -            if bounds.contains(cursor_position) { -                match event { -                    Event::Mouse(input::mouse::Event::Input { -                        state: input::ButtonState::Pressed, -                        .. -                    }) => { -                        let new_point = Point::new( -                            cursor_position.x - bounds.x, -                            cursor_position.y - bounds.y, -                        ); - -                        match self.state.pending { -                            None => { -                                self.state.pending = -                                    Some(Pending::One { from: new_point }); -                            } -                            Some(Pending::One { from }) => { -                                self.state.pending = Some(Pending::Two { -                                    from, -                                    to: new_point, -                                }); -                            } -                            Some(Pending::Two { from, to }) => { -                                self.state.pending = None; - -                                messages.push((self.on_click)(Curve { -                                    from, -                                    to, -                                    control: new_point, -                                })); -                            } -                        } -                    } -                    _ => {} -                } -            } -        } -    } - -    impl<'a, Message> Into<Element<'a, Message, Renderer>> for Bezier<'a, Message> -    where -        Message: 'static, -    { -        fn into(self) -> Element<'a, Message, Renderer> { -            Element::new(self) -        } -    } -} - -use bezier::Bezier; +//! This example showcases an interactive `Canvas` for drawing Bézier curves.  use iced::{ -    button, Align, Button, Column, Container, Element, Length, Sandbox, -    Settings, Text, +    button, Align, Button, Column, Element, Length, Sandbox, Settings, Text,  };  pub fn main() { @@ -323,6 +38,7 @@ impl Sandbox for Example {          match message {              Message::AddCurve(curve) => {                  self.curves.push(curve); +                self.bezier.request_redraw();              }              Message::Clear => {                  self.bezier = bezier::State::default(); @@ -332,7 +48,7 @@ impl Sandbox for Example {      }      fn view(&mut self) -> Element<Message> { -        let content = Column::new() +        Column::new()              .padding(20)              .spacing(20)              .align_items(Align::Center) @@ -341,22 +57,177 @@ impl Sandbox for Example {                      .width(Length::Shrink)                      .size(50),              ) -            .push(Bezier::new( -                &mut self.bezier, -                self.curves.as_slice(), -                Message::AddCurve, -            )) +            .push(self.bezier.view(&self.curves).map(Message::AddCurve))              .push(                  Button::new(&mut self.button_state, Text::new("Clear"))                      .padding(8)                      .on_press(Message::Clear), -            ); +            ) +            .into() +    } +} + +mod bezier { +    use iced::{ +        canvas::{self, Canvas, Cursor, Event, Frame, Geometry, Path, Stroke}, +        mouse, Element, Length, Point, Rectangle, +    }; + +    #[derive(Default)] +    pub struct State { +        pending: Option<Pending>, +        cache: canvas::Cache, +    } -        Container::new(content) +    impl State { +        pub fn view<'a>( +            &'a mut self, +            curves: &'a [Curve], +        ) -> Element<'a, Curve> { +            Canvas::new(Bezier { +                state: self, +                curves, +            })              .width(Length::Fill)              .height(Length::Fill) -            .center_x() -            .center_y()              .into() +        } + +        pub fn request_redraw(&mut self) { +            self.cache.clear() +        } +    } + +    struct Bezier<'a> { +        state: &'a mut State, +        curves: &'a [Curve], +    } + +    impl<'a> canvas::Program<Curve> for Bezier<'a> { +        fn update( +            &mut self, +            event: Event, +            bounds: Rectangle, +            cursor: Cursor, +        ) -> Option<Curve> { +            let cursor_position = cursor.position_in(&bounds)?; + +            match event { +                Event::Mouse(mouse_event) => match mouse_event { +                    mouse::Event::ButtonPressed(mouse::Button::Left) => { +                        match self.state.pending { +                            None => { +                                self.state.pending = Some(Pending::One { +                                    from: cursor_position, +                                }); +                                None +                            } +                            Some(Pending::One { from }) => { +                                self.state.pending = Some(Pending::Two { +                                    from, +                                    to: cursor_position, +                                }); + +                                None +                            } +                            Some(Pending::Two { from, to }) => { +                                self.state.pending = None; + +                                Some(Curve { +                                    from, +                                    to, +                                    control: cursor_position, +                                }) +                            } +                        } +                    } +                    _ => None, +                }, +            } +        } + +        fn draw(&self, bounds: Rectangle, cursor: Cursor) -> Vec<Geometry> { +            let content = +                self.state.cache.draw(bounds.size(), |frame: &mut Frame| { +                    Curve::draw_all(self.curves, frame); + +                    frame.stroke( +                        &Path::rectangle(Point::ORIGIN, frame.size()), +                        Stroke::default(), +                    ); +                }); + +            if let Some(pending) = &self.state.pending { +                let pending_curve = pending.draw(bounds, cursor); + +                vec![content, pending_curve] +            } else { +                vec![content] +            } +        } + +        fn mouse_interaction( +            &self, +            bounds: Rectangle, +            cursor: Cursor, +        ) -> mouse::Interaction { +            if cursor.is_over(&bounds) { +                mouse::Interaction::Crosshair +            } else { +                mouse::Interaction::default() +            } +        } +    } + +    #[derive(Debug, Clone, Copy)] +    pub struct Curve { +        from: Point, +        to: Point, +        control: Point, +    } + +    impl Curve { +        fn draw_all(curves: &[Curve], frame: &mut Frame) { +            let curves = Path::new(|p| { +                for curve in curves { +                    p.move_to(curve.from); +                    p.quadratic_curve_to(curve.control, curve.to); +                } +            }); + +            frame.stroke(&curves, Stroke::default().with_width(2.0)); +        } +    } + +    #[derive(Debug, Clone, Copy)] +    enum Pending { +        One { from: Point }, +        Two { from: Point, to: Point }, +    } + +    impl Pending { +        fn draw(&self, bounds: Rectangle, cursor: Cursor) -> Geometry { +            let mut frame = Frame::new(bounds.size()); + +            if let Some(cursor_position) = cursor.position_in(&bounds) { +                match *self { +                    Pending::One { from } => { +                        let line = Path::line(from, cursor_position); +                        frame.stroke(&line, Stroke::default().with_width(2.0)); +                    } +                    Pending::Two { from, to } => { +                        let curve = Curve { +                            from, +                            to, +                            control: cursor_position, +                        }; + +                        Curve::draw_all(&[curve], &mut frame); +                    } +                }; +            } + +            frame.into_geometry() +        }      }  } | 
