Fix xyzToCieLab bug and add CIE Lab DeltaE 1994 color differencing.

Fixes issue 197.
Fixes issue 198.
This commit is contained in:
castano@gmail.com 2013-09-04 01:08:39 +00:00
parent 022944e49b
commit b2b367ef5f
2 changed files with 66 additions and 5 deletions

View File

@ -132,6 +132,9 @@ float nv::averageAlphaError(const FloatImage * img, const FloatImage * ref)
} }
// Color space conversions based on:
// http://www.brucelindbloom.com/
// Assumes input is in *linear* sRGB color space. // Assumes input is in *linear* sRGB color space.
static Vector3 rgbToXyz(Vector3::Arg c) static Vector3 rgbToXyz(Vector3::Arg c)
{ {
@ -176,7 +179,7 @@ static float f(float t)
{ {
const float epsilon = powf(6.0f/29.0f, 3); const float epsilon = powf(6.0f/29.0f, 3);
if (t < epsilon) { if (t > epsilon) {
return powf(t, 1.0f/3.0f); return powf(t, 1.0f/3.0f);
} }
else { else {
@ -187,14 +190,13 @@ static float f(float t)
static float finv(float t) static float finv(float t)
{ {
if (t > 6.0f / 29.0f) { if (t > 6.0f / 29.0f) {
return powf(t, 3.0f);
}
else {
return 3.0f * powf(6.0f / 29.0f, 2) * (t - 4.0f / 29.0f); return 3.0f * powf(6.0f / 29.0f, 2) * (t - 4.0f / 29.0f);
} }
else {
return powf(t, 3.0f);
}
} }
static Vector3 xyzToCieLab(Vector3::Arg c) static Vector3 xyzToCieLab(Vector3::Arg c)
{ {
// Normalized white point. // Normalized white point.
@ -222,6 +224,12 @@ static Vector3 rgbToCieLab(Vector3::Arg c)
return xyzToCieLab(rgbToXyz(toLinear(c))); return xyzToCieLab(rgbToXyz(toLinear(c)));
} }
// h is hue-angle in radians
static Vector3 cieLabToLCh(Vector3::Arg c)
{
return Vector3(c.x, sqrtf(c.y*c.y + c.z*c.z), atan2f(c.y, c.z));
}
static void rgbToCieLab(const FloatImage * rgbImage, FloatImage * LabImage) static void rgbToCieLab(const FloatImage * rgbImage, FloatImage * LabImage)
{ {
nvDebugCheck(rgbImage != NULL && LabImage != NULL); nvDebugCheck(rgbImage != NULL && LabImage != NULL);
@ -281,6 +289,58 @@ float nv::cieLabError(const FloatImage * img0, const FloatImage * img1)
return float(error / count); return float(error / count);
} }
// Assumes input images are in linear sRGB space.
float nv::cieLab94Error(const FloatImage * img0, const FloatImage * img1)
{
if (!sameLayout(img0, img1)) return FLT_MAX;
nvDebugCheck(img0->componentCount() == 4 && img0->componentCount() == 4);
const float kL = 1;
const float kC = 1;
const float kH = 1;
const float k1 = 0.045f;
const float k2 = 0.015f;
const float sL = 1;
const float * r0 = img0->channel(0);
const float * g0 = img0->channel(1);
const float * b0 = img0->channel(2);
const float * r1 = img1->channel(0);
const float * g1 = img1->channel(1);
const float * b1 = img1->channel(2);
double error = 0.0f;
const uint count = img0->pixelCount();
for (uint i = 0; i < count; ++i)
{
Vector3 lab0 = rgbToCieLab(Vector3(r0[i], g0[i], b0[i]));
Vector3 lch0 = cieLabToLCh(lab0);
Vector3 lab1 = rgbToCieLab(Vector3(r1[i], g1[i], b1[i]));
Vector3 lch1 = cieLabToLCh(lab1);
const float sC = 1 + k1*lch0.x;
const float sH = 1 + k2*lch0.x;
// @@ Measure Delta E using the 1994 definition
Vector3 labDelta = lab0 - lab1;
Vector3 lchDelta = lch0 - lch1;
double deltaLsq = powf(lchDelta.x / (kL*sL), 2);
double deltaCsq = powf(lchDelta.y / (kC*sC), 2);
// avoid possible sqrt of negative value by computing (deltaH/(kH*sH))^2
double deltaHsq = powf(labDelta.y, 2) + powf(labDelta.z, 2) - powf(lchDelta.y, 2);
deltaHsq /= powf(kH*sH, 2);
error += sqrt(deltaLsq + deltaCsq + deltaHsq);
}
return float(error / count);
}
float nv::spatialCieLabError(const FloatImage * img0, const FloatImage * img1) float nv::spatialCieLabError(const FloatImage * img0, const FloatImage * img1)
{ {
if (img0 == NULL || img1 == NULL || img0->width() != img1->width() || img0->height() != img1->height()) { if (img0 == NULL || img1 == NULL || img0->width() != img1->width() || img0->height() != img1->height()) {

View File

@ -10,6 +10,7 @@ namespace nv
float rmsAlphaError(const FloatImage * img, const FloatImage * ref); float rmsAlphaError(const FloatImage * img, const FloatImage * ref);
float cieLabError(const FloatImage * img, const FloatImage * ref); float cieLabError(const FloatImage * img, const FloatImage * ref);
float cieLab94Error(const FloatImage * img, const FloatImage * ref);
float spatialCieLabError(const FloatImage * img, const FloatImage * ref); float spatialCieLabError(const FloatImage * img, const FloatImage * ref);
float averageColorError(const FloatImage * img, const FloatImage * ref, bool alphaWeight); float averageColorError(const FloatImage * img, const FloatImage * ref, bool alphaWeight);