use crate::{EuclideanDistance, Rgb, TaxicabDistance, TransferFunction};
use num_traits::Pow;
use std::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Neg, Sub, SubAssign};
#[repr(C)]
#[derive(Debug, Copy, Clone, PartialOrd, PartialEq)]
pub struct Oklab {
pub l: f32,
pub a: f32,
pub b: f32,
}
impl Oklab {
#[inline]
pub fn new(l: f32, a: f32, b: f32) -> Oklab {
Oklab { l, a, b }
}
#[inline]
pub fn from_srgb(rgb: Rgb<u8>) -> Oklab {
let rgb_float = rgb.to_rgb_f32();
let linearized = rgb_float.linearize(TransferFunction::Srgb);
Self::linear_rgb_to_oklab(linearized)
}
#[inline]
pub fn from_rgb(rgb: Rgb<u8>, transfer_function: TransferFunction) -> Oklab {
let rgb_float = rgb.to_rgb_f32();
let linearized = rgb_float.linearize(transfer_function);
Self::linear_rgb_to_oklab(linearized)
}
#[inline]
pub fn from_linear_rgb(rgb: Rgb<f32>) -> Oklab {
Self::linear_rgb_to_oklab(rgb)
}
#[inline]
pub fn to_srgb(&self) -> Rgb<u8> {
let linear_rgb = self.to_linear_rgb();
let transferred = linear_rgb.gamma(TransferFunction::Srgb);
transferred.to_u8()
}
#[inline]
pub fn to_rgb(&self, transfer_function: TransferFunction) -> Rgb<u8> {
let linear_rgb = self.to_linear_rgb();
let transferred = linear_rgb.gamma(transfer_function);
transferred.to_u8()
}
#[inline]
pub fn to_srgb_f32(&self) -> Rgb<f32> {
let linear_rgb = self.to_linear_rgb();
linear_rgb.gamma(TransferFunction::Srgb)
}
#[inline]
pub fn to_rgb_f32(&self, transfer_function: TransferFunction) -> Rgb<f32> {
let linear_rgb = self.to_linear_rgb();
linear_rgb.gamma(transfer_function)
}
#[inline]
fn linear_rgb_to_oklab(rgb: Rgb<f32>) -> Oklab {
let l = 0.4122214708f32 * rgb.r + 0.5363325363f32 * rgb.g + 0.0514459929f32 * rgb.b;
let m = 0.2119034982f32 * rgb.r + 0.6806995451f32 * rgb.g + 0.1073969566f32 * rgb.b;
let s = 0.0883024619f32 * rgb.r + 0.2817188376f32 * rgb.g + 0.6299787005f32 * rgb.b;
let l_ = l.cbrt();
let m_ = m.cbrt();
let s_ = s.cbrt();
Oklab {
l: 0.2104542553f32 * l_ + 0.7936177850f32 * m_ - 0.0040720468f32 * s_,
a: 1.9779984951f32 * l_ - 2.4285922050f32 * m_ + 0.4505937099f32 * s_,
b: 0.0259040371f32 * l_ + 0.7827717662f32 * m_ - 0.8086757660f32 * s_,
}
}
#[inline]
pub fn to_linear_rgb(&self) -> Rgb<f32> {
let l_ = self.l + 0.3963377774f32 * self.a + 0.2158037573f32 * self.b;
let m_ = self.l - 0.1055613458f32 * self.a - 0.0638541728f32 * self.b;
let s_ = self.l - 0.0894841775f32 * self.a - 1.2914855480f32 * self.b;
let l = l_ * l_ * l_;
let m = m_ * m_ * m_;
let s = s_ * s_ * s_;
Rgb::new(
4.0767416621f32 * l - 3.3077115913f32 * m + 0.2309699292f32 * s,
-1.2684380046f32 * l + 2.6097574011f32 * m - 0.3413193965f32 * s,
-0.0041960863f32 * l - 0.7034186147f32 * m + 1.7076147010f32 * s,
)
}
#[inline]
pub fn hybrid_distance(&self, other: Self) -> f32 {
let lax = self.l - other.l;
let dax = self.a - other.a;
let bax = self.b - other.b;
(dax * dax + bax * bax).sqrt() + lax.abs()
}
pub const fn l_range() -> (f32, f32) {
(0., 1.)
}
pub const fn a_range() -> (f32, f32) {
(-0.5, 0.5)
}
pub const fn b_range() -> (f32, f32) {
(-0.5, 0.5)
}
}
impl EuclideanDistance for Oklab {
fn euclidean_distance(&self, other: Self) -> f32 {
let lax = self.l - other.l;
let dax = self.a - other.a;
let bax = self.b - other.b;
(lax * lax + dax * dax + bax * bax).sqrt()
}
}
impl TaxicabDistance for Oklab {
fn taxicab_distance(&self, other: Self) -> f32 {
let lax = self.l - other.l;
let dax = self.a - other.a;
let bax = self.b - other.b;
lax.abs() + dax.abs() + bax.abs()
}
}
impl Add<Oklab> for Oklab {
type Output = Oklab;
#[inline]
fn add(self, rhs: Self) -> Oklab {
Oklab::new(self.l + rhs.l, self.a + rhs.a, self.b + rhs.b)
}
}
impl Add<f32> for Oklab {
type Output = Oklab;
#[inline]
fn add(self, rhs: f32) -> Oklab {
Oklab::new(self.l + rhs, self.a + rhs, self.b + rhs)
}
}
impl AddAssign<Oklab> for Oklab {
#[inline]
fn add_assign(&mut self, rhs: Oklab) {
self.l += rhs.l;
self.a += rhs.a;
self.b += rhs.b;
}
}
impl AddAssign<f32> for Oklab {
#[inline]
fn add_assign(&mut self, rhs: f32) {
self.l += rhs;
self.a += rhs;
self.b += rhs;
}
}
impl Mul<f32> for Oklab {
type Output = Oklab;
#[inline]
fn mul(self, rhs: f32) -> Self::Output {
Oklab::new(self.l * rhs, self.a * rhs, self.b * rhs)
}
}
impl Mul<Oklab> for Oklab {
type Output = Oklab;
#[inline]
fn mul(self, rhs: Oklab) -> Self::Output {
Oklab::new(self.l * rhs.l, self.a * rhs.a, self.b * rhs.b)
}
}
impl MulAssign<f32> for Oklab {
#[inline]
fn mul_assign(&mut self, rhs: f32) {
self.l *= rhs;
self.a *= rhs;
self.b *= rhs;
}
}
impl MulAssign<Oklab> for Oklab {
#[inline]
fn mul_assign(&mut self, rhs: Oklab) {
self.l *= rhs.l;
self.a *= rhs.a;
self.b *= rhs.b;
}
}
impl Sub<f32> for Oklab {
type Output = Oklab;
#[inline]
fn sub(self, rhs: f32) -> Self::Output {
Oklab::new(self.l - rhs, self.a - rhs, self.b - rhs)
}
}
impl Sub<Oklab> for Oklab {
type Output = Oklab;
#[inline]
fn sub(self, rhs: Oklab) -> Self::Output {
Oklab::new(self.l - rhs.l, self.a - rhs.a, self.b - rhs.b)
}
}
impl SubAssign<f32> for Oklab {
#[inline]
fn sub_assign(&mut self, rhs: f32) {
self.l -= rhs;
self.a -= rhs;
self.b -= rhs;
}
}
impl SubAssign<Oklab> for Oklab {
#[inline]
fn sub_assign(&mut self, rhs: Oklab) {
self.l -= rhs.l;
self.a -= rhs.a;
self.b -= rhs.b;
}
}
impl Div<f32> for Oklab {
type Output = Oklab;
#[inline]
fn div(self, rhs: f32) -> Self::Output {
Oklab::new(self.l / rhs, self.a / rhs, self.b / rhs)
}
}
impl Div<Oklab> for Oklab {
type Output = Oklab;
#[inline]
fn div(self, rhs: Oklab) -> Self::Output {
Oklab::new(self.l / rhs.l, self.a / rhs.a, self.b / rhs.b)
}
}
impl DivAssign<f32> for Oklab {
#[inline]
fn div_assign(&mut self, rhs: f32) {
self.l /= rhs;
self.a /= rhs;
self.b /= rhs;
}
}
impl DivAssign<Oklab> for Oklab {
#[inline]
fn div_assign(&mut self, rhs: Oklab) {
self.l /= rhs.l;
self.a /= rhs.a;
self.b /= rhs.b;
}
}
impl Neg for Oklab {
type Output = Oklab;
#[inline]
fn neg(self) -> Self::Output {
Oklab::new(-self.l, -self.a, -self.b)
}
}
impl Pow<f32> for Oklab {
type Output = Oklab;
#[inline]
fn pow(self, rhs: f32) -> Self::Output {
Oklab::new(self.l.powf(rhs), self.a.powf(rhs), self.b.powf(rhs))
}
}
impl Pow<Oklab> for Oklab {
type Output = Oklab;
#[inline]
fn pow(self, rhs: Oklab) -> Self::Output {
Oklab::new(self.l.powf(rhs.l), self.a.powf(rhs.a), self.b.powf(rhs.b))
}
}
impl Oklab {
#[inline]
pub fn sqrt(&self) -> Oklab {
Oklab::new(
if self.l < 0. { 0. } else { self.l.sqrt() },
if self.a < 0. { 0. } else { self.a.sqrt() },
if self.b < 0. { 0. } else { self.b.sqrt() },
)
}
#[inline]
pub fn cbrt(&self) -> Oklab {
Oklab::new(self.l.cbrt(), self.a.cbrt(), self.b.cbrt())
}
}