1 /*
2 * Copyright 2011 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/gm.h"
9 #include "include/core/SkCanvas.h"
10 #include "include/core/SkClipOp.h"
11 #include "include/core/SkColor.h"
12 #include "include/core/SkColorFilter.h"
13 #include "include/core/SkFont.h"
14 #include "include/core/SkFontTypes.h"
15 #include "include/core/SkPaint.h"
16 #include "include/core/SkPathBuilder.h"
17 #include "include/core/SkRRect.h"
18 #include "include/core/SkRect.h"
19 #include "include/core/SkScalar.h"
20 #include "include/core/SkSize.h"
21 #include "include/core/SkString.h"
22 #include "include/core/SkTypeface.h"
23 #include "include/core/SkTypes.h"
24 #include "include/effects/SkGradientShader.h"
25 #include "tools/DecodeUtils.h"
26 #include "tools/Resources.h"
27 #include "tools/ToolUtils.h"
28 #include "tools/fonts/FontToolUtils.h"
29
30 #include <string.h>
31
32 namespace skiagm {
33
34 constexpr SkColor gPathColor = SK_ColorBLACK;
35 constexpr SkColor gClipAColor = SK_ColorBLUE;
36 constexpr SkColor gClipBColor = SK_ColorRED;
37
38 class ComplexClipGM : public GM {
39 public:
ComplexClipGM(bool aaclip,bool saveLayer,bool invertDraw)40 ComplexClipGM(bool aaclip, bool saveLayer, bool invertDraw)
41 : fDoAAClip(aaclip)
42 , fDoSaveLayer(saveLayer)
43 , fInvertDraw(invertDraw) {
44 this->setBGColor(0xFFDEDFDE);
45 }
46
47 protected:
getName() const48 SkString getName() const override {
49 SkString str;
50 str.printf("complexclip_%s%s%s",
51 fDoAAClip ? "aa" : "bw",
52 fDoSaveLayer ? "_layer" : "",
53 fInvertDraw ? "_invert" : "");
54 return str;
55 }
56
getISize()57 SkISize getISize() override { return SkISize::Make(388, 780); }
58
onDraw(SkCanvas * canvas)59 void onDraw(SkCanvas* canvas) override {
60 SkPath path = SkPathBuilder()
61 .moveTo(0, 50)
62 .quadTo(0, 0, 50, 0)
63 .lineTo(175, 0)
64 .quadTo(200, 0, 200, 25)
65 .lineTo(200, 150)
66 .quadTo(200, 200, 150, 200)
67 .lineTo(0, 200)
68 .close()
69 .moveTo(50, 50)
70 .lineTo(150, 50)
71 .lineTo(150, 125)
72 .quadTo(150, 150, 125, 150)
73 .lineTo(50, 150)
74 .close()
75 .detach();
76 if (fInvertDraw) {
77 path.setFillType(SkPathFillType::kInverseEvenOdd);
78 } else {
79 path.setFillType(SkPathFillType::kEvenOdd);
80 }
81 SkPaint pathPaint;
82 pathPaint.setAntiAlias(true);
83 pathPaint.setColor(gPathColor);
84
85 SkPath clipA = SkPath::Polygon({{10, 20}, {165, 22}, {70, 105}, {165, 177}, {-5, 180}}, true);
86
87 SkPath clipB = SkPath::Polygon({{40, 10}, {190, 15}, {195, 190}, {40, 185}, {155, 100}}, true);
88
89 SkFont font(ToolUtils::DefaultPortableTypeface(), 20);
90
91 constexpr struct {
92 SkClipOp fOp;
93 const char* fName;
94 } gOps[] = { //extra spaces in names for measureText
95 {SkClipOp::kIntersect, "Isect "},
96 {SkClipOp::kDifference, "Diff " },
97 };
98
99 canvas->translate(20, 20);
100 canvas->scale(3 * SK_Scalar1 / 4, 3 * SK_Scalar1 / 4);
101
102 if (fDoSaveLayer) {
103 // We want the layer to appear symmetric relative to actual
104 // device boundaries so we need to "undo" the effect of the
105 // scale and translate
106 SkRect bounds = SkRect::MakeLTRB(
107 4.0f/3.0f * -20,
108 4.0f/3.0f * -20,
109 4.0f/3.0f * (this->getISize().fWidth - 20),
110 4.0f/3.0f * (this->getISize().fHeight - 20));
111
112 bounds.inset(100, 100);
113 SkPaint boundPaint;
114 boundPaint.setColor(SK_ColorRED);
115 boundPaint.setStyle(SkPaint::kStroke_Style);
116 canvas->drawRect(bounds, boundPaint);
117 canvas->clipRect(bounds);
118 canvas->saveLayer(&bounds, nullptr);
119 }
120
121 for (int invBits = 0; invBits < 4; ++invBits) {
122 canvas->save();
123 for (size_t op = 0; op < std::size(gOps); ++op) {
124 this->drawHairlines(canvas, path, clipA, clipB);
125
126 bool doInvA = SkToBool(invBits & 1);
127 bool doInvB = SkToBool(invBits & 2);
128 canvas->save();
129 // set clip
130 clipA.setFillType(doInvA ? SkPathFillType::kInverseEvenOdd :
131 SkPathFillType::kEvenOdd);
132 clipB.setFillType(doInvB ? SkPathFillType::kInverseEvenOdd :
133 SkPathFillType::kEvenOdd);
134 canvas->clipPath(clipA, fDoAAClip);
135 canvas->clipPath(clipB, gOps[op].fOp, fDoAAClip);
136
137 // In the inverse case we need to prevent the draw from covering the whole
138 // canvas.
139 if (fInvertDraw) {
140 SkRect rectClip = clipA.getBounds();
141 rectClip.join(path.getBounds());
142 rectClip.join(path.getBounds());
143 rectClip.outset(5, 5);
144 canvas->clipRect(rectClip);
145 }
146
147 // draw path clipped
148 canvas->drawPath(path, pathPaint);
149 canvas->restore();
150
151
152 SkPaint paint;
153 SkScalar txtX = 45;
154 paint.setColor(gClipAColor);
155 const char* aTxt = doInvA ? "InvA " : "A ";
156 canvas->drawSimpleText(aTxt, strlen(aTxt), SkTextEncoding::kUTF8, txtX, 220, font, paint);
157 txtX += font.measureText(aTxt, strlen(aTxt), SkTextEncoding::kUTF8);
158 paint.setColor(SK_ColorBLACK);
159 canvas->drawSimpleText(gOps[op].fName, strlen(gOps[op].fName), SkTextEncoding::kUTF8, txtX, 220,
160 font, paint);
161 txtX += font.measureText(gOps[op].fName, strlen(gOps[op].fName), SkTextEncoding::kUTF8);
162 paint.setColor(gClipBColor);
163 const char* bTxt = doInvB ? "InvB " : "B ";
164 canvas->drawSimpleText(bTxt, strlen(bTxt), SkTextEncoding::kUTF8, txtX, 220, font, paint);
165
166 canvas->translate(250,0);
167 }
168 canvas->restore();
169 canvas->translate(0, 250);
170 }
171
172 if (fDoSaveLayer) {
173 canvas->restore();
174 }
175 }
176 private:
drawHairlines(SkCanvas * canvas,const SkPath & path,const SkPath & clipA,const SkPath & clipB)177 void drawHairlines(SkCanvas* canvas, const SkPath& path,
178 const SkPath& clipA, const SkPath& clipB) {
179 SkPaint paint;
180 paint.setAntiAlias(true);
181 paint.setStyle(SkPaint::kStroke_Style);
182 const SkAlpha fade = 0x33;
183
184 // draw path in hairline
185 paint.setColor(gPathColor); paint.setAlpha(fade);
186 canvas->drawPath(path, paint);
187
188 // draw clips in hair line
189 paint.setColor(gClipAColor); paint.setAlpha(fade);
190 canvas->drawPath(clipA, paint);
191 paint.setColor(gClipBColor); paint.setAlpha(fade);
192 canvas->drawPath(clipB, paint);
193 }
194
195 bool fDoAAClip;
196 bool fDoSaveLayer;
197 bool fInvertDraw;
198
199 using INHERITED = GM;
200 };
201
202 //////////////////////////////////////////////////////////////////////////////
203
204 DEF_GM(return new ComplexClipGM(false, false, false);)
205 DEF_GM(return new ComplexClipGM(false, false, true);)
206 DEF_GM(return new ComplexClipGM(false, true, false);)
207 DEF_GM(return new ComplexClipGM(false, true, true);)
208 DEF_GM(return new ComplexClipGM(true, false, false);)
209 DEF_GM(return new ComplexClipGM(true, false, true);)
210 DEF_GM(return new ComplexClipGM(true, true, false);)
211 DEF_GM(return new ComplexClipGM(true, true, true);)
212 } // namespace skiagm
213
214 DEF_SIMPLE_GM(clip_shader, canvas, 840, 650) {
215 auto img = ToolUtils::GetResourceAsImage("images/yellow_rose.png");
216 auto sh = img->makeShader(SkSamplingOptions());
217
218 SkRect r = SkRect::MakeIWH(img->width(), img->height());
219 SkPaint p;
220
221 canvas->translate(10, 10);
222 canvas->drawImage(img, 0, 0);
223
224 canvas->save();
225 canvas->translate(img->width() + 10, 0);
226 canvas->clipShader(sh, SkClipOp::kIntersect);
227 p.setColor(SK_ColorRED);
228 canvas->drawRect(r, p);
229 canvas->restore();
230
231 canvas->save();
232 canvas->translate(0, img->height() + 10);
233 canvas->clipShader(sh, SkClipOp::kDifference);
234 p.setColor(SK_ColorGREEN);
235 canvas->drawRect(r, p);
236 canvas->restore();
237
238 canvas->save();
239 canvas->translate(img->width() + 10, img->height() + 10);
240 canvas->clipShader(sh, SkClipOp::kIntersect);
241 canvas->save();
242 SkMatrix lm = SkMatrix::Scale(1.0f/5, 1.0f/5);
243 canvas->clipShader(img->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
244 SkSamplingOptions(), lm));
245 canvas->drawImage(img, 0, 0);
246
247 canvas->restore();
248 canvas->restore();
249 }
250
251 DEF_SIMPLE_GM(clip_shader_layer, canvas, 430, 320) {
252 auto img = ToolUtils::GetResourceAsImage("images/yellow_rose.png");
253 auto sh = img->makeShader(SkSamplingOptions());
254
255 SkRect r = SkRect::MakeIWH(img->width(), img->height());
256
257 canvas->translate(10, 10);
258 // now add the cool clip
259 canvas->clipRect(r);
260 canvas->clipShader(sh);
261 // now draw a layer with the same image, and watch it get restored w/ the clip
262 canvas->saveLayer(&r, nullptr);
263 canvas->drawColor(0xFFFF0000);
264 canvas->restore();
265 }
266
267 DEF_SIMPLE_GM(clip_shader_nested, canvas, 256, 256) {
268 float w = 64.f;
269 float h = 64.f;
270
271 const SkColor gradColors[] = {SK_ColorBLACK, SkColorSetARGB(128, 128, 128, 128)};
272 auto s = SkGradientShader::MakeRadial({0.5f * w, 0.5f * h}, 0.1f * w, gradColors, nullptr,
273 2, SkTileMode::kRepeat, 0, nullptr);
274
275 SkPaint p;
276
277 // A large black rect affected by two gradient clips
278 canvas->save();
279 canvas->clipShader(s);
280 canvas->scale(2.f, 2.f);
281 canvas->clipShader(s);
282 canvas->drawRect(SkRect::MakeWH(w, h), p);
283 canvas->restore();
284
285 canvas->translate(0.f, 2.f * h);
286
287 // A small red rect, with no clipping
288 canvas->save();
289 p.setColor(SK_ColorRED);
290 canvas->drawRect(SkRect::MakeWH(w, h), p);
291 canvas->restore();
292
293 canvas->translate(2.f * w, -2.f * h);
294
295 // A small green rect, with clip shader and rrect clipping
296 canvas->save();
297 canvas->clipShader(s);
298 canvas->clipRRect(SkRRect::MakeRectXY(SkRect::MakeWH(w, h), 10, 10), /*doAntiAlias=*/true);
299 p.setColor(SK_ColorGREEN);
300 canvas->drawRect(SkRect::MakeWH(w, h), p);
301 canvas->restore();
302
303 canvas->translate(0.f, 2.f * h);
304
305 // A small blue rect, with clip shader and path clipping
306 SkPath starPath;
307 starPath.moveTo(0.0f, -33.3333f);
308 starPath.lineTo(9.62f, -16.6667f);
309 starPath.lineTo(28.867f, -16.6667f);
310 starPath.lineTo(19.24f, 0.0f);
311 starPath.lineTo(28.867f, 16.6667f);
312 starPath.lineTo(9.62f, 16.6667f);
313 starPath.lineTo(0.0f, 33.3333f);
314 starPath.lineTo(-9.62f, 16.6667f);
315 starPath.lineTo(-28.867f, 16.6667f);
316 starPath.lineTo(-19.24f, 0.0f);
317 starPath.lineTo(-28.867f, -16.6667f);
318 starPath.lineTo(-9.62f, -16.6667f);
319 starPath.close();
320
321 canvas->save();
322 canvas->clipShader(s);
323 canvas->translate(w/2, h/2);
324 canvas->clipPath(starPath);
325 p.setColor(SK_ColorBLUE);
326 canvas->translate(-w/2, -h/2);
327 canvas->drawRect(SkRect::MakeWH(w, h), p);
328 canvas->restore();
329 }
330
331 namespace {
332
333 // Where is canvas->concat(persp) called relative to the clipShader calls.
334 enum ConcatPerspective {
335 kConcatBeforeClips,
336 kConcatAfterClips,
337 kConcatBetweenClips
338 };
339 // Order in which clipShader(image) and clipShader(gradient) are specified; only meaningful
340 // when CanvasPerspective is kConcatBetweenClips.
341 enum ClipOrder {
342 kClipImageFirst,
343 kClipGradientFirst,
344
345 kDoesntMatter = kClipImageFirst
346 };
347 // Which shaders have perspective applied as a local matrix.
348 enum LocalMatrix {
349 kNoLocalMat,
350 kImageWithLocalMat,
351 kGradientWithLocalMat,
352 kBothWithLocalMat
353 };
354 struct Config {
355 ConcatPerspective fConcat;
356 ClipOrder fOrder;
357 LocalMatrix fLM;
358 };
359
draw_banner(SkCanvas * canvas,Config config)360 static void draw_banner(SkCanvas* canvas, Config config) {
361 SkString banner;
362 banner.append("Persp: ");
363
364 if (config.fConcat == kConcatBeforeClips || config.fLM == kBothWithLocalMat) {
365 banner.append("Both Clips");
366 } else {
367 SkASSERT((config.fConcat == kConcatBetweenClips && config.fLM == kNoLocalMat) ||
368 (config.fConcat == kConcatAfterClips && (config.fLM == kImageWithLocalMat ||
369 config.fLM == kGradientWithLocalMat)));
370 if ((config.fConcat == kConcatBetweenClips && config.fOrder == kClipImageFirst) ||
371 config.fLM == kGradientWithLocalMat) {
372 banner.append("Gradient");
373 } else {
374 SkASSERT(config.fOrder == kClipGradientFirst || config.fLM == kImageWithLocalMat);
375 banner.append("Image");
376 }
377 }
378 if (config.fLM != kNoLocalMat) {
379 banner.append(" (w/ LM, should equal top row)");
380 }
381
382 static const SkFont kFont(ToolUtils::DefaultPortableTypeface(), 12);
383 canvas->drawString(banner.c_str(), 20.f, -30.f, kFont, SkPaint());
384 };
385
386 } // namespace
387
388 DEF_SIMPLE_GM(clip_shader_persp, canvas, 1370, 1030) {
389 // Each draw has a clipShader(image-shader), a clipShader(gradient-shader), a concat(persp-mat),
390 // and each shader may or may not be wrapped with a perspective local matrix.
391
392 // Pairs of configs that should match in appearance where first config doesn't use a local
393 // matrix (top row of GM) and the second does (bottom row of GM).
394 Config matches[][2] = {
395 // Everything has perspective
396 {{kConcatBeforeClips, kDoesntMatter, kNoLocalMat},
397 {kConcatAfterClips, kDoesntMatter, kBothWithLocalMat}},
398 // Image shader has perspective
399 {{kConcatBetweenClips, kClipGradientFirst, kNoLocalMat},
400 {kConcatAfterClips, kDoesntMatter, kImageWithLocalMat}},
401 // Gradient shader has perspective
402 {{kConcatBetweenClips, kClipImageFirst, kNoLocalMat},
403 {kConcatAfterClips, kDoesntMatter, kGradientWithLocalMat}}
404 };
405
406 // The image that is drawn
407 auto img = ToolUtils::GetResourceAsImage("images/yellow_rose.png");
408 // Scale factor always applied to the image shader so that it tiles
409 SkMatrix scale = SkMatrix::Scale(1.f / 4.f, 1.f / 4.f);
410 // The perspective matrix applied wherever needed
411 SkPoint src[4];
412 SkRect::Make(img->dimensions()).toQuad(src);
413 SkPoint dst[4] = {{0, 80.f},
414 {img->width() + 28.f, -100.f},
415 {img->width() - 28.f, img->height() + 100.f},
416 {0.f, img->height() - 80.f}};
417 SkMatrix persp;
418 SkAssertResult(persp.setPolyToPoly(src, dst, 4));
419
420 SkMatrix perspScale = SkMatrix::Concat(persp, scale);
421
__anon4478d7920302(Config config) 422 auto drawConfig = [&](Config config) {
423 canvas->save();
424
425 draw_banner(canvas, config);
426
427 // Make clipShaders (possibly with local matrices)
428 bool gradLM = config.fLM == kGradientWithLocalMat || config.fLM == kBothWithLocalMat;
429 const SkColor gradColors[] = {SK_ColorBLACK, SkColorSetARGB(128, 128, 128, 128)};
430 auto gradShader = SkGradientShader::MakeRadial({0.5f * img->width(), 0.5f * img->height()},
431 0.1f * img->width(), gradColors, nullptr, 2,
432 SkTileMode::kRepeat, 0,
433 gradLM ? &persp : nullptr);
434 bool imageLM = config.fLM == kImageWithLocalMat || config.fLM == kBothWithLocalMat;
435 auto imgShader = img->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
436 SkSamplingOptions(), imageLM ? perspScale : scale);
437
438 // Perspective before any clipShader
439 if (config.fConcat == kConcatBeforeClips) {
440 canvas->concat(persp);
441 }
442
443 // First clipshader
444 canvas->clipShader(config.fOrder == kClipImageFirst ? imgShader : gradShader);
445
446 // Perspective between clipShader
447 if (config.fConcat == kConcatBetweenClips) {
448 canvas->concat(persp);
449 }
450
451 // Second clipShader
452 canvas->clipShader(config.fOrder == kClipImageFirst ? gradShader : imgShader);
453
454 // Perspective after clipShader
455 if (config.fConcat == kConcatAfterClips) {
456 canvas->concat(persp);
457 }
458
459 // Actual draw and clip boundary are the same for all configs
460 canvas->clipIRect(img->bounds());
461 canvas->clear(SK_ColorBLACK);
462 canvas->drawImage(img, 0, 0);
463
464 canvas->restore();
465 };
466
467 SkIRect grid = persp.mapRect(SkRect::Make(img->dimensions())).roundOut();
468 grid.fLeft -= 20; // manual adjust to look nicer
469
470 canvas->translate(10.f, 10.f);
471
472 for (size_t i = 0; i < std::size(matches); ++i) {
473 canvas->save();
474 canvas->translate(-grid.fLeft, -grid.fTop);
475 drawConfig(matches[i][0]);
476 canvas->translate(0.f, grid.height());
477 drawConfig(matches[i][1]);
478 canvas->restore();
479
480 canvas->translate(grid.width(), 0.f);
481 }
482 }
483
484 DEF_SIMPLE_GM(clip_shader_difference, canvas, 512, 512) {
485 auto image = ToolUtils::GetResourceAsImage("images/yellow_rose.png");
486 canvas->clear(SK_ColorGRAY);
487
488 SkRect rect = SkRect::MakeWH(256, 256);
489 SkMatrix local = SkMatrix::RectToRect(SkRect::MakeWH(image->width(), image->height()),
490 SkRect::MakeWH(64, 64));
491 auto shader = image->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
492 SkSamplingOptions(), &local);
493
494 SkPaint paint;
495 paint.setColor(SK_ColorRED);
496 paint.setAntiAlias(true);
497
498 // TL: A rectangle
499 {
500 canvas->save();
501 canvas->translate(0, 0);
502 canvas->clipShader(shader, SkClipOp::kDifference);
503 canvas->drawRect(rect, paint);
504 canvas->restore();
505 }
506 // TR: A round rectangle
507 {
508 canvas->save();
509 canvas->translate(256, 0);
510 canvas->clipShader(shader, SkClipOp::kDifference);
511 canvas->drawRRect(SkRRect::MakeRectXY(rect, 64.f, 64.f), paint);
512 canvas->restore();
513 }
514 // BL: A path
515 {
516 canvas->save();
517 canvas->translate(0, 256);
518 canvas->clipShader(shader, SkClipOp::kDifference);
519
520 SkPath path;
521 path.moveTo(0.f, 128.f);
522 path.lineTo(128.f, 256.f);
523 path.lineTo(256.f, 128.f);
524 path.lineTo(128.f, 0.f);
525
526 SkScalar d = 64.f * SK_ScalarSqrt2;
527 path.moveTo(128.f - d, 128.f - d);
528 path.lineTo(128.f - d, 128.f + d);
529 path.lineTo(128.f + d, 128.f + d);
530 path.lineTo(128.f + d, 128.f - d);
531 canvas->drawPath(path, paint);
532 canvas->restore();
533 }
534 // BR: Text
535 {
536 canvas->save();
537 canvas->translate(256, 256);
538 canvas->clipShader(shader, SkClipOp::kDifference);
539 SkFont font = SkFont(ToolUtils::DefaultPortableTypeface(), 64.f);
540 for (int y = 0; y < 4; ++y) {
541 canvas->drawString("Hello", 32.f, y * 64.f, font, paint);
542 }
543 canvas->restore();
544 }
545 }
546