2023-01-06 01:39:09 +01:00
|
|
|
// Copyright 2022 System76 <info@system76.com>
|
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
|
2023-01-06 16:18:25 +01:00
|
|
|
use super::state::{Selectable, State};
|
2023-01-03 19:35:34 +01:00
|
|
|
use super::style::StyleSheet;
|
2023-01-04 05:37:20 +01:00
|
|
|
use super::widget::{SegmentedButton, SegmentedVariant};
|
|
|
|
|
|
|
|
|
|
use iced::{Length, Rectangle, Size};
|
|
|
|
|
use iced_native::layout;
|
2023-01-03 19:35:34 +01:00
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
/// A type marker defining the horizontal variant of a [`SegmentedButton`].
|
|
|
|
|
pub struct Horizontal;
|
2023-01-03 19:35:34 +01:00
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
/// Horizontal [`SegmentedButton`].
|
2023-01-06 16:18:25 +01:00
|
|
|
pub type HorizontalSegmentedButton<'a, Selection, Message, Renderer> =
|
|
|
|
|
SegmentedButton<'a, Horizontal, Selection, Message, Renderer>;
|
2023-01-04 05:37:20 +01:00
|
|
|
|
|
|
|
|
/// Horizontal implementation of the [`SegmentedButton`].
|
2023-01-03 19:35:34 +01:00
|
|
|
#[must_use]
|
2023-01-06 16:18:25 +01:00
|
|
|
pub fn horizontal_segmented_button<Selection, Message, Renderer, Data>(
|
|
|
|
|
state: &State<Selection, Data>,
|
|
|
|
|
) -> SegmentedButton<Horizontal, Selection, Message, Renderer>
|
2023-01-03 19:35:34 +01:00
|
|
|
where
|
2023-01-06 01:39:09 +01:00
|
|
|
Renderer: iced_native::Renderer
|
|
|
|
|
+ iced_native::text::Renderer
|
|
|
|
|
+ iced_native::image::Renderer
|
|
|
|
|
+ iced_native::svg::Renderer,
|
2023-01-03 19:35:34 +01:00
|
|
|
Renderer::Theme: StyleSheet,
|
2023-01-06 16:18:25 +01:00
|
|
|
Selection: Selectable,
|
2023-01-03 19:35:34 +01:00
|
|
|
{
|
2023-01-04 05:37:20 +01:00
|
|
|
SegmentedButton::new(&state.inner)
|
2023-01-03 19:35:34 +01:00
|
|
|
}
|
|
|
|
|
|
2023-01-06 16:18:25 +01:00
|
|
|
impl<'a, Selection, Message, Renderer> SegmentedVariant
|
|
|
|
|
for SegmentedButton<'a, Horizontal, Selection, Message, Renderer>
|
2023-01-03 19:35:34 +01:00
|
|
|
where
|
2023-01-06 01:39:09 +01:00
|
|
|
Renderer: iced_native::Renderer
|
|
|
|
|
+ iced_native::text::Renderer
|
|
|
|
|
+ iced_native::image::Renderer
|
|
|
|
|
+ iced_native::svg::Renderer,
|
2023-01-03 19:35:34 +01:00
|
|
|
Renderer::Theme: StyleSheet,
|
2023-01-06 16:18:25 +01:00
|
|
|
Selection: Selectable,
|
2023-01-03 19:35:34 +01:00
|
|
|
{
|
2023-01-04 05:37:20 +01:00
|
|
|
type Renderer = Renderer;
|
2023-01-03 19:35:34 +01:00
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
fn variant_appearance(
|
|
|
|
|
theme: &<Self::Renderer as iced_native::Renderer>::Theme,
|
|
|
|
|
style: &<<Self::Renderer as iced_native::Renderer>::Theme as StyleSheet>::Style,
|
|
|
|
|
) -> super::Appearance {
|
|
|
|
|
theme.horizontal(style)
|
2023-01-03 19:35:34 +01:00
|
|
|
}
|
|
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
#[allow(clippy::cast_precision_loss)]
|
|
|
|
|
fn variant_button_bounds(&self, mut bounds: Rectangle, nth: usize) -> Rectangle {
|
|
|
|
|
let num = self.state.buttons.len();
|
|
|
|
|
if num != 0 {
|
|
|
|
|
let spacing = f32::from(self.spacing);
|
|
|
|
|
bounds.width = (bounds.width - (num as f32 * spacing) + spacing) / num as f32;
|
2023-01-03 19:35:34 +01:00
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
if nth != 0 {
|
|
|
|
|
bounds.x += (nth as f32 * bounds.width) + (nth as f32 * spacing);
|
2023-01-03 19:35:34 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
bounds
|
2023-01-03 19:35:34 +01:00
|
|
|
}
|
|
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
#[allow(clippy::cast_precision_loss)]
|
|
|
|
|
#[allow(clippy::cast_possible_truncation)]
|
|
|
|
|
#[allow(clippy::cast_sign_loss)]
|
|
|
|
|
fn variant_layout(&self, renderer: &Renderer, limits: &layout::Limits) -> layout::Node {
|
2023-01-03 19:35:34 +01:00
|
|
|
let limits = limits.width(self.width);
|
|
|
|
|
let text_size = renderer.default_size();
|
|
|
|
|
|
2023-01-04 05:37:20 +01:00
|
|
|
let (mut width, height) = self.max_button_dimensions(renderer, text_size, limits.max());
|
|
|
|
|
|
|
|
|
|
let num = self.state.buttons.len();
|
|
|
|
|
let spacing = f32::from(self.spacing);
|
|
|
|
|
|
|
|
|
|
if num != 0 {
|
|
|
|
|
width = (num as f32 * width) + (num as f32 * spacing) - spacing;
|
2023-01-03 19:35:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let size = limits
|
|
|
|
|
.height(Length::Units(height as u16))
|
|
|
|
|
.resolve(Size::new(width, height));
|
|
|
|
|
|
|
|
|
|
layout::Node::new(size)
|
|
|
|
|
}
|
|
|
|
|
}
|