blob: bb0cfb6d24106605065004f3b3a18f2aa7376f7f [file] [log] [blame]
Mike Klein8f5a7a62018-09-11 12:11:46 -04001/*
2 * Copyright 2018 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "gm.h"
9#include "SkColorSpace.h"
Mike Kleinfd007642018-09-12 13:03:06 -040010#include "SkColorSpaceXformSteps.h"
Mike Klein5e31ec62018-09-12 18:04:49 -040011#include "SkGradientShader.h"
Mike Klein8f5a7a62018-09-11 12:11:46 -040012#include "SkString.h"
13
Mike Klein5e31ec62018-09-12 18:04:49 -040014template <typename Fn>
15static void mark(SkCanvas* canvas, Fn&& fn) {
16 SkPaint alpha;
17 alpha.setAlpha(0x50);
18 canvas->saveLayer(nullptr, &alpha);
19 canvas->translate(140,40);
20 canvas->scale(2,2);
21 fn();
22 canvas->restore();
23}
24
25static void mark_good(SkCanvas* canvas) {
26 mark(canvas, [&]{
Mike Kleinfd007642018-09-12 13:03:06 -040027 SkPaint paint;
28
29 // A green circle.
30 paint.setColor(SkColorSetRGB(27, 158, 119));
Mike Klein5e31ec62018-09-12 18:04:49 -040031 canvas->drawCircle(0,0, 12, paint);
Mike Kleinfd007642018-09-12 13:03:06 -040032
33 // Cut out a check mark.
34 paint.setBlendMode(SkBlendMode::kSrc);
35 paint.setColor(0x00000000);
36 paint.setStrokeWidth(2);
37 paint.setStyle(SkPaint::kStroke_Style);
Mike Klein5e31ec62018-09-12 18:04:49 -040038 canvas->drawLine(-6, 0,
39 -1, 5, paint);
40 canvas->drawLine(-1, +5,
41 +7, -5, paint);
42 });
Mike Kleinfd007642018-09-12 13:03:06 -040043}
44
Mike Klein5e31ec62018-09-12 18:04:49 -040045static void mark_bad(SkCanvas* canvas) {
46 mark(canvas, [&] {
Mike Kleinfd007642018-09-12 13:03:06 -040047 SkPaint paint;
48
49 // A red circle.
50 paint.setColor(SkColorSetRGB(231, 41, 138));
Mike Klein5e31ec62018-09-12 18:04:49 -040051 canvas->drawCircle(0,0, 12, paint);
Mike Kleinfd007642018-09-12 13:03:06 -040052
53 // Cut out an 'X'.
54 paint.setBlendMode(SkBlendMode::kSrc);
55 paint.setColor(0x00000000);
56 paint.setStrokeWidth(2);
57 paint.setStyle(SkPaint::kStroke_Style);
Mike Klein5e31ec62018-09-12 18:04:49 -040058 canvas->drawLine(-5,-5,
59 +5,+5, paint);
60 canvas->drawLine(+5,-5,
61 -5,+5, paint);
62 });
Mike Kleinfd007642018-09-12 13:03:06 -040063}
64
65static bool nearly_equal(SkColor4f x, SkColor4f y) {
66 const float K = 0.01f;
67 return fabsf(x.fR - y.fR) < K
68 && fabsf(x.fG - y.fG) < K
69 && fabsf(x.fB - y.fB) < K
70 && fabsf(x.fA - y.fA) < K;
71}
72
73static SkString fmt(SkColor4f c) {
74 return SkStringPrintf("%.2g %.2g %.2g %.2g", c.fR, c.fG, c.fB, c.fA);
75}
76
77static SkColor4f transform(SkColor4f c, SkColorSpace* src, SkColorSpace* dst) {
78 SkColorSpaceXformSteps(src, kUnpremul_SkAlphaType,
79 dst, kUnpremul_SkAlphaType).apply(c.vec());
80 return c;
81}
82
Mike Klein5e31ec62018-09-12 18:04:49 -040083static void compare_pixel(const char* label,
84 SkCanvas* canvas, int x, int y,
85 SkColor4f color, SkColorSpace* cs) {
Mike Kleinfd007642018-09-12 13:03:06 -040086 SkPaint text;
87 text.setAntiAlias(true);
Mike Klein5e31ec62018-09-12 18:04:49 -040088 auto canvas_cs = canvas->imageInfo().refColorSpace();
Mike Klein8f5a7a62018-09-11 12:11:46 -040089
Mike Klein5e31ec62018-09-12 18:04:49 -040090 // I'm not really sure if this makes things easier or harder to follow,
91 // but we sniff the canvas to grab its current y-translate, so that (x,y)
92 // can be written in sort of chunk-relative terms.
93 const SkMatrix& m = canvas->getTotalMatrix();
94 SkASSERT(m.isTranslate());
95 SkScalar dy = m.getTranslateY();
96 SkASSERT(dy == (int)dy);
97 y += (int)dy;
Mike Klein8f5a7a62018-09-11 12:11:46 -040098
Mike Kleinfd007642018-09-12 13:03:06 -040099 SkBitmap bm;
Mike Klein5e31ec62018-09-12 18:04:49 -0400100 bm.allocPixels(SkImageInfo::Make(1,1, kRGBA_F32_SkColorType, kUnpremul_SkAlphaType, canvas_cs));
101 if (!canvas->readPixels(bm, x,y)) {
Mike Kleinfd007642018-09-12 13:03:06 -0400102 mark_good(canvas);
Mike Klein5e31ec62018-09-12 18:04:49 -0400103 canvas->drawString("can't readPixels() on this canvas :(", 100,20, text);
Mike Kleinfd007642018-09-12 13:03:06 -0400104 return;
105 }
Mike Klein8f5a7a62018-09-11 12:11:46 -0400106
Mike Kleinfd007642018-09-12 13:03:06 -0400107 SkColor4f pixel;
Mike Klein5e31ec62018-09-12 18:04:49 -0400108 memcpy(&pixel, bm.getAddr(0,0), sizeof(pixel));
Mike Klein8f5a7a62018-09-11 12:11:46 -0400109
Mike Klein5e31ec62018-09-12 18:04:49 -0400110 SkColor4f expected = transform(color,cs, canvas_cs.get());
Mike Kleinfd007642018-09-12 13:03:06 -0400111 if (canvas->imageInfo().colorType() < kRGBA_F16_SkColorType) {
112 // We can't expect normalized formats to hold values outside [0,1].
113 expected = expected.pin();
114 }
115 if (canvas->imageInfo().colorType() == kGray_8_SkColorType) {
116 // Drawing into Gray8 is known to be maybe-totally broken.
117 // TODO: update expectation here to be {lum,lum,lum,1} if we fix Gray8.
118 expected = SkColor4f{NAN, NAN, NAN, 1};
119 }
Mike Klein8f5a7a62018-09-11 12:11:46 -0400120
Mike Kleinfd007642018-09-12 13:03:06 -0400121 if (nearly_equal(pixel, expected)) {
122 mark_good(canvas);
Mike Klein8f5a7a62018-09-11 12:11:46 -0400123 } else {
Mike Kleinfd007642018-09-12 13:03:06 -0400124 mark_bad(canvas);
125 }
126
127 struct {
128 const char* label;
129 SkColor4f color;
130 } lines[] = {
Mike Klein5e31ec62018-09-12 18:04:49 -0400131 {"Pixel:" , pixel },
132 {"Expected:", expected},
Mike Kleinfd007642018-09-12 13:03:06 -0400133 };
134
135 SkAutoCanvasRestore saveRestore(canvas, true);
Mike Klein5e31ec62018-09-12 18:04:49 -0400136 canvas->drawString(label, 80,20, text);
Mike Kleinfd007642018-09-12 13:03:06 -0400137 for (auto l : lines) {
Mike Klein5e31ec62018-09-12 18:04:49 -0400138 canvas->translate(0,20);
Mike Kleinfd007642018-09-12 13:03:06 -0400139 canvas->drawString(l.label, 80,20, text);
140 canvas->drawString(fmt(l.color).c_str(), 140,20, text);
Mike Klein5e31ec62018-09-12 18:04:49 -0400141 }
142}
143
Brian Osman3f4602f2018-10-18 16:35:49 -0400144DEF_SIMPLE_GM(p3, canvas, 450, 1300) {
Mike Klein5e31ec62018-09-12 18:04:49 -0400145 auto p3 = SkColorSpace::MakeRGB(SkColorSpace::kSRGB_RenderTargetGamma,
146 SkColorSpace::kDCIP3_D65_Gamut);
Mike Kleineee83152018-10-12 13:53:03 -0400147 auto srgb = SkColorSpace::MakeSRGB();
148
149 auto p3_to_srgb = [&](SkColor4f c) {
150 SkPaint p;
151 p.setColor4f(c, p3.get());
152 return p.getColor4f();
153 };
Mike Klein5e31ec62018-09-12 18:04:49 -0400154
155 // Draw a P3 red rectangle and check the corner.
156 {
157 SkPaint paint;
158 paint.setColor4f({1,0,0,1}, p3.get());
Mike Klein5e31ec62018-09-12 18:04:49 -0400159
Mike Klein62bd12f2018-09-17 12:39:09 -0400160 canvas->drawRect({10,10,70,70}, paint);
Mike Klein5e31ec62018-09-12 18:04:49 -0400161 compare_pixel("drawRect P3 red ",
162 canvas, 10,10,
163 {1,0,0,1}, p3.get());
164 }
165
166 canvas->translate(0,80);
167
Mike Klein62bd12f2018-09-17 12:39:09 -0400168 // Draw a P3 red bitmap, using a draw.
169 {
170 SkBitmap bm;
171 bm.allocPixels(SkImageInfo::Make(60,60, kRGBA_F16_SkColorType, kPremul_SkAlphaType, p3));
172
173 SkPaint paint;
174 paint.setColor4f({1,0,0,1}, p3.get());
175 SkCanvas{bm}.drawPaint(paint);
176
177 canvas->drawBitmap(bm, 10,10);
178 compare_pixel("drawBitmap P3 red, from drawPaint",
179 canvas, 10,10,
180 {1,0,0,1}, p3.get());
181 }
182
183 canvas->translate(0,80);
184
185 // Draw a P3 red bitmap, using SkBitmap::eraseColor().
186 {
187 SkBitmap bm;
188 bm.allocPixels(SkImageInfo::Make(60,60, kRGBA_F16_SkColorType, kPremul_SkAlphaType, p3));
189
190 bm.eraseColor(0xffff0000/*in P3*/);
191
192 canvas->drawBitmap(bm, 10,10);
193 compare_pixel("drawBitmap P3 red, from SkBitmap::eraseColor()",
194 canvas, 10,10,
195 {1,0,0,1}, p3.get());
196 }
197
198 canvas->translate(0,80);
199
200 // Draw a P3 red bitmap, using SkPixmap::erase().
201 {
202 SkBitmap bm;
203 bm.allocPixels(SkImageInfo::Make(60,60, kRGBA_F16_SkColorType, kPremul_SkAlphaType, p3));
204
205 // At the moment only SkPixmap has an erase() that takes an SkColor4f.
206 SkPixmap pm;
207 SkAssertResult(bm.peekPixels(&pm));
208 SkAssertResult(pm.erase({1,0,0,1} /*in p3*/));
209
210 canvas->drawBitmap(bm, 10,10);
211 compare_pixel("drawBitmap P3 red, from SkPixmap::erase",
212 canvas, 10,10,
213 {1,0,0,1}, p3.get());
214 }
215
216 canvas->translate(0,80);
217
Brian Osman3f4602f2018-10-18 16:35:49 -0400218 // Draw a P3 red bitmap wrapped in a shader, using SkPixmap::erase().
219 {
220 SkBitmap bm;
221 bm.allocPixels(SkImageInfo::Make(60,60, kRGBA_F16_SkColorType, kPremul_SkAlphaType, p3));
222
223 // At the moment only SkPixmap has an erase() that takes an SkColor4f.
224 SkPixmap pm;
225 SkAssertResult(bm.peekPixels(&pm));
226 SkAssertResult(pm.erase({1,0,0,1} /*in p3*/));
227
228 SkPaint paint;
229 paint.setShader(SkShader::MakeBitmapShader(bm, SkShader::kRepeat_TileMode,
230 SkShader::kRepeat_TileMode));
231
232 canvas->drawRect({10,10,70,70}, paint);
233 compare_pixel("drawBitmapAsShader P3 red, from SkPixmap::erase",
234 canvas, 10,10,
235 {1,0,0,1}, p3.get());
236 }
237
238 canvas->translate(0,80);
239
Mike Kleineee83152018-10-12 13:53:03 -0400240 // Draw a gradient from P3 red to P3 green interpolating in unpremul P3, checking the corners.
Mike Klein5e31ec62018-09-12 18:04:49 -0400241 {
242
243 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
244 SkColor4f colors[] = {{1,0,0,1}, {0,1,0,1}};
245
246 SkPaint paint;
247 paint.setShader(SkGradientShader::MakeLinear(points, colors, p3,
248 nullptr, SK_ARRAY_COUNT(colors),
249 SkShader::kClamp_TileMode));
Mike Klein5e31ec62018-09-12 18:04:49 -0400250 canvas->drawRect({10,10,70,70}, paint);
Mike Klein5e31ec62018-09-12 18:04:49 -0400251 canvas->save();
Mike Kleineee83152018-10-12 13:53:03 -0400252 compare_pixel("UPM P3 gradient, P3 red",
Mike Klein5e31ec62018-09-12 18:04:49 -0400253 canvas, 10,10,
254 {1,0,0,1}, p3.get());
255
256 canvas->translate(180, 0);
257
Mike Kleineee83152018-10-12 13:53:03 -0400258 compare_pixel("UPM P3 gradient, P3 green",
Mike Klein5e31ec62018-09-12 18:04:49 -0400259 canvas, 69,69,
260 {0,1,0,1}, p3.get());
261 canvas->restore();
Mike Klein8f5a7a62018-09-11 12:11:46 -0400262 }
263
Mike Klein9fb5a532018-09-13 15:23:38 -0400264 canvas->translate(0,80);
265
Mike Kleineee83152018-10-12 13:53:03 -0400266 // Draw a gradient from P3 red to P3 green interpolating in premul P3, checking the corners.
Mike Klein983886e2018-09-20 14:10:33 -0400267 {
268
269 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
270 SkColor4f colors[] = {{1,0,0,1}, {0,1,0,1}};
271
272 SkPaint paint;
273 paint.setShader(
274 SkGradientShader::MakeLinear(points, colors, p3,
275 nullptr, SK_ARRAY_COUNT(colors),
276 SkShader::kClamp_TileMode,
277 SkGradientShader::kInterpolateColorsInPremul_Flag,
278 nullptr/*local matrix*/));
279 canvas->drawRect({10,10,70,70}, paint);
280 canvas->save();
Mike Kleineee83152018-10-12 13:53:03 -0400281 compare_pixel("PM P3 gradient, P3 red",
Mike Klein983886e2018-09-20 14:10:33 -0400282 canvas, 10,10,
283 {1,0,0,1}, p3.get());
284
285 canvas->translate(180, 0);
286
Mike Kleineee83152018-10-12 13:53:03 -0400287 compare_pixel("PM P3 gradient, P3 green",
288 canvas, 69,69,
289 {0,1,0,1}, p3.get());
290 canvas->restore();
291 }
292
293 canvas->translate(0,80);
294
295 // Draw a gradient from P3 red to P3 green interpolating in unpremul sRGB, checking the corners.
296 {
297
298 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
299 SkColor4f colors[] = {p3_to_srgb({1,0,0,1}), p3_to_srgb({0,1,0,1})};
300
301 SkPaint paint;
302 paint.setShader(SkGradientShader::MakeLinear(points, colors, srgb,
303 nullptr, SK_ARRAY_COUNT(colors),
304 SkShader::kClamp_TileMode));
305 canvas->drawRect({10,10,70,70}, paint);
306 canvas->save();
307 compare_pixel("UPM sRGB gradient, P3 red",
308 canvas, 10,10,
309 {1,0,0,1}, p3.get());
310
311 canvas->translate(180, 0);
312
313 compare_pixel("UPM sRGB gradient, P3 green",
314 canvas, 69,69,
315 {0,1,0,1}, p3.get());
316 canvas->restore();
317 }
318
319 canvas->translate(0,80);
320
321 // Draw a gradient from P3 red to P3 green interpolating in premul sRGB, checking the corners.
322 {
323
324 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
325 SkColor4f colors[] = {p3_to_srgb({1,0,0,1}), p3_to_srgb({0,1,0,1})};
326
327 SkPaint paint;
328 paint.setShader(
329 SkGradientShader::MakeLinear(points, colors, srgb,
330 nullptr, SK_ARRAY_COUNT(colors),
331 SkShader::kClamp_TileMode,
332 SkGradientShader::kInterpolateColorsInPremul_Flag,
333 nullptr/*local matrix*/));
334 canvas->drawRect({10,10,70,70}, paint);
335 canvas->save();
336 compare_pixel("PM P3 gradient, P3 red",
337 canvas, 10,10,
338 {1,0,0,1}, p3.get());
339
340 canvas->translate(180, 0);
341
342 compare_pixel("PM P3 gradient, P3 green",
Mike Klein983886e2018-09-20 14:10:33 -0400343 canvas, 69,69,
344 {0,1,0,1}, p3.get());
345 canvas->restore();
346 }
347
348 canvas->translate(0,80);
349
Mike Kleinbe569492018-09-14 09:34:21 -0400350 // Draw an A8 image with a P3 red, scaled and not, as a shader or bitmap.
Mike Klein9fb5a532018-09-13 15:23:38 -0400351 {
352 uint8_t mask[256];
353 for (int i = 0; i < 256; i++) {
354 mask[i] = 255-i;
355 }
356 SkBitmap bm;
357 bm.installPixels(SkImageInfo::MakeA8(16,16), mask, 16);
358
Mike Kleinbe569492018-09-14 09:34:21 -0400359 SkPaint as_bitmap;
360 as_bitmap.setColor4f({1,0,0,1}, p3.get());
361 as_bitmap.setFilterQuality(kLow_SkFilterQuality);
Mike Klein9fb5a532018-09-13 15:23:38 -0400362
Mike Kleinbe569492018-09-14 09:34:21 -0400363 SkPaint as_shader;
364 as_shader.setColor4f({1,0,0,1}, p3.get());
365 as_shader.setFilterQuality(kLow_SkFilterQuality);
366 as_shader.setShader(SkShader::MakeBitmapShader(bm, SkShader::kClamp_TileMode
367 , SkShader::kClamp_TileMode));
368
369 canvas->drawBitmap(bm, 10,10, &as_bitmap);
370 compare_pixel("A8 sprite bitmap P3 red",
371 canvas, 10,10,
372 {1,0,0,1}, p3.get());
373
374 canvas->translate(0, 80);
375
376 canvas->save();
377 canvas->translate(10,10);
378 canvas->drawRect({0,0,16,16}, as_shader);
379 canvas->restore();
380 compare_pixel("A8 sprite shader P3 red",
Mike Klein9fb5a532018-09-13 15:23:38 -0400381 canvas, 10,10,
382 {1,0,0,1}, p3.get());
383
384 canvas->translate(0,80);
385
Mike Kleinbe569492018-09-14 09:34:21 -0400386 canvas->drawBitmapRect(bm, {10,10,70,70}, &as_bitmap);
387 compare_pixel("A8 scaled bitmap P3 red",
388 canvas, 10,10,
389 {1,0,0,1}, p3.get());
390
391 canvas->translate(0,80);
392
393 canvas->save();
394 canvas->translate(10,10);
395 canvas->scale(3.75,3.75);
396 canvas->drawRect({0,0,16,16}, as_shader);
397 canvas->restore();
398 compare_pixel("A8 scaled shader P3 red",
Mike Klein9fb5a532018-09-13 15:23:38 -0400399 canvas, 10,10,
400 {1,0,0,1}, p3.get());
401 }
402
Mike Klein8f5a7a62018-09-11 12:11:46 -0400403 // TODO: draw P3 colors more ways
404}