1 /* 2 * Copyright (C) 2021 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.internal.graphics.cam; 18 19 import android.annotation.NonNull; 20 import android.annotation.Nullable; 21 22 import com.android.internal.graphics.ColorUtils; 23 24 /** 25 * A color appearance model, based on CAM16, extended to use L* as the lightness dimension, and 26 * coupled to a gamut mapping algorithm. Creates a color system, enables a digital design system. 27 */ 28 public class Cam { 29 // The maximum difference between the requested L* and the L* returned. 30 private static final float DL_MAX = 0.2f; 31 // The maximum color distance, in CAM16-UCS, between a requested color and the color returned. 32 private static final float DE_MAX = 1.0f; 33 // When the delta between the floor & ceiling of a binary search for chroma is less than this, 34 // the binary search terminates. 35 private static final float CHROMA_SEARCH_ENDPOINT = 0.4f; 36 // When the delta between the floor & ceiling of a binary search for J, lightness in CAM16, 37 // is less than this, the binary search terminates. 38 private static final float LIGHTNESS_SEARCH_ENDPOINT = 0.01f; 39 40 // CAM16 color dimensions, see getters for documentation. 41 private final float mHue; 42 private final float mChroma; 43 private final float mJ; 44 private final float mQ; 45 private final float mM; 46 private final float mS; 47 48 // Coordinates in UCS space. Used to determine color distance, like delta E equations in L*a*b*. 49 private final float mJstar; 50 private final float mAstar; 51 private final float mBstar; 52 53 /** Hue in CAM16 */ getHue()54 public float getHue() { 55 return mHue; 56 } 57 58 /** Chroma in CAM16 */ getChroma()59 public float getChroma() { 60 return mChroma; 61 } 62 63 /** Lightness in CAM16 */ getJ()64 public float getJ() { 65 return mJ; 66 } 67 68 /** 69 * Brightness in CAM16. 70 * 71 * <p>Prefer lightness, brightness is an absolute quantity. For example, a sheet of white paper 72 * is much brighter viewed in sunlight than in indoor light, but it is the lightest object under 73 * any lighting. 74 */ getQ()75 public float getQ() { 76 return mQ; 77 } 78 79 /** 80 * Colorfulness in CAM16. 81 * 82 * <p>Prefer chroma, colorfulness is an absolute quantity. For example, a yellow toy car is much 83 * more colorful outside than inside, but it has the same chroma in both environments. 84 */ getM()85 public float getM() { 86 return mM; 87 } 88 89 /** 90 * Saturation in CAM16. 91 * 92 * <p>Colorfulness in proportion to brightness. Prefer chroma, saturation measures colorfulness 93 * relative to the color's own brightness, where chroma is colorfulness relative to white. 94 */ getS()95 public float getS() { 96 return mS; 97 } 98 99 /** Lightness coordinate in CAM16-UCS */ getJstar()100 public float getJstar() { 101 return mJstar; 102 } 103 104 /** a* coordinate in CAM16-UCS */ getAstar()105 public float getAstar() { 106 return mAstar; 107 } 108 109 /** b* coordinate in CAM16-UCS */ getBstar()110 public float getBstar() { 111 return mBstar; 112 } 113 114 /** Construct a CAM16 color */ Cam(float hue, float chroma, float j, float q, float m, float s, float jstar, float astar, float bstar)115 Cam(float hue, float chroma, float j, float q, float m, float s, float jstar, float astar, 116 float bstar) { 117 mHue = hue; 118 mChroma = chroma; 119 mJ = j; 120 mQ = q; 121 mM = m; 122 mS = s; 123 mJstar = jstar; 124 mAstar = astar; 125 mBstar = bstar; 126 } 127 128 /** 129 * Given a hue & chroma in CAM16, L* in L*a*b*, return an ARGB integer. The chroma of the color 130 * returned may, and frequently will, be lower than requested. Assumes the color is viewed in 131 * the 132 * frame defined by the sRGB standard. 133 */ getInt(float hue, float chroma, float lstar)134 public static int getInt(float hue, float chroma, float lstar) { 135 return getInt(hue, chroma, lstar, Frame.DEFAULT); 136 } 137 138 /** 139 * Create a color appearance model from a ARGB integer representing a color. It is assumed the 140 * color was viewed in the frame defined in the sRGB standard. 141 */ 142 @NonNull fromInt(int argb)143 public static Cam fromInt(int argb) { 144 return fromIntInFrame(argb, Frame.DEFAULT); 145 } 146 147 /** 148 * Create a color appearance model from a ARGB integer representing a color, specifying the 149 * frame in which the color was viewed. Prefer Cam.fromInt. 150 */ 151 @NonNull fromIntInFrame(int argb, @NonNull Frame frame)152 public static Cam fromIntInFrame(int argb, @NonNull Frame frame) { 153 // Transform ARGB int to XYZ 154 float[] xyz = CamUtils.xyzFromInt(argb); 155 156 // Transform XYZ to 'cone'/'rgb' responses 157 float[][] matrix = CamUtils.XYZ_TO_CAM16RGB; 158 float rT = (xyz[0] * matrix[0][0]) + (xyz[1] * matrix[0][1]) + (xyz[2] * matrix[0][2]); 159 float gT = (xyz[0] * matrix[1][0]) + (xyz[1] * matrix[1][1]) + (xyz[2] * matrix[1][2]); 160 float bT = (xyz[0] * matrix[2][0]) + (xyz[1] * matrix[2][1]) + (xyz[2] * matrix[2][2]); 161 162 // Discount illuminant 163 float rD = frame.getRgbD()[0] * rT; 164 float gD = frame.getRgbD()[1] * gT; 165 float bD = frame.getRgbD()[2] * bT; 166 167 // Chromatic adaptation 168 float rAF = (float) Math.pow(frame.getFl() * Math.abs(rD) / 100.0, 0.42); 169 float gAF = (float) Math.pow(frame.getFl() * Math.abs(gD) / 100.0, 0.42); 170 float bAF = (float) Math.pow(frame.getFl() * Math.abs(bD) / 100.0, 0.42); 171 float rA = Math.signum(rD) * 400.0f * rAF / (rAF + 27.13f); 172 float gA = Math.signum(gD) * 400.0f * gAF / (gAF + 27.13f); 173 float bA = Math.signum(bD) * 400.0f * bAF / (bAF + 27.13f); 174 175 // redness-greenness 176 float a = (float) (11.0 * rA + -12.0 * gA + bA) / 11.0f; 177 // yellowness-blueness 178 float b = (float) (rA + gA - 2.0 * bA) / 9.0f; 179 180 // auxiliary components 181 float u = (20.0f * rA + 20.0f * gA + 21.0f * bA) / 20.0f; 182 float p2 = (40.0f * rA + 20.0f * gA + bA) / 20.0f; 183 184 // hue 185 float atan2 = (float) Math.atan2(b, a); 186 float atanDegrees = atan2 * 180.0f / (float) Math.PI; 187 float hue = 188 atanDegrees < 0 189 ? atanDegrees + 360.0f 190 : atanDegrees >= 360 ? atanDegrees - 360.0f : atanDegrees; 191 float hueRadians = hue * (float) Math.PI / 180.0f; 192 193 // achromatic response to color 194 float ac = p2 * frame.getNbb(); 195 196 // CAM16 lightness and brightness 197 float j = 100.0f * (float) Math.pow(ac / frame.getAw(), frame.getC() * frame.getZ()); 198 float q = 199 4.0f 200 / frame.getC() 201 * (float) Math.sqrt(j / 100.0f) 202 * (frame.getAw() + 4.0f) 203 * frame.getFlRoot(); 204 205 // CAM16 chroma, colorfulness, and saturation. 206 float huePrime = (hue < 20.14) ? hue + 360 : hue; 207 float eHue = 0.25f * (float) (Math.cos(huePrime * Math.PI / 180.0 + 2.0) + 3.8); 208 float p1 = 50000.0f / 13.0f * eHue * frame.getNc() * frame.getNcb(); 209 float t = p1 * (float) Math.sqrt(a * a + b * b) / (u + 0.305f); 210 float alpha = 211 (float) Math.pow(t, 0.9) * (float) Math.pow(1.64 - Math.pow(0.29, frame.getN()), 212 0.73); 213 // CAM16 chroma, colorfulness, saturation 214 float c = alpha * (float) Math.sqrt(j / 100.0); 215 float m = c * frame.getFlRoot(); 216 float s = 50.0f * (float) Math.sqrt((alpha * frame.getC()) / (frame.getAw() + 4.0f)); 217 218 // CAM16-UCS components 219 float jstar = (1.0f + 100.0f * 0.007f) * j / (1.0f + 0.007f * j); 220 float mstar = 1.0f / 0.0228f * (float) Math.log(1.0f + 0.0228f * m); 221 float astar = mstar * (float) Math.cos(hueRadians); 222 float bstar = mstar * (float) Math.sin(hueRadians); 223 224 return new Cam(hue, c, j, q, m, s, jstar, astar, bstar); 225 } 226 227 /** 228 * Create a CAM from lightness, chroma, and hue coordinates. It is assumed those coordinates 229 * were measured in the sRGB standard frame. 230 */ 231 @NonNull fromJch(float j, float c, float h)232 private static Cam fromJch(float j, float c, float h) { 233 return fromJchInFrame(j, c, h, Frame.DEFAULT); 234 } 235 236 /** 237 * Create a CAM from lightness, chroma, and hue coordinates, and also specify the frame in which 238 * the color is being viewed. 239 */ 240 @NonNull fromJchInFrame(float j, float c, float h, Frame frame)241 private static Cam fromJchInFrame(float j, float c, float h, Frame frame) { 242 float q = 243 4.0f 244 / frame.getC() 245 * (float) Math.sqrt(j / 100.0) 246 * (frame.getAw() + 4.0f) 247 * frame.getFlRoot(); 248 float m = c * frame.getFlRoot(); 249 float alpha = c / (float) Math.sqrt(j / 100.0); 250 float s = 50.0f * (float) Math.sqrt((alpha * frame.getC()) / (frame.getAw() + 4.0f)); 251 252 float hueRadians = h * (float) Math.PI / 180.0f; 253 float jstar = (1.0f + 100.0f * 0.007f) * j / (1.0f + 0.007f * j); 254 float mstar = 1.0f / 0.0228f * (float) Math.log(1.0 + 0.0228 * m); 255 float astar = mstar * (float) Math.cos(hueRadians); 256 float bstar = mstar * (float) Math.sin(hueRadians); 257 return new Cam(h, c, j, q, m, s, jstar, astar, bstar); 258 } 259 260 /** 261 * Distance in CAM16-UCS space between two colors. 262 * 263 * <p>Much like L*a*b* was designed to measure distance between colors, the CAM16 standard 264 * defined a color space called CAM16-UCS to measure distance between CAM16 colors. 265 */ distance(@onNull Cam other)266 public float distance(@NonNull Cam other) { 267 float dJ = getJstar() - other.getJstar(); 268 float dA = getAstar() - other.getAstar(); 269 float dB = getBstar() - other.getBstar(); 270 double dEPrime = Math.sqrt(dJ * dJ + dA * dA + dB * dB); 271 double dE = 1.41 * Math.pow(dEPrime, 0.63); 272 return (float) dE; 273 } 274 275 /** Returns perceived color as an ARGB integer, as viewed in standard sRGB frame. */ viewedInSrgb()276 public int viewedInSrgb() { 277 return viewed(Frame.DEFAULT); 278 } 279 280 /** Returns color perceived in a frame as an ARGB integer. */ viewed(@onNull Frame frame)281 public int viewed(@NonNull Frame frame) { 282 float alpha = 283 (getChroma() == 0.0 || getJ() == 0.0) 284 ? 0.0f 285 : getChroma() / (float) Math.sqrt(getJ() / 100.0); 286 287 float t = 288 (float) Math.pow(alpha / Math.pow(1.64 - Math.pow(0.29, frame.getN()), 0.73), 289 1.0 / 0.9); 290 float hRad = getHue() * (float) Math.PI / 180.0f; 291 292 float eHue = 0.25f * (float) (Math.cos(hRad + 2.0) + 3.8); 293 float ac = frame.getAw() * (float) Math.pow(getJ() / 100.0, 294 1.0 / frame.getC() / frame.getZ()); 295 float p1 = eHue * (50000.0f / 13.0f) * frame.getNc() * frame.getNcb(); 296 float p2 = (ac / frame.getNbb()); 297 298 float hSin = (float) Math.sin(hRad); 299 float hCos = (float) Math.cos(hRad); 300 301 float gamma = 302 23.0f * (p2 + 0.305f) * t / (23.0f * p1 + 11.0f * t * hCos + 108.0f * t * hSin); 303 float a = gamma * hCos; 304 float b = gamma * hSin; 305 float rA = (460.0f * p2 + 451.0f * a + 288.0f * b) / 1403.0f; 306 float gA = (460.0f * p2 - 891.0f * a - 261.0f * b) / 1403.0f; 307 float bA = (460.0f * p2 - 220.0f * a - 6300.0f * b) / 1403.0f; 308 309 float rCBase = (float) Math.max(0, (27.13 * Math.abs(rA)) / (400.0 - Math.abs(rA))); 310 float rC = Math.signum(rA) * (100.0f / frame.getFl()) * (float) Math.pow(rCBase, 311 1.0 / 0.42); 312 float gCBase = (float) Math.max(0, (27.13 * Math.abs(gA)) / (400.0 - Math.abs(gA))); 313 float gC = Math.signum(gA) * (100.0f / frame.getFl()) * (float) Math.pow(gCBase, 314 1.0 / 0.42); 315 float bCBase = (float) Math.max(0, (27.13 * Math.abs(bA)) / (400.0 - Math.abs(bA))); 316 float bC = Math.signum(bA) * (100.0f / frame.getFl()) * (float) Math.pow(bCBase, 317 1.0 / 0.42); 318 float rF = rC / frame.getRgbD()[0]; 319 float gF = gC / frame.getRgbD()[1]; 320 float bF = bC / frame.getRgbD()[2]; 321 322 323 float[][] matrix = CamUtils.CAM16RGB_TO_XYZ; 324 float x = (rF * matrix[0][0]) + (gF * matrix[0][1]) + (bF * matrix[0][2]); 325 float y = (rF * matrix[1][0]) + (gF * matrix[1][1]) + (bF * matrix[1][2]); 326 float z = (rF * matrix[2][0]) + (gF * matrix[2][1]) + (bF * matrix[2][2]); 327 328 int argb = ColorUtils.XYZToColor(x, y, z); 329 return argb; 330 } 331 332 /** 333 * Given a hue & chroma in CAM16, L* in L*a*b*, and the frame in which the color will be 334 * viewed, 335 * return an ARGB integer. 336 * 337 * <p>The chroma of the color returned may, and frequently will, be lower than requested. This 338 * is 339 * a fundamental property of color that cannot be worked around by engineering. For example, a 340 * red 341 * hue, with high chroma, and high L* does not exist: red hues have a maximum chroma below 10 342 * in 343 * light shades, creating pink. 344 */ getInt(float hue, float chroma, float lstar, @NonNull Frame frame)345 public static int getInt(float hue, float chroma, float lstar, @NonNull Frame frame) { 346 // This is a crucial routine for building a color system, CAM16 itself is not sufficient. 347 // 348 // * Why these dimensions? 349 // Hue and chroma from CAM16 are used because they're the most accurate measures of those 350 // quantities. L* from L*a*b* is used because it correlates with luminance, luminance is 351 // used to measure contrast for a11y purposes, thus providing a key constraint on what 352 // colors 353 // can be used. 354 // 355 // * Why is this routine required to build a color system? 356 // In all perceptually accurate color spaces (i.e. L*a*b* and later), `chroma` may be 357 // impossible for a given `hue` and `lstar`. 358 // For example, a high chroma light red does not exist - chroma is limited to below 10 at 359 // light red shades, we call that pink. High chroma light green does exist, but not dark 360 // Also, when converting from another color space to RGB, the color may not be able to be 361 // represented in RGB. In those cases, the conversion process ends with RGB values 362 // outside 0-255 363 // The vast majority of color libraries surveyed simply round to 0 to 255. That is not an 364 // option for this library, as it distorts the expected luminance, and thus the expected 365 // contrast needed for a11y 366 // 367 // * What does this routine do? 368 // Dealing with colors in one color space not fitting inside RGB is, loosely referred to as 369 // gamut mapping or tone mapping. These algorithms are traditionally idiosyncratic, there is 370 // no universal answer. However, because the intent of this library is to build a system for 371 // digital design, and digital design uses luminance to measure contrast/a11y, we have one 372 // very important constraint that leads to an objective algorithm: the L* of the returned 373 // color _must_ match the requested L*. 374 // 375 // Intuitively, if the color must be distorted to fit into the RGB gamut, and the L* 376 // requested *must* be fulfilled, than the hue or chroma of the returned color will need 377 // to be different from the requested hue/chroma. 378 // 379 // After exploring both options, it was more intuitive that if the requested chroma could 380 // not be reached, it used the highest possible chroma. The alternative was finding the 381 // closest hue where the requested chroma could be reached, but that is not nearly as 382 // intuitive, as the requested hue is so fundamental to the color description. 383 384 // If the color doesn't have meaningful chroma, return a gray with the requested Lstar. 385 // 386 // Yellows are very chromatic at L = 100, and blues are very chromatic at L = 0. All the 387 // other hues are white at L = 100, and black at L = 0. To preserve consistency for users of 388 // this system, it is better to simply return white at L* > 99, and black and L* < 0. 389 if (frame == Frame.DEFAULT) { 390 // If the viewing conditions are the same as the default sRGB-like viewing conditions, 391 // skip to using HctSolver: it uses geometrical insights to find the closest in-gamut 392 // match to hue/chroma/lstar. 393 return HctSolver.solveToInt(hue, chroma, lstar); 394 } 395 396 if (chroma < 1.0 || Math.round(lstar) <= 0.0 || Math.round(lstar) >= 100.0) { 397 return CamUtils.intFromLstar(lstar); 398 } 399 400 hue = hue < 0 ? 0 : Math.min(360, hue); 401 402 // The highest chroma possible. Updated as binary search proceeds. 403 float high = chroma; 404 405 // The guess for the current binary search iteration. Starts off at the highest chroma, 406 // thus, if a color is possible at the requested chroma, the search can stop after one try. 407 float mid = chroma; 408 float low = 0.0f; 409 boolean isFirstLoop = true; 410 411 Cam answer = null; 412 413 while (Math.abs(low - high) >= CHROMA_SEARCH_ENDPOINT) { 414 // Given the current chroma guess, mid, and the desired hue, find J, lightness in 415 // CAM16 color space, that creates a color with L* = `lstar` in the L*a*b* color space. 416 Cam possibleAnswer = findCamByJ(hue, mid, lstar); 417 418 if (isFirstLoop) { 419 if (possibleAnswer != null) { 420 return possibleAnswer.viewed(frame); 421 } else { 422 // If this binary search iteration was the first iteration, and this point 423 // has been reached, it means the requested chroma was not available at the 424 // requested hue and L*. 425 // Proceed to a traditional binary search that starts at the midpoint between 426 // the requested chroma and 0. 427 isFirstLoop = false; 428 mid = low + (high - low) / 2.0f; 429 continue; 430 } 431 } 432 433 if (possibleAnswer == null) { 434 // There isn't a CAM16 J that creates a color with L* `lstar`. Try a lower chroma. 435 high = mid; 436 } else { 437 answer = possibleAnswer; 438 // It is possible to create a color. Try higher chroma. 439 low = mid; 440 } 441 442 mid = low + (high - low) / 2.0f; 443 } 444 445 // There was no answer: meaning, for the desired hue, there was no chroma low enough to 446 // generate a color with the desired L*. 447 // All values of L* are possible when there is 0 chroma. Return a color with 0 chroma, i.e. 448 // a shade of gray, with the desired L*. 449 if (answer == null) { 450 return CamUtils.intFromLstar(lstar); 451 } 452 453 return answer.viewed(frame); 454 } 455 456 // Find J, lightness in CAM16 color space, that creates a color with L* = `lstar` in the L*a*b* 457 // color space. 458 // 459 // Returns null if no J could be found that generated a color with L* `lstar`. 460 @Nullable findCamByJ(float hue, float chroma, float lstar)461 private static Cam findCamByJ(float hue, float chroma, float lstar) { 462 float low = 0.0f; 463 float high = 100.0f; 464 float mid = 0.0f; 465 float bestdL = 1000.0f; 466 float bestdE = 1000.0f; 467 468 Cam bestCam = null; 469 while (Math.abs(low - high) > LIGHTNESS_SEARCH_ENDPOINT) { 470 mid = low + (high - low) / 2; 471 // Create the intended CAM color 472 Cam camBeforeClip = Cam.fromJch(mid, chroma, hue); 473 // Convert the CAM color to RGB. If the color didn't fit in RGB, during the conversion, 474 // the initial RGB values will be outside 0 to 255. The final RGB values are clipped to 475 // 0 to 255, distorting the intended color. 476 int clipped = camBeforeClip.viewedInSrgb(); 477 float clippedLstar = CamUtils.lstarFromInt(clipped); 478 float dL = Math.abs(lstar - clippedLstar); 479 480 // If the clipped color's L* is within error margin... 481 if (dL < DL_MAX) { 482 // ...check if the CAM equivalent of the clipped color is far away from intended CAM 483 // color. For the intended color, use lightness and chroma from the clipped color, 484 // and the intended hue. Callers are wondering what the lightness is, they know 485 // chroma may be distorted, so the only concern here is if the hue slipped too far. 486 Cam camClipped = Cam.fromInt(clipped); 487 float dE = camClipped.distance( 488 Cam.fromJch(camClipped.getJ(), camClipped.getChroma(), hue)); 489 if (dE <= DE_MAX) { 490 bestdL = dL; 491 bestdE = dE; 492 bestCam = camClipped; 493 } 494 } 495 496 // If there's no error at all, there's no need to search more. 497 // 498 // Note: this happens much more frequently than expected, but this is a very delicate 499 // property which relies on extremely precise sRGB <=> XYZ calculations, as well as fine 500 // tuning of the constants that determine error margins and when the binary search can 501 // terminate. 502 if (bestdL == 0 && bestdE == 0) { 503 break; 504 } 505 506 if (clippedLstar < lstar) { 507 low = mid; 508 } else { 509 high = mid; 510 } 511 } 512 513 return bestCam; 514 } 515 516 } 517