blob: ec9f7e3e6df7902001edcee3e016ea8a55aa9396 [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"
28#include "include/private/SkTDArray.h"
scroggo@google.com32f20eb2012-06-29 17:28:50 +000029
Ben Wagner7fde8e12019-05-01 17:28:53 -040030#include <utility>
31
Mike Reed7d954ad2016-10-28 15:42:34 -040032static sk_sp<SkShader> make_shader(SkBlendMode mode) {
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000033 SkPoint pts[2];
34 SkColor colors[2];
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000035
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000036 pts[0].set(0, 0);
37 pts[1].set(SkIntToScalar(100), 0);
38 colors[0] = SK_ColorRED;
39 colors[1] = SK_ColorBLUE;
Mike Reedfae8fce2019-04-03 10:27:45 -040040 auto shaderA = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000041
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000042 pts[0].set(0, 0);
43 pts[1].set(0, SkIntToScalar(100));
44 colors[0] = SK_ColorBLACK;
45 colors[1] = SkColorSetARGB(0x80, 0, 0, 0);
Mike Reedfae8fce2019-04-03 10:27:45 -040046 auto shaderB = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000047
Mike Reedc8bea7d2019-04-09 13:55:36 -040048 return SkShaders::Blend(mode, std::move(shaderA), std::move(shaderB));
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000049}
50
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000051class ComposeShaderGM : public skiagm::GM {
scroggo@google.com32f20eb2012-06-29 17:28:50 +000052public:
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000053 ComposeShaderGM() {
Mike Reed7d954ad2016-10-28 15:42:34 -040054 fShader = make_shader(SkBlendMode::kDstIn);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000055 }
56
scroggo@google.com32f20eb2012-06-29 17:28:50 +000057protected:
mtklein36352bf2015-03-25 18:17:31 -070058 SkString onShortName() override {
scroggo@google.com32f20eb2012-06-29 17:28:50 +000059 return SkString("composeshader");
60 }
61
mtklein36352bf2015-03-25 18:17:31 -070062 SkISize onISize() override {
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000063 return SkISize::Make(120, 120);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000064 }
65
mtklein36352bf2015-03-25 18:17:31 -070066 void onDraw(SkCanvas* canvas) override {
scroggo@google.com32f20eb2012-06-29 17:28:50 +000067 SkPaint paint;
scroggo@google.com32f20eb2012-06-29 17:28:50 +000068 paint.setColor(SK_ColorGREEN);
Mike Reed3661bc92017-02-22 13:21:42 -050069 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000070 paint.setShader(fShader);
Mike Reed3661bc92017-02-22 13:21:42 -050071 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
scroggo@google.com32f20eb2012-06-29 17:28:50 +000072 }
73
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000074protected:
reed2ad1aa62016-03-09 09:50:50 -080075 sk_sp<SkShader> fShader;
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000076
77private:
scroggo@google.com32f20eb2012-06-29 17:28:50 +000078 typedef GM INHERITED ;
79};
Mike Reed27738e02018-02-02 14:36:05 -050080DEF_GM( return new ComposeShaderGM; )
scroggo@google.com32f20eb2012-06-29 17:28:50 +000081
commit-bot@chromium.org58e42872014-05-05 18:23:54 +000082class ComposeShaderAlphaGM : public skiagm::GM {
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000083public:
84 ComposeShaderAlphaGM() {}
85
86protected:
mtklein36352bf2015-03-25 18:17:31 -070087 SkString onShortName() override {
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000088 return SkString("composeshader_alpha");
89 }
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +000090
mtklein36352bf2015-03-25 18:17:31 -070091 SkISize onISize() override {
wangyixd7059582015-09-03 08:32:22 -070092 return SkISize::Make(750, 220);
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +000093 }
reed@google.com775bf662014-05-05 19:37:22 +000094
mtklein36352bf2015-03-25 18:17:31 -070095 void onDraw(SkCanvas* canvas) override {
reed2ad1aa62016-03-09 09:50:50 -080096 sk_sp<SkShader> shaders[] = {
Mike Reed7d954ad2016-10-28 15:42:34 -040097 make_shader(SkBlendMode::kDstIn),
98 make_shader(SkBlendMode::kSrcOver),
reed2ad1aa62016-03-09 09:50:50 -080099 };
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000100
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000101 SkPaint paint;
102 paint.setColor(SK_ColorGREEN);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000103
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000104 const SkRect r = SkRect::MakeXYWH(5, 5, 100, 100);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000105
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000106 for (size_t y = 0; y < SK_ARRAY_COUNT(shaders); ++y) {
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000107 canvas->save();
108 for (int alpha = 0xFF; alpha > 0; alpha -= 0x28) {
Mike Reed9407e242019-02-15 16:13:57 -0500109 paint.setAlphaf(1.0f);
halcanary96fcdcc2015-08-27 07:41:13 -0700110 paint.setShader(nullptr);
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000111 canvas->drawRect(r, paint);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000112
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000113 paint.setAlpha(alpha);
reed2ad1aa62016-03-09 09:50:50 -0800114 paint.setShader(shaders[y]);
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000115 canvas->drawRect(r, paint);
skia.committer@gmail.comedda70e2014-05-06 03:06:53 +0000116
commit-bot@chromium.org58e42872014-05-05 18:23:54 +0000117 canvas->translate(r.width() + 5, 0);
118 }
119 canvas->restore();
120 canvas->translate(0, r.height() + 5);
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000121 }
122 }
wangyixd7059582015-09-03 08:32:22 -0700123
124private:
125 typedef GM INHERITED ;
126};
Mike Reed27738e02018-02-02 14:36:05 -0500127DEF_GM( return new ComposeShaderAlphaGM; )
wangyixd7059582015-09-03 08:32:22 -0700128
129// creates a square bitmap with red background and a green circle in the center
130static void draw_color_bm(SkBitmap* bm, int length) {
131 SkPaint paint;
132 paint.setColor(SK_ColorGREEN);
133
134 bm->allocN32Pixels(length, length);
135 bm->eraseColor(SK_ColorRED);
136
137 SkCanvas canvas(*bm);
138 canvas.drawCircle(SkIntToScalar(length/2), SkIntToScalar(length/2), SkIntToScalar(length/2),
139 paint);
140}
141
142// creates a square alpha8 bitmap with transparent background and an opaque circle in the center
143static void draw_alpha8_bm(SkBitmap* bm, int length) {
144 SkPaint circlePaint;
145 circlePaint.setColor(SK_ColorBLACK);
146
147 bm->allocPixels(SkImageInfo::MakeA8(length, length));
148 bm->eraseColor(SK_ColorTRANSPARENT);
149
150 SkCanvas canvas(*bm);
151 canvas.drawCircle(SkIntToScalar(length/2), SkIntToScalar(length/2), SkIntToScalar(length/4),
152 circlePaint);
153}
154
155// creates a linear gradient shader
reed2ad1aa62016-03-09 09:50:50 -0800156static sk_sp<SkShader> make_linear_gradient_shader(int length) {
wangyixd7059582015-09-03 08:32:22 -0700157 SkPoint pts[2];
158 SkColor colors[2];
159 pts[0].set(0, 0);
160 pts[1].set(SkIntToScalar(length), 0);
161 colors[0] = SK_ColorBLUE;
162 colors[1] = SkColorSetARGB(0, 0, 0, 0xFF);
Mike Reedfae8fce2019-04-03 10:27:45 -0400163 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
wangyixd7059582015-09-03 08:32:22 -0700164}
165
166
167class ComposeShaderBitmapGM : public skiagm::GM {
168public:
herbb10fe492016-01-08 13:48:43 -0800169 ComposeShaderBitmapGM() {}
caryclark6df61152016-01-04 14:17:47 -0800170
herbb10fe492016-01-08 13:48:43 -0800171protected:
caryclark6df61152016-01-04 14:17:47 -0800172 void onOnceBeforeDraw() override {
wangyixd7059582015-09-03 08:32:22 -0700173 draw_color_bm(&fColorBitmap, squareLength);
174 draw_alpha8_bm(&fAlpha8Bitmap, squareLength);
175 SkMatrix s;
176 s.reset();
Mike Reed50acf8f2019-04-08 13:20:23 -0400177 fColorBitmapShader = fColorBitmap.makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat, &s);
178 fAlpha8BitmapShader = fAlpha8Bitmap.makeShader(SkTileMode::kRepeat,
179 SkTileMode::kRepeat, &s);
wangyixd7059582015-09-03 08:32:22 -0700180 fLinearGradientShader = make_linear_gradient_shader(squareLength);
181 }
caryclark6df61152016-01-04 14:17:47 -0800182
wangyixd7059582015-09-03 08:32:22 -0700183 SkString onShortName() override {
184 return SkString("composeshader_bitmap");
185 }
186
187 SkISize onISize() override {
188 return SkISize::Make(7 * (squareLength + 5), 2 * (squareLength + 5));
189 }
190
191 void onDraw(SkCanvas* canvas) override {
Mike Reed7d954ad2016-10-28 15:42:34 -0400192 SkBlendMode mode = SkBlendMode::kDstOver;
wangyixd7059582015-09-03 08:32:22 -0700193
reed2ad1aa62016-03-09 09:50:50 -0800194 sk_sp<SkShader> shaders[] = {
195 // gradient should appear over color bitmap
Mike Reedc8bea7d2019-04-09 13:55:36 -0400196 SkShaders::Blend(mode, fLinearGradientShader, fColorBitmapShader),
reed2ad1aa62016-03-09 09:50:50 -0800197 // gradient should appear over alpha8 bitmap colorized by the paint color
Mike Reedc8bea7d2019-04-09 13:55:36 -0400198 SkShaders::Blend(mode, fLinearGradientShader, fAlpha8BitmapShader),
reed2ad1aa62016-03-09 09:50:50 -0800199 };
wangyixd7059582015-09-03 08:32:22 -0700200
201 SkPaint paint;
202 paint.setColor(SK_ColorYELLOW);
203
204 const SkRect r = SkRect::MakeXYWH(0, 0, SkIntToScalar(squareLength),
205 SkIntToScalar(squareLength));
206
207 for (size_t y = 0; y < SK_ARRAY_COUNT(shaders); ++y) {
wangyixd7059582015-09-03 08:32:22 -0700208 canvas->save();
209 for (int alpha = 0xFF; alpha > 0; alpha -= 0x28) {
210 paint.setAlpha(alpha);
reed2ad1aa62016-03-09 09:50:50 -0800211 paint.setShader(shaders[y]);
wangyixd7059582015-09-03 08:32:22 -0700212 canvas->drawRect(r, paint);
213
214 canvas->translate(r.width() + 5, 0);
215 }
216 canvas->restore();
217 canvas->translate(0, r.height() + 5);
218 }
219 }
halcanary9d524f22016-03-29 09:03:52 -0700220
wangyixd7059582015-09-03 08:32:22 -0700221private:
reed8681e602016-02-15 08:27:14 -0800222 /** This determines the length and width of the bitmaps used in the ComposeShaders. Values
wangyixd7059582015-09-03 08:32:22 -0700223 * above 20 may cause an SkASSERT to fail in SkSmallAllocator. However, larger values will
224 * work in a release build. You can change this parameter and then compile a release build
225 * to have this GM draw larger bitmaps for easier visual inspection.
226 */
mtkleindbfd7ab2016-09-01 11:24:54 -0700227 static constexpr int squareLength = 20;
wangyixd7059582015-09-03 08:32:22 -0700228
229 SkBitmap fColorBitmap;
230 SkBitmap fAlpha8Bitmap;
reed2ad1aa62016-03-09 09:50:50 -0800231 sk_sp<SkShader> fColorBitmapShader;
232 sk_sp<SkShader> fAlpha8BitmapShader;
233 sk_sp<SkShader> fLinearGradientShader;
wangyixd7059582015-09-03 08:32:22 -0700234
235 typedef GM INHERITED;
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000236};
Mike Reed27738e02018-02-02 14:36:05 -0500237DEF_GM( return new ComposeShaderBitmapGM; )
commit-bot@chromium.orgec7d6732014-05-05 16:19:08 +0000238
caryclarkc2ad65e2016-08-15 12:03:33 -0700239DEF_SIMPLE_GM(composeshader_bitmap2, canvas, 200, 200) {
240 int width = 255;
241 int height = 255;
242 SkTDArray<uint8_t> dst8Storage;
243 dst8Storage.setCount(width * height);
244 SkTDArray<uint32_t> dst32Storage;
245 dst32Storage.setCount(width * height * sizeof(int32_t));
246 for (int y = 0; y < height; ++y) {
247 for (int x = 0; x < width; ++x) {
248 dst8Storage[y * width + x] = (y + x) / 2;
249 dst32Storage[y * width + x] = SkPackARGB32(0xFF, x, y, 0);
250 }
251 }
252 SkPaint paint;
253 paint.setAntiAlias(true);
254 paint.setColor(SK_ColorBLUE);
255 SkRect r = {0, 0, SkIntToScalar(width), SkIntToScalar(height)};
256 canvas->drawRect(r, paint);
257 SkBitmap skBitmap, skMask;
258 SkImageInfo imageInfo = SkImageInfo::Make(width, height,
259 SkColorType::kN32_SkColorType, kPremul_SkAlphaType);
260 skBitmap.installPixels(imageInfo, dst32Storage.begin(), width * sizeof(int32_t),
Mike Reed086a4272017-07-18 10:53:11 -0400261 nullptr, nullptr);
caryclarkc2ad65e2016-08-15 12:03:33 -0700262 imageInfo = SkImageInfo::Make(width, height,
263 SkColorType::kAlpha_8_SkColorType, kPremul_SkAlphaType);
Mike Reed086a4272017-07-18 10:53:11 -0400264 skMask.installPixels(imageInfo, dst8Storage.begin(), width, nullptr, nullptr);
caryclarkc2ad65e2016-08-15 12:03:33 -0700265 sk_sp<SkImage> skSrc = SkImage::MakeFromBitmap(skBitmap);
caryclarkc2ad65e2016-08-15 12:03:33 -0700266 sk_sp<SkImage> skMaskImage = SkImage::MakeFromBitmap(skMask);
caryclarkc2ad65e2016-08-15 12:03:33 -0700267 paint.setShader(
Mike Reedc8bea7d2019-04-09 13:55:36 -0400268 SkShaders::Blend(SkBlendMode::kSrcIn, skMaskImage->makeShader(), skSrc->makeShader()));
caryclarkc2ad65e2016-08-15 12:03:33 -0700269 canvas->drawRect(r, paint);
270}
271
Mike Reed27738e02018-02-02 14:36:05 -0500272///////////////////////////////////////////////////////////////////////////////////////////////////
scroggo@google.com32f20eb2012-06-29 17:28:50 +0000273
Mike Reed27738e02018-02-02 14:36:05 -0500274static sk_sp<SkShader> make_src_shader(SkScalar size) {
275 const SkPoint pts[] = { { 0, 0 }, { 0, size } };
276 const SkColor colors[] = { 0xFF0000FF, 0x000000FF };
Mike Reedfae8fce2019-04-03 10:27:45 -0400277 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
Mike Reed27738e02018-02-02 14:36:05 -0500278}
279
280static sk_sp<SkShader> make_dst_shader(SkScalar size) {
281 const SkPoint pts[] = { { 0, 0 }, { size, 0 } };
282 const SkColor colors[] = { SK_ColorRED, 0x00FF0000 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400283 return SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkTileMode::kClamp);
Mike Reed27738e02018-02-02 14:36:05 -0500284}
285
286const SkScalar gCellSize = 100;
287
288static void draw_cell(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
289 SkBlendMode mode, SkAlpha alpha) {
290 const SkRect r = SkRect::MakeWH(gCellSize, gCellSize);
291 SkPaint p;
292 p.setAlpha(alpha);
293
294 SkAutoCanvasRestore acr(canvas, false);
295 canvas->saveLayer(&r, &p);
296 p.setAlpha(0xFF);
297
298 p.setShader(dst);
299 p.setBlendMode(SkBlendMode::kSrc);
300 canvas->drawRect(r, p);
301
302 p.setShader(src);
303 p.setBlendMode(mode);
304 canvas->drawRect(r, p);
305}
306
307static void draw_composed(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
308 SkBlendMode mode, SkAlpha alpha) {
309 SkPaint p;
310 p.setAlpha(alpha);
Mike Reedc8bea7d2019-04-09 13:55:36 -0400311 p.setShader(SkShaders::Blend(mode, dst, src));
Mike Reed27738e02018-02-02 14:36:05 -0500312 canvas->drawRect(SkRect::MakeWH(gCellSize, gCellSize), p);
313}
314
315static void draw_pair(SkCanvas* canvas, sk_sp<SkShader> src, sk_sp<SkShader> dst,
316 SkBlendMode mode) {
317 SkAutoCanvasRestore acr(canvas, true);
318
319 const SkScalar gap = 4;
320 SkRect r = SkRect::MakeWH(2 * gCellSize + gap, 2 * gCellSize + gap);
Mike Reed1d08d722018-02-02 16:15:49 -0500321 r.outset(gap + 1.5f, gap + 1.5f);
Mike Reed27738e02018-02-02 14:36:05 -0500322 SkPaint p;
323 p.setStyle(SkPaint::kStroke_Style);
324 canvas->drawRect(r, p); // border
325
326 SkAlpha alpha = 0xFF;
327 for (int y = 0; y < 2; ++y) {
328 draw_cell(canvas, src, dst, mode, alpha);
329 canvas->save();
330 canvas->translate(gCellSize + gap, 0);
331 draw_composed(canvas, src, dst, mode, alpha);
332 canvas->restore();
333
334 canvas->translate(0, gCellSize + gap);
335 alpha = 0x80;
336 }
337}
338
339DEF_SIMPLE_GM(composeshader_grid, canvas, 882, 882) {
340 auto src = make_src_shader(gCellSize);
341 auto dst = make_dst_shader(gCellSize);
342
Mike Reed1d08d722018-02-02 16:15:49 -0500343 const SkScalar margin = 15;
Mike Reed27738e02018-02-02 14:36:05 -0500344 const SkScalar dx = 2*gCellSize + margin;
345 const SkScalar dy = 2*gCellSize + margin;
346
347 canvas->translate(margin, margin);
348 canvas->save();
349 for (int m = 0; m < 16; ++m) {
350 SkBlendMode mode = static_cast<SkBlendMode>(m);
351 draw_pair(canvas, src, dst, mode);
352 if ((m % 4) == 3) {
353 canvas->restore();
354 canvas->translate(0, dy);
355 canvas->save();
356 } else {
357 canvas->translate(dx, 0);
358 }
359 }
360 canvas->restore();
361}