diff options
author | 2024-03-07 00:14:41 +0100 | |
---|---|---|
committer | 2024-03-07 00:14:41 +0100 | |
commit | 905f2160e6eb7504f52d9bd62c7bfa42c8ec2902 (patch) | |
tree | 9d5915ab355facbca08b6be3d7eac03c7b5d9acc /core | |
parent | 7c4bf70023a8092faad9630c2c87fbf41bd6ab76 (diff) | |
download | iced-905f2160e6eb7504f52d9bd62c7bfa42c8ec2902.tar.gz iced-905f2160e6eb7504f52d9bd62c7bfa42c8ec2902.tar.bz2 iced-905f2160e6eb7504f52d9bd62c7bfa42c8ec2902.zip |
Move `Theme` type to `iced_core`
Diffstat (limited to 'core')
-rw-r--r-- | core/Cargo.toml | 3 | ||||
-rw-r--r-- | core/src/color.rs | 6 | ||||
-rw-r--r-- | core/src/lib.rs | 2 | ||||
-rw-r--r-- | core/src/theme.rs | 221 | ||||
-rw-r--r-- | core/src/theme/palette.rs | 625 |
5 files changed, 849 insertions, 8 deletions
diff --git a/core/Cargo.toml b/core/Cargo.toml index 2360e822..e71b75ae 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -19,9 +19,8 @@ smol_str.workspace = true thiserror.workspace = true web-time.workspace = true xxhash-rust.workspace = true - palette.workspace = true -palette.optional = true +once_cell.workspace = true [target.'cfg(windows)'.dependencies] raw-window-handle.workspace = true diff --git a/core/src/color.rs b/core/src/color.rs index 6526e220..da40ca15 100644 --- a/core/src/color.rs +++ b/core/src/color.rs @@ -1,4 +1,3 @@ -#[cfg(feature = "palette")] use palette::rgb::{Srgb, Srgba}; /// A color in the `sRGB` color space. @@ -210,7 +209,6 @@ macro_rules! color { }}; } -#[cfg(feature = "palette")] /// Converts from palette's `Rgba` type to a [`Color`]. impl From<Srgba> for Color { fn from(rgba: Srgba) -> Self { @@ -218,7 +216,6 @@ impl From<Srgba> for Color { } } -#[cfg(feature = "palette")] /// Converts from [`Color`] to palette's `Rgba` type. impl From<Color> for Srgba { fn from(c: Color) -> Self { @@ -226,7 +223,6 @@ impl From<Color> for Srgba { } } -#[cfg(feature = "palette")] /// Converts from palette's `Rgb` type to a [`Color`]. impl From<Srgb> for Color { fn from(rgb: Srgb) -> Self { @@ -234,7 +230,6 @@ impl From<Srgb> for Color { } } -#[cfg(feature = "palette")] /// Converts from [`Color`] to palette's `Rgb` type. impl From<Color> for Srgb { fn from(c: Color) -> Self { @@ -242,7 +237,6 @@ impl From<Color> for Srgb { } } -#[cfg(feature = "palette")] #[cfg(test)] mod tests { use super::*; diff --git a/core/src/lib.rs b/core/src/lib.rs index 002336ee..d076413e 100644 --- a/core/src/lib.rs +++ b/core/src/lib.rs @@ -30,6 +30,7 @@ pub mod overlay; pub mod renderer; pub mod svg; pub mod text; +pub mod theme; pub mod time; pub mod touch; pub mod widget; @@ -76,6 +77,7 @@ pub use shadow::Shadow; pub use shell::Shell; pub use size::Size; pub use text::Text; +pub use theme::Theme; pub use transformation::Transformation; pub use vector::Vector; pub use widget::Widget; diff --git a/core/src/theme.rs b/core/src/theme.rs new file mode 100644 index 00000000..21ba2a37 --- /dev/null +++ b/core/src/theme.rs @@ -0,0 +1,221 @@ +//! Use the built-in theme and styles. +pub mod palette; + +pub use palette::Palette; + +use std::fmt; +use std::sync::Arc; + +/// A built-in theme. +#[derive(Debug, Clone, PartialEq, Default)] +pub enum Theme { + /// The built-in light variant. + #[default] + Light, + /// The built-in dark variant. + Dark, + /// The built-in Dracula variant. + Dracula, + /// The built-in Nord variant. + Nord, + /// The built-in Solarized Light variant. + SolarizedLight, + /// The built-in Solarized Dark variant. + SolarizedDark, + /// The built-in Gruvbox Light variant. + GruvboxLight, + /// The built-in Gruvbox Dark variant. + GruvboxDark, + /// The built-in Catppuccin Latte variant. + CatppuccinLatte, + /// The built-in Catppuccin Frappé variant. + CatppuccinFrappe, + /// The built-in Catppuccin Macchiato variant. + CatppuccinMacchiato, + /// The built-in Catppuccin Mocha variant. + CatppuccinMocha, + /// The built-in Tokyo Night variant. + TokyoNight, + /// The built-in Tokyo Night Storm variant. + TokyoNightStorm, + /// The built-in Tokyo Night Light variant. + TokyoNightLight, + /// The built-in Kanagawa Wave variant. + KanagawaWave, + /// The built-in Kanagawa Dragon variant. + KanagawaDragon, + /// The built-in Kanagawa Lotus variant. + KanagawaLotus, + /// The built-in Moonfly variant. + Moonfly, + /// The built-in Nightfly variant. + Nightfly, + /// The built-in Oxocarbon variant. + Oxocarbon, + /// A [`Theme`] that uses a [`Custom`] palette. + Custom(Arc<Custom>), +} + +impl Theme { + /// A list with all the defined themes. + pub const ALL: &'static [Self] = &[ + Self::Light, + Self::Dark, + Self::Dracula, + Self::Nord, + Self::SolarizedLight, + Self::SolarizedDark, + Self::GruvboxLight, + Self::GruvboxDark, + Self::CatppuccinLatte, + Self::CatppuccinFrappe, + Self::CatppuccinMacchiato, + Self::CatppuccinMocha, + Self::TokyoNight, + Self::TokyoNightStorm, + Self::TokyoNightLight, + Self::KanagawaWave, + Self::KanagawaDragon, + Self::KanagawaLotus, + Self::Moonfly, + Self::Nightfly, + Self::Oxocarbon, + ]; + + /// Creates a new custom [`Theme`] from the given [`Palette`]. + pub fn custom(name: String, palette: Palette) -> Self { + Self::custom_with_fn(name, palette, palette::Extended::generate) + } + + /// Creates a new custom [`Theme`] from the given [`Palette`], with + /// a custom generator of a [`palette::Extended`]. + pub fn custom_with_fn( + name: String, + palette: Palette, + generate: impl FnOnce(Palette) -> palette::Extended, + ) -> Self { + Self::Custom(Arc::new(Custom::with_fn(name, palette, generate))) + } + + /// Returns the [`Palette`] of the [`Theme`]. + pub fn palette(&self) -> Palette { + match self { + Self::Light => Palette::LIGHT, + Self::Dark => Palette::DARK, + Self::Dracula => Palette::DRACULA, + Self::Nord => Palette::NORD, + Self::SolarizedLight => Palette::SOLARIZED_LIGHT, + Self::SolarizedDark => Palette::SOLARIZED_DARK, + Self::GruvboxLight => Palette::GRUVBOX_LIGHT, + Self::GruvboxDark => Palette::GRUVBOX_DARK, + Self::CatppuccinLatte => Palette::CATPPUCCIN_LATTE, + Self::CatppuccinFrappe => Palette::CATPPUCCIN_FRAPPE, + Self::CatppuccinMacchiato => Palette::CATPPUCCIN_MACCHIATO, + Self::CatppuccinMocha => Palette::CATPPUCCIN_MOCHA, + Self::TokyoNight => Palette::TOKYO_NIGHT, + Self::TokyoNightStorm => Palette::TOKYO_NIGHT_STORM, + Self::TokyoNightLight => Palette::TOKYO_NIGHT_LIGHT, + Self::KanagawaWave => Palette::KANAGAWA_WAVE, + Self::KanagawaDragon => Palette::KANAGAWA_DRAGON, + Self::KanagawaLotus => Palette::KANAGAWA_LOTUS, + Self::Moonfly => Palette::MOONFLY, + Self::Nightfly => Palette::NIGHTFLY, + Self::Oxocarbon => Palette::OXOCARBON, + Self::Custom(custom) => custom.palette, + } + } + + /// Returns the [`palette::Extended`] of the [`Theme`]. + pub fn extended_palette(&self) -> &palette::Extended { + match self { + Self::Light => &palette::EXTENDED_LIGHT, + Self::Dark => &palette::EXTENDED_DARK, + Self::Dracula => &palette::EXTENDED_DRACULA, + Self::Nord => &palette::EXTENDED_NORD, + Self::SolarizedLight => &palette::EXTENDED_SOLARIZED_LIGHT, + Self::SolarizedDark => &palette::EXTENDED_SOLARIZED_DARK, + Self::GruvboxLight => &palette::EXTENDED_GRUVBOX_LIGHT, + Self::GruvboxDark => &palette::EXTENDED_GRUVBOX_DARK, + Self::CatppuccinLatte => &palette::EXTENDED_CATPPUCCIN_LATTE, + Self::CatppuccinFrappe => &palette::EXTENDED_CATPPUCCIN_FRAPPE, + Self::CatppuccinMacchiato => { + &palette::EXTENDED_CATPPUCCIN_MACCHIATO + } + Self::CatppuccinMocha => &palette::EXTENDED_CATPPUCCIN_MOCHA, + Self::TokyoNight => &palette::EXTENDED_TOKYO_NIGHT, + Self::TokyoNightStorm => &palette::EXTENDED_TOKYO_NIGHT_STORM, + Self::TokyoNightLight => &palette::EXTENDED_TOKYO_NIGHT_LIGHT, + Self::KanagawaWave => &palette::EXTENDED_KANAGAWA_WAVE, + Self::KanagawaDragon => &palette::EXTENDED_KANAGAWA_DRAGON, + Self::KanagawaLotus => &palette::EXTENDED_KANAGAWA_LOTUS, + Self::Moonfly => &palette::EXTENDED_MOONFLY, + Self::Nightfly => &palette::EXTENDED_NIGHTFLY, + Self::Oxocarbon => &palette::EXTENDED_OXOCARBON, + Self::Custom(custom) => &custom.extended, + } + } +} + +impl fmt::Display for Theme { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + Self::Light => write!(f, "Light"), + Self::Dark => write!(f, "Dark"), + Self::Dracula => write!(f, "Dracula"), + Self::Nord => write!(f, "Nord"), + Self::SolarizedLight => write!(f, "Solarized Light"), + Self::SolarizedDark => write!(f, "Solarized Dark"), + Self::GruvboxLight => write!(f, "Gruvbox Light"), + Self::GruvboxDark => write!(f, "Gruvbox Dark"), + Self::CatppuccinLatte => write!(f, "Catppuccin Latte"), + Self::CatppuccinFrappe => write!(f, "Catppuccin Frappé"), + Self::CatppuccinMacchiato => write!(f, "Catppuccin Macchiato"), + Self::CatppuccinMocha => write!(f, "Catppuccin Mocha"), + Self::TokyoNight => write!(f, "Tokyo Night"), + Self::TokyoNightStorm => write!(f, "Tokyo Night Storm"), + Self::TokyoNightLight => write!(f, "Tokyo Night Light"), + Self::KanagawaWave => write!(f, "Kanagawa Wave"), + Self::KanagawaDragon => write!(f, "Kanagawa Dragon"), + Self::KanagawaLotus => write!(f, "Kanagawa Lotus"), + Self::Moonfly => write!(f, "Moonfly"), + Self::Nightfly => write!(f, "Nightfly"), + Self::Oxocarbon => write!(f, "Oxocarbon"), + Self::Custom(custom) => custom.fmt(f), + } + } +} + +/// A [`Theme`] with a customized [`Palette`]. +#[derive(Debug, Clone, PartialEq)] +pub struct Custom { + name: String, + palette: Palette, + extended: palette::Extended, +} + +impl Custom { + /// Creates a [`Custom`] theme from the given [`Palette`]. + pub fn new(name: String, palette: Palette) -> Self { + Self::with_fn(name, palette, palette::Extended::generate) + } + + /// Creates a [`Custom`] theme from the given [`Palette`] with + /// a custom generator of a [`palette::Extended`]. + pub fn with_fn( + name: String, + palette: Palette, + generate: impl FnOnce(Palette) -> palette::Extended, + ) -> Self { + Self { + name, + palette, + extended: generate(palette), + } + } +} + +impl fmt::Display for Custom { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + write!(f, "{}", self.name) + } +} diff --git a/core/src/theme/palette.rs b/core/src/theme/palette.rs new file mode 100644 index 00000000..985a54a8 --- /dev/null +++ b/core/src/theme/palette.rs @@ -0,0 +1,625 @@ +//! Define the colors of a theme. +use crate::{color, Color}; + +use once_cell::sync::Lazy; +use palette::color_difference::Wcag21RelativeContrast; +use palette::rgb::Rgb; +use palette::{FromColor, Hsl, Mix}; + +/// A color palette. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Palette { + /// The background [`Color`] of the [`Palette`]. + pub background: Color, + /// The text [`Color`] of the [`Palette`]. + pub text: Color, + /// The primary [`Color`] of the [`Palette`]. + pub primary: Color, + /// The success [`Color`] of the [`Palette`]. + pub success: Color, + /// The danger [`Color`] of the [`Palette`]. + pub danger: Color, +} + +impl Palette { + /// The built-in light variant of a [`Palette`]. + pub const LIGHT: Self = Self { + background: Color::WHITE, + text: Color::BLACK, + primary: Color::from_rgb( + 0x5E as f32 / 255.0, + 0x7C as f32 / 255.0, + 0xE2 as f32 / 255.0, + ), + success: Color::from_rgb( + 0x12 as f32 / 255.0, + 0x66 as f32 / 255.0, + 0x4F as f32 / 255.0, + ), + danger: Color::from_rgb( + 0xC3 as f32 / 255.0, + 0x42 as f32 / 255.0, + 0x3F as f32 / 255.0, + ), + }; + + /// The built-in dark variant of a [`Palette`]. + pub const DARK: Self = Self { + background: Color::from_rgb( + 0x20 as f32 / 255.0, + 0x22 as f32 / 255.0, + 0x25 as f32 / 255.0, + ), + text: Color::from_rgb(0.90, 0.90, 0.90), + primary: Color::from_rgb( + 0x5E as f32 / 255.0, + 0x7C as f32 / 255.0, + 0xE2 as f32 / 255.0, + ), + success: Color::from_rgb( + 0x12 as f32 / 255.0, + 0x66 as f32 / 255.0, + 0x4F as f32 / 255.0, + ), + danger: Color::from_rgb( + 0xC3 as f32 / 255.0, + 0x42 as f32 / 255.0, + 0x3F as f32 / 255.0, + ), + }; + + /// The built-in [Dracula] variant of a [`Palette`]. + /// + /// [Dracula]: https://draculatheme.com + pub const DRACULA: Self = Self { + background: color!(0x282A36), // BACKGROUND + text: color!(0xf8f8f2), // FOREGROUND + primary: color!(0xbd93f9), // PURPLE + success: color!(0x50fa7b), // GREEN + danger: color!(0xff5555), // RED + }; + + /// The built-in [Nord] variant of a [`Palette`]. + /// + /// [Nord]: https://www.nordtheme.com/docs/colors-and-palettes + pub const NORD: Self = Self { + background: color!(0x2e3440), // nord0 + text: color!(0xeceff4), // nord6 + primary: color!(0x8fbcbb), // nord7 + success: color!(0xa3be8c), // nord14 + danger: color!(0xbf616a), // nord11 + }; + + /// The built-in [Solarized] Light variant of a [`Palette`]. + /// + /// [Solarized]: https://ethanschoonover.com/solarized + pub const SOLARIZED_LIGHT: Self = Self { + background: color!(0xfdf6e3), // base3 + text: color!(0x657b83), // base00 + primary: color!(0x2aa198), // cyan + success: color!(0x859900), // green + danger: color!(0xdc322f), // red + }; + + /// The built-in [Solarized] Dark variant of a [`Palette`]. + /// + /// [Solarized]: https://ethanschoonover.com/solarized + pub const SOLARIZED_DARK: Self = Self { + background: color!(0x002b36), // base03 + text: color!(0x839496), // base0 + primary: color!(0x2aa198), // cyan + success: color!(0x859900), // green + danger: color!(0xdc322f), // red + }; + + /// The built-in [Gruvbox] Light variant of a [`Palette`]. + /// + /// [Gruvbox]: https://github.com/morhetz/gruvbox + pub const GRUVBOX_LIGHT: Self = Self { + background: color!(0xfbf1c7), // light BG_0 + text: color!(0x282828), // light FG0_29 + primary: color!(0x458588), // light BLUE_4 + success: color!(0x98971a), // light GREEN_2 + danger: color!(0xcc241d), // light RED_1 + }; + + /// The built-in [Gruvbox] Dark variant of a [`Palette`]. + /// + /// [Gruvbox]: https://github.com/morhetz/gruvbox + pub const GRUVBOX_DARK: Self = Self { + background: color!(0x282828), // dark BG_0 + text: color!(0xfbf1c7), // dark FG0_29 + primary: color!(0x458588), // dark BLUE_4 + success: color!(0x98971a), // dark GREEN_2 + danger: color!(0xcc241d), // dark RED_1 + }; + + /// The built-in [Catppuccin] Latte variant of a [`Palette`]. + /// + /// [Catppuccin]: https://github.com/catppuccin/catppuccin + pub const CATPPUCCIN_LATTE: Self = Self { + background: color!(0xeff1f5), // Base + text: color!(0x4c4f69), // Text + primary: color!(0x1e66f5), // Blue + success: color!(0x40a02b), // Green + danger: color!(0xd20f39), // Red + }; + + /// The built-in [Catppuccin] Frappé variant of a [`Palette`]. + /// + /// [Catppuccin]: https://github.com/catppuccin/catppuccin + pub const CATPPUCCIN_FRAPPE: Self = Self { + background: color!(0x303446), // Base + text: color!(0xc6d0f5), // Text + primary: color!(0x8caaee), // Blue + success: color!(0xa6d189), // Green + danger: color!(0xe78284), // Red + }; + + /// The built-in [Catppuccin] Macchiato variant of a [`Palette`]. + /// + /// [Catppuccin]: https://github.com/catppuccin/catppuccin + pub const CATPPUCCIN_MACCHIATO: Self = Self { + background: color!(0x24273a), // Base + text: color!(0xcad3f5), // Text + primary: color!(0x8aadf4), // Blue + success: color!(0xa6da95), // Green + danger: color!(0xed8796), // Red + }; + + /// The built-in [Catppuccin] Mocha variant of a [`Palette`]. + /// + /// [Catppuccin]: https://github.com/catppuccin/catppuccin + pub const CATPPUCCIN_MOCHA: Self = Self { + background: color!(0x1e1e2e), // Base + text: color!(0xcdd6f4), // Text + primary: color!(0x89b4fa), // Blue + success: color!(0xa6e3a1), // Green + danger: color!(0xf38ba8), // Red + }; + + /// The built-in [Tokyo Night] variant of a [`Palette`]. + /// + /// [Tokyo Night]: https://github.com/enkia/tokyo-night-vscode-theme + pub const TOKYO_NIGHT: Self = Self { + background: color!(0x1a1b26), // Background (Night) + text: color!(0x9aa5ce), // Text + primary: color!(0x2ac3de), // Blue + success: color!(0x9ece6a), // Green + danger: color!(0xf7768e), // Red + }; + + /// The built-in [Tokyo Night] Storm variant of a [`Palette`]. + /// + /// [Tokyo Night]: https://github.com/enkia/tokyo-night-vscode-theme + pub const TOKYO_NIGHT_STORM: Self = Self { + background: color!(0x24283b), // Background (Storm) + text: color!(0x9aa5ce), // Text + primary: color!(0x2ac3de), // Blue + success: color!(0x9ece6a), // Green + danger: color!(0xf7768e), // Red + }; + + /// The built-in [Tokyo Night] Light variant of a [`Palette`]. + /// + /// [Tokyo Night]: https://github.com/enkia/tokyo-night-vscode-theme + pub const TOKYO_NIGHT_LIGHT: Self = Self { + background: color!(0xd5d6db), // Background + text: color!(0x565a6e), // Text + primary: color!(0x166775), // Blue + success: color!(0x485e30), // Green + danger: color!(0x8c4351), // Red + }; + + /// The built-in [Kanagawa] Wave variant of a [`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 + success: color!(0x76946A), // Autumn Green + danger: color!(0xC34043), // Autumn Red + }; + + /// The built-in [Kanagawa] Dragon variant of a [`Palette`]. + /// + /// [Kanagawa]: https://github.com/rebelot/kanagawa.nvim + pub const KANAGAWA_DRAGON: Self = Self { + background: color!(0x181616), // Dragon Black 3 + text: color!(0xc5c9c5), // Dragon White + primary: color!(0x223249), // Wave Blue 1 + success: color!(0x8a9a7b), // Dragon Green 2 + danger: color!(0xc4746e), // Dragon Red + }; + + /// The built-in [Kanagawa] Lotus variant of a [`Palette`]. + /// + /// [Kanagawa]: https://github.com/rebelot/kanagawa.nvim + pub const KANAGAWA_LOTUS: Self = Self { + background: color!(0xf2ecbc), // Lotus White 3 + text: color!(0x545464), // Lotus Ink 1 + primary: color!(0xc9cbd1), // Lotus Violet 3 + success: color!(0x6f894e), // Lotus Green + danger: color!(0xc84053), // Lotus Red + }; + + /// The built-in [Moonfly] variant of a [`Palette`]. + /// + /// [Moonfly]: https://github.com/bluz71/vim-moonfly-colors + pub const MOONFLY: Self = Self { + background: color!(0x080808), // Background + text: color!(0xbdbdbd), // Foreground + primary: color!(0x80a0ff), // Blue (normal) + success: color!(0x8cc85f), // Green (normal) + danger: color!(0xff5454), // Red (normal) + }; + + /// The built-in [Nightfly] variant of a [`Palette`]. + /// + /// [Nightfly]: https://github.com/bluz71/vim-nightfly-colors + pub const NIGHTFLY: Self = Self { + background: color!(0x011627), // Background + text: color!(0xbdc1c6), // Foreground + primary: color!(0x82aaff), // Blue (normal) + success: color!(0xa1cd5e), // Green (normal) + danger: color!(0xfc514e), // Red (normal) + }; + + /// The built-in [Oxocarbon] variant of a [`Palette`]. + /// + /// [Oxocarbon]: https://github.com/nyoom-engineering/oxocarbon.nvim + pub const OXOCARBON: Self = Self { + background: color!(0x232323), + text: color!(0xd0d0d0), + primary: color!(0x00b4ff), + success: color!(0x00c15a), + danger: color!(0xf62d0f), + }; +} + +/// An extended set of colors generated from a [`Palette`]. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Extended { + /// The set of background colors. + pub background: Background, + /// The set of primary colors. + pub primary: Primary, + /// The set of secondary colors. + pub secondary: Secondary, + /// The set of success colors. + pub success: Success, + /// The set of danger colors. + pub danger: Danger, + /// Whether the palette is dark or not. + pub is_dark: bool, +} + +/// The built-in light variant of an [`Extended`] palette. +pub static EXTENDED_LIGHT: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::LIGHT)); + +/// The built-in dark variant of an [`Extended`] palette. +pub static EXTENDED_DARK: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::DARK)); + +/// The built-in Dracula variant of an [`Extended`] palette. +pub static EXTENDED_DRACULA: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::DRACULA)); + +/// The built-in Nord variant of an [`Extended`] palette. +pub static EXTENDED_NORD: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::NORD)); + +/// The built-in Solarized Light variant of an [`Extended`] palette. +pub static EXTENDED_SOLARIZED_LIGHT: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::SOLARIZED_LIGHT)); + +/// The built-in Solarized Dark variant of an [`Extended`] palette. +pub static EXTENDED_SOLARIZED_DARK: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::SOLARIZED_DARK)); + +/// The built-in Gruvbox Light variant of an [`Extended`] palette. +pub static EXTENDED_GRUVBOX_LIGHT: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::GRUVBOX_LIGHT)); + +/// The built-in Gruvbox Dark variant of an [`Extended`] palette. +pub static EXTENDED_GRUVBOX_DARK: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::GRUVBOX_DARK)); + +/// The built-in Catppuccin Latte variant of an [`Extended`] palette. +pub static EXTENDED_CATPPUCCIN_LATTE: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::CATPPUCCIN_LATTE)); + +/// The built-in Catppuccin Frappé variant of an [`Extended`] palette. +pub static EXTENDED_CATPPUCCIN_FRAPPE: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::CATPPUCCIN_FRAPPE)); + +/// The built-in Catppuccin Macchiato variant of an [`Extended`] palette. +pub static EXTENDED_CATPPUCCIN_MACCHIATO: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::CATPPUCCIN_MACCHIATO)); + +/// The built-in Catppuccin Mocha variant of an [`Extended`] palette. +pub static EXTENDED_CATPPUCCIN_MOCHA: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::CATPPUCCIN_MOCHA)); + +/// The built-in Tokyo Night variant of an [`Extended`] palette. +pub static EXTENDED_TOKYO_NIGHT: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::TOKYO_NIGHT)); + +/// The built-in Tokyo Night Storm variant of an [`Extended`] palette. +pub static EXTENDED_TOKYO_NIGHT_STORM: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::TOKYO_NIGHT_STORM)); + +/// The built-in Tokyo Night variant of an [`Extended`] palette. +pub static EXTENDED_TOKYO_NIGHT_LIGHT: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::TOKYO_NIGHT_LIGHT)); + +/// The built-in Kanagawa Wave variant of an [`Extended`] palette. +pub static EXTENDED_KANAGAWA_WAVE: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::KANAGAWA_WAVE)); + +/// The built-in Kanagawa Dragon variant of an [`Extended`] palette. +pub static EXTENDED_KANAGAWA_DRAGON: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::KANAGAWA_DRAGON)); + +/// The built-in Kanagawa Lotus variant of an [`Extended`] palette. +pub static EXTENDED_KANAGAWA_LOTUS: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::KANAGAWA_LOTUS)); + +/// The built-in Moonfly variant of an [`Extended`] palette. +pub static EXTENDED_MOONFLY: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::MOONFLY)); + +/// The built-in Nightfly variant of an [`Extended`] palette. +pub static EXTENDED_NIGHTFLY: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::NIGHTFLY)); + +/// The built-in Oxocarbon variant of an [`Extended`] palette. +pub static EXTENDED_OXOCARBON: Lazy<Extended> = + Lazy::new(|| Extended::generate(Palette::OXOCARBON)); + +impl Extended { + /// Generates an [`Extended`] palette from a simple [`Palette`]. + pub fn generate(palette: Palette) -> Self { + Self { + background: Background::new(palette.background, palette.text), + primary: Primary::generate( + palette.primary, + palette.background, + palette.text, + ), + secondary: Secondary::generate(palette.background, palette.text), + success: Success::generate( + palette.success, + palette.background, + palette.text, + ), + danger: Danger::generate( + palette.danger, + palette.background, + palette.text, + ), + is_dark: is_dark(palette.background), + } + } +} + +/// A pair of background and text colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Pair { + /// The background color. + pub color: Color, + + /// The text color. + /// + /// It's guaranteed to be readable on top of the background [`color`]. + /// + /// [`color`]: Self::color + pub text: Color, +} + +impl Pair { + /// Creates a new [`Pair`] from a background [`Color`] and some text [`Color`]. + pub fn new(color: Color, text: Color) -> Self { + Self { + color, + text: readable(color, text), + } + } +} + +/// A set of background colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Background { + /// The base background color. + pub base: Pair, + /// A weaker version of the base background color. + pub weak: Pair, + /// A stronger version of the base background color. + pub strong: 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); + + Self { + base: Pair::new(base, text), + weak: Pair::new(weak, text), + strong: Pair::new(strong, text), + } + } +} + +/// A set of primary colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Primary { + /// The base primary color. + pub base: Pair, + /// A weaker version of the base primary color. + pub weak: Pair, + /// A stronger version of the base primary color. + pub strong: Pair, +} + +impl Primary { + /// Generates a set of [`Primary`] colors from the base, background, and text colors. + pub fn generate(base: Color, background: Color, text: Color) -> Self { + let weak = mix(base, background, 0.4); + let strong = deviate(base, 0.1); + + Self { + base: Pair::new(base, text), + weak: Pair::new(weak, text), + strong: Pair::new(strong, text), + } + } +} + +/// A set of secondary colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Secondary { + /// The base secondary color. + pub base: Pair, + /// A weaker version of the base secondary color. + pub weak: Pair, + /// A stronger version of the base secondary color. + pub strong: Pair, +} + +impl Secondary { + /// Generates a set of [`Secondary`] colors from the base and text colors. + pub fn generate(base: Color, text: Color) -> Self { + let base = mix(base, text, 0.2); + let weak = mix(base, text, 0.1); + let strong = mix(base, text, 0.3); + + Self { + base: Pair::new(base, text), + weak: Pair::new(weak, text), + strong: Pair::new(strong, text), + } + } +} + +/// A set of success colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Success { + /// The base success color. + pub base: Pair, + /// A weaker version of the base success color. + pub weak: Pair, + /// A stronger version of the base success color. + pub strong: Pair, +} + +impl Success { + /// Generates a set of [`Success`] colors from the base, background, and text colors. + pub fn generate(base: Color, background: Color, text: Color) -> Self { + let weak = mix(base, background, 0.4); + let strong = deviate(base, 0.1); + + Self { + base: Pair::new(base, text), + weak: Pair::new(weak, text), + strong: Pair::new(strong, text), + } + } +} + +/// A set of danger colors. +#[derive(Debug, Clone, Copy, PartialEq)] +pub struct Danger { + /// The base danger color. + pub base: Pair, + /// A weaker version of the base danger color. + pub weak: Pair, + /// A stronger version of the base danger color. + pub strong: Pair, +} + +impl Danger { + /// Generates a set of [`Danger`] colors from the base, background, and text colors. + pub fn generate(base: Color, background: Color, text: Color) -> Self { + let weak = mix(base, background, 0.4); + let strong = deviate(base, 0.1); + + Self { + base: Pair::new(base, text), + weak: Pair::new(weak, text), + strong: Pair::new(strong, text), + } + } +} + +fn darken(color: Color, amount: f32) -> Color { + let mut hsl = to_hsl(color); + + hsl.lightness = if hsl.lightness - amount < 0.0 { + 0.0 + } else { + hsl.lightness - amount + }; + + from_hsl(hsl) +} + +fn lighten(color: Color, amount: f32) -> Color { + let mut hsl = to_hsl(color); + + hsl.lightness = if hsl.lightness + amount > 1.0 { + 1.0 + } else { + hsl.lightness + amount + }; + + from_hsl(hsl) +} + +fn deviate(color: Color, amount: f32) -> Color { + if is_dark(color) { + lighten(color, amount) + } else { + darken(color, amount) + } +} + +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(); + + let mixed = a_lin.mix(b_lin, factor); + Rgb::from_linear(mixed).into() +} + +fn readable(background: Color, text: Color) -> Color { + if is_readable(background, text) { + text + } else if is_dark(background) { + Color::WHITE + } else { + Color::BLACK + } +} + +fn is_dark(color: Color) -> bool { + to_hsl(color).lightness < 0.6 +} + +fn is_readable(a: Color, b: Color) -> bool { + let a_srgb = Rgb::from(a); + let b_srgb = Rgb::from(b); + + a_srgb.has_enhanced_contrast_text(b_srgb) +} + +fn to_hsl(color: Color) -> Hsl { + Hsl::from_color(Rgb::from(color)) +} + +fn from_hsl(hsl: Hsl) -> Color { + Rgb::from_color(hsl).into() +} |