diff options
| author | pml68 <contact@pml68.dev> | 2025-04-08 18:12:10 +0200 |
|---|---|---|
| committer | pml68 <contact@pml68.dev> | 2025-04-15 23:49:01 +0200 |
| commit | 00d296b230e41dbfaf33d1ba374beadf9d113948 (patch) | |
| tree | 321c1f3c40f8de8e665b830c67cf116ff29017e3 /material_theme/src/scrollable.rs | |
| parent | feat(material_theme): implement Catalog for iced_dialog (`dialog` feature) (diff) | |
| download | iced-builder-00d296b230e41dbfaf33d1ba374beadf9d113948.tar.gz | |
feat(material_theme): impl `scrollable::Catalog`
Diffstat (limited to '')
| -rw-r--r-- | material_theme/src/scrollable.rs | 107 |
1 files changed, 107 insertions, 0 deletions
diff --git a/material_theme/src/scrollable.rs b/material_theme/src/scrollable.rs new file mode 100644 index 0000000..c2dde67 --- /dev/null +++ b/material_theme/src/scrollable.rs @@ -0,0 +1,107 @@ +use iced_widget::core::{Border, border}; +use iced_widget::scrollable::{ + Catalog, Rail, Scroller, Status, Style, StyleFn, +}; + +use super::Theme; +use super::container::surface_container; +use super::utils::mix; +use crate::utils::{HOVERED_LAYER_OPACITY, PRESSED_LAYER_OPACITY}; + +impl Catalog for Theme { + type Class<'a> = StyleFn<'a, Self>; + + fn default<'a>() -> Self::Class<'a> { + Box::new(default) + } + + fn style(&self, class: &Self::Class<'_>, status: Status) -> Style { + class(self, status) + } +} + +pub fn default(theme: &Theme, status: Status) -> Style { + let colors = theme.colorscheme.surface; + + let rail = Rail { + background: None, + scroller: Scroller { + color: colors.on_surface, + border: border::rounded(400), + }, + border: Border::default(), + }; + + let style = Style { + container: surface_container(theme), + vertical_rail: rail, + horizontal_rail: rail, + gap: None, + }; + + match status { + Status::Active { .. } => style, + Status::Hovered { + is_horizontal_scrollbar_hovered, + is_vertical_scrollbar_hovered, + .. + } => { + let hovered_rail = Rail { + scroller: Scroller { + color: mix( + colors.on_surface, + colors.color, + HOVERED_LAYER_OPACITY, + ), + border: border::rounded(400), + }, + ..rail + }; + + Style { + horizontal_rail: if is_horizontal_scrollbar_hovered { + hovered_rail + } else { + rail + }, + vertical_rail: if is_vertical_scrollbar_hovered { + hovered_rail + } else { + rail + }, + ..style + } + } + Status::Dragged { + is_horizontal_scrollbar_dragged, + is_vertical_scrollbar_dragged, + .. + } => { + let dragged_rail = Rail { + scroller: Scroller { + color: mix( + colors.on_surface, + colors.color, + PRESSED_LAYER_OPACITY, + ), + border: border::rounded(400), + }, + ..rail + }; + + Style { + horizontal_rail: if is_horizontal_scrollbar_dragged { + dragged_rail + } else { + rail + }, + vertical_rail: if is_vertical_scrollbar_dragged { + dragged_rail + } else { + rail + }, + ..style + } + } + } +} |
