colorutils_rs/
hsl.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
/*
 * // Copyright 2024 (c) the Radzivon Bartoshyk. All rights reserved.
 * //
 * // Use of this source code is governed by a BSD-style
 * // license that can be found in the LICENSE file.
 */
use crate::rgb::Rgb;

#[repr(C)]
#[derive(Debug, Copy, Clone, PartialOrd, PartialEq)]
/// Represents *HSL* (hue, saturation, lightness) colorspace, H ∈ [0, 360f32], s ∈ [0f32, 1f32], v ∈ [0f32, 1f32]
pub struct Hsl {
    /// Hue H ∈ [0, 360f32]
    pub h: f32,
    /// Saturation s ∈ [0, 1f32]
    pub s: f32,
    /// Lightness v ∈ [0, 1f32]
    pub l: f32,
}

impl Hsl {
    #[inline]
    pub fn new(h: u16, s: u16, l: u16) -> Hsl {
        Hsl {
            h: h as f32,
            s: s as f32 * (1f32 / 100f32),
            l: l as f32 * (1f32 / 100f32),
        }
    }

    #[inline]
    pub fn from_components(h: f32, s: f32, l: f32) -> Hsl {
        Hsl { h, s, l }
    }

    #[inline]
    pub fn from_rgb(rgb: Rgb<u8>) -> Hsl {
        rgb2hsl(rgb.r, rgb.g, rgb.b)
    }

    #[inline]
    pub fn to_rgb8(&self) -> Rgb<u8> {
        let c = (1f32 - (2f32 * self.l - 1f32).abs()) * self.s;
        let x = c * (1f32 - ((self.h / 60f32) % 2f32 - 1f32).abs());
        let m = self.l - c / 2f32;

        let (r, g, b) = if self.h >= 0f32 && self.h < 60f32 {
            (c, x, 0f32)
        } else if self.h >= 60f32 && self.h < 120f32 {
            (x, c, 0f32)
        } else if self.h >= 120f32 && self.h < 180f32 {
            (0f32, c, x)
        } else if self.h >= 180f32 && self.h < 240f32 {
            (0f32, x, c)
        } else if self.h >= 240f32 && self.h < 300f32 {
            (x, 0f32, c)
        } else {
            (c, 0f32, x)
        };

        Rgb::<u8> {
            r: ((r + m) * 255f32).round() as u8,
            g: ((g + m) * 255f32).round() as u8,
            b: ((b + m) * 255f32).round() as u8,
        }
    }

    #[inline]
    pub fn to_rgb(&self) -> Rgb<u8> {
        self.to_rgb8()
    }

    #[inline]
    pub fn get_saturation(&self) -> u16 {
        ((self.s * 100f32) as u16).min(100u16)
    }

    #[inline]
    pub fn get_lightness(&self) -> u16 {
        ((self.l * 100f32) as u16).min(100u16)
    }

    #[inline]
    pub fn get_hue(&self) -> u16 {
        (self.h as u16).min(360)
    }
}

#[inline]
fn rgb2hsl(o_r: u8, o_g: u8, o_b: u8) -> Hsl {
    let r = o_r as f32 / 255f32;
    let g = o_g as f32 / 255f32;
    let b = o_b as f32 / 255f32;

    let c_max = r.max(g).max(b);
    let c_min = r.min(g).min(b);
    let delta = c_max - c_min;

    let mut h = if delta == 0f32 {
        0f32
    } else if c_max == r {
        60f32 * (((g - b) / delta) % 6f32)
    } else if c_max == g {
        60f32 * (((b - r) / delta) + 2f32)
    } else {
        60f32 * (((r - g) / delta) + 4f32)
    };

    if h < 0f32 {
        h += 360f32;
    }

    let l = 0.5f32 * (c_max + c_min);
    let s = delta / (1f32 - (2f32 * l - 1f32).abs());

    Hsl { h, s, l }
}