#![allow(non_snake_case)]
#![allow(non_upper_case_globals)]
use crate::image::ToRGB;
use crate::image::RGBAPLU;
use crate::image::RGBLU;
use imgref::*;
#[cfg(not(feature = "threads"))]
use crate::lieon as rayon;
use rayon::prelude::*;
const D65x: f32 = 0.9505;
const D65y: f32 = 1.0;
const D65z: f32 = 1.089;
pub type GBitmap = ImgVec<f32>;
pub(crate) trait ToLAB {
fn to_lab(&self) -> (f32, f32, f32);
}
#[inline(always)]
fn fma_matrix(r: f32, rx: f32, g: f32, gx: f32, b: f32, bx: f32) -> f32 {
b.mul_add(bx, g.mul_add(gx, r * rx))
}
impl ToLAB for RGBLU {
fn to_lab(&self) -> (f32, f32, f32) {
let fx = fma_matrix(self.r, 0.4124 / D65x, self.g, 0.3576 / D65x, self.b, 0.1805 / D65x);
let fy = fma_matrix(self.r, 0.2126 / D65y, self.g, 0.7152 / D65y, self.b, 0.0722 / D65y);
let fz = fma_matrix(self.r, 0.0193 / D65z, self.g, 0.1192 / D65z, self.b, 0.9505 / D65z);
let epsilon: f32 = 216. / 24389.;
let k = 24389. / (27. * 116.); let X = if fx > epsilon { cbrt_poly(fx) - 16. / 116. } else { k * fx };
let Y = if fy > epsilon { cbrt_poly(fy) - 16. / 116. } else { k * fy };
let Z = if fz > epsilon { cbrt_poly(fz) - 16. / 116. } else { k * fz };
let lab = (
(Y * 1.05f32), (500.0 / 220.0f32).mul_add(X - Y, 86.2 / 220.0f32),
(200.0 / 220.0f32).mul_add(Y - Z, 107.9 / 220.0f32),
);
debug_assert!(lab.0 <= 1.0 && lab.1 <= 1.0 && lab.2 <= 1.0);
lab
}
}
fn cbrt_poly(x: f32) -> f32 {
let poly = [0.2, 1.51, -0.5];
let y = (poly[2] * x + poly[1]) * x + poly[0];
let y3 = y*y*y;
let y = y * (y3 + 2. * x) / (2. * y3 + x);
let y3 = y*y*y;
let y = y * (y3 + 2. * x) / (2. * y3 + x);
debug_assert!(y < 1.001);
debug_assert!(x < 216. / 24389. || y >= 16. / 116.);
y
}
pub trait ToLABBitmap {
fn to_lab(&self) -> Vec<GBitmap>;
}
impl ToLABBitmap for ImgVec<RGBAPLU> {
#[inline(always)]
fn to_lab(&self) -> Vec<GBitmap> {
self.as_ref().to_lab()
}
}
impl ToLABBitmap for ImgVec<RGBLU> {
#[inline(always)]
fn to_lab(&self) -> Vec<GBitmap> {
self.as_ref().to_lab()
}
}
impl ToLABBitmap for GBitmap {
#[inline(never)]
fn to_lab(&self) -> Vec<GBitmap> {
let width = self.width();
assert!(width > 0);
let height = self.height();
let area = width * height;
let mut out = Vec::with_capacity(area);
out.spare_capacity_mut().par_chunks_exact_mut(width).take(height).enumerate().for_each(|(y, out_row)| {
let in_row = &self.rows().nth(y).unwrap()[0..width];
let out_row = &mut out_row[0..width];
let epsilon: f32 = 216. / 24389.;
for x in 0..width {
let fy = in_row[x];
let Y = if fy > epsilon { fy.cbrt() - 16. / 116. } else { ((24389. / 27.) / 116.) * fy };
out_row[x].write(Y * 1.16);
}
});
unsafe { out.set_len(area) };
vec![Img::new(out, width, height)]
}
}
#[inline(never)]
fn rgb_to_lab<T: Copy + Sync + Send + 'static, F>(img: ImgRef<'_, T>, cb: F) -> Vec<GBitmap>
where F: Fn(T, usize) -> (f32, f32, f32) + Sync + Send + 'static
{
let width = img.width();
assert!(width > 0);
let height = img.height();
let area = width * height;
let mut out_l = Vec::with_capacity(area);
let mut out_a = Vec::with_capacity(area);
let mut out_b = Vec::with_capacity(area);
out_l.spare_capacity_mut().par_chunks_exact_mut(width).take(height).zip(
out_a.spare_capacity_mut().par_chunks_exact_mut(width).take(height).zip(
out_b.spare_capacity_mut().par_chunks_exact_mut(width).take(height))
).enumerate()
.for_each(|(y, (l_row, (a_row, b_row)))| {
let in_row = &img.rows().nth(y).unwrap()[0..width];
let l_row = &mut l_row[0..width];
let a_row = &mut a_row[0..width];
let b_row = &mut b_row[0..width];
for x in 0..width {
let n = (x+11) ^ (y+11);
let (l,a,b) = cb(in_row[x], n);
l_row[x].write(l);
a_row[x].write(a);
b_row[x].write(b);
}
});
unsafe { out_l.set_len(area) };
unsafe { out_a.set_len(area) };
unsafe { out_b.set_len(area) };
vec![
Img::new(out_l, width, height),
Img::new(out_a, width, height),
Img::new(out_b, width, height),
]
}
impl ToLABBitmap for ImgRef<'_, RGBAPLU> {
#[inline]
fn to_lab(&self) -> Vec<GBitmap> {
rgb_to_lab(*self, |px, n|{
px.to_rgb(n).to_lab()
})
}
}
impl ToLABBitmap for ImgRef<'_, RGBLU> {
#[inline]
fn to_lab(&self) -> Vec<GBitmap> {
rgb_to_lab(*self, |px, _n|{
px.to_lab()
})
}
}
#[test]
fn cbrts1() {
let mut totaldiff = 0.;
let mut maxdiff: f64 = 0.;
for i in (0..=10001).rev() {
let x = (i as f64 / 10001.) as f32;
let a = cbrt_poly(x);
let actual = a * a * a;
let expected = x;
let absdiff = (expected as f64 - actual as f64).abs();
assert!(absdiff < 0.0002, "{expected} - {actual} = {} @ {x}", expected - actual);
if i % 400 == 0 {
println!("{:+0.3}", (expected - actual)*255.);
}
totaldiff += absdiff;
maxdiff = maxdiff.max(absdiff);
}
println!("1={totaldiff:0.6}; {maxdiff:0.8}");
assert!(totaldiff < 0.0025, "{totaldiff}");
}
#[test]
fn cbrts2() {
let mut totaldiff = 0.;
let mut maxdiff: f64 = 0.;
for i in (2000..=10001).rev() {
let x = i as f64 / 10001.;
let actual = cbrt_poly(x as f32) as f64;
let expected = x.cbrt();
let absdiff = (expected - actual).abs();
totaldiff += absdiff;
maxdiff = maxdiff.max(absdiff);
assert!(absdiff < 0.0000005, "{expected} - {actual} = {} @ {x}", expected - actual);
}
println!("2={totaldiff:0.6}; {maxdiff:0.8}");
assert!(totaldiff < 0.0025, "{totaldiff}");
}