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
use core::ColorInterface;
use HSL;
#[derive(Debug, Copy, Clone, PartialEq)]
pub struct RGB {
r: u8,
g: u8,
b: u8,
}
impl RGB {
pub fn new(r: u8, g: u8, b: u8) -> RGB {
RGB { r, g, b }
}
pub fn unpack(&self) -> (u8, u8, u8) {
(self.r, self.g, self.b)
}
pub fn rgb_to_hsl(&self) -> HSL {
let (r, g, b) = self.unpack();
let r = r as f32 / 255.0;
let g = g as f32 / 255.0;
let b = b as f32 / 255.0;
let max = r.max(g).max(b);
let min = r.min(g).min(b);
let mut h: f32 = 0.0;
let mut s: f32 = 0.0;
let l = (max + min) / 2.0;
if max != min {
let d = max - min;
s = if l > 0.5 { d / (2.0 - max - min) } else { d / (max + min) };
if max == r {
h = (g - b) / d + (if g < b { 6.0 } else { 0.0 });
} else if max == g {
h = (b - r) / d + 2.0;
} else {
h = (r - g) / d + 4.0;
}
h /= 6.0;
}
return HSL::new(h, s, l);
}
}
impl ColorInterface for RGB {
fn to_color_str(&self) -> String {
format!("{};{};{}", self.r, self.g, self.b)
}
fn to_hsl(&self) -> HSL { self.rgb_to_hsl() }
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_rgb_2_hsl_1() {
let hsl = HSL::new(0.69934636, 0.49999997, 0.60);
let rgb = RGB::new(122, 102, 204);
assert_eq!(hsl, rgb.rgb_to_hsl());
}
#[test]
fn test_rgb_2_hsl_2() {
let hsl = HSL::new(0.0, 0.0, 0.60);
let rgb = RGB::new(153, 153, 153);
assert_eq!(hsl, rgb.rgb_to_hsl());
}
#[test]
fn test_rgb_2_hsl_3() {
let hsl = HSL::new(0.7012987, 0.50326794, 0.30);
let rgb = RGB::new(54, 38, 115);
assert_eq!(hsl, rgb.rgb_to_hsl());
}
#[test]
fn test_rgb_2_hsl_4() {
let hsl = HSL::new(0.08333334, 1.0, 0.6862745);
let rgb = RGB::new(255, 175, 95);
assert_eq!(hsl, rgb.rgb_to_hsl());
}
}