diff --git a/CHANGELOG.md b/CHANGELOG.md index f24dd3e7b..498b24d9b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,8 +28,10 @@ Please only add new entries below the [Unreleased](#unreleased---releasedate) he ### Features - **core**: Added `grab_pointer` to grabs all the pointer input. (#pr @wjian23) +- **widgets**: Added the widget of Slider (#pr @wjian23) ### Fixed + - **core**: fix mismatch of providers. (#pr @wjian23) - **core**: Added DeclarerWithSubscription to let Widget `Expanded` accept pipe value. (#pr @wjian23) diff --git a/test_cases/ribir_widgets/slider/tests/slider_widgets_with_default_by_wgpu.png b/test_cases/ribir_widgets/slider/tests/slider_widgets_with_default_by_wgpu.png new file mode 100644 index 000000000..8865e256c Binary files /dev/null and b/test_cases/ribir_widgets/slider/tests/slider_widgets_with_default_by_wgpu.png differ diff --git a/test_cases/ribir_widgets/slider/tests/slider_widgets_with_material_by_wgpu.png b/test_cases/ribir_widgets/slider/tests/slider_widgets_with_material_by_wgpu.png new file mode 100644 index 000000000..b32a28132 Binary files /dev/null and b/test_cases/ribir_widgets/slider/tests/slider_widgets_with_material_by_wgpu.png differ diff --git a/themes/material/src/classes.rs b/themes/material/src/classes.rs index 3d3f93eaf..f21604b8e 100644 --- a/themes/material/src/classes.rs +++ b/themes/material/src/classes.rs @@ -4,6 +4,7 @@ mod checkbox_cls; mod progress_cls; mod radio_cls; mod scrollbar_cls; +mod slider_cls; mod tooltips_cls; pub fn initd_classes() -> Classes { @@ -14,6 +15,7 @@ pub fn initd_classes() -> Classes { progress_cls::init(&mut classes); checkbox_cls::init(&mut classes); tooltips_cls::init(&mut classes); + slider_cls::init(&mut classes); classes } diff --git a/themes/material/src/classes/slider_cls.rs b/themes/material/src/classes/slider_cls.rs new file mode 100644 index 000000000..095345306 --- /dev/null +++ b/themes/material/src/classes/slider_cls.rs @@ -0,0 +1,137 @@ +use ribir_core::prelude::*; +use ribir_widgets::prelude::*; + +use crate::md; + +const INDICATOR_HEIGHT: f32 = 44.; +const TRACK_HEIGHT: f32 = 16.; +const TRACK_WIDTH: f32 = 4.; + +const SMALL_RADIUS: f32 = 2.; +const LARGE_RADIUS: f32 = 8.; +const STOP_INDICATOR_MARGIN: EdgeInsets = EdgeInsets::horizontal(6.); +const STOP_INDICATOR_SIZE: Size = Size::new(4., 4.); + +class_names! { + BASE_SLIDER_TRACK, +} + +macro_rules! stop_indicator_class { + ($($field: ident: $value: expr),* ) => { + style_class! { + v_align: VAlign::Center, + border_radius: Radius::all(SMALL_RADIUS), + margin: STOP_INDICATOR_MARGIN, + clamp: BoxClamp::fixed_size(STOP_INDICATOR_SIZE), + $($field: $value),* + } + }; +} + +pub(super) fn init(classes: &mut Classes) { + classes.insert(BASE_SLIDER_TRACK, |w| { + fn_widget! { + let flex = Provider::of::>(BuildCtx::get()).unwrap(); + part_writer!(&mut flex.flex).transition( + EasingTransition { + easing: easing::LinearEasing, + duration: md::easing::duration::SHORT2, + }); + let w = FatObj::new(w); + @ $w { + clamp: BoxClamp::fixed_height(TRACK_HEIGHT), + } + } + .into_widget() + }); + + classes.insert( + SLIDER_CONTAINER, + style_class!( + clamp: BoxClamp::fixed_height(INDICATOR_HEIGHT) + ), + ); + classes.insert(SLIDER_ACTIVE_TRACK, |w| { + fn_widget! { + let w = FatObj::new(w); + @ $w { + class: BASE_SLIDER_TRACK, + background: Palette::of(BuildCtx::get()).primary(), + border_radius: Radius::new(LARGE_RADIUS, SMALL_RADIUS, LARGE_RADIUS, SMALL_RADIUS), + } + } + .into_widget() + }); + + classes.insert(SLIDER_INACTIVE_TRACK, |w| { + fn_widget! { + let w = FatObj::new(w); + @ $w { + class: BASE_SLIDER_TRACK, + background: Palette::of(BuildCtx::get()).secondary_container(), + border_radius: Radius::new(SMALL_RADIUS, LARGE_RADIUS, SMALL_RADIUS, LARGE_RADIUS), + } + } + .into_widget() + }); + + classes.insert(SLIDER_INDICATOR, |w| { + fn_widget! { + let w = FatObj::new(w); + @ Cursor { + cursor: CursorIcon::Pointer, + @ $w { + v_align: VAlign::Center, + background: Palette::of(BuildCtx::get()).primary(), + margin: EdgeInsets::horizontal(6.), + clamp: BoxClamp::fixed_size(Size::new(TRACK_WIDTH, INDICATOR_HEIGHT)), + } + } + } + .into_widget() + }); + + classes.insert(RANGE_SLIDER_INACTIVE_TRACK_LEFT, |w| { + fn_widget! { + let w = FatObj::new(w); + @ $w { + class: BASE_SLIDER_TRACK, + border_radius: Radius::new(LARGE_RADIUS, SMALL_RADIUS, LARGE_RADIUS, SMALL_RADIUS), + background: Palette::of(BuildCtx::get()).secondary_container(), + } + } + .into_widget() + }); + + classes.insert(RANGE_SLIDER_INACTIVE_TRACK_RIGHT, |w| { + fn_widget! { + let w = FatObj::new(w); + @ $w { + class: BASE_SLIDER_TRACK, + border_radius: Radius::new(SMALL_RADIUS, LARGE_RADIUS, SMALL_RADIUS, LARGE_RADIUS,), + background: Palette::of(BuildCtx::get()).secondary_container(), + } + } + .into_widget() + }); + + classes.insert(RANGE_SLIDER_ACTIVE_TRACK, |w| { + fn_widget! { + let w = FatObj::new(w); + @ $w { + class: BASE_SLIDER_TRACK, + border_radius: Radius::all(SMALL_RADIUS), + background: Palette::of(BuildCtx::get()).primary(), + } + } + .into_widget() + }); + + classes.insert(STOP_INDICATOR_ACTIVE, stop_indicator_class! { + background: Palette::of(BuildCtx::get()).on_primary() + }); + + classes.insert(STOP_INDICATOR_INACTIVE, stop_indicator_class! { + background: Palette::of(BuildCtx::get()).on_secondary_container() + }); +} diff --git a/widgets/src/lib.rs b/widgets/src/lib.rs index ab5b0f5a7..5db001e7a 100644 --- a/widgets/src/lib.rs +++ b/widgets/src/lib.rs @@ -14,6 +14,7 @@ pub mod path; pub mod progress; pub mod radio; pub mod scrollbar; +pub mod slider; pub mod tabs; pub mod text_field; @@ -22,6 +23,6 @@ pub mod prelude { pub use super::{ avatar::*, buttons::*, checkbox::*, common_widget::*, divider::*, grid_view::*, icon::*, input::*, label::*, layout::*, link::*, lists::*, path::*, progress::*, radio::*, scrollbar::*, - tabs::*, text_field::*, transform_box::*, + slider::*, tabs::*, text_field::*, transform_box::*, }; } diff --git a/widgets/src/slider.rs b/widgets/src/slider.rs new file mode 100644 index 000000000..b7e62114f --- /dev/null +++ b/widgets/src/slider.rs @@ -0,0 +1,389 @@ +use std::{mem::swap, ops::Range}; + +use ribir_core::prelude::*; + +use crate::{ + layout::{Expanded, JustifyContent, Row}, + prelude::Stack, +}; + +class_names! { + #[doc = "Class name for the slider container"] + SLIDER_CONTAINER, + #[doc = "Class name for the slider indicator"] + SLIDER_INDICATOR, + #[doc = "Class name for the slider track"] + SLIDER_ACTIVE_TRACK, + #[doc = "Class name for the slider inactive track"] + SLIDER_INACTIVE_TRACK, + #[doc = "Class name for the left inactive track of range slider"] + RANGE_SLIDER_INACTIVE_TRACK_LEFT, + #[doc = "Class name for the right inactive track of range slider"] + RANGE_SLIDER_INACTIVE_TRACK_RIGHT, + #[doc = "Class name for the active track of range slider"] + RANGE_SLIDER_ACTIVE_TRACK, + #[doc="Class name for the active stop indicator"] + STOP_INDICATOR_ACTIVE, + #[doc="Class name for the inactive stop indicator"] + STOP_INDICATOR_INACTIVE, +} + +#[derive(Declare)] +pub struct Slider { + pub value: f32, + #[declare(default = 100.)] + pub max: f32, + #[declare(default = 0.)] + pub min: f32, + #[declare(default)] + pub divisions: Option, +} + +impl Slider { + fn set_to(&mut self, mut v: f32) { + v = v.clamp(0., 1.); + if let Some(divisions) = self.divisions { + if divisions > 0 { + v = (v * divisions as f32).round() / (divisions as f32); + } + } + + self.value = (self.min + v * (self.max - self.min)).clamp(self.min, self.max); + } + + fn ratio(&self) -> f32 { + if self.max == self.min { + return 1.; + } + let mut v = (self.value - self.min) / (self.max - self.min); + v = v.clamp(0., 1.); + if let Some(divisions) = self.divisions { + if divisions > 0 { + v = (v * divisions as f32).round() / (divisions as f32) + } + } + v + } + + fn validate(mut this: WriteRef) { + if this.max < this.min { + let Self { max, min, .. } = &mut *this; + swap(max, min); + } + + if this.value < this.min { + this.value = this.min; + } + + if this.value > this.max { + this.value = this.max; + } + } + + fn stop_indicator_track(&self) -> Option> { + let divisions = self.divisions?; + if divisions == 0 { + return None; + } + let active = (self.ratio() * divisions as f32) as usize; + Some(stop_indicator_track(divisions + 1, 0..active, vec![active])) + } +} + +fn precision(min: f32, max: f32) -> usize { + ((max - min).log10().floor() - 2.).min(-2.).abs() as usize +} + +impl Compose for Slider { + fn compose(this: impl StateWriter) -> Widget<'static> { + fn_widget! { + let u = this.modifies().subscribe(move |_| Slider::validate($this.write())); + + let mut row = @Row { align_items: Align::Center }; + let track1 = @Expanded { flex: pipe!($this.ratio())}; + let track2 = @Expanded { flex: pipe!($this.ratio()).map(|v| 1. - v)}; + + let drag_handle = Stateful::new(None); + @ Stack { + class: SLIDER_CONTAINER, + @ $row { + v_align: VAlign::Center, + on_tap: move |e| { + let width = $row.layout_size().width; + $this.write().set_to(e.position().x / width); + }, + on_disposed: move |_| u.unsubscribe(), + @ { + Provider::new(Box::new(track1.clone_writer())).with_child( + fn_widget!{ @$track1 { @Void { class: SLIDER_ACTIVE_TRACK } } } + ) + } + @Void { + class: SLIDER_INDICATOR , + on_tap: move |e| e.stop_propagation(), + on_pointer_down: move |e| { + if let Some(handle) = e.window().grab_pointer(e.current_target()) { + *$drag_handle.write() = Some( + (handle.guard(), e.global_pos().x, $this.ratio()) + ); + } + }, + on_pointer_move: move|e| if let Some((_, pos, ratio)) = $drag_handle.as_ref() { + let width = $row.layout_size().width; + let val = ratio + (e.global_pos().x - pos) / width; + $this.write().set_to(val); + }, + on_pointer_up: move |_| { + $drag_handle.write().take(); + }, + tooltips: pipe!($this.value).map(move |v| { + let precision = precision($this.min, $this.max); + format!("{:.1$}", v, precision) + }), + } + @ { + Provider::new(Box::new(track2.clone_writer())).with_child( + fn_widget! { @ $track2 { @Void { class: SLIDER_INACTIVE_TRACK } } } + ) + } + } + + @{ pipe!($this.stop_indicator_track()) } + } + } + .into_widget() + } +} + +#[derive(Declare)] +pub struct RangeSlider { + pub start: f32, + pub end: f32, + #[declare(default = 100.)] + pub max: f32, + #[declare(default = 0.)] + pub min: f32, + #[declare(default)] + pub divisions: Option, +} + +impl RangeSlider { + fn set_ratio(&mut self, mut ratio: f32) { + ratio = ratio.clamp(0., 1.); + let val = self.convert_ratio(ratio); + if (self.start - val).abs() < (self.end - val).abs() { + self.start = val; + } else { + self.end = val; + } + } + + fn set_start_ratio(&mut self, ratio: f32) { + self.start = self + .convert_ratio(ratio) + .min(self.end) + .max(self.min); + } + + fn set_end_ratio(&mut self, ratio: f32) { + self.end = self + .convert_ratio(ratio) + .max(self.start) + .min(self.max); + } + + fn convert_ratio(&self, mut ratio: f32) -> f32 { + if let Some(divisions) = self.divisions { + if divisions > 1 { + ratio = (ratio * divisions as f32).round() / (divisions as f32); + } + } + self.min + ratio * (self.max - self.min) + } + + fn ratio(&self, v: f32) -> f32 { + if self.max == self.min { + return 1.; + } + let mut v = (v - self.min) / (self.max - self.min); + v = v.clamp(0., 1.); + if let Some(divisions) = self.divisions { + if divisions > 0 { + v = (v * divisions as f32).round() / (divisions as f32); + } + } + v + } + + fn start_ratio(&self) -> f32 { self.ratio(self.start) } + + fn end_ratio(&self) -> f32 { self.ratio(self.end) } + + fn validate(mut this: WriteRef) { + if this.max < this.min { + let Self { max, min, .. } = &mut *this; + swap(max, min); + } + + if this.start > this.end { + let Self { start, end, .. } = &mut *this; + swap(start, end); + } + + if this.start < this.min { + this.start = this.min; + } + + if this.end > this.max { + this.end = this.max; + } + } + + fn stop_indicator_track(&self) -> Option> { + let divisions = self.divisions?; + if divisions == 0 { + return None; + } + let start = (self.start_ratio() * divisions as f32) as usize; + let end = (self.end_ratio() * divisions as f32) as usize; + Some(stop_indicator_track(divisions + 1, start..end + 1, vec![start, end])) + } +} + +impl Compose for RangeSlider { + fn compose(this: impl StateWriter) -> Widget<'static> { + fn_widget! { + let u = this.modifies().subscribe(move |_| RangeSlider::validate($this.write())); + + let mut row = @Row { align_items: Align::Center }; + let track1 = @Expanded { flex: pipe!($this.start_ratio()) }; + let track2 = @Expanded { flex: pipe!($this.end_ratio() - $this.start_ratio()) }; + let track3 = @Expanded { flex: pipe!($this.end_ratio()).map(|v| 1. - v) }; + let drag_handle1 = Stateful::new(None); + let drag_handle2 = Stateful::new(None); + @Stack { + class: SLIDER_CONTAINER, + @ $row { + v_align: VAlign::Center, + on_tap: move |e| { + let width = $row.layout_size().width; + $this.write().set_ratio(e.position().x / width); + }, + on_disposed: move |_| u.unsubscribe(), + @ { + Provider::new(Box::new(track1.clone_writer())).with_child( + fn_widget!{ @ $track1 { @Void { class: RANGE_SLIDER_INACTIVE_TRACK_LEFT } } } + ) + } + @Void { + class: SLIDER_INDICATOR, + tooltips: pipe!($this.start).map(move |v| { + let precision = precision($this.min, $this.max); + format!("{:.1$}", v, precision) + }), + on_tap: move |e| e.stop_propagation(), + on_pointer_down: move |e| { + if let Some(handle) = e.window().grab_pointer(e.current_target()) { + *$drag_handle1.write() = Some( + (handle.guard(), e.global_pos().x, $this.start_ratio()) + ); + } + }, + on_pointer_move: move |e| { + if let Some((_, pos, ratio)) = $drag_handle1.as_ref() { + let width = $row.layout_size().width; + let val = ratio + (e.global_pos().x - pos) / width; + $this.write().set_start_ratio(val); + } + }, + on_pointer_up: move |_| { $drag_handle1.write().take(); } + } + @ { + Provider::new(Box::new(track2.clone_writer())).with_child( + fn_widget! { @ $track2 { @Void { class: RANGE_SLIDER_ACTIVE_TRACK } } } + ) + } + @Void { + class: SLIDER_INDICATOR, + tooltips: pipe!($this.end).map(move |v| { + let precision = precision($this.min, $this.max); + format!("{:.1$}", v, precision) + }), + on_tap: move |e| e.stop_propagation(), + on_pointer_down: move |e| { + if let Some(handle) = e.window().grab_pointer(e.current_target()) { + *$drag_handle2.write() = Some( + (handle.guard(), e.global_pos().x, $this.end_ratio()) + ); + } + }, + on_pointer_move: move |e| { + if let Some((_, pos, ratio)) = $drag_handle2.as_ref() { + let width = $row.layout_size().width; + let val = ratio + (e.global_pos().x - pos) / width; + $this.write().set_end_ratio(val); + } + }, + on_pointer_up: move |_| { $drag_handle2.write().take(); } + } + @ { + Provider::new(Box::new(track3.clone_writer())).with_child( + fn_widget! { @ $track3 { @Void { class: RANGE_SLIDER_INACTIVE_TRACK_RIGHT } } } + ) + } + } + @{ pipe!($this.stop_indicator_track()) } + } + } + .into_widget() + } +} + +fn stop_indicator_track(cnt: usize, actives: Range, filter: Vec) -> Widget<'static> { + fn_widget!( + @IgnorePointer { + ignore: true, + @Row { + v_align: VAlign::Center, + align_items: Align::Center, + justify_content: JustifyContent::SpaceBetween, + @ { + (0..cnt).map(move |i| { + @Void { + class: if actives.contains(&i) { + STOP_INDICATOR_ACTIVE + } else { + STOP_INDICATOR_INACTIVE + }, + visible: !filter.contains(&i), + } + }) + } + } + } + ) + .into_widget() +} + +#[cfg(test)] +mod tests { + use ribir_core::test_helper::*; + use ribir_dev_helper::*; + + use super::*; + use crate::prelude::*; + + widget_image_tests!( + slider_widgets, + WidgetTester::new(self::column! { + justify_content: JustifyContent::SpaceAround, + align_items: Align::Center, + @Slider { value: 32. } + @Slider { value: 32., divisions: Some(10) } + @RangeSlider { start: 10., end: 73. } + @RangeSlider { start: 10., end: 73., divisions: Some(10) } + }) + .with_wnd_size(Size::new(300., 200.)) + .with_comparison(0.002) + ); +}