Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(615)

Side by Side Diff: ui/gfx/color_transform.cc

Issue 2203663002: ColorTransform, transforms colors from one color space to another (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: use floats Created 4 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « ui/gfx/color_transform.h ('k') | ui/gfx/color_transform_unittest.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(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
OLDNEW
« no previous file with comments | « ui/gfx/color_transform.h ('k') | ui/gfx/color_transform_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698