summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLibravatar Héctor <hector@hecrj.dev>2025-02-23 07:10:39 +0100
committerLibravatar GitHub <noreply@github.com>2025-02-23 07:10:39 +0100
commitfd1cfc05eae2d2527fc5fe0a4b30b133283c7095 (patch)
tree84179bc878ad1a92493fdb970344e056b4c4675e
parentf1ed99cb47997e1d194a41e7cdf2846f8eb5f8fa (diff)
parent873311558f1b96f7a40ee73ddb270c396607b9bb (diff)
downloadiced-fd1cfc05eae2d2527fc5fe0a4b30b133283c7095.tar.gz
iced-fd1cfc05eae2d2527fc5fe0a4b30b133283c7095.tar.bz2
iced-fd1cfc05eae2d2527fc5fe0a4b30b133283c7095.zip
Merge pull request #2811 from iced-rs/palette-tweaks
Tweak `Palette` Generation
-rw-r--r--core/src/theme/palette.rs63
-rw-r--r--examples/styling/src/main.rs29
-rw-r--r--examples/todos/snapshots/creates_a_new_task.sha2562
-rw-r--r--widget/src/button.rs4
-rw-r--r--widget/src/checkbox.rs27
-rw-r--r--widget/src/container.rs4
-rw-r--r--widget/src/progress_bar.rs5
-rw-r--r--widget/src/slider.rs8
-rw-r--r--widget/src/text_input.rs4
9 files changed, 100 insertions, 46 deletions
diff --git a/core/src/theme/palette.rs b/core/src/theme/palette.rs
index b4acaa83..ed49063c 100644
--- a/core/src/theme/palette.rs
+++ b/core/src/theme/palette.rs
@@ -29,7 +29,7 @@ impl Palette {
pub const LIGHT: Self = Self {
background: Color::WHITE,
text: Color::BLACK,
- primary: color!(0x5e7ce2),
+ primary: color!(0x5865F2),
success: color!(0x12664f),
warning: color!(0xffc14e),
danger: color!(0xc3423f),
@@ -37,9 +37,9 @@ impl Palette {
/// The built-in dark variant of a [`Palette`].
pub const DARK: Self = Self {
- background: color!(0x202225),
+ background: color!(0x2B2D31),
text: Color::from_rgb(0.90, 0.90, 0.90),
- primary: color!(0x5e7ce2),
+ primary: color!(0x5865F2),
success: color!(0x12664f),
warning: color!(0xffc14e),
danger: color!(0xc3423f),
@@ -205,9 +205,9 @@ impl Palette {
///
/// [Kanagawa]: https://github.com/rebelot/kanagawa.nvim
pub const KANAGAWA_WAVE: Self = Self {
- background: color!(0x363646), // Sumi Ink 3
- text: color!(0xCD7BA), // Fuji White
- primary: color!(0x2D4F67), // Wave Blue 2
+ background: color!(0x1f1f28), // Sumi Ink 3
+ text: color!(0xDCD7BA), // Fuji White
+ primary: color!(0x7FB4CA), // Wave Blue
success: color!(0x76946A), // Autumn Green
warning: color!(0xff9e3b), // Ronin Yellow
danger: color!(0xC34043), // Autumn Red
@@ -231,7 +231,7 @@ impl Palette {
pub const KANAGAWA_LOTUS: Self = Self {
background: color!(0xf2ecbc), // Lotus White 3
text: color!(0x545464), // Lotus Ink 1
- primary: color!(0xc9cbd1), // Lotus Violet 3
+ primary: color!(0x4d699b), // Lotus Blue
success: color!(0x6f894e), // Lotus Green
warning: color!(0xe98a00), // Lotus Orange 2
danger: color!(0xc84053), // Lotus Red
@@ -453,22 +453,30 @@ impl Pair {
pub struct Background {
/// The base background color.
pub base: Pair,
+ /// The weakest version of the base background color.
+ pub weakest: Pair,
/// A weaker version of the base background color.
pub weak: Pair,
/// A stronger version of the base background color.
pub strong: Pair,
+ /// The strongest version of the base background color.
+ pub strongest: Pair,
}
impl Background {
/// Generates a set of [`Background`] colors from the base and text colors.
pub fn new(base: Color, text: Color) -> Self {
- let weak = mix(base, text, 0.15);
- let strong = mix(base, text, 0.40);
+ let weakest = deviate(base, 0.03);
+ let weak = muted(deviate(base, 0.1));
+ let strong = muted(deviate(base, 0.2));
+ let strongest = muted(deviate(base, 0.3));
Self {
base: Pair::new(base, text),
+ weakest: Pair::new(weakest, text),
weak: Pair::new(weak, text),
strong: Pair::new(strong, text),
+ strongest: Pair::new(strongest, text),
}
}
}
@@ -627,10 +635,18 @@ fn deviate(color: Color, amount: f32) -> Color {
if is_dark(color) {
lighten(color, amount)
} else {
- darken(color, amount)
+ darken(color, amount * 0.7)
}
}
+fn muted(color: Color) -> Color {
+ let mut hsl = to_hsl(color);
+
+ hsl.saturation = hsl.saturation.min(0.5);
+
+ from_hsl(hsl)
+}
+
fn mix(a: Color, b: Color, factor: f32) -> Color {
let a_lin = Rgb::from(a).into_linear();
let b_lin = Rgb::from(b).into_linear();
@@ -641,16 +657,25 @@ fn mix(a: Color, b: Color, factor: f32) -> Color {
fn readable(background: Color, text: Color) -> Color {
if is_readable(background, text) {
- text
- } else {
- let white_contrast = relative_contrast(background, Color::WHITE);
- let black_contrast = relative_contrast(background, Color::BLACK);
+ return text;
+ }
- if white_contrast >= black_contrast {
- Color::WHITE
- } else {
- Color::BLACK
- }
+ let improve = if is_dark(background) { lighten } else { darken };
+
+ // TODO: Compute factor from relative contrast value
+ let candidate = improve(text, 0.1);
+
+ if is_readable(background, candidate) {
+ return candidate;
+ }
+
+ let white_contrast = relative_contrast(background, Color::WHITE);
+ let black_contrast = relative_contrast(background, Color::BLACK);
+
+ if white_contrast >= black_contrast {
+ mix(Color::WHITE, background, 0.05)
+ } else {
+ mix(Color::BLACK, background, 0.05)
}
}
diff --git a/examples/styling/src/main.rs b/examples/styling/src/main.rs
index 594be4a7..fce2b162 100644
--- a/examples/styling/src/main.rs
+++ b/examples/styling/src/main.rs
@@ -1,8 +1,8 @@
use iced::keyboard;
use iced::widget::{
- button, center, checkbox, column, horizontal_rule, pick_list, progress_bar,
- row, scrollable, slider, text, text_input, toggler, vertical_rule,
- vertical_space,
+ button, center, checkbox, column, container, horizontal_rule, pick_list,
+ progress_bar, row, scrollable, slider, text, text_input, toggler,
+ vertical_rule, vertical_space,
};
use iced::{Center, Element, Fill, Subscription, Theme};
@@ -90,9 +90,9 @@ impl Styling {
let danger = styled_button("Danger").style(button::danger);
let slider =
- slider(0.0..=100.0, self.slider_value, Message::SliderChanged);
+ || slider(0.0..=100.0, self.slider_value, Message::SliderChanged);
- let progress_bar = progress_bar(0.0..=100.0, self.slider_value);
+ let progress_bar = || progress_bar(0.0..=100.0, self.slider_value);
let scrollable = scrollable(column![
"Scroll me!",
@@ -110,6 +110,20 @@ impl Styling {
.on_toggle(Message::TogglerToggled)
.spacing(10);
+ let card = {
+ container(
+ column![
+ text("Card Example").size(24),
+ slider(),
+ progress_bar(),
+ ]
+ .spacing(20),
+ )
+ .width(Fill)
+ .padding(20)
+ .style(container::bordered_box)
+ };
+
let content = column![
choose_theme,
horizontal_rule(38),
@@ -117,8 +131,8 @@ impl Styling {
row![primary, success, warning, danger]
.spacing(10)
.align_y(Center),
- slider,
- progress_bar,
+ slider(),
+ progress_bar(),
row![
scrollable,
vertical_rule(38),
@@ -127,6 +141,7 @@ impl Styling {
.spacing(10)
.height(100)
.align_y(Center),
+ card
]
.spacing(20)
.padding(20)
diff --git a/examples/todos/snapshots/creates_a_new_task.sha256 b/examples/todos/snapshots/creates_a_new_task.sha256
index 193132c5..cba60539 100644
--- a/examples/todos/snapshots/creates_a_new_task.sha256
+++ b/examples/todos/snapshots/creates_a_new_task.sha256
@@ -1 +1 @@
-3160686067cb7c738802009cdf2f3c5f5a5bd8c89ada70517388b7adbe64c313 \ No newline at end of file
+0ef7325a79ce31c83759529ed478c8b4848c40c2867193624ef9673b9e21ff53 \ No newline at end of file
diff --git a/widget/src/button.rs b/widget/src/button.rs
index 1b51065d..d4500888 100644
--- a/widget/src/button.rs
+++ b/widget/src/button.rs
@@ -591,12 +591,12 @@ impl Catalog for Theme {
/// A primary button; denoting a main action.
pub fn primary(theme: &Theme, status: Status) -> Style {
let palette = theme.extended_palette();
- let base = styled(palette.primary.strong);
+ let base = styled(palette.primary.base);
match status {
Status::Active | Status::Pressed => base,
Status::Hovered => Style {
- background: Some(Background::Color(palette.primary.base.color)),
+ background: Some(Background::Color(palette.primary.strong.color)),
..base
},
Status::Disabled => disabled(base),
diff --git a/widget/src/checkbox.rs b/widget/src/checkbox.rs
index 6ed3e080..3c1ef276 100644
--- a/widget/src/checkbox.rs
+++ b/widget/src/checkbox.rs
@@ -555,18 +555,21 @@ pub fn primary(theme: &Theme, status: Status) -> Style {
match status {
Status::Active { is_checked } => styled(
palette.primary.strong.text,
+ palette.background.strongest.color,
palette.background.base,
- palette.primary.strong,
+ palette.primary.base,
is_checked,
),
Status::Hovered { is_checked } => styled(
palette.primary.strong.text,
+ palette.background.strongest.color,
palette.background.weak,
- palette.primary.base,
+ palette.primary.strong,
is_checked,
),
Status::Disabled { is_checked } => styled(
palette.primary.strong.text,
+ palette.background.weak.color,
palette.background.weak,
palette.background.strong,
is_checked,
@@ -581,18 +584,21 @@ pub fn secondary(theme: &Theme, status: Status) -> Style {
match status {
Status::Active { is_checked } => styled(
palette.background.base.text,
+ palette.background.strongest.color,
palette.background.base,
palette.background.strong,
is_checked,
),
Status::Hovered { is_checked } => styled(
palette.background.base.text,
+ palette.background.strongest.color,
palette.background.weak,
palette.background.strong,
is_checked,
),
Status::Disabled { is_checked } => styled(
palette.background.strong.color,
+ palette.background.weak.color,
palette.background.weak,
palette.background.weak,
is_checked,
@@ -607,18 +613,21 @@ pub fn success(theme: &Theme, status: Status) -> Style {
match status {
Status::Active { is_checked } => styled(
palette.success.base.text,
+ palette.background.weak.color,
palette.background.base,
palette.success.base,
is_checked,
),
Status::Hovered { is_checked } => styled(
palette.success.base.text,
+ palette.background.strongest.color,
palette.background.weak,
- palette.success.base,
+ palette.success.strong,
is_checked,
),
Status::Disabled { is_checked } => styled(
palette.success.base.text,
+ palette.background.weak.color,
palette.background.weak,
palette.success.weak,
is_checked,
@@ -633,18 +642,21 @@ pub fn danger(theme: &Theme, status: Status) -> Style {
match status {
Status::Active { is_checked } => styled(
palette.danger.base.text,
+ palette.background.strongest.color,
palette.background.base,
palette.danger.base,
is_checked,
),
Status::Hovered { is_checked } => styled(
palette.danger.base.text,
+ palette.background.strongest.color,
palette.background.weak,
- palette.danger.base,
+ palette.danger.strong,
is_checked,
),
Status::Disabled { is_checked } => styled(
palette.danger.base.text,
+ palette.background.weak.color,
palette.background.weak,
palette.danger.weak,
is_checked,
@@ -654,6 +666,7 @@ pub fn danger(theme: &Theme, status: Status) -> Style {
fn styled(
icon_color: Color,
+ border_color: Color,
base: palette::Pair,
accent: palette::Pair,
is_checked: bool,
@@ -668,7 +681,11 @@ fn styled(
border: Border {
radius: 2.0.into(),
width: 1.0,
- color: accent.color,
+ color: if is_checked {
+ accent.color
+ } else {
+ border_color
+ },
},
text_color: None,
}
diff --git a/widget/src/container.rs b/widget/src/container.rs
index da5b436f..82774186 100644
--- a/widget/src/container.rs
+++ b/widget/src/container.rs
@@ -703,10 +703,10 @@ pub fn bordered_box(theme: &Theme) -> Style {
let palette = theme.extended_palette();
Style {
- background: Some(palette.background.weak.color.into()),
+ background: Some(palette.background.weakest.color.into()),
border: Border {
width: 1.0,
- radius: 0.0.into(),
+ radius: 5.0.into(),
color: palette.background.strong.color,
},
..Style::default()
diff --git a/widget/src/progress_bar.rs b/widget/src/progress_bar.rs
index 554e8a44..a9b4eb65 100644
--- a/widget/src/progress_bar.rs
+++ b/widget/src/progress_bar.rs
@@ -288,10 +288,7 @@ impl Catalog for Theme {
pub fn primary(theme: &Theme) -> Style {
let palette = theme.extended_palette();
- styled(
- palette.background.strong.color,
- palette.primary.strong.color,
- )
+ styled(palette.background.strong.color, palette.primary.base.color)
}
/// The secondary style of a [`ProgressBar`].
diff --git a/widget/src/slider.rs b/widget/src/slider.rs
index 1908abc9..1a2f8b9d 100644
--- a/widget/src/slider.rs
+++ b/widget/src/slider.rs
@@ -670,14 +670,14 @@ pub fn default(theme: &Theme, status: Status) -> Style {
let palette = theme.extended_palette();
let color = match status {
- Status::Active => palette.primary.strong.color,
- Status::Hovered => palette.primary.base.color,
- Status::Dragged => palette.primary.strong.color,
+ Status::Active => palette.primary.base.color,
+ Status::Hovered => palette.primary.strong.color,
+ Status::Dragged => palette.primary.weak.color,
};
Style {
rail: Rail {
- backgrounds: (color.into(), palette.secondary.base.color.into()),
+ backgrounds: (color.into(), palette.background.strong.color.into()),
width: 4.0,
border: Border {
radius: 2.0.into(),
diff --git a/widget/src/text_input.rs b/widget/src/text_input.rs
index 697d0d64..5b6ee9e0 100644
--- a/widget/src/text_input.rs
+++ b/widget/src/text_input.rs
@@ -1802,10 +1802,10 @@ pub fn default(theme: &Theme, status: Status) -> Style {
border: Border {
radius: 2.0.into(),
width: 1.0,
- color: palette.background.strong.color,
+ color: palette.background.strongest.color,
},
icon: palette.background.weak.text,
- placeholder: palette.background.strong.color,
+ placeholder: palette.background.strongest.color,
value: palette.background.base.text,
selection: palette.primary.weak.color,
};