blob: 8b35506d58e23f343e4fd97cecc0195e384141f4 [file] [log] [blame]
mike@reedtribe.orge51755f2011-12-10 19:36:56 +00001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
9#include "include/core/SkCanvas.h"
Ben Wagnerd1701ba2019-04-30 13:44:26 -040010#include "include/core/SkColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkFont.h"
Ben Wagnerd1701ba2019-04-30 13:44:26 -040012#include "include/core/SkFontTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "include/core/SkImage.h"
Ben Wagnerd1701ba2019-04-30 13:44:26 -040014#include "include/core/SkImageFilter.h"
15#include "include/core/SkPaint.h"
16#include "include/core/SkPoint.h"
17#include "include/core/SkRect.h"
18#include "include/core/SkRefCnt.h"
19#include "include/core/SkScalar.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050020#include "include/core/SkShader.h"
Ben Wagnerd1701ba2019-04-30 13:44:26 -040021#include "include/core/SkSize.h"
22#include "include/core/SkString.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "include/core/SkSurface.h"
Ben Wagnerd1701ba2019-04-30 13:44:26 -040024#include "include/core/SkTileMode.h"
25#include "include/core/SkTypeface.h"
26#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "include/effects/SkGradientShader.h"
Michael Ludwig898bbfa2019-08-02 15:21:23 -040028#include "include/effects/SkImageFilters.h"
Kevin Lubickc5963822023-07-17 08:23:31 -040029#include "include/effects/SkRuntimeEffect.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050030#include "tools/ToolUtils.h"
Kevin Lubicke836c3a2023-10-20 06:55:35 -040031#include "tools/fonts/FontToolUtils.h"
fmalitaad7cb812016-09-29 12:25:26 -070032
Ben Wagnerd1701ba2019-04-30 13:44:26 -040033#include <utility>
34
Mike Reed26ea9752021-07-08 10:43:31 -040035static sk_sp<SkImage> make_src(int w, int h) {
Kevin Lubick5c93acf2023-05-09 12:11:43 -040036 sk_sp<SkSurface> surface(SkSurfaces::Raster(SkImageInfo::MakeN32Premul(w, h)));
fmalitaad7cb812016-09-29 12:25:26 -070037 SkCanvas* canvas = surface->getCanvas();
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000038
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000039 SkPaint paint;
Mike Reed26ea9752021-07-08 10:43:31 -040040 SkPoint pts[] = { {0, 0}, {SkIntToScalar(w), SkIntToScalar(h)} };
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000041 SkColor colors[] = {
senorblanco@chromium.org35c733c2013-05-28 19:43:05 +000042 SK_ColorTRANSPARENT, SK_ColorGREEN, SK_ColorCYAN,
43 SK_ColorRED, SK_ColorMAGENTA, SK_ColorWHITE,
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000044 };
Herb Derbyc37b3862022-06-21 09:49:17 -040045 paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -040046 SkTileMode::kClamp));
fmalitaad7cb812016-09-29 12:25:26 -070047 canvas->drawPaint(paint);
48 return surface->makeImageSnapshot();
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000049}
50
Mike Reed26ea9752021-07-08 10:43:31 -040051static sk_sp<SkImage> make_dst(int w, int h) {
Kevin Lubick5c93acf2023-05-09 12:11:43 -040052 sk_sp<SkSurface> surface(SkSurfaces::Raster(SkImageInfo::MakeN32Premul(w, h)));
fmalitaad7cb812016-09-29 12:25:26 -070053 SkCanvas* canvas = surface->getCanvas();
54
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000055 SkPaint paint;
Mike Reed26ea9752021-07-08 10:43:31 -040056 SkPoint pts[] = { {0, SkIntToScalar(h)}, {SkIntToScalar(w), 0} };
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000057 SkColor colors[] = {
caryclarkdfcb7ab2015-07-17 09:39:16 -070058 SK_ColorBLUE, SK_ColorYELLOW, SK_ColorBLACK, SK_ColorGREEN,
Mike Kleind46dce32018-08-16 10:17:03 -040059 SK_ColorGRAY,
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000060 };
Herb Derbyc37b3862022-06-21 09:49:17 -040061 paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, std::size(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -040062 SkTileMode::kClamp));
fmalitaad7cb812016-09-29 12:25:26 -070063 canvas->drawPaint(paint);
64 return surface->makeImageSnapshot();
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000065}
66
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000067static void show_k_text(SkCanvas* canvas, SkScalar x, SkScalar y, const SkScalar k[]) {
Kevin Lubicke836c3a2023-10-20 06:55:35 -040068 SkFont font(ToolUtils::DefaultPortableTypeface(), 24);
Mike Reed17c574a2018-12-12 18:10:38 -050069 font.setEdging(SkFont::Edging::kAntiAlias);
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000070 SkPaint paint;
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000071 paint.setAntiAlias(true);
72 for (int i = 0; i < 4; ++i) {
73 SkString str;
74 str.appendScalar(k[i]);
Ben Wagner51e15a62019-05-07 15:38:46 -040075 SkScalar width = font.measureText(str.c_str(), str.size(), SkTextEncoding::kUTF8);
Hal Canary89a644b2019-01-07 09:36:09 -050076 canvas->drawString(str, x, y + font.getSize(), font, paint);
reed@google.comeb0fa292011-12-12 22:01:06 +000077 x += width + SkIntToScalar(10);
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000078 }
79}
80
81class ArithmodeGM : public skiagm::GM {
Leandro Lovisolo24fa2112023-08-15 19:05:17 +000082 SkString getName() const override { return SkString("arithmode"); }
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000083
Leandro Lovisolo8f023882023-08-15 21:13:52 +000084 SkISize getISize() override { return {640, 572}; }
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000085
Hal Canarybd865e22019-07-18 11:51:19 -040086 void onDraw(SkCanvas* canvas) override {
Mike Reed26ea9752021-07-08 10:43:31 -040087 constexpr int WW = 100,
88 HH = 32;
89
90 sk_sp<SkImage> src = make_src(WW, HH);
91 sk_sp<SkImage> dst = make_dst(WW, HH);
Michael Ludwig642f2182023-05-30 20:57:39 -040092 sk_sp<SkImageFilter> srcFilter = SkImageFilters::Image(src, {SkFilterMode::kLinear});
93 sk_sp<SkImageFilter> dstFilter = SkImageFilters::Image(dst, {SkFilterMode::kLinear});
rmistry@google.comae933ce2012-08-23 18:19:56 +000094
mtkleindbfd7ab2016-09-01 11:24:54 -070095 constexpr SkScalar one = SK_Scalar1;
96 constexpr SkScalar K[] = {
mike@reedtribe.orge51755f2011-12-10 19:36:56 +000097 0, 0, 0, 0,
reed@google.comeb0fa292011-12-12 22:01:06 +000098 0, 0, 0, one,
99 0, one, 0, 0,
100 0, 0, one, 0,
101 0, one, one, 0,
102 0, one, -one, 0,
103 0, one/2, one/2, 0,
104 0, one/2, one/2, one/4,
105 0, one/2, one/2, -one/4,
106 one/4, one/2, one/2, 0,
107 -one/4, one/2, one/2, 0,
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000108 };
rmistry@google.comae933ce2012-08-23 18:19:56 +0000109
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000110 const SkScalar* k = K;
Herb Derbyc37b3862022-06-21 09:49:17 -0400111 const SkScalar* stop = k + std::size(K);
Michael Ludwigbbfb84f2023-05-18 16:01:05 -0400112 // Many of the Arithmetic filters have a 4th coefficient that's not zero, which means they
113 // affect transparent black. 'rect' is used as a crop filter to make sure they don't
114 // overwrite each other.
fmalitaad7cb812016-09-29 12:25:26 -0700115 const SkRect rect = SkRect::MakeWH(WW, HH);
116 SkScalar gap = SkIntToScalar(WW + 20);
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000117 while (k < stop) {
fmalitaad7cb812016-09-29 12:25:26 -0700118 {
119 SkAutoCanvasRestore acr(canvas, true);
120 canvas->drawImage(src, 0, 0);
121 canvas->translate(gap, 0);
122 canvas->drawImage(dst, 0, 0);
123 canvas->translate(gap, 0);
124 SkPaint paint;
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400125 paint.setImageFilter(SkImageFilters::Arithmetic(k[0], k[1], k[2], k[3], true,
Michael Ludwigbbfb84f2023-05-18 16:01:05 -0400126 dstFilter, srcFilter, rect));
127 canvas->saveLayer(nullptr, &paint);
fmalitaad7cb812016-09-29 12:25:26 -0700128 canvas->restore();
129
130 canvas->translate(gap, 0);
131 show_k_text(canvas, 0, 0, k);
132 }
133
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000134 k += 4;
fmalitaad7cb812016-09-29 12:25:26 -0700135 canvas->translate(0, HH + 12);
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000136 }
ericrkaf96fce2015-10-19 14:41:11 -0700137
138 // Draw two special cases to test enforcePMColor. In these cases, we
139 // draw the dst bitmap twice, the first time it is halved and inverted,
140 // leading to invalid premultiplied colors. If we enforcePMColor, these
141 // invalid values should be clamped, and will not contribute to the
142 // second draw.
143 for (int i = 0; i < 2; i++) {
144 const bool enforcePMColor = (i == 0);
ericrkaf96fce2015-10-19 14:41:11 -0700145
fmalitaad7cb812016-09-29 12:25:26 -0700146 {
147 SkAutoCanvasRestore acr(canvas, true);
148 canvas->translate(gap, 0);
149 canvas->drawImage(dst, 0, 0);
150 canvas->translate(gap, 0);
ericrkaf96fce2015-10-19 14:41:11 -0700151
fmalitaad7cb812016-09-29 12:25:26 -0700152 sk_sp<SkImageFilter> bg =
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400153 SkImageFilters::Arithmetic(0, 0, -one / 2, 1, enforcePMColor, dstFilter,
154 nullptr, nullptr);
fmalitaad7cb812016-09-29 12:25:26 -0700155 SkPaint p;
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400156 p.setImageFilter(SkImageFilters::Arithmetic(0, one / 2, -one, 1, true,
Michael Ludwigbbfb84f2023-05-18 16:01:05 -0400157 std::move(bg), dstFilter, rect));
158 canvas->saveLayer(nullptr, &p);
fmalitaad7cb812016-09-29 12:25:26 -0700159 canvas->restore();
160 canvas->translate(gap, 0);
161
162 // Label
Kevin Lubicke836c3a2023-10-20 06:55:35 -0400163 SkFont font(ToolUtils::DefaultPortableTypeface(), 24);
fmalitaad7cb812016-09-29 12:25:26 -0700164 SkString str(enforcePMColor ? "enforcePM" : "no enforcePM");
Mike Reed4de2f1f2019-01-05 16:35:13 -0500165 canvas->drawString(str, 0, font.getSize(), font, SkPaint());
fmalitaad7cb812016-09-29 12:25:26 -0700166 }
167 canvas->translate(0, HH + 12);
ericrkaf96fce2015-10-19 14:41:11 -0700168 }
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000169 }
170
171private:
John Stiles7571f9e2020-09-02 22:42:33 -0400172 using INHERITED = GM;
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000173};
Mike Reed26ea9752021-07-08 10:43:31 -0400174DEF_GM( return new ArithmodeGM; )
mike@reedtribe.orge51755f2011-12-10 19:36:56 +0000175
176///////////////////////////////////////////////////////////////////////////////
177
Mike Reed26ea9752021-07-08 10:43:31 -0400178#include "include/effects/SkBlenders.h"
179
John Stilese1452102022-09-15 14:46:28 -0400180class ArithmodeBlenderGM : public skiagm::GM {
181 float fK1, fK2, fK3, fK4;
182 sk_sp<SkImage> fSrc, fDst, fChecker;
183 sk_sp<SkShader> fSrcShader, fDstShader;
184 sk_sp<SkRuntimeEffect> fRuntimeEffect;
Mike Reed26ea9752021-07-08 10:43:31 -0400185
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000186 SkString getName() const override { return SkString("arithmode_blender"); }
Mike Reed26ea9752021-07-08 10:43:31 -0400187
John Stilesf25d0f12022-09-16 17:27:21 -0400188 static constexpr int W = 200;
189 static constexpr int H = 200;
Mike Reed26ea9752021-07-08 10:43:31 -0400190
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000191 SkISize getISize() override { return {(W + 30) * 2, (H + 30) * 4}; }
Mike Reed26ea9752021-07-08 10:43:31 -0400192
193 void onOnceBeforeDraw() override {
John Stilese1452102022-09-15 14:46:28 -0400194 // Prepare a runtime effect for this blend.
195 static constexpr char kShader[] = R"(
196 uniform shader srcImage;
197 uniform shader dstImage;
198 uniform blender arithBlend;
199 half4 main(float2 xy) {
200 return arithBlend.eval(srcImage.eval(xy), dstImage.eval(xy));
201 }
202 )";
203 auto [effect, error] = SkRuntimeEffect::MakeForShader(SkString(kShader));
204 SkASSERT(effect);
205 fRuntimeEffect = effect;
206
207 // Start with interesting K-values, in case we're drawn without calling onAnimate().
Mike Reed26ea9752021-07-08 10:43:31 -0400208 fK1 = -0.25f;
209 fK2 = 0.25f;
210 fK3 = 0.25f;
211 fK4 = 0;
212
213 fSrc = make_src(W, H);
214 fDst = make_dst(W, H);
John Stilese1452102022-09-15 14:46:28 -0400215 fSrcShader = fSrc->makeShader(SkTileMode::kClamp, SkTileMode::kClamp, SkSamplingOptions());
216 fDstShader = fDst->makeShader(SkTileMode::kClamp, SkTileMode::kClamp, SkSamplingOptions());
Mike Reed26ea9752021-07-08 10:43:31 -0400217
Mike Reeda2a85e42021-07-26 15:29:43 -0400218 fChecker = ToolUtils::create_checkerboard_image(W, H, 0xFFBBBBBB, 0xFFEEEEEE, 8);
Mike Reed26ea9752021-07-08 10:43:31 -0400219 }
220
221 bool onAnimate(double nanos) override {
222 double theta = nanos * 1e-6 * 0.001;
223 fK1 = sin(theta + 0) * 0.25;
224 fK2 = cos(theta + 1) * 0.25;
225 fK3 = sin(theta + 2) * 0.25;
226 fK4 = 0.5;
227 return true;
228 }
229
230 void onDraw(SkCanvas* canvas) override {
231 const SkRect rect = SkRect::MakeWH(W, H);
232
233 canvas->drawImage(fSrc, 10, 10);
John Stilesf25d0f12022-09-16 17:27:21 -0400234 canvas->drawImage(fDst, 10, 10 + H + 10);
Mike Reed26ea9752021-07-08 10:43:31 -0400235
John Stilese1452102022-09-15 14:46:28 -0400236 SkSamplingOptions sampling;
237 sk_sp<SkBlender> blender = SkBlenders::Arithmetic(fK1, fK2, fK3, fK4,
238 /*enforcePremul=*/true);
John Stilesf25d0f12022-09-16 17:27:21 -0400239 canvas->translate(10 + W + 10, 10);
Mike Reed26ea9752021-07-08 10:43:31 -0400240
John Stilese1452102022-09-15 14:46:28 -0400241 // All three images drawn below should appear identical.
Mike Reeda2a85e42021-07-26 15:29:43 -0400242 // Draw via blend step
John Stilesf25d0f12022-09-16 17:27:21 -0400243 SkPaint blenderPaint;
John Stilese1452102022-09-15 14:46:28 -0400244 canvas->drawImage(fChecker, 0, 0);
Mike Reed26ea9752021-07-08 10:43:31 -0400245 canvas->saveLayer(&rect, nullptr);
246 canvas->drawImage(fDst, 0, 0);
John Stilesf25d0f12022-09-16 17:27:21 -0400247 blenderPaint.setBlender(blender);
248 canvas->drawImage(fSrc, 0, 0, sampling, &blenderPaint);
Mike Reed26ea9752021-07-08 10:43:31 -0400249 canvas->restore();
Mike Reeda2a85e42021-07-26 15:29:43 -0400250
John Stilesf25d0f12022-09-16 17:27:21 -0400251 canvas->translate(0, 10 + H);
Mike Reeda2a85e42021-07-26 15:29:43 -0400252
John Stilesf25d0f12022-09-16 17:27:21 -0400253 // Draw via SkImageFilters::Blend (should appear the same as above)
254 SkPaint imageFilterPaint;
John Stilese1452102022-09-15 14:46:28 -0400255 canvas->drawImage(fChecker, 0, 0);
John Stilesf25d0f12022-09-16 17:27:21 -0400256 imageFilterPaint.setImageFilter(
John Stilese1452102022-09-15 14:46:28 -0400257 SkImageFilters::Blend(blender,
258 /*background=*/nullptr,
259 /*foreground=*/SkImageFilters::Image(fSrc, sampling)));
John Stilesf25d0f12022-09-16 17:27:21 -0400260 canvas->drawImage(fDst, 0, 0, sampling, &imageFilterPaint);
John Stilese1452102022-09-15 14:46:28 -0400261
John Stilesf25d0f12022-09-16 17:27:21 -0400262 canvas->translate(0, 10 + H);
263
264 // Draw via SkShaders::Blend (should still appear the same as above)
265 SkPaint shaderBlendPaint;
266 canvas->drawImage(fChecker, 0, 0);
267 shaderBlendPaint.setShader(SkShaders::Blend(blender, fDstShader, fSrcShader));
268 canvas->drawRect(rect, shaderBlendPaint);
269
270 canvas->translate(0, 10 + H);
John Stilese1452102022-09-15 14:46:28 -0400271
272 // Draw via runtime effect (should still appear the same as above)
273 SkPaint runtimePaint;
274 canvas->drawImage(fChecker, 0, 0);
275 SkRuntimeEffect::ChildPtr children[] = {fSrcShader, fDstShader, blender};
276 runtimePaint.setShader(fRuntimeEffect->makeShader(/*uniforms=*/{}, children));
277 canvas->drawRect(rect, runtimePaint);
Mike Reed26ea9752021-07-08 10:43:31 -0400278 }
279
280private:
281 using INHERITED = GM;
282};
John Stilese1452102022-09-15 14:46:28 -0400283DEF_GM( return new ArithmodeBlenderGM; )