blob: 99b95f1a31a10cda3cb83cce1ced3a0f5b3f4582 [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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -04009#include "include/core/SkBitmap.h"
10#include "include/core/SkCanvas.h"
11#include "include/core/SkColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkColorSpace.h"
13#include "include/core/SkFont.h"
Mike Reed607a3822021-01-24 19:49:21 -050014#include "include/core/SkImage.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040015#include "include/core/SkImageInfo.h"
16#include "include/core/SkMatrix.h"
17#include "include/core/SkPaint.h"
18#include "include/core/SkPathEffect.h"
19#include "include/core/SkPixmap.h"
20#include "include/core/SkPoint.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040021#include "include/core/SkShader.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "include/core/SkString.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040023#include "include/core/SkTileMode.h"
24#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050025#include "include/effects/SkDashPathEffect.h"
26#include "include/effects/SkGradientShader.h"
Kevin Lubick46572b42023-01-18 13:11:06 -050027#include "include/private/base/SkTPin.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050028#include "src/core/SkColorSpaceXformSteps.h"
Brian Osman699841c2023-01-04 10:32:03 -050029#include "src/core/SkImageInfoPriv.h"
Kevin Lubickbca43ec2023-10-30 10:11:22 -040030#include "tools/fonts/FontToolUtils.h"
Mike Klein8f5a7a62018-09-11 12:11:46 -040031
Ben Wagner7fde8e12019-05-01 17:28:53 -040032#include <math.h>
33#include <string.h>
34
Mike Kleinfd007642018-09-12 13:03:06 -040035static bool nearly_equal(SkColor4f x, SkColor4f y) {
36 const float K = 0.01f;
37 return fabsf(x.fR - y.fR) < K
38 && fabsf(x.fG - y.fG) < K
39 && fabsf(x.fB - y.fB) < K
40 && fabsf(x.fA - y.fA) < K;
41}
42
43static SkString fmt(SkColor4f c) {
44 return SkStringPrintf("%.2g %.2g %.2g %.2g", c.fR, c.fG, c.fB, c.fA);
45}
46
47static SkColor4f transform(SkColor4f c, SkColorSpace* src, SkColorSpace* dst) {
48 SkColorSpaceXformSteps(src, kUnpremul_SkAlphaType,
49 dst, kUnpremul_SkAlphaType).apply(c.vec());
50 return c;
51}
52
Mike Klein5e31ec62018-09-12 18:04:49 -040053static void compare_pixel(const char* label,
54 SkCanvas* canvas, int x, int y,
55 SkColor4f color, SkColorSpace* cs) {
Hal Canarydf2d27e2019-01-08 09:38:02 -050056 SkPaint paint;
Kevin Lubickbca43ec2023-10-30 10:11:22 -040057 SkFont font = ToolUtils::DefaultPortableFont();
Mike Klein5e31ec62018-09-12 18:04:49 -040058 auto canvas_cs = canvas->imageInfo().refColorSpace();
Mike Klein8f5a7a62018-09-11 12:11:46 -040059
Mike Klein5e31ec62018-09-12 18:04:49 -040060 // I'm not really sure if this makes things easier or harder to follow,
61 // but we sniff the canvas to grab its current y-translate, so that (x,y)
62 // can be written in sort of chunk-relative terms.
63 const SkMatrix& m = canvas->getTotalMatrix();
John Stilesc766dd52021-06-18 12:22:56 -040064 SkASSERT(m.isScaleTranslate());
Mike Klein5e31ec62018-09-12 18:04:49 -040065 SkScalar dy = m.getTranslateY();
66 SkASSERT(dy == (int)dy);
67 y += (int)dy;
Mike Klein8f5a7a62018-09-11 12:11:46 -040068
Mike Kleinfd007642018-09-12 13:03:06 -040069 SkBitmap bm;
Mike Klein5e31ec62018-09-12 18:04:49 -040070 bm.allocPixels(SkImageInfo::Make(1,1, kRGBA_F32_SkColorType, kUnpremul_SkAlphaType, canvas_cs));
71 if (!canvas->readPixels(bm, x,y)) {
Mike Kleinc9eace82018-10-31 10:49:38 -040072 MarkGMGood(canvas, 140,40);
Hal Canarydf2d27e2019-01-08 09:38:02 -050073 canvas->drawString("can't readPixels() on this canvas :(", 100,20, font, paint);
Mike Kleinfd007642018-09-12 13:03:06 -040074 return;
75 }
Mike Klein8f5a7a62018-09-11 12:11:46 -040076
Mike Kleinfd007642018-09-12 13:03:06 -040077 SkColor4f pixel;
Mike Klein5e31ec62018-09-12 18:04:49 -040078 memcpy(&pixel, bm.getAddr(0,0), sizeof(pixel));
Mike Klein8f5a7a62018-09-11 12:11:46 -040079
Mike Klein5e31ec62018-09-12 18:04:49 -040080 SkColor4f expected = transform(color,cs, canvas_cs.get());
Mike Klein21e82882020-07-17 14:01:18 -050081 if (SkColorTypeIsNormalized(canvas->imageInfo().colorType())) {
Mike Kleinfd007642018-09-12 13:03:06 -040082 // We can't expect normalized formats to hold values outside [0,1].
Brian Osmand5ea9982018-10-31 15:59:49 -040083 for (int i = 0; i < 4; ++i) {
84 expected[i] = SkTPin(expected[i], 0.0f, 1.0f);
85 }
Mike Kleinfd007642018-09-12 13:03:06 -040086 }
87 if (canvas->imageInfo().colorType() == kGray_8_SkColorType) {
88 // Drawing into Gray8 is known to be maybe-totally broken.
89 // TODO: update expectation here to be {lum,lum,lum,1} if we fix Gray8.
90 expected = SkColor4f{NAN, NAN, NAN, 1};
91 }
Mike Klein8f5a7a62018-09-11 12:11:46 -040092
Mike Kleinfd007642018-09-12 13:03:06 -040093 if (nearly_equal(pixel, expected)) {
Mike Kleinc9eace82018-10-31 10:49:38 -040094 MarkGMGood(canvas, 140,40);
Mike Klein8f5a7a62018-09-11 12:11:46 -040095 } else {
Mike Kleinc9eace82018-10-31 10:49:38 -040096 MarkGMBad(canvas, 140,40);
Mike Kleinfd007642018-09-12 13:03:06 -040097 }
98
99 struct {
100 const char* label;
101 SkColor4f color;
102 } lines[] = {
Mike Klein5e31ec62018-09-12 18:04:49 -0400103 {"Pixel:" , pixel },
104 {"Expected:", expected},
Mike Kleinfd007642018-09-12 13:03:06 -0400105 };
106
107 SkAutoCanvasRestore saveRestore(canvas, true);
Hal Canarydf2d27e2019-01-08 09:38:02 -0500108 canvas->drawString(label, 80,20, font, paint);
Mike Kleinfd007642018-09-12 13:03:06 -0400109 for (auto l : lines) {
Mike Klein5e31ec62018-09-12 18:04:49 -0400110 canvas->translate(0,20);
Hal Canarydf2d27e2019-01-08 09:38:02 -0500111 canvas->drawString(l.label, 80,20, font, paint);
112 canvas->drawString(fmt(l.color).c_str(), 140,20, font, paint);
Mike Klein5e31ec62018-09-12 18:04:49 -0400113 }
114}
115
Brian Osman3f4602f2018-10-18 16:35:49 -0400116DEF_SIMPLE_GM(p3, canvas, 450, 1300) {
Mike Kleinb147ace2020-01-16 11:11:06 -0600117 auto p3 = SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, SkNamedGamut::kDisplayP3);
Mike Kleineee83152018-10-12 13:53:03 -0400118 auto srgb = SkColorSpace::MakeSRGB();
119
120 auto p3_to_srgb = [&](SkColor4f c) {
121 SkPaint p;
122 p.setColor4f(c, p3.get());
123 return p.getColor4f();
124 };
Mike Klein5e31ec62018-09-12 18:04:49 -0400125
126 // Draw a P3 red rectangle and check the corner.
127 {
128 SkPaint paint;
129 paint.setColor4f({1,0,0,1}, p3.get());
Mike Klein5e31ec62018-09-12 18:04:49 -0400130
Mike Klein62bd12f2018-09-17 12:39:09 -0400131 canvas->drawRect({10,10,70,70}, paint);
Mike Klein5e31ec62018-09-12 18:04:49 -0400132 compare_pixel("drawRect P3 red ",
133 canvas, 10,10,
134 {1,0,0,1}, p3.get());
135 }
136
137 canvas->translate(0,80);
138
Mike Klein62bd12f2018-09-17 12:39:09 -0400139 // Draw a P3 red bitmap, using a draw.
140 {
141 SkBitmap bm;
142 bm.allocPixels(SkImageInfo::Make(60,60, kRGBA_F16_SkColorType, kPremul_SkAlphaType, p3));
143
144 SkPaint paint;
145 paint.setColor4f({1,0,0,1}, p3.get());
146 SkCanvas{bm}.drawPaint(paint);
147
Mike Reed607a3822021-01-24 19:49:21 -0500148 canvas->drawImage(bm.asImage(), 10,10);
Mike Klein62bd12f2018-09-17 12:39:09 -0400149 compare_pixel("drawBitmap P3 red, from drawPaint",
150 canvas, 10,10,
151 {1,0,0,1}, p3.get());
152 }
153
154 canvas->translate(0,80);
155
Mike Klein4dc70ec2019-03-01 09:06:22 -0600156 // TODO(mtklein): sample and check the middle points of these gradients too.
157
Mike Kleineee83152018-10-12 13:53:03 -0400158 // Draw a gradient from P3 red to P3 green interpolating in unpremul P3, checking the corners.
Mike Klein5e31ec62018-09-12 18:04:49 -0400159 {
160
161 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
162 SkColor4f colors[] = {{1,0,0,1}, {0,1,0,1}};
163
164 SkPaint paint;
165 paint.setShader(SkGradientShader::MakeLinear(points, colors, p3,
Herb Derbyc37b3862022-06-21 09:49:17 -0400166 nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -0400167 SkTileMode::kClamp));
Mike Klein5e31ec62018-09-12 18:04:49 -0400168 canvas->drawRect({10,10,70,70}, paint);
Mike Klein5e31ec62018-09-12 18:04:49 -0400169 canvas->save();
Mike Kleineee83152018-10-12 13:53:03 -0400170 compare_pixel("UPM P3 gradient, P3 red",
Mike Klein5e31ec62018-09-12 18:04:49 -0400171 canvas, 10,10,
172 {1,0,0,1}, p3.get());
173
174 canvas->translate(180, 0);
175
Mike Kleineee83152018-10-12 13:53:03 -0400176 compare_pixel("UPM P3 gradient, P3 green",
Mike Klein5e31ec62018-09-12 18:04:49 -0400177 canvas, 69,69,
178 {0,1,0,1}, p3.get());
179 canvas->restore();
Mike Klein8f5a7a62018-09-11 12:11:46 -0400180 }
181
Mike Klein9fb5a532018-09-13 15:23:38 -0400182 canvas->translate(0,80);
183
Mike Kleineee83152018-10-12 13:53:03 -0400184 // Draw a gradient from P3 red to P3 green interpolating in premul P3, checking the corners.
Mike Klein983886e2018-09-20 14:10:33 -0400185 {
186
187 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
188 SkColor4f colors[] = {{1,0,0,1}, {0,1,0,1}};
189
190 SkPaint paint;
191 paint.setShader(
192 SkGradientShader::MakeLinear(points, colors, p3,
Herb Derbyc37b3862022-06-21 09:49:17 -0400193 nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -0400194 SkTileMode::kClamp,
Mike Klein983886e2018-09-20 14:10:33 -0400195 SkGradientShader::kInterpolateColorsInPremul_Flag,
196 nullptr/*local matrix*/));
197 canvas->drawRect({10,10,70,70}, paint);
198 canvas->save();
Mike Kleineee83152018-10-12 13:53:03 -0400199 compare_pixel("PM P3 gradient, P3 red",
Mike Klein983886e2018-09-20 14:10:33 -0400200 canvas, 10,10,
201 {1,0,0,1}, p3.get());
202
203 canvas->translate(180, 0);
204
Mike Kleineee83152018-10-12 13:53:03 -0400205 compare_pixel("PM P3 gradient, P3 green",
206 canvas, 69,69,
207 {0,1,0,1}, p3.get());
208 canvas->restore();
209 }
210
211 canvas->translate(0,80);
212
213 // Draw a gradient from P3 red to P3 green interpolating in unpremul sRGB, checking the corners.
214 {
215
216 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
217 SkColor4f colors[] = {p3_to_srgb({1,0,0,1}), p3_to_srgb({0,1,0,1})};
218
219 SkPaint paint;
220 paint.setShader(SkGradientShader::MakeLinear(points, colors, srgb,
Herb Derbyc37b3862022-06-21 09:49:17 -0400221 nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -0400222 SkTileMode::kClamp));
Mike Kleineee83152018-10-12 13:53:03 -0400223 canvas->drawRect({10,10,70,70}, paint);
224 canvas->save();
225 compare_pixel("UPM sRGB gradient, P3 red",
226 canvas, 10,10,
227 {1,0,0,1}, p3.get());
228
229 canvas->translate(180, 0);
230
231 compare_pixel("UPM sRGB gradient, P3 green",
232 canvas, 69,69,
233 {0,1,0,1}, p3.get());
234 canvas->restore();
235 }
236
237 canvas->translate(0,80);
238
239 // Draw a gradient from P3 red to P3 green interpolating in premul sRGB, checking the corners.
240 {
241
242 SkPoint points[] = {{10.5,10.5}, {69.5,69.5}};
243 SkColor4f colors[] = {p3_to_srgb({1,0,0,1}), p3_to_srgb({0,1,0,1})};
244
245 SkPaint paint;
246 paint.setShader(
247 SkGradientShader::MakeLinear(points, colors, srgb,
Herb Derbyc37b3862022-06-21 09:49:17 -0400248 nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -0400249 SkTileMode::kClamp,
Mike Kleineee83152018-10-12 13:53:03 -0400250 SkGradientShader::kInterpolateColorsInPremul_Flag,
251 nullptr/*local matrix*/));
252 canvas->drawRect({10,10,70,70}, paint);
253 canvas->save();
Mike Klein4dc70ec2019-03-01 09:06:22 -0600254 compare_pixel("PM sRGB gradient, P3 red",
Mike Kleineee83152018-10-12 13:53:03 -0400255 canvas, 10,10,
256 {1,0,0,1}, p3.get());
257
258 canvas->translate(180, 0);
259
Mike Klein4dc70ec2019-03-01 09:06:22 -0600260 compare_pixel("PM sRGB gradient, P3 green",
Mike Klein983886e2018-09-20 14:10:33 -0400261 canvas, 69,69,
262 {0,1,0,1}, p3.get());
263 canvas->restore();
264 }
265
266 canvas->translate(0,80);
267
Mike Kleinb0243b22019-01-30 11:43:47 -0500268 // Leon's blue -> green -> red gradient, interpolating in premul.
269 {
270 SkPoint points[] = {{10.5,10.5}, {10.5,69.5}};
271 SkColor4f colors[] = { {0,0,1,1}, {0,1,0,1}, {1,0,0,1} };
272
273 SkPaint paint;
274 paint.setShader(
275 SkGradientShader::MakeLinear(points, colors, p3,
Herb Derbyc37b3862022-06-21 09:49:17 -0400276 nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -0400277 SkTileMode::kClamp,
Mike Kleinb0243b22019-01-30 11:43:47 -0500278 SkGradientShader::kInterpolateColorsInPremul_Flag,
279 nullptr/*local matrix*/));
280 canvas->drawRect({10,10,70,70}, paint);
281 canvas->save();
282 compare_pixel("Leon's gradient, P3 blue",
283 canvas, 10,10,
284 {0,0,1,1}, p3.get());
285
286 canvas->translate(180, 0);
287
288 compare_pixel("Leon's gradient, P3 red",
289 canvas, 10,69,
290 {1,0,0,1}, p3.get());
291 canvas->restore();
292 }
293
294 canvas->translate(0,80);
295
Mike Kleinbe569492018-09-14 09:34:21 -0400296 // Draw an A8 image with a P3 red, scaled and not, as a shader or bitmap.
Mike Klein9fb5a532018-09-13 15:23:38 -0400297 {
298 uint8_t mask[256];
299 for (int i = 0; i < 256; i++) {
300 mask[i] = 255-i;
301 }
Mike Reed607a3822021-01-24 19:49:21 -0500302
Mike Klein9fb5a532018-09-13 15:23:38 -0400303 SkBitmap bm;
304 bm.installPixels(SkImageInfo::MakeA8(16,16), mask, 16);
305
Mike Kleinbe569492018-09-14 09:34:21 -0400306 SkPaint as_bitmap;
307 as_bitmap.setColor4f({1,0,0,1}, p3.get());
Mike Reed607a3822021-01-24 19:49:21 -0500308 SkSamplingOptions sampling(SkFilterMode::kLinear);
Mike Klein9fb5a532018-09-13 15:23:38 -0400309
Mike Kleinbe569492018-09-14 09:34:21 -0400310 SkPaint as_shader;
311 as_shader.setColor4f({1,0,0,1}, p3.get());
Mike Reed607a3822021-01-24 19:49:21 -0500312 as_shader.setShader(bm.makeShader(sampling));
Mike Kleinbe569492018-09-14 09:34:21 -0400313
Mike Reed607a3822021-01-24 19:49:21 -0500314 canvas->drawImage(bm.asImage(), 10,10, sampling, &as_bitmap);
Mike Kleinbe569492018-09-14 09:34:21 -0400315 compare_pixel("A8 sprite bitmap P3 red",
316 canvas, 10,10,
317 {1,0,0,1}, p3.get());
318
319 canvas->translate(0, 80);
320
321 canvas->save();
322 canvas->translate(10,10);
323 canvas->drawRect({0,0,16,16}, as_shader);
324 canvas->restore();
325 compare_pixel("A8 sprite shader P3 red",
Mike Klein9fb5a532018-09-13 15:23:38 -0400326 canvas, 10,10,
327 {1,0,0,1}, p3.get());
328
329 canvas->translate(0,80);
330
Mike Reed607a3822021-01-24 19:49:21 -0500331 canvas->drawImageRect(bm.asImage(), {10,10,70,70}, sampling, &as_bitmap);
Mike Kleinbe569492018-09-14 09:34:21 -0400332 compare_pixel("A8 scaled bitmap P3 red",
333 canvas, 10,10,
334 {1,0,0,1}, p3.get());
335
336 canvas->translate(0,80);
337
338 canvas->save();
339 canvas->translate(10,10);
340 canvas->scale(3.75,3.75);
341 canvas->drawRect({0,0,16,16}, as_shader);
342 canvas->restore();
343 compare_pixel("A8 scaled shader P3 red",
Mike Klein9fb5a532018-09-13 15:23:38 -0400344 canvas, 10,10,
345 {1,0,0,1}, p3.get());
346 }
347
Mike Klein8f5a7a62018-09-11 12:11:46 -0400348 // TODO: draw P3 colors more ways
349}
Brian Osmane3caf2d2018-11-21 13:48:36 -0500350
351DEF_SIMPLE_GM(p3_ovals, canvas, 450, 320) {
Mike Kleinb147ace2020-01-16 11:11:06 -0600352 auto p3 = SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, SkNamedGamut::kDisplayP3);
Brian Osmane3caf2d2018-11-21 13:48:36 -0500353
354 // Test cases that exercise each Op in GrOvalOpFactory.cpp
355
356 // Draw a circle and check the center (CircleOp)
357 {
358 SkPaint paint;
359 paint.setAntiAlias(true);
360 paint.setColor4f({ 1,0,0,1 }, p3.get());
361
362 canvas->drawCircle(40, 40, 30, paint);
363 compare_pixel("drawCircle P3 red ",
364 canvas, 40, 40,
365 { 1,0,0,1 }, p3.get());
366 }
367
368 canvas->translate(0, 80);
369
370 // Draw an oval and check the center (EllipseOp)
371 {
372 SkPaint paint;
373 paint.setAntiAlias(true);
374 paint.setColor4f({ 1,0,0,1 }, p3.get());
375
376 canvas->drawOval({ 20,10,60,70 }, paint);
377 compare_pixel("drawOval P3 red ",
378 canvas, 40, 40,
379 { 1,0,0,1 }, p3.get());
380 }
381
382 canvas->translate(0, 80);
383
384 // Draw a butt-capped dashed circle and check the top of the stroke (ButtCappedDashedCircleOp)
385 {
386 SkPaint paint;
387 paint.setAntiAlias(true);
388 paint.setColor4f({ 1,0,0,1 }, p3.get());
389 paint.setStyle(SkPaint::kStroke_Style);
390 float intervals[] = { 70, 10 };
391 paint.setPathEffect(SkDashPathEffect::Make(intervals, 2, 0));
392 paint.setStrokeWidth(10);
393
394 canvas->drawCircle(40, 40, 30, paint);
395 compare_pixel("drawDashedCircle P3 red ",
Brian Osman975197c2018-11-21 15:47:38 -0500396 canvas, 40, 10,
Brian Osmane3caf2d2018-11-21 13:48:36 -0500397 { 1,0,0,1 }, p3.get());
398 }
399
400 canvas->translate(0, 80);
401
402 // Draw an oval with rotation and check the center (DIEllipseOp)
403 {
404 SkPaint paint;
405 paint.setAntiAlias(true);
406 paint.setColor4f({ 1,0,0,1 }, p3.get());
407
408 canvas->save();
409 canvas->translate(40, 40);
410 canvas->rotate(45);
411 canvas->drawOval({ -20,-30,20,30 }, paint);
412 canvas->restore();
413 compare_pixel("drawRotatedOval P3 red ",
414 canvas, 40, 40,
415 { 1,0,0,1 }, p3.get());
416 }
417
418 canvas->translate(0, 80);
419}