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
118
119
120
121
122
123
124
125
126
127
128
129
#[cfg(not(feature = "no_std"))]
use std::fmt;
#[derive(Copy, Clone, Default)]
#[repr(packed)]
pub struct Color {
pub data: u32,
}
impl Color {
pub const fn rgb(r: u8, g: u8, b: u8) -> Self {
Color {
data: 0xFF00_0000 | ((r as u32) << 16) | ((g as u32) << 8) | (b as u32),
}
}
pub const fn rgba(r: u8, g: u8, b: u8, a: u8) -> Self {
Color {
data: ((a as u32) << 24) | ((r as u32) << 16) | ((g as u32) << 8) | (b as u32),
}
}
pub fn r(self) -> u8 {
((self.data & 0x00FF_0000) >> 16) as u8
}
pub fn g(self) -> u8 {
((self.data & 0x0000_FF00) >> 8) as u8
}
pub fn b(self) -> u8 {
(self.data & 0x0000_00FF) as u8
}
pub fn a(self) -> u8 {
((self.data & 0xFF00_0000) >> 24) as u8
}
pub fn interpolate(start_color: Color, end_color: Color, scale: f64) -> Color {
let r = Color::interp(start_color.r(), end_color.r(), scale);
let g = Color::interp(start_color.g(), end_color.g(), scale);
let b = Color::interp(start_color.b(), end_color.b(), scale);
let a = Color::interp(start_color.a(), end_color.a(), scale);
Color::rgba(r, g, b, a)
}
fn interp(start_color: u8, end_color: u8, scale: f64) -> u8 {
(end_color as f64 - start_color as f64).mul_add(scale, start_color as f64) as u8
}
}
impl ToString for Color {
fn to_string(&self) -> String {
if self.a() == 0 {
return String::from("transparent");
}
let data = self.data;
let mut color = format!("#{:x}", data);
color.remove(1);
color.remove(1);
color
}
}
impl From<&str> for Color {
fn from(s: &str) -> Color {
if s == "transparent" {
return Color::rgba(0, 0, 0, 0);
}
let clean_hex = s.trim_start_matches('#');
match clean_hex.len() {
6 | 8 => {
let mut x = match u32::from_str_radix(&clean_hex, 16) {
Ok(x) => x,
Err(_) => 0,
};
if clean_hex.len() == 6 {
x |= 0xFF_000_000;
}
Color { data: x }
}
_ => Color { data: 0 },
}
}
}
impl From<String> for Color {
fn from(s: String) -> Color {
Color::from(s.as_str())
}
}
impl PartialEq for Color {
fn eq(&self, other: &Color) -> bool {
self.r() == other.r() && self.g() == other.g() && self.b() == other.b()
}
}
#[cfg(not(feature = "no_std"))]
impl fmt::Debug for Color {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
write!(f, "{:#010X}", { self.data })
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn partial_eq() {
assert_eq!(true, Color::rgb(1, 2, 3) == Color::rgba(1, 2, 3, 200));
assert_eq!(false, Color::rgb(1, 2, 3) == Color::rgba(11, 2, 3, 200));
assert_eq!(true, Color::rgba(1, 2, 3, 200) == Color::rgba(1, 2, 3, 200));
}
}