1use crate::{LinearRgb, round_dp};
2use core::fmt;
3
4#[derive(Debug, Clone, Copy, PartialEq)]
11pub struct A98Rgb {
12 pub red: f64,
13 pub green: f64,
14 pub blue: f64,
15 pub alpha: f32,
16}
17
18impl A98Rgb {
19 pub fn new(red: f64, green: f64, blue: f64, alpha: f32) -> Self {
20 Self {
21 red: red.clamp(0.0, 1.0),
22 green: green.clamp(0.0, 1.0),
23 blue: blue.clamp(0.0, 1.0),
24 alpha: alpha.clamp(0.0, 100.0),
25 }
26 }
27}
28
29impl fmt::Display for A98Rgb {
30 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
31 let Self { red, green, blue, alpha } = self;
32 write!(f, "color(a98-rgb {} {}% {}%", round_dp(*red, 2), round_dp(*green, 2), round_dp(*blue, 2))?;
33 if *alpha < 100.0 {
34 write!(f, " / {}", round_dp(*alpha as f64, 2))?;
35 }
36 write!(f, ")")
37 }
38}
39
40impl From<LinearRgb> for A98Rgb {
41 fn from(value: LinearRgb) -> Self {
42 let LinearRgb { red, green, blue, alpha } = value;
43 const INV_GAMMA: f64 = 256.0 / 563.0;
44 let gamma_red = red.signum() * red.abs().powf(INV_GAMMA);
45 let gamma_green = green.signum() * green.abs().powf(INV_GAMMA);
46 let gamma_blue = blue.signum() * blue.abs().powf(INV_GAMMA);
47 A98Rgb::new(gamma_red, gamma_green, gamma_blue, alpha)
48 }
49}
50
51impl From<A98Rgb> for LinearRgb {
52 fn from(value: A98Rgb) -> Self {
53 let A98Rgb { red, green, blue, alpha } = value;
54 const GAMMA: f64 = 563.0 / 256.0;
55 let linear_red = red.signum() * red.abs().powf(GAMMA);
56 let linear_green = green.signum() * green.abs().powf(GAMMA);
57 let linear_blue = blue.signum() * blue.abs().powf(GAMMA);
58 LinearRgb::new(linear_red, linear_green, linear_blue, alpha)
59 }
60}