use dioxus::prelude::*;
use freya_elements::elements as dioxus_elements;
use freya_elements::events::{keyboard::Key, KeyboardEvent, MouseEvent, WheelEvent};
use freya_hooks::{
theme_with, use_applied_theme, use_focus, use_node, ScrollBarThemeWith, ScrollViewThemeWith,
};
use crate::{
get_container_size, get_corrected_scroll_position, get_scroll_position_from_cursor,
get_scroll_position_from_wheel, get_scrollbar_pos_and_size, is_scrollbar_visible,
manage_key_event, Axis, ScrollBar, ScrollThumb, SCROLLBAR_SIZE, SCROLL_SPEED_MULTIPLIER,
};
#[derive(Props)]
pub struct ScrollViewProps<'a> {
#[props(optional)]
pub theme: Option<ScrollViewThemeWith>,
pub children: Element<'a>,
#[props(default = "vertical".to_string(), into)]
pub direction: String,
#[props(default = true, into)]
pub show_scrollbar: bool,
#[props(default = true, into)]
pub scroll_with_arrows: bool,
}
#[allow(non_snake_case)]
pub fn ScrollView<'a>(cx: Scope<'a, ScrollViewProps<'a>>) -> Element {
let clicking_scrollbar = use_ref::<Option<(Axis, f64)>>(cx, || None);
let clicking_shift = use_ref(cx, || false);
let clicking_alt = use_ref(cx, || false);
let scrolled_y = use_ref(cx, || 0);
let scrolled_x = use_ref(cx, || 0);
let (node_ref, size) = use_node(cx);
let focus = use_focus(cx);
let theme = use_applied_theme!(cx, &cx.props.theme, scroll_view);
let padding = &theme.padding;
let user_container_width = &theme.width;
let user_container_height = &theme.height;
let user_direction = &cx.props.direction;
let show_scrollbar = cx.props.show_scrollbar;
let scroll_with_arrows = cx.props.scroll_with_arrows;
let vertical_scrollbar_is_visible =
is_scrollbar_visible(show_scrollbar, size.inner.height, size.area.height());
let horizontal_scrollbar_is_visible =
is_scrollbar_visible(show_scrollbar, size.inner.width, size.area.width());
let container_width = get_container_size(vertical_scrollbar_is_visible);
let container_height = get_container_size(horizontal_scrollbar_is_visible);
let corrected_scrolled_y = get_corrected_scroll_position(
size.inner.height,
size.area.height(),
*scrolled_y.read() as f32,
);
let corrected_scrolled_x = get_corrected_scroll_position(
size.inner.width,
size.area.width(),
*scrolled_x.read() as f32,
);
let (scrollbar_y, scrollbar_height) =
get_scrollbar_pos_and_size(size.inner.height, size.area.height(), corrected_scrolled_y);
let (scrollbar_x, scrollbar_width) =
get_scrollbar_pos_and_size(size.inner.width, size.area.width(), corrected_scrolled_x);
let onwheel = move |e: WheelEvent| {
let speed_multiplier = if *clicking_alt.read() {
SCROLL_SPEED_MULTIPLIER
} else {
1.0
};
if !*clicking_shift.read() {
let wheel_y = e.get_delta_y() as f32 * speed_multiplier;
let scroll_position_y = get_scroll_position_from_wheel(
wheel_y,
size.inner.height,
size.area.height(),
corrected_scrolled_y,
);
scrolled_y.with_mut(|y| *y = scroll_position_y);
}
let wheel_x = if *clicking_shift.read() {
e.get_delta_y() as f32
} else {
e.get_delta_x() as f32
} * speed_multiplier;
let scroll_position_x = get_scroll_position_from_wheel(
wheel_x,
size.inner.width,
size.area.width(),
corrected_scrolled_x,
);
scrolled_x.with_mut(|x| *x = scroll_position_x);
focus.focus();
};
let onmouseover = move |e: MouseEvent| {
let clicking_scrollbar = clicking_scrollbar.read();
if let Some((Axis::Y, y)) = *clicking_scrollbar {
let coordinates = e.get_element_coordinates();
let cursor_y = coordinates.y - y - size.area.min_y() as f64;
let scroll_position = get_scroll_position_from_cursor(
cursor_y as f32,
size.inner.height,
size.area.height(),
);
scrolled_y.with_mut(|y| *y = scroll_position);
} else if let Some((Axis::X, x)) = *clicking_scrollbar {
let coordinates = e.get_element_coordinates();
let cursor_x = coordinates.x - x - size.area.min_x() as f64;
let scroll_position = get_scroll_position_from_cursor(
cursor_x as f32,
size.inner.width,
size.area.width(),
);
scrolled_x.with_mut(|x| *x = scroll_position);
}
if clicking_scrollbar.is_some() {
focus.focus();
}
};
let onkeydown = move |e: KeyboardEvent| {
if !focus.is_focused() {
return;
}
match &e.key {
Key::Shift => {
clicking_shift.set(true);
}
Key::Alt => {
clicking_alt.set(true);
}
k => {
if !scroll_with_arrows
&& (k == &Key::ArrowUp
|| k == &Key::ArrowRight
|| k == &Key::ArrowDown
|| k == &Key::ArrowLeft)
{
return;
}
let x = corrected_scrolled_x;
let y = corrected_scrolled_y;
let inner_height = size.inner.height;
let inner_width = size.inner.width;
let viewport_height = size.area.height();
let viewport_width = size.area.width();
let (x, y) = manage_key_event(
e,
(x, y),
inner_height,
inner_width,
viewport_height,
viewport_width,
);
scrolled_x.set(x as i32);
scrolled_y.set(y as i32);
}
};
};
let onkeyup = |e: KeyboardEvent| {
if e.key == Key::Shift {
clicking_shift.set(false);
} else if e.key == Key::Alt {
clicking_alt.set(false);
}
};
let onmousedown_y = |e: MouseEvent| {
let coordinates = e.get_element_coordinates();
*clicking_scrollbar.write() = Some((Axis::Y, coordinates.y));
};
let onmousedown_x = |e: MouseEvent| {
let coordinates = e.get_element_coordinates();
*clicking_scrollbar.write() = Some((Axis::X, coordinates.x));
};
let onclick = |_: MouseEvent| {
if clicking_scrollbar.read().is_some() {
*clicking_scrollbar.write() = None;
}
};
let horizontal_scrollbar_size = if horizontal_scrollbar_is_visible {
SCROLLBAR_SIZE
} else {
0
};
let vertical_scrollbar_size = if vertical_scrollbar_is_visible {
SCROLLBAR_SIZE
} else {
0
};
let is_scrolling_x = clicking_scrollbar
.read()
.as_ref()
.map(|f| f.0 == Axis::X)
.unwrap_or_default();
let is_scrolling_y = clicking_scrollbar
.read()
.as_ref()
.map(|f| f.0 == Axis::Y)
.unwrap_or_default();
render!(
rect {
role: "scrollView",
overflow: "clip",
direction: "horizontal",
width: "{user_container_width}",
height: "{user_container_height}",
onglobalclick: onclick,
onglobalmouseover: onmouseover,
onkeydown: onkeydown,
onkeyup: onkeyup,
rect {
direction: "vertical",
width: "{container_width}",
height: "{container_height}",
rect {
overflow: "clip",
padding: "{padding}",
height: "100%",
width: "100%",
direction: "{user_direction}",
offset_y: "{corrected_scrolled_y}",
offset_x: "{corrected_scrolled_x}",
reference: node_ref,
onwheel: onwheel,
&cx.props.children
}
ScrollBar {
width: "100%",
height: "{horizontal_scrollbar_size}",
theme: theme_with!(ScrollBarTheme {
offset_x: scrollbar_x.to_string().into(),
}),
clicking_scrollbar: is_scrolling_x,
ScrollThumb {
clicking_scrollbar: is_scrolling_x,
onmousedown: onmousedown_x,
width: "{scrollbar_width}",
height: "100%"
}
}
}
ScrollBar {
width: "{vertical_scrollbar_size}",
height: "100%",
theme: theme_with!(ScrollBarTheme {
offset_y: scrollbar_y.to_string().into(),
}),
clicking_scrollbar: is_scrolling_y,
ScrollThumb {
clicking_scrollbar: is_scrolling_y,
onmousedown: onmousedown_y,
width: "100%",
height: "{scrollbar_height}"
}
}
}
)
}