| OLD | NEW |
| (Empty) | |
| 1 // Copyright (c) 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "ui/gfx/color_transform.h" |
| 6 |
| 7 #include <vector> |
| 8 |
| 9 #include "base/logging.h" |
| 10 #include "ui/gfx/color_space.h" |
| 11 #include "ui/gfx/transform.h" |
| 12 |
| 13 namespace gfx { |
| 14 |
| 15 Transform Invert(const Transform& t) { |
| 16 Transform ret = t; |
| 17 if (!t.GetInverse(&ret)) { |
| 18 LOG(ERROR) << "Inverse should alsways be possible."; |
| 19 } |
| 20 return ret; |
| 21 } |
| 22 |
| 23 ColorTransform::TriStim Map(const Transform& t, ColorTransform::TriStim color) { |
| 24 t.TransformPoint(&color); |
| 25 return color; |
| 26 } |
| 27 |
| 28 ColorTransform::TriStim Xy2xyz(float x, float y) { |
| 29 return ColorTransform::TriStim(x, y, 1.0f - x - y); |
| 30 } |
| 31 |
| 32 void GetPrimaries(ColorSpace::PrimaryID id, |
| 33 ColorTransform::TriStim primaries[4]) { |
| 34 switch (id) { |
| 35 default: |
| 36 // If we don't know, assume BT709 |
| 37 |
| 38 case ColorSpace::PrimaryID::BT709: |
| 39 // Red |
| 40 primaries[0] = Xy2xyz(0.640f, 0.330f); |
| 41 // Green |
| 42 primaries[1] = Xy2xyz(0.300f, 0.600f); |
| 43 // Blue |
| 44 primaries[2] = Xy2xyz(0.150f, 0.060f); |
| 45 // Whitepoint (D65f) |
| 46 primaries[3] = Xy2xyz(0.3127f, 0.3290f); |
| 47 break; |
| 48 |
| 49 case ColorSpace::PrimaryID::BT470M: |
| 50 // Red |
| 51 primaries[0] = Xy2xyz(0.67f, 0.33f); |
| 52 // Green |
| 53 primaries[1] = Xy2xyz(0.21f, 0.71f); |
| 54 // Blue |
| 55 primaries[2] = Xy2xyz(0.14f, 0.08f); |
| 56 // Whitepoint |
| 57 primaries[3] = Xy2xyz(0.31f, 0.316f); |
| 58 break; |
| 59 |
| 60 case ColorSpace::PrimaryID::BT470BG: |
| 61 // Red |
| 62 primaries[0] = Xy2xyz(0.64f, 0.33f); |
| 63 // Green |
| 64 primaries[1] = Xy2xyz(0.29f, 0.60f); |
| 65 // Blue |
| 66 primaries[2] = Xy2xyz(0.15f, 0.06f); |
| 67 // Whitepoint (D65f) |
| 68 primaries[3] = Xy2xyz(0.3127f, 0.3290f); |
| 69 break; |
| 70 |
| 71 case ColorSpace::PrimaryID::SMPTE170M: |
| 72 case ColorSpace::PrimaryID::SMPTE240M: |
| 73 // Red |
| 74 primaries[0] = Xy2xyz(0.630f, 0.340f); |
| 75 // Green |
| 76 primaries[1] = Xy2xyz(0.310f, 0.595f); |
| 77 // Blue |
| 78 primaries[2] = Xy2xyz(0.155f, 0.070f); |
| 79 // Whitepoint (D65f) |
| 80 primaries[3] = Xy2xyz(0.3127f, 0.3290f); |
| 81 break; |
| 82 |
| 83 case ColorSpace::PrimaryID::FILM: |
| 84 // Red |
| 85 primaries[0] = Xy2xyz(0.681f, 0.319f); |
| 86 // Green |
| 87 primaries[1] = Xy2xyz(0.243f, 0.692f); |
| 88 // Blue |
| 89 primaries[2] = Xy2xyz(0.145f, 0.049f); |
| 90 // Whitepoint (Cf) |
| 91 primaries[3] = Xy2xyz(0.310f, 0.136f); |
| 92 break; |
| 93 |
| 94 case ColorSpace::PrimaryID::BT2020: |
| 95 // Red |
| 96 primaries[0] = Xy2xyz(0.708f, 0.292f); |
| 97 // Green |
| 98 primaries[1] = Xy2xyz(0.170f, 0.797f); |
| 99 // Blue |
| 100 primaries[2] = Xy2xyz(0.131f, 0.046f); |
| 101 // Whitepoint (D65f) |
| 102 primaries[3] = Xy2xyz(0.3127f, 0.3290f); |
| 103 break; |
| 104 |
| 105 case ColorSpace::PrimaryID::SMPTEST428_1: |
| 106 // X |
| 107 primaries[0] = Xy2xyz(1.0f, 0.0f); |
| 108 // Y |
| 109 primaries[1] = Xy2xyz(0.0f, 1.0f); |
| 110 // Z |
| 111 primaries[2] = Xy2xyz(0.0f, 0.0f); |
| 112 // Whitepoint (Ef) |
| 113 primaries[3] = Xy2xyz(1.0f / 3.0f, 1.0f / 3.0f); |
| 114 break; |
| 115 |
| 116 case ColorSpace::PrimaryID::SMPTEST431_2: |
| 117 // Red |
| 118 primaries[0] = Xy2xyz(0.680f, 0.320f); |
| 119 // Green |
| 120 primaries[1] = Xy2xyz(0.265f, 0.690f); |
| 121 // Blue |
| 122 primaries[2] = Xy2xyz(0.150f, 0.060f); |
| 123 // Whitepoint |
| 124 primaries[3] = Xy2xyz(0.314f, 0.351f); |
| 125 break; |
| 126 |
| 127 case ColorSpace::PrimaryID::SMPTEST432_1: |
| 128 // Red |
| 129 primaries[0] = Xy2xyz(0.680f, 0.320f); |
| 130 // Green |
| 131 primaries[1] = Xy2xyz(0.265f, 0.690f); |
| 132 // Blue |
| 133 primaries[2] = Xy2xyz(0.150f, 0.060f); |
| 134 // Whitepoint (D65f) |
| 135 primaries[3] = Xy2xyz(0.3127f, 0.3290f); |
| 136 break; |
| 137 |
| 138 case ColorSpace::PrimaryID::XYZ_D50: |
| 139 // X |
| 140 primaries[0] = Xy2xyz(1.0f, 0.0f); |
| 141 // Y |
| 142 primaries[1] = Xy2xyz(0.0f, 1.0f); |
| 143 // Z |
| 144 primaries[2] = Xy2xyz(0.0f, 0.0f); |
| 145 // D50 |
| 146 primaries[3] = Xy2xyz(0.34567f, 0.35850f); |
| 147 break; |
| 148 } |
| 149 } |
| 150 |
| 151 GFX_EXPORT Transform GetPrimaryMatrix(ColorSpace::PrimaryID id) { |
| 152 ColorTransform::TriStim primaries[4]; |
| 153 GetPrimaries(id, primaries); |
| 154 ColorTransform::TriStim WXYZ(primaries[3].x() / primaries[3].y(), 1.0f, |
| 155 primaries[3].z() / primaries[3].y()); |
| 156 |
| 157 Transform ret( |
| 158 primaries[0].x(), primaries[1].x(), primaries[2].x(), 0.0f, // 1 |
| 159 primaries[0].y(), primaries[1].y(), primaries[2].y(), 0.0f, // 2 |
| 160 primaries[0].z(), primaries[1].z(), primaries[2].z(), 0.0f, // 3 |
| 161 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 162 |
| 163 ColorTransform::TriStim conv = Map(Invert(ret), WXYZ); |
| 164 ret.Scale3d(conv.x(), conv.y(), conv.z()); |
| 165 |
| 166 // Chromatic adaptation. |
| 167 Transform bradford(0.8951000f, 0.2664000f, -0.1614000f, 0.0f, // 1 |
| 168 -0.7502000f, 1.7135000f, 0.0367000f, 0.0f, // 2 |
| 169 0.0389000f, -0.0685000f, 1.0296000f, 0.0f, // 3 |
| 170 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 171 |
| 172 ColorTransform::TriStim D50(0.9642f, 1.0f, 0.8249f); |
| 173 ColorTransform::TriStim source_response = Map(bradford, WXYZ); |
| 174 ColorTransform::TriStim dest_response = Map(bradford, D50); |
| 175 |
| 176 Transform adapter; |
| 177 adapter.Scale3d(dest_response.x() / source_response.x(), |
| 178 dest_response.y() / source_response.y(), |
| 179 dest_response.z() / source_response.z()); |
| 180 |
| 181 return bradford * adapter * Invert(bradford) * ret; |
| 182 } |
| 183 |
| 184 GFX_EXPORT float FromLinear(ColorSpace::TransferID id, float v) { |
| 185 switch (id) { |
| 186 default: |
| 187 case ColorSpace::TransferID::BT709: |
| 188 case ColorSpace::TransferID::SMPTE170M: |
| 189 case ColorSpace::TransferID::BT2020_10: |
| 190 case ColorSpace::TransferID::BT2020_12: { |
| 191 v = fmax(0.0f, v); |
| 192 float a = 1.099296826809442f; |
| 193 float b = 0.018053968510807f; |
| 194 if (v <= b) { |
| 195 return 4.5f * v; |
| 196 } else { |
| 197 return a * powf(v, 0.45f) - (a - 1.0f); |
| 198 } |
| 199 } |
| 200 |
| 201 case ColorSpace::TransferID::GAMMA22: |
| 202 v = fmax(0.0f, v); |
| 203 return powf(v, 1.0f / 2.2f); |
| 204 |
| 205 case ColorSpace::TransferID::GAMMA28: |
| 206 v = fmax(0.0f, v); |
| 207 return powf(v, 1.0f / 2.8f); |
| 208 |
| 209 case ColorSpace::TransferID::SMPTE240M: { |
| 210 v = fmax(0.0f, v); |
| 211 float a = 1.11157219592173128753f; |
| 212 float b = 0.02282158552944503135f; |
| 213 if (v <= b) { |
| 214 return 4.0f * v; |
| 215 } else { |
| 216 return a * powf(v, 0.45f) - (a - 1.0f); |
| 217 } |
| 218 } |
| 219 |
| 220 case ColorSpace::TransferID::LINEAR: |
| 221 return v; |
| 222 |
| 223 case ColorSpace::TransferID::LOG: |
| 224 if (v < 0.01f) |
| 225 return 0.0f; |
| 226 return 1.0f + log(v) / log(10.0f) / 2.0f; |
| 227 |
| 228 case ColorSpace::TransferID::LOG_SQRT: |
| 229 if (v < sqrt(10.0f) / 1000.0f) |
| 230 return 0.0f; |
| 231 return 1.0f + log(v) / log(10.0f) / 2.5f; |
| 232 |
| 233 case ColorSpace::TransferID::IEC61966_2_4: { |
| 234 float a = 1.099296826809442f; |
| 235 float b = 0.018053968510807f; |
| 236 if (v < -b) { |
| 237 return -a * powf(-v, 0.45f) + (a - 1.0f); |
| 238 } else if (v <= b) { |
| 239 return 4.5f * v; |
| 240 } else { |
| 241 return a * powf(v, 0.45f) - (a - 1.0f); |
| 242 } |
| 243 } |
| 244 |
| 245 case ColorSpace::TransferID::BT1361_ECG: { |
| 246 float a = 1.099f; |
| 247 float b = 0.018f; |
| 248 float l = 0.0045f; |
| 249 if (v < -l) { |
| 250 return -(a * powf(-4.0f * v, 0.45f) + (a - 1.0f)) / 4.0f; |
| 251 } else if (v <= b) { |
| 252 return 4.5f * v; |
| 253 } else { |
| 254 return a * powf(v, 0.45f) - (a - 1.0f); |
| 255 } |
| 256 } |
| 257 |
| 258 case ColorSpace::TransferID::IEC61966_2_1: { // SRGB |
| 259 v = fmax(0.0f, v); |
| 260 float a = 1.055f; |
| 261 float b = 0.0031308f; |
| 262 if (v < b) { |
| 263 return 12.92f * v; |
| 264 } else { |
| 265 return a * powf(v, 1.0f / 2.4f) - (a - 1.0f); |
| 266 } |
| 267 } |
| 268 case ColorSpace::TransferID::SMPTEST2084: { |
| 269 v = fmax(0.0f, v); |
| 270 float m1 = (2610.0f / 4096.0f) / 4.0f; |
| 271 float m2 = (2523.0f / 4096.0f) * 128.0f; |
| 272 float c1 = 3424.0f / 4096.0f; |
| 273 float c2 = (2413.0f / 4096.0f) * 32.0f; |
| 274 float c3 = (2392.0f / 4096.0f) * 32.0f; |
| 275 return powf((c1 + c2 * powf(v, m1)) / (1.0f + c3 * powf(v, m1)), m2); |
| 276 } |
| 277 |
| 278 case ColorSpace::TransferID::SMPTEST428_1: |
| 279 v = fmax(0.0f, v); |
| 280 return powf(48.0f * v + 52.37f, 1.0f / 2.6f); |
| 281 |
| 282 // Chrome-specific values below |
| 283 case ColorSpace::TransferID::GAMMA24: |
| 284 v = fmax(0.0f, v); |
| 285 return powf(v, 1.0f / 2.4f); |
| 286 } |
| 287 } |
| 288 |
| 289 GFX_EXPORT float ToLinear(ColorSpace::TransferID id, float v) { |
| 290 switch (id) { |
| 291 default: |
| 292 case ColorSpace::TransferID::BT709: |
| 293 case ColorSpace::TransferID::SMPTE170M: |
| 294 case ColorSpace::TransferID::BT2020_10: |
| 295 case ColorSpace::TransferID::BT2020_12: { |
| 296 v = fmax(0.0f, v); |
| 297 float a = 1.099296826809442f; |
| 298 float b = 0.018053968510807f; |
| 299 if (v < FromLinear(ColorSpace::TransferID::BT709, b)) { |
| 300 return v / 4.5f; |
| 301 } else { |
| 302 return powf((v + a - 1.0f) / a, 1.0f / 0.45f); |
| 303 } |
| 304 } |
| 305 |
| 306 case ColorSpace::TransferID::GAMMA22: |
| 307 v = fmax(0.0f, v); |
| 308 return powf(v, 2.2f); |
| 309 |
| 310 case ColorSpace::TransferID::GAMMA28: |
| 311 v = fmax(0.0f, v); |
| 312 return powf(v, 2.8f); |
| 313 |
| 314 case ColorSpace::TransferID::SMPTE240M: { |
| 315 v = fmax(0.0f, v); |
| 316 float a = 1.11157219592173128753f; |
| 317 float b = 0.02282158552944503135f; |
| 318 if (v <= FromLinear(ColorSpace::TransferID::SMPTE240M, b)) { |
| 319 return v / 4.0f; |
| 320 } else { |
| 321 return powf((v + a - 1.0f) / a, 1.0f / 0.45f); |
| 322 } |
| 323 } |
| 324 |
| 325 case ColorSpace::TransferID::LINEAR: |
| 326 return v; |
| 327 |
| 328 case ColorSpace::TransferID::LOG: |
| 329 if (v < 0.0f) |
| 330 return 0.0f; |
| 331 return powf(10.0f, (v - 1.0f) * 2.0f); |
| 332 |
| 333 case ColorSpace::TransferID::LOG_SQRT: |
| 334 if (v < 0.0f) |
| 335 return 0.0f; |
| 336 return powf(10.0f, (v - 1.0f) * 2.5f); |
| 337 |
| 338 case ColorSpace::TransferID::IEC61966_2_4: { |
| 339 float a = 1.099296826809442f; |
| 340 float b = 0.018053968510807f; |
| 341 if (v < FromLinear(ColorSpace::TransferID::IEC61966_2_4, -a)) { |
| 342 return -powf((a - 1.0f - v) / a, 1.0f / 0.45f); |
| 343 } else if (v <= FromLinear(ColorSpace::TransferID::IEC61966_2_4, b)) { |
| 344 return v / 4.5f; |
| 345 } else { |
| 346 return powf((v + a - 1.0f) / a, 1.0f / 0.45f); |
| 347 } |
| 348 } |
| 349 |
| 350 case ColorSpace::TransferID::BT1361_ECG: { |
| 351 float a = 1.099f; |
| 352 float b = 0.018f; |
| 353 float l = 0.0045f; |
| 354 if (v < FromLinear(ColorSpace::TransferID::BT1361_ECG, -l)) { |
| 355 return -powf((1.0f - a - v * 4.0f) / a, 1.0f / 0.45f) / 4.0f; |
| 356 } else if (v <= FromLinear(ColorSpace::TransferID::BT1361_ECG, b)) { |
| 357 return v / 4.5f; |
| 358 } else { |
| 359 return powf((v + a - 1.0f) / a, 1.0f / 0.45f); |
| 360 } |
| 361 } |
| 362 |
| 363 case ColorSpace::TransferID::IEC61966_2_1: { // SRGB |
| 364 v = fmax(0.0f, v); |
| 365 float a = 1.055f; |
| 366 float b = 0.0031308f; |
| 367 if (v < FromLinear(ColorSpace::TransferID::IEC61966_2_1, b)) { |
| 368 return v / 12.92f; |
| 369 } else { |
| 370 return powf((v + a - 1.0f) / a, 2.4f); |
| 371 } |
| 372 } |
| 373 |
| 374 case ColorSpace::TransferID::SMPTEST2084: { |
| 375 v = fmax(0.0f, v); |
| 376 float m1 = (2610.0f / 4096.0f) / 4.0f; |
| 377 float m2 = (2523.0f / 4096.0f) * 128.0f; |
| 378 float c1 = 3424.0f / 4096.0f; |
| 379 float c2 = (2413.0f / 4096.0f) * 32.0f; |
| 380 float c3 = (2392.0f / 4096.0f) * 32.0f; |
| 381 return powf( |
| 382 fmax(powf(v, 1.0f / m2) - c1, 0) / (c2 - c3 * powf(v, 1.0f / m2)), |
| 383 1.0f / m1); |
| 384 } |
| 385 |
| 386 case ColorSpace::TransferID::SMPTEST428_1: |
| 387 return (powf(v, 2.6f) - 52.37f) / 48.0f; |
| 388 |
| 389 // Chrome-specific values below |
| 390 case ColorSpace::TransferID::GAMMA24: |
| 391 v = fmax(0.0f, v); |
| 392 return powf(v, 2.4f); |
| 393 } |
| 394 } |
| 395 |
| 396 GFX_EXPORT Transform GetTransferMatrix(ColorSpace::MatrixID id) { |
| 397 float Kr = 0.0f, Kb = 0.0f; |
| 398 switch (id) { |
| 399 case ColorSpace::MatrixID::RGB: |
| 400 return Transform(); |
| 401 |
| 402 case ColorSpace::MatrixID::BT709: |
| 403 case ColorSpace::MatrixID::UNSPECIFIED: |
| 404 case ColorSpace::MatrixID::RESERVED: |
| 405 Kr = 0.2126f; |
| 406 Kb = 0.0722f; |
| 407 break; |
| 408 |
| 409 case ColorSpace::MatrixID::FCC: |
| 410 Kr = 0.30f; |
| 411 Kb = 0.11f; |
| 412 break; |
| 413 |
| 414 case ColorSpace::MatrixID::BT470BG: |
| 415 case ColorSpace::MatrixID::SMPTE170M: |
| 416 Kr = 0.299f; |
| 417 Kb = 0.144f; |
| 418 break; |
| 419 |
| 420 case ColorSpace::MatrixID::SMPTE240M: |
| 421 Kr = 0.212f; |
| 422 Kb = 0.087f; |
| 423 break; |
| 424 |
| 425 case ColorSpace::MatrixID::YCOCG: |
| 426 return Transform(0.25f, 0.5f, 0.25f, 0.5f, // 1 |
| 427 -0.25f, 0.5f, -0.25f, 0.5f, // 2 |
| 428 0.5f, 0.0f, -0.5f, 0.0f, // 3 |
| 429 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 430 |
| 431 // TODO(hubbe): Check if the CL equation is right. |
| 432 case ColorSpace::MatrixID::BT2020_NCL: |
| 433 case ColorSpace::MatrixID::BT2020_CL: |
| 434 Kr = 0.2627f; |
| 435 Kb = 0.0593f; |
| 436 break; |
| 437 |
| 438 case ColorSpace::MatrixID::YDZDX: |
| 439 return Transform(0.0f, 1.0f, 0.0f, 0.0f, // 1 |
| 440 0.0f, -0.5f, 0.986566f / 2.0f, 0.5f, // 2 |
| 441 0.5f, -0.991902f / 2.0f, 0.0f, 0.5f, // 3 |
| 442 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 443 } |
| 444 float u_m = 0.5f / (1.0f - Kb); |
| 445 float v_m = 0.5f / (1.0f - Kr); |
| 446 return Transform( |
| 447 Kr, 1.0f - Kr - Kb, Kb, 0.0f, // 1 |
| 448 u_m * -Kr, u_m * -(1.0f - Kr - Kb), u_m * (1.0f - Kb), 0.5f, // 2 |
| 449 v_m * (1.0f - Kr), v_m * -(1.0f - Kr - Kb), v_m * -Kb, 0.5f, // 3 |
| 450 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 451 } |
| 452 |
| 453 Transform GetRangeAdjustMatrix(ColorSpace::RangeID range, |
| 454 ColorSpace::MatrixID matrix) { |
| 455 switch (range) { |
| 456 case ColorSpace::RangeID::FULL: |
| 457 return Transform(); |
| 458 |
| 459 case ColorSpace::RangeID::LIMITED: |
| 460 break; |
| 461 } |
| 462 switch (matrix) { |
| 463 case ColorSpace::MatrixID::RGB: |
| 464 case ColorSpace::MatrixID::YCOCG: |
| 465 return Transform(255.0f / 219.0f, 0.0f, 0.0f, -16.0f / 219.0f, // 1 |
| 466 0.0f, 255.0f / 219.0f, 0.0f, -16.0f / 219.0f, // 2 |
| 467 0.0f, 0.0f, 255.0f / 219.0f, -16.0f / 219.0f, // 3 |
| 468 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 469 |
| 470 case ColorSpace::MatrixID::BT709: |
| 471 case ColorSpace::MatrixID::UNSPECIFIED: |
| 472 case ColorSpace::MatrixID::RESERVED: |
| 473 case ColorSpace::MatrixID::FCC: |
| 474 case ColorSpace::MatrixID::BT470BG: |
| 475 case ColorSpace::MatrixID::SMPTE170M: |
| 476 case ColorSpace::MatrixID::SMPTE240M: |
| 477 case ColorSpace::MatrixID::BT2020_NCL: |
| 478 case ColorSpace::MatrixID::BT2020_CL: |
| 479 case ColorSpace::MatrixID::YDZDX: |
| 480 return Transform(255.0f / 219.0f, 0.0f, 0.0f, -16.0f / 219.0f, // 1 |
| 481 0.0f, 255.0f / 224.0f, 0.0f, -15.5f / 224.0f, // 2 |
| 482 0.0f, 0.0f, 255.0f / 224.0f, -15.5f / 224.0f, // 3 |
| 483 0.0f, 0.0f, 0.0f, 1.0f); // 4 |
| 484 } |
| 485 NOTREACHED(); |
| 486 return Transform(); |
| 487 } |
| 488 |
| 489 class ColorSpaceToColorSpaceTransform : public ColorTransform { |
| 490 public: |
| 491 ColorSpaceToColorSpaceTransform(const ColorSpace& from, |
| 492 const ColorSpace& to, |
| 493 Intent intent) |
| 494 : from_(from), to_(to) { |
| 495 if (intent == Intent::PERCEPTUAL) { |
| 496 switch (from_.transfer_) { |
| 497 case ColorSpace::TransferID::UNSPECIFIED: |
| 498 case ColorSpace::TransferID::BT709: |
| 499 case ColorSpace::TransferID::SMPTE170M: |
| 500 // See SMPTE 1886 |
| 501 from_.transfer_ = ColorSpace::TransferID::GAMMA24; |
| 502 break; |
| 503 |
| 504 default: // Do nothing |
| 505 break; |
| 506 } |
| 507 |
| 508 // TODO(hubbe): shrink gamuts here (never stretch gamuts) |
| 509 } |
| 510 |
| 511 Transform* from_transfer_matrix = |
| 512 from_.matrix_ == ColorSpace::MatrixID::BT2020_CL ? &b_ : &a_; |
| 513 Transform* to_transfer_matrix = |
| 514 to_.matrix_ == ColorSpace::MatrixID::BT2020_CL ? &b_ : &c_; |
| 515 |
| 516 c_ *= Invert(GetRangeAdjustMatrix(to_.range_, to_.matrix_)); |
| 517 *to_transfer_matrix *= GetTransferMatrix(to_.matrix_); |
| 518 b_ *= Invert(GetPrimaryMatrix(to_.primaries_)); |
| 519 b_ *= GetPrimaryMatrix(from_.primaries_); |
| 520 *from_transfer_matrix *= Invert(GetTransferMatrix(from_.matrix_)); |
| 521 a_ *= GetRangeAdjustMatrix(from_.range_, from_.matrix_); |
| 522 } |
| 523 |
| 524 void transform(TriStim* colors, size_t num) override { |
| 525 for (size_t i = 0; i < num; i++) { |
| 526 TriStim c = colors[i]; |
| 527 a_.TransformPoint(&c); |
| 528 c.set_x(ToLinear(from_.transfer_, c.x())); |
| 529 c.set_y(ToLinear(from_.transfer_, c.y())); |
| 530 c.set_z(ToLinear(from_.transfer_, c.z())); |
| 531 b_.TransformPoint(&c); |
| 532 c.set_x(FromLinear(to_.transfer_, c.x())); |
| 533 c.set_y(FromLinear(to_.transfer_, c.y())); |
| 534 c.set_z(FromLinear(to_.transfer_, c.z())); |
| 535 c_.TransformPoint(&c); |
| 536 colors[i] = c; |
| 537 } |
| 538 } |
| 539 |
| 540 private: |
| 541 ColorSpace from_; |
| 542 ColorSpace to_; |
| 543 |
| 544 // a_ -> tolinear -> b_ -> fromlinear -> c_; |
| 545 Transform a_; |
| 546 Transform b_; |
| 547 Transform c_; |
| 548 }; |
| 549 |
| 550 std::unique_ptr<ColorTransform> ColorTransform::NewColorTransform( |
| 551 const ColorSpace& from, |
| 552 const ColorSpace& to, |
| 553 Intent intent) { |
| 554 // TODO(Hubbe): Check if from and/or to can be mapped to ICC profiles and |
| 555 // provide better transforms in those cases. |
| 556 return std::unique_ptr<ColorTransform>( |
| 557 new ColorSpaceToColorSpaceTransform(from, to, intent)); |
| 558 } |
| 559 |
| 560 } // namespace gfx |
| OLD | NEW |