Implement mouse-based pane resizing for PaneGrid
This commit is contained in:
parent
db441a64b1
commit
f08cb4ad56
@ -22,6 +22,16 @@ impl Point {
|
|||||||
pub const fn new(x: f32, y: f32) -> Self {
|
pub const fn new(x: f32, y: f32) -> Self {
|
||||||
Self { x, y }
|
Self { x, y }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Computes the distance to another [`Point`].
|
||||||
|
///
|
||||||
|
/// [`Point`]: struct.Point.html
|
||||||
|
pub fn distance(&self, to: Point) -> f32 {
|
||||||
|
let a = self.x - to.x;
|
||||||
|
let b = self.y - to.y;
|
||||||
|
|
||||||
|
f32::sqrt(a * a + b * b)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<[f32; 2]> for Point {
|
impl From<[f32; 2]> for Point {
|
||||||
|
|||||||
@ -21,6 +21,12 @@ pub enum MouseCursor {
|
|||||||
|
|
||||||
/// The cursor is over a text widget.
|
/// The cursor is over a text widget.
|
||||||
Text,
|
Text,
|
||||||
|
|
||||||
|
/// The cursor is resizing a widget horizontally.
|
||||||
|
ResizingHorizontally,
|
||||||
|
|
||||||
|
/// The cursor is resizing a widget vertically.
|
||||||
|
ResizingVertically,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for MouseCursor {
|
impl Default for MouseCursor {
|
||||||
|
|||||||
@ -14,7 +14,7 @@ pub use state::{Focus, State};
|
|||||||
use crate::{
|
use crate::{
|
||||||
input::{keyboard, mouse, ButtonState},
|
input::{keyboard, mouse, ButtonState},
|
||||||
layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Size,
|
layout, Clipboard, Element, Event, Hasher, Layout, Length, Point, Size,
|
||||||
Widget,
|
Vector, Widget,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[allow(missing_debug_implementations)]
|
#[allow(missing_debug_implementations)]
|
||||||
@ -26,6 +26,7 @@ pub struct PaneGrid<'a, Message, Renderer> {
|
|||||||
height: Length,
|
height: Length,
|
||||||
spacing: u16,
|
spacing: u16,
|
||||||
on_drag: Option<Box<dyn Fn(DragEvent) -> Message>>,
|
on_drag: Option<Box<dyn Fn(DragEvent) -> Message>>,
|
||||||
|
on_resize: Option<Box<dyn Fn(ResizeEvent) -> Message>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> {
|
impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> {
|
||||||
@ -67,6 +68,7 @@ impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> {
|
|||||||
height: Length::Fill,
|
height: Length::Fill,
|
||||||
spacing: 0,
|
spacing: 0,
|
||||||
on_drag: None,
|
on_drag: None,
|
||||||
|
on_resize: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -101,6 +103,14 @@ impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> {
|
|||||||
self.on_drag = Some(Box::new(f));
|
self.on_drag = Some(Box::new(f));
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn on_resize(
|
||||||
|
mut self,
|
||||||
|
f: impl Fn(ResizeEvent) -> Message + 'static,
|
||||||
|
) -> Self {
|
||||||
|
self.on_resize = Some(Box::new(f));
|
||||||
|
self
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
@ -110,6 +120,12 @@ pub enum DragEvent {
|
|||||||
Canceled { pane: Pane },
|
Canceled { pane: Pane },
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct ResizeEvent {
|
||||||
|
pub split: Split,
|
||||||
|
pub ratio: f32,
|
||||||
|
}
|
||||||
|
|
||||||
impl<'a, Message, Renderer> Widget<Message, Renderer>
|
impl<'a, Message, Renderer> Widget<Message, Renderer>
|
||||||
for PaneGrid<'a, Message, Renderer>
|
for PaneGrid<'a, Message, Renderer>
|
||||||
where
|
where
|
||||||
@ -178,7 +194,7 @@ where
|
|||||||
if let Some(((pane, _), _)) = clicked_region.next() {
|
if let Some(((pane, _), _)) = clicked_region.next() {
|
||||||
match &self.on_drag {
|
match &self.on_drag {
|
||||||
Some(on_drag) if self.modifiers.alt => {
|
Some(on_drag) if self.modifiers.alt => {
|
||||||
self.state.drag(pane);
|
self.state.pick_pane(pane);
|
||||||
|
|
||||||
messages.push(on_drag(DragEvent::Picked {
|
messages.push(on_drag(DragEvent::Picked {
|
||||||
pane: *pane,
|
pane: *pane,
|
||||||
@ -193,7 +209,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
ButtonState::Released => {
|
ButtonState::Released => {
|
||||||
if let Some(pane) = self.state.dragged() {
|
if let Some(pane) = self.state.picked_pane() {
|
||||||
self.state.focus(&pane);
|
self.state.focus(&pane);
|
||||||
|
|
||||||
if let Some(on_drag) = &self.on_drag {
|
if let Some(on_drag) = &self.on_drag {
|
||||||
@ -220,13 +236,101 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
Event::Mouse(mouse::Event::Input {
|
||||||
|
button: mouse::Button::Right,
|
||||||
|
state,
|
||||||
|
}) if self.on_resize.is_some()
|
||||||
|
&& self.state.picked_pane().is_none()
|
||||||
|
&& self.modifiers.alt =>
|
||||||
|
{
|
||||||
|
match state {
|
||||||
|
ButtonState::Pressed => {
|
||||||
|
let bounds = layout.bounds();
|
||||||
|
|
||||||
|
let splits = self.state.splits(
|
||||||
|
f32::from(self.spacing),
|
||||||
|
Size::new(bounds.width, bounds.height),
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut sorted_splits: Vec<_> = splits.iter().collect();
|
||||||
|
let offset = Vector::new(bounds.x, bounds.y);
|
||||||
|
|
||||||
|
sorted_splits.sort_by_key(
|
||||||
|
|(_, (axis, rectangle, ratio))| {
|
||||||
|
let center = match axis {
|
||||||
|
Axis::Horizontal => Point::new(
|
||||||
|
rectangle.x + rectangle.width / 2.0,
|
||||||
|
rectangle.y + rectangle.height * ratio,
|
||||||
|
),
|
||||||
|
|
||||||
|
Axis::Vertical => Point::new(
|
||||||
|
rectangle.x + rectangle.width * ratio,
|
||||||
|
rectangle.y + rectangle.height / 2.0,
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
cursor_position
|
||||||
|
.distance(center + offset)
|
||||||
|
.round()
|
||||||
|
as u32
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some((split, (axis, _, _))) =
|
||||||
|
sorted_splits.first()
|
||||||
|
{
|
||||||
|
self.state.pick_split(split, *axis);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ButtonState::Released => {
|
||||||
|
self.state.drop_split();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Event::Mouse(mouse::Event::CursorMoved { .. }) => {
|
||||||
|
if let Some(on_resize) = &self.on_resize {
|
||||||
|
if let Some((split, _)) = self.state.picked_split() {
|
||||||
|
let bounds = layout.bounds();
|
||||||
|
|
||||||
|
let splits = self.state.splits(
|
||||||
|
f32::from(self.spacing),
|
||||||
|
Size::new(bounds.width, bounds.height),
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some((axis, rectangle, _)) = splits.get(&split) {
|
||||||
|
let ratio = match axis {
|
||||||
|
Axis::Horizontal => {
|
||||||
|
let position = cursor_position.x - bounds.x
|
||||||
|
+ rectangle.x;
|
||||||
|
|
||||||
|
(position / (rectangle.x + rectangle.width))
|
||||||
|
.max(0.1)
|
||||||
|
.min(0.9)
|
||||||
|
}
|
||||||
|
Axis::Vertical => {
|
||||||
|
let position = cursor_position.y - bounds.y
|
||||||
|
+ rectangle.y;
|
||||||
|
|
||||||
|
(position
|
||||||
|
/ (rectangle.y + rectangle.height))
|
||||||
|
.max(0.1)
|
||||||
|
.min(0.9)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
messages
|
||||||
|
.push(on_resize(ResizeEvent { split, ratio }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Event::Keyboard(keyboard::Event::Input { modifiers, .. }) => {
|
Event::Keyboard(keyboard::Event::Input { modifiers, .. }) => {
|
||||||
*self.modifiers = modifiers;
|
*self.modifiers = modifiers;
|
||||||
}
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.state.dragged().is_none() {
|
if self.state.picked_pane().is_none() {
|
||||||
{
|
{
|
||||||
self.elements.iter_mut().zip(layout.children()).for_each(
|
self.elements.iter_mut().zip(layout.children()).for_each(
|
||||||
|((_, pane), layout)| {
|
|((_, pane), layout)| {
|
||||||
@ -254,7 +358,8 @@ where
|
|||||||
renderer.draw(
|
renderer.draw(
|
||||||
defaults,
|
defaults,
|
||||||
&self.elements,
|
&self.elements,
|
||||||
self.state.dragged(),
|
self.state.picked_pane(),
|
||||||
|
self.state.picked_split().map(|(_, axis)| axis),
|
||||||
layout,
|
layout,
|
||||||
cursor_position,
|
cursor_position,
|
||||||
)
|
)
|
||||||
@ -297,6 +402,7 @@ pub trait Renderer: crate::Renderer + Sized {
|
|||||||
defaults: &Self::Defaults,
|
defaults: &Self::Defaults,
|
||||||
content: &[(Pane, Element<'_, Message, Self>)],
|
content: &[(Pane, Element<'_, Message, Self>)],
|
||||||
dragging: Option<Pane>,
|
dragging: Option<Pane>,
|
||||||
|
resizing: Option<Axis>,
|
||||||
layout: Layout<'_>,
|
layout: Layout<'_>,
|
||||||
cursor_position: Point,
|
cursor_position: Point,
|
||||||
) -> Self::Output;
|
) -> Self::Output;
|
||||||
|
|||||||
@ -55,6 +55,25 @@ impl Node {
|
|||||||
f(self);
|
f(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn resize(&mut self, split: &Split, percentage: f32) -> bool {
|
||||||
|
match self {
|
||||||
|
Node::Split {
|
||||||
|
id, ratio, a, b, ..
|
||||||
|
} => {
|
||||||
|
if id == split {
|
||||||
|
*ratio = (percentage * 1_000_000.0).round() as u32;
|
||||||
|
|
||||||
|
true
|
||||||
|
} else if a.resize(split, percentage) {
|
||||||
|
true
|
||||||
|
} else {
|
||||||
|
b.resize(split, percentage)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Node::Pane(_) => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn remove(&mut self, pane: &Pane) -> Option<Pane> {
|
pub fn remove(&mut self, pane: &Pane) -> Option<Pane> {
|
||||||
match self {
|
match self {
|
||||||
Node::Split { a, b, .. } => {
|
Node::Split { a, b, .. } => {
|
||||||
@ -93,6 +112,27 @@ impl Node {
|
|||||||
regions
|
regions
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn splits(
|
||||||
|
&self,
|
||||||
|
spacing: f32,
|
||||||
|
size: Size,
|
||||||
|
) -> HashMap<Split, (Axis, Rectangle, f32)> {
|
||||||
|
let mut splits = HashMap::new();
|
||||||
|
|
||||||
|
self.compute_splits(
|
||||||
|
spacing / 2.0,
|
||||||
|
&Rectangle {
|
||||||
|
x: 0.0,
|
||||||
|
y: 0.0,
|
||||||
|
width: size.width,
|
||||||
|
height: size.height,
|
||||||
|
},
|
||||||
|
&mut splits,
|
||||||
|
);
|
||||||
|
|
||||||
|
splits
|
||||||
|
}
|
||||||
|
|
||||||
pub fn pane(&self) -> Option<Pane> {
|
pub fn pane(&self) -> Option<Pane> {
|
||||||
match self {
|
match self {
|
||||||
Node::Split { .. } => None,
|
Node::Split { .. } => None,
|
||||||
@ -129,4 +169,31 @@ impl Node {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn compute_splits(
|
||||||
|
&self,
|
||||||
|
halved_spacing: f32,
|
||||||
|
current: &Rectangle,
|
||||||
|
splits: &mut HashMap<Split, (Axis, Rectangle, f32)>,
|
||||||
|
) {
|
||||||
|
match self {
|
||||||
|
Node::Split {
|
||||||
|
axis,
|
||||||
|
ratio,
|
||||||
|
a,
|
||||||
|
b,
|
||||||
|
id,
|
||||||
|
} => {
|
||||||
|
let ratio = *ratio as f32 / 1_000_000.0;
|
||||||
|
let (region_a, region_b) =
|
||||||
|
axis.split(current, ratio, halved_spacing);
|
||||||
|
|
||||||
|
let _ = splits.insert(*id, (*axis, *current, ratio));
|
||||||
|
|
||||||
|
a.compute_splits(halved_spacing, ®ion_a, splits);
|
||||||
|
b.compute_splits(halved_spacing, ®ion_b, splits);
|
||||||
|
}
|
||||||
|
Node::Pane(_) => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -134,6 +134,10 @@ impl<T> State<T> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn resize(&mut self, split: &Split, percentage: f32) {
|
||||||
|
let _ = self.internal.layout.resize(split, percentage);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn close(&mut self, pane: &Pane) -> Option<T> {
|
pub fn close(&mut self, pane: &Pane) -> Option<T> {
|
||||||
if let Some(sibling) = self.internal.layout.remove(pane) {
|
if let Some(sibling) = self.internal.layout.remove(pane) {
|
||||||
self.focus(&sibling);
|
self.focus(&sibling);
|
||||||
@ -153,14 +157,25 @@ pub struct Internal {
|
|||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
pub enum Action {
|
pub enum Action {
|
||||||
Idle { focus: Option<Pane> },
|
Idle {
|
||||||
Dragging { pane: Pane },
|
focus: Option<Pane>,
|
||||||
|
},
|
||||||
|
Dragging {
|
||||||
|
pane: Pane,
|
||||||
|
},
|
||||||
|
Resizing {
|
||||||
|
split: Split,
|
||||||
|
axis: Axis,
|
||||||
|
focus: Option<Pane>,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Action {
|
impl Action {
|
||||||
pub fn focus(&self) -> Option<(Pane, Focus)> {
|
pub fn focus(&self) -> Option<(Pane, Focus)> {
|
||||||
match self {
|
match self {
|
||||||
Action::Idle { focus } => focus.map(|pane| (pane, Focus::Idle)),
|
Action::Idle { focus } | Action::Resizing { focus, .. } => {
|
||||||
|
focus.map(|pane| (pane, Focus::Idle))
|
||||||
|
}
|
||||||
Action::Dragging { pane } => Some((*pane, Focus::Dragging)),
|
Action::Dragging { pane } => Some((*pane, Focus::Dragging)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -171,13 +186,20 @@ impl Internal {
|
|||||||
self.action
|
self.action
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn dragged(&self) -> Option<Pane> {
|
pub fn picked_pane(&self) -> Option<Pane> {
|
||||||
match self.action {
|
match self.action {
|
||||||
Action::Dragging { pane } => Some(pane),
|
Action::Dragging { pane } => Some(pane),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn picked_split(&self) -> Option<(Split, Axis)> {
|
||||||
|
match self.action {
|
||||||
|
Action::Resizing { split, axis, .. } => Some((split, axis)),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn regions(
|
pub fn regions(
|
||||||
&self,
|
&self,
|
||||||
spacing: f32,
|
spacing: f32,
|
||||||
@ -186,14 +208,47 @@ impl Internal {
|
|||||||
self.layout.regions(spacing, size)
|
self.layout.regions(spacing, size)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn splits(
|
||||||
|
&self,
|
||||||
|
spacing: f32,
|
||||||
|
size: Size,
|
||||||
|
) -> HashMap<Split, (Axis, Rectangle, f32)> {
|
||||||
|
self.layout.splits(spacing, size)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn focus(&mut self, pane: &Pane) {
|
pub fn focus(&mut self, pane: &Pane) {
|
||||||
self.action = Action::Idle { focus: Some(*pane) };
|
self.action = Action::Idle { focus: Some(*pane) };
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn drag(&mut self, pane: &Pane) {
|
pub fn pick_pane(&mut self, pane: &Pane) {
|
||||||
self.action = Action::Dragging { pane: *pane };
|
self.action = Action::Dragging { pane: *pane };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn pick_split(&mut self, split: &Split, axis: Axis) {
|
||||||
|
// TODO: Obtain `axis` from layout itself. Maybe we should implement
|
||||||
|
// `Node::find_split`
|
||||||
|
if self.picked_pane().is_some() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let focus = self.action.focus().map(|(pane, _)| pane);
|
||||||
|
|
||||||
|
self.action = Action::Resizing {
|
||||||
|
split: *split,
|
||||||
|
axis,
|
||||||
|
focus,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn drop_split(&mut self) {
|
||||||
|
match self.action {
|
||||||
|
Action::Resizing { focus, .. } => {
|
||||||
|
self.action = Action::Idle { focus };
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn unfocus(&mut self) {
|
pub fn unfocus(&mut self) {
|
||||||
self.action = Action::Idle { focus: None };
|
self.action = Action::Idle { focus: None };
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
use crate::{Primitive, Renderer};
|
use crate::{Primitive, Renderer};
|
||||||
use iced_native::{
|
use iced_native::{
|
||||||
pane_grid::{self, Pane},
|
pane_grid::{self, Axis, Pane},
|
||||||
Element, Layout, MouseCursor, Point, Rectangle, Vector,
|
Element, Layout, MouseCursor, Point, Rectangle, Vector,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -10,6 +10,7 @@ impl pane_grid::Renderer for Renderer {
|
|||||||
defaults: &Self::Defaults,
|
defaults: &Self::Defaults,
|
||||||
content: &[(Pane, Element<'_, Message, Self>)],
|
content: &[(Pane, Element<'_, Message, Self>)],
|
||||||
dragging: Option<Pane>,
|
dragging: Option<Pane>,
|
||||||
|
resizing: Option<Axis>,
|
||||||
layout: Layout<'_>,
|
layout: Layout<'_>,
|
||||||
cursor_position: Point,
|
cursor_position: Point,
|
||||||
) -> Self::Output {
|
) -> Self::Output {
|
||||||
@ -70,6 +71,11 @@ impl pane_grid::Renderer for Renderer {
|
|||||||
Primitive::Group { primitives },
|
Primitive::Group { primitives },
|
||||||
if dragging.is_some() {
|
if dragging.is_some() {
|
||||||
MouseCursor::Grabbing
|
MouseCursor::Grabbing
|
||||||
|
} else if let Some(axis) = resizing {
|
||||||
|
match axis {
|
||||||
|
Axis::Horizontal => MouseCursor::ResizingHorizontally,
|
||||||
|
Axis::Vertical => MouseCursor::ResizingVertically,
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
mouse_cursor
|
mouse_cursor
|
||||||
},
|
},
|
||||||
|
|||||||
@ -116,6 +116,10 @@ pub fn mouse_cursor(mouse_cursor: MouseCursor) -> winit::window::CursorIcon {
|
|||||||
MouseCursor::Grab => winit::window::CursorIcon::Grab,
|
MouseCursor::Grab => winit::window::CursorIcon::Grab,
|
||||||
MouseCursor::Grabbing => winit::window::CursorIcon::Grabbing,
|
MouseCursor::Grabbing => winit::window::CursorIcon::Grabbing,
|
||||||
MouseCursor::Text => winit::window::CursorIcon::Text,
|
MouseCursor::Text => winit::window::CursorIcon::Text,
|
||||||
|
MouseCursor::ResizingHorizontally => {
|
||||||
|
winit::window::CursorIcon::EwResize
|
||||||
|
}
|
||||||
|
MouseCursor::ResizingVertically => winit::window::CursorIcon::NsResize,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user