Merge pull request #1912 from tarkah/feat/scrollable-alignment
Add scrollable alignment option
This commit is contained in:
commit
21bd51426d
2 changed files with 165 additions and 67 deletions
|
|
@ -20,6 +20,7 @@ struct ScrollableDemo {
|
||||||
scrollbar_margin: u16,
|
scrollbar_margin: u16,
|
||||||
scroller_width: u16,
|
scroller_width: u16,
|
||||||
current_scroll_offset: scrollable::RelativeOffset,
|
current_scroll_offset: scrollable::RelativeOffset,
|
||||||
|
alignment: scrollable::Alignment,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Eq, PartialEq, Copy)]
|
#[derive(Debug, Clone, Eq, PartialEq, Copy)]
|
||||||
|
|
@ -32,6 +33,7 @@ enum Direction {
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
enum Message {
|
enum Message {
|
||||||
SwitchDirection(Direction),
|
SwitchDirection(Direction),
|
||||||
|
AlignmentChanged(scrollable::Alignment),
|
||||||
ScrollbarWidthChanged(u16),
|
ScrollbarWidthChanged(u16),
|
||||||
ScrollbarMarginChanged(u16),
|
ScrollbarMarginChanged(u16),
|
||||||
ScrollerWidthChanged(u16),
|
ScrollerWidthChanged(u16),
|
||||||
|
|
@ -54,6 +56,7 @@ impl Application for ScrollableDemo {
|
||||||
scrollbar_margin: 0,
|
scrollbar_margin: 0,
|
||||||
scroller_width: 10,
|
scroller_width: 10,
|
||||||
current_scroll_offset: scrollable::RelativeOffset::START,
|
current_scroll_offset: scrollable::RelativeOffset::START,
|
||||||
|
alignment: scrollable::Alignment::Start,
|
||||||
},
|
},
|
||||||
Command::none(),
|
Command::none(),
|
||||||
)
|
)
|
||||||
|
|
@ -74,6 +77,15 @@ impl Application for ScrollableDemo {
|
||||||
self.current_scroll_offset,
|
self.current_scroll_offset,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
Message::AlignmentChanged(alignment) => {
|
||||||
|
self.current_scroll_offset = scrollable::RelativeOffset::START;
|
||||||
|
self.alignment = alignment;
|
||||||
|
|
||||||
|
scrollable::snap_to(
|
||||||
|
SCROLLABLE_ID.clone(),
|
||||||
|
self.current_scroll_offset,
|
||||||
|
)
|
||||||
|
}
|
||||||
Message::ScrollbarWidthChanged(width) => {
|
Message::ScrollbarWidthChanged(width) => {
|
||||||
self.scrollbar_width = width;
|
self.scrollbar_width = width;
|
||||||
|
|
||||||
|
|
@ -165,10 +177,33 @@ impl Application for ScrollableDemo {
|
||||||
.spacing(10)
|
.spacing(10)
|
||||||
.width(Length::Fill);
|
.width(Length::Fill);
|
||||||
|
|
||||||
let scroll_controls =
|
let scroll_alignment_controls = column(vec![
|
||||||
row![scroll_slider_controls, scroll_orientation_controls]
|
text("Scrollable alignment:").into(),
|
||||||
.spacing(20)
|
radio(
|
||||||
.width(Length::Fill);
|
"Start",
|
||||||
|
scrollable::Alignment::Start,
|
||||||
|
Some(self.alignment),
|
||||||
|
Message::AlignmentChanged,
|
||||||
|
)
|
||||||
|
.into(),
|
||||||
|
radio(
|
||||||
|
"End",
|
||||||
|
scrollable::Alignment::End,
|
||||||
|
Some(self.alignment),
|
||||||
|
Message::AlignmentChanged,
|
||||||
|
)
|
||||||
|
.into(),
|
||||||
|
])
|
||||||
|
.spacing(10)
|
||||||
|
.width(Length::Fill);
|
||||||
|
|
||||||
|
let scroll_controls = row![
|
||||||
|
scroll_slider_controls,
|
||||||
|
scroll_orientation_controls,
|
||||||
|
scroll_alignment_controls
|
||||||
|
]
|
||||||
|
.spacing(20)
|
||||||
|
.width(Length::Fill);
|
||||||
|
|
||||||
let scroll_to_end_button = || {
|
let scroll_to_end_button = || {
|
||||||
button("Scroll to end")
|
button("Scroll to end")
|
||||||
|
|
@ -204,7 +239,8 @@ impl Application for ScrollableDemo {
|
||||||
Properties::new()
|
Properties::new()
|
||||||
.width(self.scrollbar_width)
|
.width(self.scrollbar_width)
|
||||||
.margin(self.scrollbar_margin)
|
.margin(self.scrollbar_margin)
|
||||||
.scroller_width(self.scroller_width),
|
.scroller_width(self.scroller_width)
|
||||||
|
.alignment(self.alignment),
|
||||||
))
|
))
|
||||||
.id(SCROLLABLE_ID.clone())
|
.id(SCROLLABLE_ID.clone())
|
||||||
.on_scroll(Message::Scrolled),
|
.on_scroll(Message::Scrolled),
|
||||||
|
|
@ -228,7 +264,8 @@ impl Application for ScrollableDemo {
|
||||||
Properties::new()
|
Properties::new()
|
||||||
.width(self.scrollbar_width)
|
.width(self.scrollbar_width)
|
||||||
.margin(self.scrollbar_margin)
|
.margin(self.scrollbar_margin)
|
||||||
.scroller_width(self.scroller_width),
|
.scroller_width(self.scroller_width)
|
||||||
|
.alignment(self.alignment),
|
||||||
))
|
))
|
||||||
.style(theme::Scrollable::custom(ScrollbarCustomStyle))
|
.style(theme::Scrollable::custom(ScrollbarCustomStyle))
|
||||||
.id(SCROLLABLE_ID.clone())
|
.id(SCROLLABLE_ID.clone())
|
||||||
|
|
@ -269,7 +306,8 @@ impl Application for ScrollableDemo {
|
||||||
let properties = Properties::new()
|
let properties = Properties::new()
|
||||||
.width(self.scrollbar_width)
|
.width(self.scrollbar_width)
|
||||||
.margin(self.scrollbar_margin)
|
.margin(self.scrollbar_margin)
|
||||||
.scroller_width(self.scroller_width);
|
.scroller_width(self.scroller_width)
|
||||||
|
.alignment(self.alignment);
|
||||||
|
|
||||||
scrollable::Direction::Both {
|
scrollable::Direction::Both {
|
||||||
horizontal: properties,
|
horizontal: properties,
|
||||||
|
|
|
||||||
|
|
@ -143,6 +143,7 @@ pub struct Properties {
|
||||||
width: f32,
|
width: f32,
|
||||||
margin: f32,
|
margin: f32,
|
||||||
scroller_width: f32,
|
scroller_width: f32,
|
||||||
|
alignment: Alignment,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Properties {
|
impl Default for Properties {
|
||||||
|
|
@ -151,6 +152,7 @@ impl Default for Properties {
|
||||||
width: 10.0,
|
width: 10.0,
|
||||||
margin: 0.0,
|
margin: 0.0,
|
||||||
scroller_width: 10.0,
|
scroller_width: 10.0,
|
||||||
|
alignment: Alignment::Start,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -178,6 +180,22 @@ impl Properties {
|
||||||
self.scroller_width = scroller_width.into().0.max(0.0);
|
self.scroller_width = scroller_width.into().0.max(0.0);
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets the alignment of the [`Scrollable`] .
|
||||||
|
pub fn alignment(mut self, alignment: Alignment) -> Self {
|
||||||
|
self.alignment = alignment;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Alignment of the scrollable's content relative to it's [`Viewport`] in one direction.
|
||||||
|
#[derive(Debug, Clone, Copy, Default, PartialEq, Eq)]
|
||||||
|
pub enum Alignment {
|
||||||
|
/// Content is aligned to the start of the [`Viewport`].
|
||||||
|
#[default]
|
||||||
|
Start,
|
||||||
|
/// Content is aligned to the end of the [`Viewport`]
|
||||||
|
End,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, Message, Renderer> Widget<Message, Renderer>
|
impl<'a, Message, Renderer> Widget<Message, Renderer>
|
||||||
|
|
@ -268,7 +286,7 @@ where
|
||||||
cursor,
|
cursor,
|
||||||
clipboard,
|
clipboard,
|
||||||
shell,
|
shell,
|
||||||
&self.direction,
|
self.direction,
|
||||||
&self.on_scroll,
|
&self.on_scroll,
|
||||||
|event, layout, cursor, clipboard, shell| {
|
|event, layout, cursor, clipboard, shell| {
|
||||||
self.content.as_widget_mut().on_event(
|
self.content.as_widget_mut().on_event(
|
||||||
|
|
@ -300,7 +318,7 @@ where
|
||||||
theme,
|
theme,
|
||||||
layout,
|
layout,
|
||||||
cursor,
|
cursor,
|
||||||
&self.direction,
|
self.direction,
|
||||||
&self.style,
|
&self.style,
|
||||||
|renderer, layout, cursor, viewport| {
|
|renderer, layout, cursor, viewport| {
|
||||||
self.content.as_widget().draw(
|
self.content.as_widget().draw(
|
||||||
|
|
@ -328,7 +346,7 @@ where
|
||||||
tree.state.downcast_ref::<State>(),
|
tree.state.downcast_ref::<State>(),
|
||||||
layout,
|
layout,
|
||||||
cursor,
|
cursor,
|
||||||
&self.direction,
|
self.direction,
|
||||||
|layout, cursor, viewport| {
|
|layout, cursor, viewport| {
|
||||||
self.content.as_widget().mouse_interaction(
|
self.content.as_widget().mouse_interaction(
|
||||||
&tree.children[0],
|
&tree.children[0],
|
||||||
|
|
@ -358,13 +376,12 @@ where
|
||||||
let bounds = layout.bounds();
|
let bounds = layout.bounds();
|
||||||
let content_layout = layout.children().next().unwrap();
|
let content_layout = layout.children().next().unwrap();
|
||||||
let content_bounds = content_layout.bounds();
|
let content_bounds = content_layout.bounds();
|
||||||
let offset = tree.state.downcast_ref::<State>().offset(
|
let translation = tree
|
||||||
&self.direction,
|
.state
|
||||||
bounds,
|
.downcast_ref::<State>()
|
||||||
content_bounds,
|
.translation(self.direction, bounds, content_bounds);
|
||||||
);
|
|
||||||
|
|
||||||
overlay.translate(Vector::new(-offset.x, -offset.y))
|
overlay.translate(Vector::new(-translation.x, -translation.y))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -467,7 +484,7 @@ pub fn update<Message>(
|
||||||
cursor: mouse::Cursor,
|
cursor: mouse::Cursor,
|
||||||
clipboard: &mut dyn Clipboard,
|
clipboard: &mut dyn Clipboard,
|
||||||
shell: &mut Shell<'_, Message>,
|
shell: &mut Shell<'_, Message>,
|
||||||
direction: &Direction,
|
direction: Direction,
|
||||||
on_scroll: &Option<Box<dyn Fn(Viewport) -> Message + '_>>,
|
on_scroll: &Option<Box<dyn Fn(Viewport) -> Message + '_>>,
|
||||||
update_content: impl FnOnce(
|
update_content: impl FnOnce(
|
||||||
Event,
|
Event,
|
||||||
|
|
@ -495,7 +512,7 @@ pub fn update<Message>(
|
||||||
{
|
{
|
||||||
mouse::Cursor::Available(
|
mouse::Cursor::Available(
|
||||||
cursor_position
|
cursor_position
|
||||||
+ state.offset(direction, bounds, content_bounds),
|
+ state.translation(direction, bounds, content_bounds),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
_ => mouse::Cursor::Unavailable,
|
_ => mouse::Cursor::Unavailable,
|
||||||
|
|
@ -535,7 +552,7 @@ pub fn update<Message>(
|
||||||
mouse::ScrollDelta::Pixels { x, y } => Vector::new(x, y),
|
mouse::ScrollDelta::Pixels { x, y } => Vector::new(x, y),
|
||||||
};
|
};
|
||||||
|
|
||||||
state.scroll(delta, bounds, content_bounds);
|
state.scroll(delta, direction, bounds, content_bounds);
|
||||||
|
|
||||||
notify_on_scroll(state, on_scroll, bounds, content_bounds, shell);
|
notify_on_scroll(state, on_scroll, bounds, content_bounds, shell);
|
||||||
|
|
||||||
|
|
@ -566,7 +583,7 @@ pub fn update<Message>(
|
||||||
cursor_position.y - scroll_box_touched_at.y,
|
cursor_position.y - scroll_box_touched_at.y,
|
||||||
);
|
);
|
||||||
|
|
||||||
state.scroll(delta, bounds, content_bounds);
|
state.scroll(delta, direction, bounds, content_bounds);
|
||||||
|
|
||||||
state.scroll_area_touched_at = Some(cursor_position);
|
state.scroll_area_touched_at = Some(cursor_position);
|
||||||
|
|
||||||
|
|
@ -748,7 +765,7 @@ pub fn mouse_interaction(
|
||||||
state: &State,
|
state: &State,
|
||||||
layout: Layout<'_>,
|
layout: Layout<'_>,
|
||||||
cursor: mouse::Cursor,
|
cursor: mouse::Cursor,
|
||||||
direction: &Direction,
|
direction: Direction,
|
||||||
content_interaction: impl FnOnce(
|
content_interaction: impl FnOnce(
|
||||||
Layout<'_>,
|
Layout<'_>,
|
||||||
mouse::Cursor,
|
mouse::Cursor,
|
||||||
|
|
@ -771,13 +788,13 @@ pub fn mouse_interaction(
|
||||||
{
|
{
|
||||||
mouse::Interaction::Idle
|
mouse::Interaction::Idle
|
||||||
} else {
|
} else {
|
||||||
let offset = state.offset(direction, bounds, content_bounds);
|
let translation = state.translation(direction, bounds, content_bounds);
|
||||||
|
|
||||||
let cursor = match cursor_over_scrollable {
|
let cursor = match cursor_over_scrollable {
|
||||||
Some(cursor_position)
|
Some(cursor_position)
|
||||||
if !(mouse_over_x_scrollbar || mouse_over_y_scrollbar) =>
|
if !(mouse_over_x_scrollbar || mouse_over_y_scrollbar) =>
|
||||||
{
|
{
|
||||||
mouse::Cursor::Available(cursor_position + offset)
|
mouse::Cursor::Available(cursor_position + translation)
|
||||||
}
|
}
|
||||||
_ => mouse::Cursor::Unavailable,
|
_ => mouse::Cursor::Unavailable,
|
||||||
};
|
};
|
||||||
|
|
@ -786,8 +803,8 @@ pub fn mouse_interaction(
|
||||||
content_layout,
|
content_layout,
|
||||||
cursor,
|
cursor,
|
||||||
&Rectangle {
|
&Rectangle {
|
||||||
y: bounds.y + offset.y,
|
y: bounds.y + translation.y,
|
||||||
x: bounds.x + offset.x,
|
x: bounds.x + translation.x,
|
||||||
..bounds
|
..bounds
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
@ -801,7 +818,7 @@ pub fn draw<Renderer>(
|
||||||
theme: &Renderer::Theme,
|
theme: &Renderer::Theme,
|
||||||
layout: Layout<'_>,
|
layout: Layout<'_>,
|
||||||
cursor: mouse::Cursor,
|
cursor: mouse::Cursor,
|
||||||
direction: &Direction,
|
direction: Direction,
|
||||||
style: &<Renderer::Theme as StyleSheet>::Style,
|
style: &<Renderer::Theme as StyleSheet>::Style,
|
||||||
draw_content: impl FnOnce(&mut Renderer, Layout<'_>, mouse::Cursor, &Rectangle),
|
draw_content: impl FnOnce(&mut Renderer, Layout<'_>, mouse::Cursor, &Rectangle),
|
||||||
) where
|
) where
|
||||||
|
|
@ -818,13 +835,13 @@ pub fn draw<Renderer>(
|
||||||
let (mouse_over_y_scrollbar, mouse_over_x_scrollbar) =
|
let (mouse_over_y_scrollbar, mouse_over_x_scrollbar) =
|
||||||
scrollbars.is_mouse_over(cursor);
|
scrollbars.is_mouse_over(cursor);
|
||||||
|
|
||||||
let offset = state.offset(direction, bounds, content_bounds);
|
let translation = state.translation(direction, bounds, content_bounds);
|
||||||
|
|
||||||
let cursor = match cursor_over_scrollable {
|
let cursor = match cursor_over_scrollable {
|
||||||
Some(cursor_position)
|
Some(cursor_position)
|
||||||
if !(mouse_over_x_scrollbar || mouse_over_y_scrollbar) =>
|
if !(mouse_over_x_scrollbar || mouse_over_y_scrollbar) =>
|
||||||
{
|
{
|
||||||
mouse::Cursor::Available(cursor_position + offset)
|
mouse::Cursor::Available(cursor_position + translation)
|
||||||
}
|
}
|
||||||
_ => mouse::Cursor::Unavailable,
|
_ => mouse::Cursor::Unavailable,
|
||||||
};
|
};
|
||||||
|
|
@ -833,15 +850,15 @@ pub fn draw<Renderer>(
|
||||||
if scrollbars.active() {
|
if scrollbars.active() {
|
||||||
renderer.with_layer(bounds, |renderer| {
|
renderer.with_layer(bounds, |renderer| {
|
||||||
renderer.with_translation(
|
renderer.with_translation(
|
||||||
Vector::new(-offset.x, -offset.y),
|
Vector::new(-translation.x, -translation.y),
|
||||||
|renderer| {
|
|renderer| {
|
||||||
draw_content(
|
draw_content(
|
||||||
renderer,
|
renderer,
|
||||||
content_layout,
|
content_layout,
|
||||||
cursor,
|
cursor,
|
||||||
&Rectangle {
|
&Rectangle {
|
||||||
y: bounds.y + offset.y,
|
y: bounds.y + translation.y,
|
||||||
x: bounds.x + offset.x,
|
x: bounds.x + translation.x,
|
||||||
..bounds
|
..bounds
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
@ -932,8 +949,8 @@ pub fn draw<Renderer>(
|
||||||
content_layout,
|
content_layout,
|
||||||
cursor,
|
cursor,
|
||||||
&Rectangle {
|
&Rectangle {
|
||||||
x: bounds.x + offset.x,
|
x: bounds.x + translation.x,
|
||||||
y: bounds.y + offset.y,
|
y: bounds.y + translation.y,
|
||||||
..bounds
|
..bounds
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
@ -1040,6 +1057,20 @@ impl Offset {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn translation(
|
||||||
|
self,
|
||||||
|
viewport: f32,
|
||||||
|
content: f32,
|
||||||
|
alignment: Alignment,
|
||||||
|
) -> f32 {
|
||||||
|
let offset = self.absolute(viewport, content);
|
||||||
|
|
||||||
|
match alignment {
|
||||||
|
Alignment::Start => offset,
|
||||||
|
Alignment::End => ((content - viewport).max(0.0) - offset).max(0.0),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The current [`Viewport`] of the [`Scrollable`].
|
/// The current [`Viewport`] of the [`Scrollable`].
|
||||||
|
|
@ -1086,9 +1117,30 @@ impl State {
|
||||||
pub fn scroll(
|
pub fn scroll(
|
||||||
&mut self,
|
&mut self,
|
||||||
delta: Vector<f32>,
|
delta: Vector<f32>,
|
||||||
|
direction: Direction,
|
||||||
bounds: Rectangle,
|
bounds: Rectangle,
|
||||||
content_bounds: Rectangle,
|
content_bounds: Rectangle,
|
||||||
) {
|
) {
|
||||||
|
let horizontal_alignment = direction
|
||||||
|
.horizontal()
|
||||||
|
.map(|p| p.alignment)
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let vertical_alignment = direction
|
||||||
|
.vertical()
|
||||||
|
.map(|p| p.alignment)
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let align = |alignment: Alignment, delta: f32| match alignment {
|
||||||
|
Alignment::Start => delta,
|
||||||
|
Alignment::End => -delta,
|
||||||
|
};
|
||||||
|
|
||||||
|
let delta = Vector::new(
|
||||||
|
align(horizontal_alignment, delta.x),
|
||||||
|
align(vertical_alignment, delta.y),
|
||||||
|
);
|
||||||
|
|
||||||
if bounds.height < content_bounds.height {
|
if bounds.height < content_bounds.height {
|
||||||
self.offset_y = Offset::Absolute(
|
self.offset_y = Offset::Absolute(
|
||||||
(self.offset_y.absolute(bounds.height, content_bounds.height)
|
(self.offset_y.absolute(bounds.height, content_bounds.height)
|
||||||
|
|
@ -1157,22 +1209,30 @@ impl State {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the scrolling offset of the [`State`], given a [`Direction`],
|
/// Returns the scrolling translation of the [`State`], given a [`Direction`],
|
||||||
/// the bounds of the [`Scrollable`] and its contents.
|
/// the bounds of the [`Scrollable`] and its contents.
|
||||||
pub fn offset(
|
fn translation(
|
||||||
&self,
|
&self,
|
||||||
direction: &Direction,
|
direction: Direction,
|
||||||
bounds: Rectangle,
|
bounds: Rectangle,
|
||||||
content_bounds: Rectangle,
|
content_bounds: Rectangle,
|
||||||
) -> Vector {
|
) -> Vector {
|
||||||
Vector::new(
|
Vector::new(
|
||||||
if direction.horizontal().is_some() {
|
if let Some(horizontal) = direction.horizontal() {
|
||||||
self.offset_x.absolute(bounds.width, content_bounds.width)
|
self.offset_x.translation(
|
||||||
|
bounds.width,
|
||||||
|
content_bounds.width,
|
||||||
|
horizontal.alignment,
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
0.0
|
0.0
|
||||||
},
|
},
|
||||||
if direction.vertical().is_some() {
|
if let Some(vertical) = direction.vertical() {
|
||||||
self.offset_y.absolute(bounds.height, content_bounds.height)
|
self.offset_y.translation(
|
||||||
|
bounds.height,
|
||||||
|
content_bounds.height,
|
||||||
|
vertical.alignment,
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
0.0
|
0.0
|
||||||
},
|
},
|
||||||
|
|
@ -1197,11 +1257,11 @@ impl Scrollbars {
|
||||||
/// Create y and/or x scrollbar(s) if content is overflowing the [`Scrollable`] bounds.
|
/// Create y and/or x scrollbar(s) if content is overflowing the [`Scrollable`] bounds.
|
||||||
fn new(
|
fn new(
|
||||||
state: &State,
|
state: &State,
|
||||||
direction: &Direction,
|
direction: Direction,
|
||||||
bounds: Rectangle,
|
bounds: Rectangle,
|
||||||
content_bounds: Rectangle,
|
content_bounds: Rectangle,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let offset = state.offset(direction, bounds, content_bounds);
|
let translation = state.translation(direction, bounds, content_bounds);
|
||||||
|
|
||||||
let show_scrollbar_x = direction
|
let show_scrollbar_x = direction
|
||||||
.horizontal()
|
.horizontal()
|
||||||
|
|
@ -1216,6 +1276,7 @@ impl Scrollbars {
|
||||||
width,
|
width,
|
||||||
margin,
|
margin,
|
||||||
scroller_width,
|
scroller_width,
|
||||||
|
..
|
||||||
} = *vertical;
|
} = *vertical;
|
||||||
|
|
||||||
// Adjust the height of the vertical scrollbar if the horizontal scrollbar
|
// Adjust the height of the vertical scrollbar if the horizontal scrollbar
|
||||||
|
|
@ -1247,7 +1308,7 @@ impl Scrollbars {
|
||||||
let ratio = bounds.height / content_bounds.height;
|
let ratio = bounds.height / content_bounds.height;
|
||||||
// min height for easier grabbing with super tall content
|
// min height for easier grabbing with super tall content
|
||||||
let scroller_height = (bounds.height * ratio).max(2.0);
|
let scroller_height = (bounds.height * ratio).max(2.0);
|
||||||
let scroller_offset = offset.y * ratio;
|
let scroller_offset = translation.y * ratio;
|
||||||
|
|
||||||
let scroller_bounds = Rectangle {
|
let scroller_bounds = Rectangle {
|
||||||
x: bounds.x + bounds.width
|
x: bounds.x + bounds.width
|
||||||
|
|
@ -1265,6 +1326,7 @@ impl Scrollbars {
|
||||||
scroller: internals::Scroller {
|
scroller: internals::Scroller {
|
||||||
bounds: scroller_bounds,
|
bounds: scroller_bounds,
|
||||||
},
|
},
|
||||||
|
alignment: vertical.alignment,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
|
|
@ -1275,6 +1337,7 @@ impl Scrollbars {
|
||||||
width,
|
width,
|
||||||
margin,
|
margin,
|
||||||
scroller_width,
|
scroller_width,
|
||||||
|
..
|
||||||
} = *horizontal;
|
} = *horizontal;
|
||||||
|
|
||||||
// Need to adjust the width of the horizontal scrollbar if the vertical scrollbar
|
// Need to adjust the width of the horizontal scrollbar if the vertical scrollbar
|
||||||
|
|
@ -1306,7 +1369,7 @@ impl Scrollbars {
|
||||||
let ratio = bounds.width / content_bounds.width;
|
let ratio = bounds.width / content_bounds.width;
|
||||||
// min width for easier grabbing with extra wide content
|
// min width for easier grabbing with extra wide content
|
||||||
let scroller_length = (bounds.width * ratio).max(2.0);
|
let scroller_length = (bounds.width * ratio).max(2.0);
|
||||||
let scroller_offset = offset.x * ratio;
|
let scroller_offset = translation.x * ratio;
|
||||||
|
|
||||||
let scroller_bounds = Rectangle {
|
let scroller_bounds = Rectangle {
|
||||||
x: (scrollbar_bounds.x + scroller_offset - scrollbar_y_width)
|
x: (scrollbar_bounds.x + scroller_offset - scrollbar_y_width)
|
||||||
|
|
@ -1324,6 +1387,7 @@ impl Scrollbars {
|
||||||
scroller: internals::Scroller {
|
scroller: internals::Scroller {
|
||||||
bounds: scroller_bounds,
|
bounds: scroller_bounds,
|
||||||
},
|
},
|
||||||
|
alignment: horizontal.alignment,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
|
|
@ -1390,18 +1454,14 @@ impl Scrollbars {
|
||||||
pub(super) mod internals {
|
pub(super) mod internals {
|
||||||
use crate::core::{Point, Rectangle};
|
use crate::core::{Point, Rectangle};
|
||||||
|
|
||||||
/// The scrollbar of a [`Scrollable`].
|
use super::Alignment;
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
#[derive(Debug, Copy, Clone)]
|
||||||
pub struct Scrollbar {
|
pub struct Scrollbar {
|
||||||
/// The total bounds of the [`Scrollbar`], including the scrollbar, the scroller,
|
|
||||||
/// and the scrollbar margin.
|
|
||||||
pub total_bounds: Rectangle,
|
pub total_bounds: Rectangle,
|
||||||
|
|
||||||
/// The bounds of just the [`Scrollbar`].
|
|
||||||
pub bounds: Rectangle,
|
pub bounds: Rectangle,
|
||||||
|
|
||||||
/// The state of this scrollbar's [`Scroller`].
|
|
||||||
pub scroller: Scroller,
|
pub scroller: Scroller,
|
||||||
|
pub alignment: Alignment,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Scrollbar {
|
impl Scrollbar {
|
||||||
|
|
@ -1416,15 +1476,14 @@ pub(super) mod internals {
|
||||||
grabbed_at: f32,
|
grabbed_at: f32,
|
||||||
cursor_position: Point,
|
cursor_position: Point,
|
||||||
) -> f32 {
|
) -> f32 {
|
||||||
if cursor_position.x < 0.0 && cursor_position.y < 0.0 {
|
let percentage = (cursor_position.y
|
||||||
// cursor position is unavailable! Set to either end or beginning of scrollbar depending
|
- self.bounds.y
|
||||||
// on where the thumb currently is in the track
|
- self.scroller.bounds.height * grabbed_at)
|
||||||
(self.scroller.bounds.y / self.total_bounds.height).round()
|
/ (self.bounds.height - self.scroller.bounds.height);
|
||||||
} else {
|
|
||||||
(cursor_position.y
|
match self.alignment {
|
||||||
- self.bounds.y
|
Alignment::Start => percentage,
|
||||||
- self.scroller.bounds.height * grabbed_at)
|
Alignment::End => 1.0 - percentage,
|
||||||
/ (self.bounds.height - self.scroller.bounds.height)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1434,13 +1493,14 @@ pub(super) mod internals {
|
||||||
grabbed_at: f32,
|
grabbed_at: f32,
|
||||||
cursor_position: Point,
|
cursor_position: Point,
|
||||||
) -> f32 {
|
) -> f32 {
|
||||||
if cursor_position.x < 0.0 && cursor_position.y < 0.0 {
|
let percentage = (cursor_position.x
|
||||||
(self.scroller.bounds.x / self.total_bounds.width).round()
|
- self.bounds.x
|
||||||
} else {
|
- self.scroller.bounds.width * grabbed_at)
|
||||||
(cursor_position.x
|
/ (self.bounds.width - self.scroller.bounds.width);
|
||||||
- self.bounds.x
|
|
||||||
- self.scroller.bounds.width * grabbed_at)
|
match self.alignment {
|
||||||
/ (self.bounds.width - self.scroller.bounds.width)
|
Alignment::Start => percentage,
|
||||||
|
Alignment::End => 1.0 - percentage,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue