mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 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 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "gm/gm.h" |
| 9 | #include "include/core/SkCanvas.h" |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 10 | #include "include/core/SkColor.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 11 | #include "include/core/SkFont.h" |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 12 | #include "include/core/SkFontTypes.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 13 | #include "include/core/SkImage.h" |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 14 | #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 Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 20 | #include "include/core/SkShader.h" |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 21 | #include "include/core/SkSize.h" |
| 22 | #include "include/core/SkString.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 23 | #include "include/core/SkSurface.h" |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 24 | #include "include/core/SkTileMode.h" |
| 25 | #include "include/core/SkTypeface.h" |
| 26 | #include "include/core/SkTypes.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 27 | #include "include/effects/SkGradientShader.h" |
Michael Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 28 | #include "include/effects/SkImageFilters.h" |
Kevin Lubick | c596382 | 2023-07-17 08:23:31 -0400 | [diff] [blame] | 29 | #include "include/effects/SkRuntimeEffect.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 30 | #include "tools/ToolUtils.h" |
Kevin Lubick | e836c3a | 2023-10-20 06:55:35 -0400 | [diff] [blame] | 31 | #include "tools/fonts/FontToolUtils.h" |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 32 | |
Ben Wagner | d1701ba | 2019-04-30 13:44:26 -0400 | [diff] [blame] | 33 | #include <utility> |
| 34 | |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 35 | static sk_sp<SkImage> make_src(int w, int h) { |
Kevin Lubick | 5c93acf | 2023-05-09 12:11:43 -0400 | [diff] [blame] | 36 | sk_sp<SkSurface> surface(SkSurfaces::Raster(SkImageInfo::MakeN32Premul(w, h))); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 37 | SkCanvas* canvas = surface->getCanvas(); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 38 | |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 39 | SkPaint paint; |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 40 | SkPoint pts[] = { {0, 0}, {SkIntToScalar(w), SkIntToScalar(h)} }; |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 41 | SkColor colors[] = { |
senorblanco@chromium.org | 35c733c | 2013-05-28 19:43:05 +0000 | [diff] [blame] | 42 | SK_ColorTRANSPARENT, SK_ColorGREEN, SK_ColorCYAN, |
| 43 | SK_ColorRED, SK_ColorMAGENTA, SK_ColorWHITE, |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 44 | }; |
Herb Derby | c37b386 | 2022-06-21 09:49:17 -0400 | [diff] [blame] | 45 | paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, std::size(colors), |
Mike Reed | fae8fce | 2019-04-03 10:27:45 -0400 | [diff] [blame] | 46 | SkTileMode::kClamp)); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 47 | canvas->drawPaint(paint); |
| 48 | return surface->makeImageSnapshot(); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 49 | } |
| 50 | |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 51 | static sk_sp<SkImage> make_dst(int w, int h) { |
Kevin Lubick | 5c93acf | 2023-05-09 12:11:43 -0400 | [diff] [blame] | 52 | sk_sp<SkSurface> surface(SkSurfaces::Raster(SkImageInfo::MakeN32Premul(w, h))); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 53 | SkCanvas* canvas = surface->getCanvas(); |
| 54 | |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 55 | SkPaint paint; |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 56 | SkPoint pts[] = { {0, SkIntToScalar(h)}, {SkIntToScalar(w), 0} }; |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 57 | SkColor colors[] = { |
caryclark | dfcb7ab | 2015-07-17 09:39:16 -0700 | [diff] [blame] | 58 | SK_ColorBLUE, SK_ColorYELLOW, SK_ColorBLACK, SK_ColorGREEN, |
Mike Klein | d46dce3 | 2018-08-16 10:17:03 -0400 | [diff] [blame] | 59 | SK_ColorGRAY, |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 60 | }; |
Herb Derby | c37b386 | 2022-06-21 09:49:17 -0400 | [diff] [blame] | 61 | paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, std::size(colors), |
Mike Reed | fae8fce | 2019-04-03 10:27:45 -0400 | [diff] [blame] | 62 | SkTileMode::kClamp)); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 63 | canvas->drawPaint(paint); |
| 64 | return surface->makeImageSnapshot(); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 65 | } |
| 66 | |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 67 | static void show_k_text(SkCanvas* canvas, SkScalar x, SkScalar y, const SkScalar k[]) { |
Kevin Lubick | e836c3a | 2023-10-20 06:55:35 -0400 | [diff] [blame] | 68 | SkFont font(ToolUtils::DefaultPortableTypeface(), 24); |
Mike Reed | 17c574a | 2018-12-12 18:10:38 -0500 | [diff] [blame] | 69 | font.setEdging(SkFont::Edging::kAntiAlias); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 70 | SkPaint paint; |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 71 | paint.setAntiAlias(true); |
| 72 | for (int i = 0; i < 4; ++i) { |
| 73 | SkString str; |
| 74 | str.appendScalar(k[i]); |
Ben Wagner | 51e15a6 | 2019-05-07 15:38:46 -0400 | [diff] [blame] | 75 | SkScalar width = font.measureText(str.c_str(), str.size(), SkTextEncoding::kUTF8); |
Hal Canary | 89a644b | 2019-01-07 09:36:09 -0500 | [diff] [blame] | 76 | canvas->drawString(str, x, y + font.getSize(), font, paint); |
reed@google.com | eb0fa29 | 2011-12-12 22:01:06 +0000 | [diff] [blame] | 77 | x += width + SkIntToScalar(10); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 78 | } |
| 79 | } |
| 80 | |
| 81 | class ArithmodeGM : public skiagm::GM { |
Leandro Lovisolo | 24fa211 | 2023-08-15 19:05:17 +0000 | [diff] [blame] | 82 | SkString getName() const override { return SkString("arithmode"); } |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 83 | |
Leandro Lovisolo | 8f02388 | 2023-08-15 21:13:52 +0000 | [diff] [blame] | 84 | SkISize getISize() override { return {640, 572}; } |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 85 | |
Hal Canary | bd865e2 | 2019-07-18 11:51:19 -0400 | [diff] [blame] | 86 | void onDraw(SkCanvas* canvas) override { |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 87 | 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 Ludwig | 642f218 | 2023-05-30 20:57:39 -0400 | [diff] [blame] | 92 | sk_sp<SkImageFilter> srcFilter = SkImageFilters::Image(src, {SkFilterMode::kLinear}); |
| 93 | sk_sp<SkImageFilter> dstFilter = SkImageFilters::Image(dst, {SkFilterMode::kLinear}); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 94 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 95 | constexpr SkScalar one = SK_Scalar1; |
| 96 | constexpr SkScalar K[] = { |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 97 | 0, 0, 0, 0, |
reed@google.com | eb0fa29 | 2011-12-12 22:01:06 +0000 | [diff] [blame] | 98 | 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.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 108 | }; |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 109 | |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 110 | const SkScalar* k = K; |
Herb Derby | c37b386 | 2022-06-21 09:49:17 -0400 | [diff] [blame] | 111 | const SkScalar* stop = k + std::size(K); |
Michael Ludwig | bbfb84f | 2023-05-18 16:01:05 -0400 | [diff] [blame] | 112 | // 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. |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 115 | const SkRect rect = SkRect::MakeWH(WW, HH); |
| 116 | SkScalar gap = SkIntToScalar(WW + 20); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 117 | while (k < stop) { |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 118 | { |
| 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 Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 125 | paint.setImageFilter(SkImageFilters::Arithmetic(k[0], k[1], k[2], k[3], true, |
Michael Ludwig | bbfb84f | 2023-05-18 16:01:05 -0400 | [diff] [blame] | 126 | dstFilter, srcFilter, rect)); |
| 127 | canvas->saveLayer(nullptr, &paint); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 128 | canvas->restore(); |
| 129 | |
| 130 | canvas->translate(gap, 0); |
| 131 | show_k_text(canvas, 0, 0, k); |
| 132 | } |
| 133 | |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 134 | k += 4; |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 135 | canvas->translate(0, HH + 12); |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 136 | } |
ericrk | af96fce | 2015-10-19 14:41:11 -0700 | [diff] [blame] | 137 | |
| 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); |
ericrk | af96fce | 2015-10-19 14:41:11 -0700 | [diff] [blame] | 145 | |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 146 | { |
| 147 | SkAutoCanvasRestore acr(canvas, true); |
| 148 | canvas->translate(gap, 0); |
| 149 | canvas->drawImage(dst, 0, 0); |
| 150 | canvas->translate(gap, 0); |
ericrk | af96fce | 2015-10-19 14:41:11 -0700 | [diff] [blame] | 151 | |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 152 | sk_sp<SkImageFilter> bg = |
Michael Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 153 | SkImageFilters::Arithmetic(0, 0, -one / 2, 1, enforcePMColor, dstFilter, |
| 154 | nullptr, nullptr); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 155 | SkPaint p; |
Michael Ludwig | 898bbfa | 2019-08-02 15:21:23 -0400 | [diff] [blame] | 156 | p.setImageFilter(SkImageFilters::Arithmetic(0, one / 2, -one, 1, true, |
Michael Ludwig | bbfb84f | 2023-05-18 16:01:05 -0400 | [diff] [blame] | 157 | std::move(bg), dstFilter, rect)); |
| 158 | canvas->saveLayer(nullptr, &p); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 159 | canvas->restore(); |
| 160 | canvas->translate(gap, 0); |
| 161 | |
| 162 | // Label |
Kevin Lubick | e836c3a | 2023-10-20 06:55:35 -0400 | [diff] [blame] | 163 | SkFont font(ToolUtils::DefaultPortableTypeface(), 24); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 164 | SkString str(enforcePMColor ? "enforcePM" : "no enforcePM"); |
Mike Reed | 4de2f1f | 2019-01-05 16:35:13 -0500 | [diff] [blame] | 165 | canvas->drawString(str, 0, font.getSize(), font, SkPaint()); |
fmalita | ad7cb81 | 2016-09-29 12:25:26 -0700 | [diff] [blame] | 166 | } |
| 167 | canvas->translate(0, HH + 12); |
ericrk | af96fce | 2015-10-19 14:41:11 -0700 | [diff] [blame] | 168 | } |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 169 | } |
| 170 | |
| 171 | private: |
John Stiles | 7571f9e | 2020-09-02 22:42:33 -0400 | [diff] [blame] | 172 | using INHERITED = GM; |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 173 | }; |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 174 | DEF_GM( return new ArithmodeGM; ) |
mike@reedtribe.org | e51755f | 2011-12-10 19:36:56 +0000 | [diff] [blame] | 175 | |
| 176 | /////////////////////////////////////////////////////////////////////////////// |
| 177 | |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 178 | #include "include/effects/SkBlenders.h" |
| 179 | |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 180 | class 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 Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 185 | |
Leandro Lovisolo | 24fa211 | 2023-08-15 19:05:17 +0000 | [diff] [blame] | 186 | SkString getName() const override { return SkString("arithmode_blender"); } |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 187 | |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 188 | static constexpr int W = 200; |
| 189 | static constexpr int H = 200; |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 190 | |
Leandro Lovisolo | 8f02388 | 2023-08-15 21:13:52 +0000 | [diff] [blame] | 191 | SkISize getISize() override { return {(W + 30) * 2, (H + 30) * 4}; } |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 192 | |
| 193 | void onOnceBeforeDraw() override { |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 194 | // 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 Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 208 | 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 Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 215 | fSrcShader = fSrc->makeShader(SkTileMode::kClamp, SkTileMode::kClamp, SkSamplingOptions()); |
| 216 | fDstShader = fDst->makeShader(SkTileMode::kClamp, SkTileMode::kClamp, SkSamplingOptions()); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 217 | |
Mike Reed | a2a85e4 | 2021-07-26 15:29:43 -0400 | [diff] [blame] | 218 | fChecker = ToolUtils::create_checkerboard_image(W, H, 0xFFBBBBBB, 0xFFEEEEEE, 8); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 219 | } |
| 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 Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 234 | canvas->drawImage(fDst, 10, 10 + H + 10); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 235 | |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 236 | SkSamplingOptions sampling; |
| 237 | sk_sp<SkBlender> blender = SkBlenders::Arithmetic(fK1, fK2, fK3, fK4, |
| 238 | /*enforcePremul=*/true); |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 239 | canvas->translate(10 + W + 10, 10); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 240 | |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 241 | // All three images drawn below should appear identical. |
Mike Reed | a2a85e4 | 2021-07-26 15:29:43 -0400 | [diff] [blame] | 242 | // Draw via blend step |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 243 | SkPaint blenderPaint; |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 244 | canvas->drawImage(fChecker, 0, 0); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 245 | canvas->saveLayer(&rect, nullptr); |
| 246 | canvas->drawImage(fDst, 0, 0); |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 247 | blenderPaint.setBlender(blender); |
| 248 | canvas->drawImage(fSrc, 0, 0, sampling, &blenderPaint); |
Mike Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 249 | canvas->restore(); |
Mike Reed | a2a85e4 | 2021-07-26 15:29:43 -0400 | [diff] [blame] | 250 | |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 251 | canvas->translate(0, 10 + H); |
Mike Reed | a2a85e4 | 2021-07-26 15:29:43 -0400 | [diff] [blame] | 252 | |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 253 | // Draw via SkImageFilters::Blend (should appear the same as above) |
| 254 | SkPaint imageFilterPaint; |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 255 | canvas->drawImage(fChecker, 0, 0); |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 256 | imageFilterPaint.setImageFilter( |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 257 | SkImageFilters::Blend(blender, |
| 258 | /*background=*/nullptr, |
| 259 | /*foreground=*/SkImageFilters::Image(fSrc, sampling))); |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 260 | canvas->drawImage(fDst, 0, 0, sampling, &imageFilterPaint); |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 261 | |
John Stiles | f25d0f1 | 2022-09-16 17:27:21 -0400 | [diff] [blame] | 262 | 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 Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 271 | |
| 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 Reed | 26ea975 | 2021-07-08 10:43:31 -0400 | [diff] [blame] | 278 | } |
| 279 | |
| 280 | private: |
| 281 | using INHERITED = GM; |
| 282 | }; |
John Stiles | e145210 | 2022-09-15 14:46:28 -0400 | [diff] [blame] | 283 | DEF_GM( return new ArithmodeBlenderGM; ) |