diff options
| author | 2020-03-20 12:10:52 +0100 | |
|---|---|---|
| committer | 2020-03-20 12:10:52 +0100 | |
| commit | f7ec679fec1b69c6dc5bc12d60627629f086bf22 (patch) | |
| tree | 58ec4dd11e50f11ef85b972fee211930e7a0e7c0 /wgpu | |
| parent | 93f5640a2dc06d8f1bf2b0d033d45c62f8985380 (diff) | |
| parent | fb744a338c1b7566a3db9a3d24c03729b4858217 (diff) | |
| download | iced-f7ec679fec1b69c6dc5bc12d60627629f086bf22.tar.gz iced-f7ec679fec1b69c6dc5bc12d60627629f086bf22.tar.bz2 iced-f7ec679fec1b69c6dc5bc12d60627629f086bf22.zip | |
Merge pull request #224 from hecrj/feature/panes-widget
Pane grid widget
Diffstat (limited to 'wgpu')
| -rw-r--r-- | wgpu/src/renderer/widget.rs | 1 | ||||
| -rw-r--r-- | wgpu/src/renderer/widget/pane_grid.rs | 92 | ||||
| -rw-r--r-- | wgpu/src/renderer/widget/scrollable.rs | 14 | ||||
| -rw-r--r-- | wgpu/src/widget.rs | 3 | ||||
| -rw-r--r-- | wgpu/src/widget/canvas/layer/cache.rs | 14 | ||||
| -rw-r--r-- | wgpu/src/widget/pane_grid.rs | 17 | 
6 files changed, 133 insertions, 8 deletions
| diff --git a/wgpu/src/renderer/widget.rs b/wgpu/src/renderer/widget.rs index 84f908e7..37421fbe 100644 --- a/wgpu/src/renderer/widget.rs +++ b/wgpu/src/renderer/widget.rs @@ -2,6 +2,7 @@ mod button;  mod checkbox;  mod column;  mod container; +mod pane_grid;  mod progress_bar;  mod radio;  mod row; diff --git a/wgpu/src/renderer/widget/pane_grid.rs b/wgpu/src/renderer/widget/pane_grid.rs new file mode 100644 index 00000000..2d201fec --- /dev/null +++ b/wgpu/src/renderer/widget/pane_grid.rs @@ -0,0 +1,92 @@ +use crate::{Primitive, Renderer}; +use iced_native::{ +    pane_grid::{self, Axis, Pane}, +    Element, Layout, MouseCursor, Point, Rectangle, Vector, +}; + +impl pane_grid::Renderer for Renderer { +    fn draw<Message>( +        &mut self, +        defaults: &Self::Defaults, +        content: &[(Pane, Element<'_, Message, Self>)], +        dragging: Option<Pane>, +        resizing: Option<Axis>, +        layout: Layout<'_>, +        cursor_position: Point, +    ) -> Self::Output { +        let pane_cursor_position = if dragging.is_some() { +            // TODO: Remove once cursor availability is encoded in the type +            // system +            Point::new(-1.0, -1.0) +        } else { +            cursor_position +        }; + +        let mut mouse_cursor = MouseCursor::OutOfBounds; +        let mut dragged_pane = None; + +        let mut panes: Vec<_> = content +            .iter() +            .zip(layout.children()) +            .enumerate() +            .map(|(i, ((id, pane), layout))| { +                let (primitive, new_mouse_cursor) = +                    pane.draw(self, defaults, layout, pane_cursor_position); + +                if new_mouse_cursor > mouse_cursor { +                    mouse_cursor = new_mouse_cursor; +                } + +                if Some(*id) == dragging { +                    dragged_pane = Some((i, layout)); +                } + +                primitive +            }) +            .collect(); + +        let primitives = if let Some((index, layout)) = dragged_pane { +            let pane = panes.remove(index); +            let bounds = layout.bounds(); + +            // TODO: Fix once proper layering is implemented. +            // This is a pretty hacky way to achieve layering. +            let clip = Primitive::Clip { +                bounds: Rectangle { +                    x: cursor_position.x - bounds.width / 2.0, +                    y: cursor_position.y - bounds.height / 2.0, +                    width: bounds.width + 0.5, +                    height: bounds.height + 0.5, +                }, +                offset: Vector::new(0, 0), +                content: Box::new(Primitive::Cached { +                    origin: Point::new( +                        cursor_position.x - bounds.x - bounds.width / 2.0, +                        cursor_position.y - bounds.y - bounds.height / 2.0, +                    ), +                    cache: std::sync::Arc::new(pane), +                }), +            }; + +            panes.push(clip); + +            panes +        } else { +            panes +        }; + +        ( +            Primitive::Group { primitives }, +            if dragging.is_some() { +                MouseCursor::Grabbing +            } else if let Some(axis) = resizing { +                match axis { +                    Axis::Horizontal => MouseCursor::ResizingVertically, +                    Axis::Vertical => MouseCursor::ResizingHorizontally, +                } +            } else { +                mouse_cursor +            }, +        ) +    } +} diff --git a/wgpu/src/renderer/widget/scrollable.rs b/wgpu/src/renderer/widget/scrollable.rs index bfee7411..732523e3 100644 --- a/wgpu/src/renderer/widget/scrollable.rs +++ b/wgpu/src/renderer/widget/scrollable.rs @@ -58,14 +58,14 @@ impl scrollable::Renderer for Renderer {          style_sheet: &Self::Style,          (content, mouse_cursor): Self::Output,      ) -> Self::Output { -        let clip = Primitive::Clip { -            bounds, -            offset: Vector::new(0, offset), -            content: Box::new(content), -        }; -          (              if let Some(scrollbar) = scrollbar { +                let clip = Primitive::Clip { +                    bounds, +                    offset: Vector::new(0, offset), +                    content: Box::new(content), +                }; +                  let style = if state.is_scroller_grabbed() {                      style_sheet.dragging()                  } else if is_mouse_over_scrollbar { @@ -115,7 +115,7 @@ impl scrollable::Renderer for Renderer {                      primitives: vec![clip, scrollbar, scroller],                  }              } else { -                clip +                content              },              if is_mouse_over_scrollbar || state.is_scroller_grabbed() {                  MouseCursor::Idle diff --git a/wgpu/src/widget.rs b/wgpu/src/widget.rs index 73cce7e2..b39f2d91 100644 --- a/wgpu/src/widget.rs +++ b/wgpu/src/widget.rs @@ -10,6 +10,7 @@  pub mod button;  pub mod checkbox;  pub mod container; +pub mod pane_grid;  pub mod progress_bar;  pub mod radio;  pub mod scrollable; @@ -23,6 +24,8 @@ pub use checkbox::Checkbox;  #[doc(no_inline)]  pub use container::Container;  #[doc(no_inline)] +pub use pane_grid::PaneGrid; +#[doc(no_inline)]  pub use progress_bar::ProgressBar;  #[doc(no_inline)]  pub use radio::Radio; diff --git a/wgpu/src/widget/canvas/layer/cache.rs b/wgpu/src/widget/canvas/layer/cache.rs index 6b69f01e..f7002459 100644 --- a/wgpu/src/widget/canvas/layer/cache.rs +++ b/wgpu/src/widget/canvas/layer/cache.rs @@ -19,7 +19,6 @@ pub struct Cache<T: Drawable> {      state: RefCell<State>,  } -#[derive(Debug)]  enum State {      Empty,      Filled { @@ -97,3 +96,16 @@ where          primitive      }  } + +impl std::fmt::Debug for State { +    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { +        match self { +            State::Empty => write!(f, "Empty"), +            State::Filled { primitive, bounds } => f +                .debug_struct("Filled") +                .field("primitive", primitive) +                .field("bounds", bounds) +                .finish(), +        } +    } +} diff --git a/wgpu/src/widget/pane_grid.rs b/wgpu/src/widget/pane_grid.rs new file mode 100644 index 00000000..7bc2f7c5 --- /dev/null +++ b/wgpu/src/widget/pane_grid.rs @@ -0,0 +1,17 @@ +//! Let your users split regions of your application and organize layout dynamically. +//! +//! [](https://gfycat.com/mixedflatjellyfish) +use crate::Renderer; + +pub use iced_native::pane_grid::{ +    Axis, Direction, DragEvent, Focus, KeyPressEvent, Pane, ResizeEvent, Split, +    State, +}; + +/// A collection of panes distributed using either vertical or horizontal splits +/// to completely fill the space available. +/// +/// [](https://gfycat.com/mixedflatjellyfish) +/// +/// This is an alias of an `iced_native` pane grid with an `iced_wgpu::Renderer`. +pub type PaneGrid<'a, Message> = iced_native::PaneGrid<'a, Message, Renderer>; | 
