chromashift/
xyzd50.rs

1use crate::{XyzD65, round_dp};
2use core::fmt;
3
4/// A colour expressed as X, Y and Z values, expressed in the CIE XYZ tristimulus colour space, with an explicit D50
5/// white point.
6/// The components are:
7/// - X - a number between 0.0 and 100.0
8/// - Y - a number between 0.0 and 100.0
9/// - Z - a number between 0.0 and 100.0
10/// - Alpha - a number between 0.0 and 100.0
11#[derive(Debug, Clone, Copy, PartialEq)]
12pub struct XyzD50 {
13	pub x: f64,
14	pub y: f64,
15	pub z: f64,
16	pub alpha: f32,
17}
18
19impl XyzD50 {
20	pub fn new(x: f64, y: f64, z: f64, alpha: f32) -> Self {
21		Self { x: x.clamp(0.0, 100.0), y: y.clamp(0.0, 100.0), z: z.clamp(0.0, 100.0), alpha: alpha.clamp(0.0, 100.0) }
22	}
23}
24
25impl fmt::Display for XyzD50 {
26	fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
27		let Self { x, y, z, alpha } = self;
28		write!(f, "color(xyz-d50 {} {}% {}%", round_dp(*x, 4), round_dp(*y, 4), round_dp(*z, 4))?;
29		if *alpha < 100.0 {
30			write!(f, " / {}", round_dp(*alpha as f64, 2))?;
31		}
32		write!(f, ")")
33	}
34}
35
36impl From<XyzD50> for XyzD65 {
37	fn from(value: XyzD50) -> Self {
38		let XyzD50 { x, y, z, alpha } = value;
39		let x_d65 = x * 0.955473421488075 + y * (-0.02309845494876471) + z * 0.06325924320057072;
40		let y_d65 = x * (-0.0283697093338637) + y * 1.0099953980813041 + z * 0.021041441191917323;
41		let z_d65 = x * 0.012314014864481998 + y * (-0.020507649298898964) + z * 1.330365926242124;
42		XyzD65::new(x_d65, y_d65, z_d65, alpha)
43	}
44}
45
46impl From<XyzD65> for XyzD50 {
47	fn from(value: XyzD65) -> Self {
48		let XyzD65 { x, y, z, alpha } = value;
49		let x_d50 = x * 1.0479297925449969 + y * 0.022946870601609652 + z * (-0.05019226628920524);
50		let y_d50 = x * 0.02962780877005599 + y * 0.9904344267538799 + z * (-0.017073799063418826);
51		let z_d50 = x * (-0.009243040646204504) + y * 0.015055191490298152 + z * 0.7518742814281371;
52		XyzD50::new(x_d50, y_d50, z_d50, alpha)
53	}
54}