1use crate::{ToAlpha, XyzD50, round_dp};
2use core::fmt;
3
4const D50X: f64 = 96.4220;
5const D50Y: f64 = 100.0;
6const D50Z: f64 = 82.5210;
7
8#[derive(Debug, Clone, Copy, PartialEq)]
15pub struct Lab {
16 pub lightness: f64,
17 pub a: f64,
18 pub b: f64,
19 pub alpha: f32,
20}
21
22impl Lab {
23 pub fn new(lightness: f64, a: f64, b: f64, alpha: f32) -> Self {
24 Self { lightness, a, b, alpha: alpha.clamp(0.0, 100.0) }
25 }
26}
27
28impl ToAlpha for Lab {
29 fn to_alpha(&self) -> f32 {
30 self.alpha
31 }
32}
33
34impl fmt::Display for Lab {
35 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
36 let Self { lightness, a, b, alpha } = self;
37 write!(f, "lab({} {} {}", round_dp(*lightness, 2), round_dp(*a, 3), round_dp(*b, 3))?;
38 if *alpha < 100.0 {
39 write!(f, " / {}", round_dp(*alpha as f64, 2))?;
40 }
41 write!(f, ")")
42 }
43}
44
45impl From<XyzD50> for Lab {
46 fn from(value: XyzD50) -> Self {
47 let XyzD50 { x, y, z, alpha } = value;
48 let x = x / D50X;
49 let y = y / D50Y;
50 let z = z / D50Z;
51 let epsilon = 216.0 / 24389.0; let kappa = 24389.0 / 27.0; let fx = if x > epsilon { x.cbrt() } else { (kappa * x + 16.0) / 116.0 };
54 let fy = if y > epsilon { y.cbrt() } else { (kappa * y + 16.0) / 116.0 };
55 let fz = if z > epsilon { z.cbrt() } else { (kappa * z + 16.0) / 116.0 };
56 let lightness = 116.0 * fy - 16.0;
57 let a = 500.0 * (fx - fy);
58 let b = 200.0 * (fy - fz);
59 Lab::new(lightness, a, b, alpha)
60 }
61}
62
63impl From<Lab> for XyzD50 {
64 fn from(value: Lab) -> Self {
65 let Lab { lightness, a, b, alpha } = value;
66 let epsilon = 216.0 / 24389.0; let kappa = 24389.0 / 27.0; let fy = (lightness + 16.0) / 116.0;
69 let fx = a / 500.0 + fy;
70 let fz = fy - b / 200.0;
71 let x = if fx.powi(3) > epsilon { fx.powi(3) } else { (116.0 * fx - 16.0) / kappa };
72 let y = if lightness > kappa * epsilon { ((lightness + 16.0) / 116.0).powi(3) } else { lightness / kappa };
73 let z = if fz.powi(3) > epsilon { fz.powi(3) } else { (116.0 * fz - 16.0) / kappa };
74 XyzD50::new(x * D50X, y * D50Y, z * D50Z, alpha)
75 }
76}