blob: 56fd4b5e6bb48ad0eb49be4565f4153306d7f74d [file] [log] [blame]
scroggo@google.com32f20eb2012-06-29 17:28:50 +00001/*
2 * Copyright 2012 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/SkBlendMode.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040012#include "include/core/SkColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "include/core/SkColorPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05: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/SkPoint.h"
19#include "include/core/SkRect.h"
20#include "include/core/SkRefCnt.h"
21#include "include/core/SkScalar.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "include/core/SkShader.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040023#include "include/core/SkSize.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050024#include "include/core/SkString.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040025#include "include/core/SkTileMode.h"
26#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "include/effects/SkGradientShader.h"
Kevin Lubickdc6cc022023-01-13 11:24:27 -050028#include "include/private/base/SkTDArray.h"
Kevin Lubick1b3aa8b2023-01-19 14:03:31 -050029#include "src/base/SkTLazy.h"
Kevin Lubick9b028372023-10-05 15:04:54 -040030#include "tools/GpuToolUtils.h"
Robert Phillipseae18482022-05-03 09:25:38 -040031#include "tools/ToolUtils.h"
scroggo@google.com32f20eb2012-06-29 17:28:50 +000032
Ben Wagner7fde8e12019-05-01 17:28:53 -040033#include <utility>
34
Mike Reed7d954ad2016-10-28 15:42:34 -040035static sk_sp<SkShader> make_shader(SkBlendMode mode) {
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000036 SkPoint pts[2];
37 SkColor colors[2];
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000038
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000039 pts[0].set(0, 0);
40 pts[1].set(SkIntToScalar(100), 0);
41 colors[0] = SK_ColorRED;
42 colors[1] = SK_ColorBLUE;
Mike Reedfae8fce2019-04-03 10:27:45 -040043 auto shaderA = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000044
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000045 pts[0].set(0, 0);
46 pts[1].set(0, SkIntToScalar(100));
47 colors[0] = SK_ColorBLACK;
48 colors[1] = SkColorSetARGB(0x80, 0, 0, 0);
Mike Reedfae8fce2019-04-03 10:27:45 -040049 auto shaderB = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000050
Mike Reedc8bea7d2019-04-09 13:55:36 -040051 return SkShaders::Blend(mode, std::move(shaderA), std::move(shaderB));
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000052}
53
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000054class ComposeShaderGM : public skiagm::GM {
John Stilese6a05a42020-11-05 10:05:19 -050055protected:
56 void onOnceBeforeDraw() override {
Mike Reed7d954ad2016-10-28 15:42:34 -040057 fShader = make_shader(SkBlendMode::kDstIn);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000058 }
59
Leandro Lovisolo24fa2112023-08-15 19:05:17 +000060 SkString getName() const override { return SkString("composeshader"); }
scroggo@google.com32f20eb2012-06-29 17:28:50 +000061
Leandro Lovisolo8f023882023-08-15 21:13:52 +000062 SkISize getISize() override { return SkISize::Make(120, 120); }
scroggo@google.com32f20eb2012-06-29 17:28:50 +000063
mtklein36352bf2015-03-25 18:17:31 -070064 void onDraw(SkCanvas* canvas) override {
scroggo@google.com32f20eb2012-06-29 17:28:50 +000065 SkPaint paint;
scroggo@google.com32f20eb2012-06-29 17:28:50 +000066 paint.setColor(SK_ColorGREEN);
Mike Reed3661bc92017-02-22 13:21:42 -050067 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000068 paint.setShader(fShader);
Mike Reed3661bc92017-02-22 13:21:42 -050069 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000070 }
71
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000072protected:
reed2ad1aa62016-03-09 09:50:50 -080073 sk_sp<SkShader> fShader;
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000074
75private:
scroggo@google.com32f20eb2012-06-29 17:28:50 +000076 typedef GM INHERITED ;
77};
Mike Reed27738e02018-02-02 14:36:05 -050078DEF_GM( return new ComposeShaderGM; )
scroggo@google.com32f20eb2012-06-29 17:28:50 +000079
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000080class ComposeShaderAlphaGM : public skiagm::GM {
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000081public:
82 ComposeShaderAlphaGM() {}
83
84protected:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +000085 SkString getName() const override { return SkString("composeshader_alpha"); }
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000086
Leandro Lovisolo8f023882023-08-15 21:13:52 +000087 SkISize getISize() override { return SkISize::Make(750, 220); }
reed@google.com775bf662014-05-05 19:37:22 +000088
mtklein36352bf2015-03-25 18:17:31 -070089 void onDraw(SkCanvas* canvas) override {
reed2ad1aa62016-03-09 09:50:50 -080090 sk_sp<SkShader> shaders[] = {
Mike Reed7d954ad2016-10-28 15:42:34 -040091 make_shader(SkBlendMode::kDstIn),
92 make_shader(SkBlendMode::kSrcOver),
reed2ad1aa62016-03-09 09:50:50 -080093 };
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000094
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000095 SkPaint paint;
96 paint.setColor(SK_ColorGREEN);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000097
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000098 const SkRect r = SkRect::MakeXYWH(5, 5, 100, 100);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000099
Herb Derbyc37b3862022-06-21 09:49:17 -0400100 for (size_t y = 0; y < std::size(shaders); ++y) {
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000101 canvas->save();
102 for (int alpha = 0xFF; alpha > 0; alpha -= 0x28) {
Mike Reed9407e242019-02-15 16:13:57 -0500103 paint.setAlphaf(1.0f);
halcanary96fcdcc2015-08-27 07:41:13 -0700104 paint.setShader(nullptr);
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000105 canvas->drawRect(r, paint);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000106
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000107 paint.setAlpha(alpha);
reed2ad1aa62016-03-09 09:50:50 -0800108 paint.setShader(shaders[y]);
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000109 canvas->drawRect(r, paint);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000110
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000111 canvas->translate(r.width() + 5, 0);
112 }
113 canvas->restore();
114 canvas->translate(0, r.height() + 5);
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000115 }
116 }
wangyixd7059582015-09-03 08:32:22 -0700117
118private:
119 typedef GM INHERITED ;
120};
Mike Reed27738e02018-02-02 14:36:05 -0500121DEF_GM( return new ComposeShaderAlphaGM; )
wangyixd7059582015-09-03 08:32:22 -0700122
123// creates a square bitmap with red background and a green circle in the center
124static void draw_color_bm(SkBitmap* bm, int length) {
125 SkPaint paint;
126 paint.setColor(SK_ColorGREEN);
127
128 bm->allocN32Pixels(length, length);
129 bm->eraseColor(SK_ColorRED);
130
131 SkCanvas canvas(*bm);
132 canvas.drawCircle(SkIntToScalar(length/2), SkIntToScalar(length/2), SkIntToScalar(length/2),
133 paint);
134}
135
136// creates a square alpha8 bitmap with transparent background and an opaque circle in the center
137static void draw_alpha8_bm(SkBitmap* bm, int length) {
138 SkPaint circlePaint;
139 circlePaint.setColor(SK_ColorBLACK);
140
141 bm->allocPixels(SkImageInfo::MakeA8(length, length));
142 bm->eraseColor(SK_ColorTRANSPARENT);
143
144 SkCanvas canvas(*bm);
145 canvas.drawCircle(SkIntToScalar(length/2), SkIntToScalar(length/2), SkIntToScalar(length/4),
146 circlePaint);
147}
148
149// creates a linear gradient shader
reed2ad1aa62016-03-09 09:50:50 -0800150static sk_sp<SkShader> make_linear_gradient_shader(int length) {
wangyixd7059582015-09-03 08:32:22 -0700151 SkPoint pts[2];
152 SkColor colors[2];
153 pts[0].set(0, 0);
154 pts[1].set(SkIntToScalar(length), 0);
155 colors[0] = SK_ColorBLUE;
156 colors[1] = SkColorSetARGB(0, 0, 0, 0xFF);
Mike Reedfae8fce2019-04-03 10:27:45 -0400157 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
wangyixd7059582015-09-03 08:32:22 -0700158}
159
160
161class ComposeShaderBitmapGM : public skiagm::GM {
162public:
Florin Malita9ab449a2019-07-09 13:22:50 -0400163 ComposeShaderBitmapGM(bool use_lm) : fUseLocalMatrix(use_lm) {}
caryclark6df61152016-01-04 14:17:47 -0800164
herbb10fe492016-01-08 13:48:43 -0800165protected:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000166 SkString getName() const override {
Florin Malita9ab449a2019-07-09 13:22:50 -0400167 return SkStringPrintf("composeshader_bitmap%s", fUseLocalMatrix ? "_lm" : "");
wangyixd7059582015-09-03 08:32:22 -0700168 }
169
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000170 SkISize getISize() override {
wangyixd7059582015-09-03 08:32:22 -0700171 return SkISize::Make(7 * (squareLength + 5), 2 * (squareLength + 5));
172 }
173
174 void onDraw(SkCanvas* canvas) override {
Robert Phillipseae18482022-05-03 09:25:38 -0400175 if (!fInitialized) {
176 draw_color_bm(&fColorBitmap, squareLength);
Kevin Lubick77472bf2023-03-24 07:11:17 -0400177 sk_sp<SkImage> img = SkImages::RasterFromBitmap(fColorBitmap);
Robert Phillipseae18482022-05-03 09:25:38 -0400178 img = ToolUtils::MakeTextureImage(canvas, std::move(img));
Kevin Lubickc2ec5d02022-10-17 15:49:00 -0400179 if (img) {
180 fColorBitmapShader = img->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
181 SkSamplingOptions(), SkMatrix::I());
182 }
Robert Phillipseae18482022-05-03 09:25:38 -0400183 draw_alpha8_bm(&fAlpha8Bitmap, squareLength);
Kevin Lubick77472bf2023-03-24 07:11:17 -0400184 img = SkImages::RasterFromBitmap(fAlpha8Bitmap);
Robert Phillipseae18482022-05-03 09:25:38 -0400185 img = ToolUtils::MakeTextureImage(canvas, std::move(img));
Kevin Lubickc2ec5d02022-10-17 15:49:00 -0400186 if (img) {
187 fAlpha8BitmapShader = fAlpha8Bitmap.makeShader(SkTileMode::kRepeat,
188 SkTileMode::kRepeat,
189 SkSamplingOptions(),
190 SkMatrix::I());
191 }
Robert Phillipseae18482022-05-03 09:25:38 -0400192 fLinearGradientShader = make_linear_gradient_shader(squareLength);
193 fInitialized = true;
194 }
195
Mike Reed7d954ad2016-10-28 15:42:34 -0400196 SkBlendMode mode = SkBlendMode::kDstOver;
wangyixd7059582015-09-03 08:32:22 -0700197
Mike Reed1f607332020-05-21 12:11:27 -0400198 SkMatrix lm = SkMatrix::Translate(0, squareLength * 0.5f);
Florin Malita9ab449a2019-07-09 13:22:50 -0400199
reed2ad1aa62016-03-09 09:50:50 -0800200 sk_sp<SkShader> shaders[] = {
201 // gradient should appear over color bitmap
Kevin Lubick988ce042020-03-25 13:13:20 -0400202 SkShaders::Blend(mode, fLinearGradientShader, fColorBitmapShader),
reed2ad1aa62016-03-09 09:50:50 -0800203 // gradient should appear over alpha8 bitmap colorized by the paint color
Kevin Lubick988ce042020-03-25 13:13:20 -0400204 SkShaders::Blend(mode, fLinearGradientShader, fAlpha8BitmapShader),
reed2ad1aa62016-03-09 09:50:50 -0800205 };
Kevin Lubick988ce042020-03-25 13:13:20 -0400206 if (fUseLocalMatrix) {
Herb Derbyc37b3862022-06-21 09:49:17 -0400207 for (unsigned i = 0; i < std::size(shaders); ++i) {
Kevin Lubickc2ec5d02022-10-17 15:49:00 -0400208 shaders[i] = shaders[i] ? shaders[i]->makeWithLocalMatrix(lm) : nullptr;
Kevin Lubick988ce042020-03-25 13:13:20 -0400209 }
210 }
wangyixd7059582015-09-03 08:32:22 -0700211
212 SkPaint paint;
213 paint.setColor(SK_ColorYELLOW);
214
Kevin Lubick988ce042020-03-25 13:13:20 -0400215 const SkRect r = SkRect::MakeIWH(squareLength, squareLength);
wangyixd7059582015-09-03 08:32:22 -0700216
Herb Derbyc37b3862022-06-21 09:49:17 -0400217 for (size_t y = 0; y < std::size(shaders); ++y) {
wangyixd7059582015-09-03 08:32:22 -0700218 canvas->save();
219 for (int alpha = 0xFF; alpha > 0; alpha -= 0x28) {
220 paint.setAlpha(alpha);
reed2ad1aa62016-03-09 09:50:50 -0800221 paint.setShader(shaders[y]);
wangyixd7059582015-09-03 08:32:22 -0700222 canvas->drawRect(r, paint);
223
224 canvas->translate(r.width() + 5, 0);
225 }
226 canvas->restore();
227 canvas->translate(0, r.height() + 5);
228 }
229 }
halcanary9d524f22016-03-29 09:03:52 -0700230
wangyixd7059582015-09-03 08:32:22 -0700231private:
reed8681e602016-02-15 08:27:14 -0800232 /** This determines the length and width of the bitmaps used in the ComposeShaders. Values
wangyixd7059582015-09-03 08:32:22 -0700233 * above 20 may cause an SkASSERT to fail in SkSmallAllocator. However, larger values will
234 * work in a release build. You can change this parameter and then compile a release build
235 * to have this GM draw larger bitmaps for easier visual inspection.
236 */
Brian Salomon9fa47cc2021-10-08 18:48:26 -0400237 inline static constexpr int squareLength = 20;
wangyixd7059582015-09-03 08:32:22 -0700238
Florin Malita9ab449a2019-07-09 13:22:50 -0400239 const bool fUseLocalMatrix;
240
Robert Phillipseae18482022-05-03 09:25:38 -0400241 bool fInitialized = false;
wangyixd7059582015-09-03 08:32:22 -0700242 SkBitmap fColorBitmap;
243 SkBitmap fAlpha8Bitmap;
reed2ad1aa62016-03-09 09:50:50 -0800244 sk_sp<SkShader> fColorBitmapShader;
245 sk_sp<SkShader> fAlpha8BitmapShader;
246 sk_sp<SkShader> fLinearGradientShader;
wangyixd7059582015-09-03 08:32:22 -0700247
John Stiles7571f9e2020-09-02 22:42:33 -0400248 using INHERITED = GM;
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000249};
Florin Malita9ab449a2019-07-09 13:22:50 -0400250DEF_GM( return new ComposeShaderBitmapGM(false); )
251DEF_GM( return new ComposeShaderBitmapGM(true); )
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000252
caryclarkc2ad65e2016-08-15 12:03:33 -0700253DEF_SIMPLE_GM(composeshader_bitmap2, canvas, 200, 200) {
254 int width = 255;
255 int height = 255;
256 SkTDArray<uint8_t> dst8Storage;
Herb Derbyebfa2b72022-09-30 14:08:00 -0400257 dst8Storage.resize(width * height);
caryclarkc2ad65e2016-08-15 12:03:33 -0700258 SkTDArray<uint32_t> dst32Storage;
Herb Derbyebfa2b72022-09-30 14:08:00 -0400259 dst32Storage.resize(width * height * sizeof(int32_t));
caryclarkc2ad65e2016-08-15 12:03:33 -0700260 for (int y = 0; y < height; ++y) {
261 for (int x = 0; x < width; ++x) {
262 dst8Storage[y * width + x] = (y + x) / 2;
263 dst32Storage[y * width + x] = SkPackARGB32(0xFF, x, y, 0);
264 }
265 }
266 SkPaint paint;
267 paint.setAntiAlias(true);
268 paint.setColor(SK_ColorBLUE);
269 SkRect r = {0, 0, SkIntToScalar(width), SkIntToScalar(height)};
270 canvas->drawRect(r, paint);
271 SkBitmap skBitmap, skMask;
272 SkImageInfo imageInfo = SkImageInfo::Make(width, height,
273 SkColorType::kN32_SkColorType, kPremul_SkAlphaType);
274 skBitmap.installPixels(imageInfo, dst32Storage.begin(), width * sizeof(int32_t),
Mike Reed086a4272017-07-18 10:53:11 -0400275 nullptr, nullptr);
caryclarkc2ad65e2016-08-15 12:03:33 -0700276 imageInfo = SkImageInfo::Make(width, height,
277 SkColorType::kAlpha_8_SkColorType, kPremul_SkAlphaType);
Mike Reed086a4272017-07-18 10:53:11 -0400278 skMask.installPixels(imageInfo, dst8Storage.begin(), width, nullptr, nullptr);
Mike Reedac9f0c92020-12-23 10:11:33 -0500279 sk_sp<SkImage> skSrc = skBitmap.asImage();
280 sk_sp<SkImage> skMaskImage = skMask.asImage();
caryclarkc2ad65e2016-08-15 12:03:33 -0700281 paint.setShader(
Mike Reedb612b6c2020-12-08 21:58:35 -0500282 SkShaders::Blend(SkBlendMode::kSrcIn,
283 skMaskImage->makeShader(SkSamplingOptions()),
284 skSrc->makeShader(SkSamplingOptions())));
caryclarkc2ad65e2016-08-15 12:03:33 -0700285 canvas->drawRect(r, paint);
286}
287
Mike Reed27738e02018-02-02 14:36:05 -0500288///////////////////////////////////////////////////////////////////////////////////////////////////
scroggo@google.com32f20eb2012-06-29 17:28:50 +0000289
Mike Reed27738e02018-02-02 14:36:05 -0500290static sk_sp<SkShader> make_src_shader(SkScalar size) {
291 const SkPoint pts[] = { { 0, 0 }, { 0, size } };
292 const SkColor colors[] = { 0xFF0000FF, 0x000000FF };
Mike Reedfae8fce2019-04-03 10:27:45 -0400293 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
Mike Reed27738e02018-02-02 14:36:05 -0500294}
295
296static sk_sp<SkShader> make_dst_shader(SkScalar size) {
297 const SkPoint pts[] = { { 0, 0 }, { size, 0 } };
298 const SkColor colors[] = { SK_ColorRED, 0x00FF0000 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400299 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
Mike Reed27738e02018-02-02 14:36:05 -0500300}
301
302const SkScalar gCellSize = 100;
303
304static void draw_cell(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
305 SkBlendMode mode, SkAlpha alpha) {
306 const SkRect r = SkRect::MakeWH(gCellSize, gCellSize);
307 SkPaint p;
308 p.setAlpha(alpha);
309
310 SkAutoCanvasRestore acr(canvas, false);
311 canvas->saveLayer(&r, &p);
312 p.setAlpha(0xFF);
313
314 p.setShader(dst);
315 p.setBlendMode(SkBlendMode::kSrc);
316 canvas->drawRect(r, p);
317
318 p.setShader(src);
319 p.setBlendMode(mode);
320 canvas->drawRect(r, p);
321}
322
323static void draw_composed(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
324 SkBlendMode mode, SkAlpha alpha) {
325 SkPaint p;
326 p.setAlpha(alpha);
Mike Reedc8bea7d2019-04-09 13:55:36 -0400327 p.setShader(SkShaders::Blend(mode, dst, src));
Mike Reed27738e02018-02-02 14:36:05 -0500328 canvas->drawRect(SkRect::MakeWH(gCellSize, gCellSize), p);
329}
330
331static void draw_pair(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
332 SkBlendMode mode) {
333 SkAutoCanvasRestore acr(canvas, true);
334
335 const SkScalar gap = 4;
336 SkRect r = SkRect::MakeWH(2 * gCellSize + gap, 2 * gCellSize + gap);
Mike Reed1d08d722018-02-02 16:15:49 -0500337 r.outset(gap + 1.5f, gap + 1.5f);
Mike Reed27738e02018-02-02 14:36:05 -0500338 SkPaint p;
339 p.setStyle(SkPaint::kStroke_Style);
340 canvas->drawRect(r, p); // border
341
342 SkAlpha alpha = 0xFF;
343 for (int y = 0; y < 2; ++y) {
344 draw_cell(canvas, src, dst, mode, alpha);
345 canvas->save();
346 canvas->translate(gCellSize + gap, 0);
347 draw_composed(canvas, src, dst, mode, alpha);
348 canvas->restore();
349
350 canvas->translate(0, gCellSize + gap);
351 alpha = 0x80;
352 }
353}
354
355DEF_SIMPLE_GM(composeshader_grid, canvas, 882, 882) {
356 auto src = make_src_shader(gCellSize);
357 auto dst = make_dst_shader(gCellSize);
358
Mike Reed1d08d722018-02-02 16:15:49 -0500359 const SkScalar margin = 15;
Mike Reed27738e02018-02-02 14:36:05 -0500360 const SkScalar dx = 2*gCellSize + margin;
361 const SkScalar dy = 2*gCellSize + margin;
362
363 canvas->translate(margin, margin);
364 canvas->save();
365 for (int m = 0; m < 16; ++m) {
366 SkBlendMode mode = static_cast<SkBlendMode>(m);
367 draw_pair(canvas, src, dst, mode);
368 if ((m % 4) == 3) {
369 canvas->restore();
370 canvas->translate(0, dy);
371 canvas->save();
372 } else {
373 canvas->translate(dx, 0);
374 }
375 }
376 canvas->restore();
377}