blob: 1f244e950c55c53b02a268d4619e62f8a317555c [file] [log] [blame]
Florin Malita325ea322018-04-04 14:17:30 -04001/*
2 * Copyright 2018 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"
Mike Reedac9f0c92020-12-23 10:11:33 -05009#include "include/core/SkBitmap.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040010#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"
13#include "include/core/SkImage.h"
14#include "include/core/SkImageInfo.h"
15#include "include/core/SkMatrix.h"
16#include "include/core/SkPaint.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"
21#include "include/core/SkSurface.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040022#include "include/core/SkTypes.h"
Michael Ludwig5c51f5f2020-06-02 12:55:10 -040023#include "include/effects/SkGradientShader.h"
Kevin Lubick8b741882023-10-06 11:41:38 -040024#include "tools/DecodeUtils.h"
Kevin Lubick9b028372023-10-05 15:04:54 -040025#include "tools/GpuToolUtils.h"
Michael Ludwig5c51f5f2020-06-02 12:55:10 -040026#include "tools/Resources.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "tools/ToolUtils.h"
Brian Salomonf87ffc72022-11-02 15:30:32 -040028#include "tools/timer/TimeUtils.h"
Florin Malita325ea322018-04-04 14:17:30 -040029
30static sk_sp<SkImage> make_image(SkCanvas* rootCanvas) {
31 static constexpr SkScalar kSize = 50;
32 SkImageInfo info = SkImageInfo::MakeN32Premul(kSize, kSize);
Kevin Lubick5c93acf2023-05-09 12:11:43 -040033 auto surface(SkSurfaces::Raster(info));
Florin Malita325ea322018-04-04 14:17:30 -040034
35 SkPaint p;
36 p.setAntiAlias(true);
37 p.setColor(SK_ColorGREEN);
38
39 surface->getCanvas()->drawCircle(kSize / 2, kSize / 2, kSize / 2, p);
40
41 p.setStyle(SkPaint::kStroke_Style);
42 p.setColor(SK_ColorRED);
43 surface->getCanvas()->drawLine(kSize * .25f, kSize * .50f, kSize * .75f, kSize * .50f, p);
44 surface->getCanvas()->drawLine(kSize * .50f, kSize * .25f, kSize * .50f, kSize * .75f, p);
45
Robert Phillipseae18482022-05-03 09:25:38 -040046 sk_sp<SkImage> img = surface->makeImageSnapshot();
47 return ToolUtils::MakeTextureImage(rootCanvas, std::move(img));
Florin Malita325ea322018-04-04 14:17:30 -040048}
49
50DEF_SIMPLE_GM(localmatrixshader_nested, canvas, 450, 1200) {
51 auto image = make_image(canvas);
Robert Phillipsd3600f82022-05-03 10:13:10 -040052 if (!image) {
53 return;
54 }
Florin Malita325ea322018-04-04 14:17:30 -040055
56 using FactoryT = sk_sp<SkShader> (*)(const sk_sp<SkImage>&,
57 const SkMatrix& inner,
58 const SkMatrix& outer);
59 static const FactoryT gFactories[] = {
60 // SkLocalMatrixShader(SkImageShader(inner), outer)
61 [](const sk_sp<SkImage>& img, const SkMatrix& inner, const SkMatrix& outer) {
Mike Reedb612b6c2020-12-08 21:58:35 -050062 return img->makeShader(SkSamplingOptions(), inner)->makeWithLocalMatrix(outer);
Florin Malita325ea322018-04-04 14:17:30 -040063 },
64
65 // SkLocalMatrixShader(SkLocalMatrixShader(SkImageShader(I), inner), outer)
66 [](const sk_sp<SkImage>& img, const SkMatrix& inner, const SkMatrix& outer) {
Mike Reedb612b6c2020-12-08 21:58:35 -050067 return img->makeShader(SkSamplingOptions())->makeWithLocalMatrix(inner)->makeWithLocalMatrix(outer);
Florin Malita325ea322018-04-04 14:17:30 -040068 },
69
70 // SkLocalMatrixShader(SkComposeShader(SkImageShader(inner)), outer)
71 [](const sk_sp<SkImage>& img, const SkMatrix& inner, const SkMatrix& outer) {
Mike Reedc8bea7d2019-04-09 13:55:36 -040072 return SkShaders::Blend(SkBlendMode::kSrcOver,
73 SkShaders::Color(SK_ColorTRANSPARENT),
Mike Reedb612b6c2020-12-08 21:58:35 -050074 img->makeShader(SkSamplingOptions(), inner))
Florin Malita325ea322018-04-04 14:17:30 -040075 ->makeWithLocalMatrix(outer);
76 },
77
78 // SkLocalMatrixShader(SkComposeShader(SkLocalMatrixShader(SkImageShader(I), inner)), outer)
79 [](const sk_sp<SkImage>& img, const SkMatrix& inner, const SkMatrix& outer) {
Mike Reedc8bea7d2019-04-09 13:55:36 -040080 return SkShaders::Blend(SkBlendMode::kSrcOver,
81 SkShaders::Color(SK_ColorTRANSPARENT),
Mike Reedb612b6c2020-12-08 21:58:35 -050082 img->makeShader(SkSamplingOptions())->makeWithLocalMatrix(inner))
Florin Malita325ea322018-04-04 14:17:30 -040083 ->makeWithLocalMatrix(outer);
84 },
85 };
86
Brian Salomonf87ffc72022-11-02 15:30:32 -040087 static const auto outer = SkMatrix::Scale(2, 2),
88 inner = SkMatrix::Translate(20, 20);
Florin Malita325ea322018-04-04 14:17:30 -040089
90 SkPaint border;
91 border.setAntiAlias(true);
92 border.setStyle(SkPaint::kStroke_Style);
93
94 auto rect = SkRect::Make(image->bounds());
Brian Salomonf87ffc72022-11-02 15:30:32 -040095 SkAssertResult(SkMatrix::Concat(outer, inner).mapRect(&rect));
Florin Malita325ea322018-04-04 14:17:30 -040096
97 const auto drawColumn = [&]() {
98 SkAutoCanvasRestore acr(canvas, true);
99 for (const auto& f : gFactories) {
100 SkPaint p;
101 p.setShader(f(image, inner, outer));
102
103 canvas->drawRect(rect, p);
104 canvas->drawRect(rect, border);
105
106 canvas->translate(0, rect.height() * 1.5f);
107 }
108 };
109
110 drawColumn();
111
112 {
113 SkAutoCanvasRestore acr(canvas, true);
Herb Derbyc37b3862022-06-21 09:49:17 -0400114 canvas->translate(0, rect.height() * std::size(gFactories) * 1.5f);
Florin Malita325ea322018-04-04 14:17:30 -0400115 drawColumn();
116 }
117
118 canvas->translate(rect.width() * 1.5f, 0);
119 canvas->scale(2, 2);
120 drawColumn();
121}
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400122
123DEF_SIMPLE_GM(localmatrixshader_persp, canvas, 542, 266) {
Kevin Lubick8b741882023-10-06 11:41:38 -0400124 auto image = ToolUtils::GetResourceAsImage("images/yellow_rose.png");
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400125
126 SkBitmap downsized;
127 downsized.allocPixels(image->imageInfo().makeWH(128, 128));
Mike Reed5ec22382021-01-14 21:59:01 -0500128 image->scalePixels(downsized.pixmap(), SkSamplingOptions(SkFilterMode::kLinear));
Mike Reedac9f0c92020-12-23 10:11:33 -0500129 image = downsized.asImage();
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400130 SkRect imgRect = SkRect::MakeIWH(image->width(), image->height());
131
132 // scale matrix
133 SkMatrix scale = SkMatrix::Scale(1.f / 5.f, 1.f / 5.f);
134
135 // perspective matrix
136 SkPoint src[4];
137 imgRect.toQuad(src);
138 SkPoint dst[4] = {{0, 10.f},
139 {image->width() + 28.f, -100.f},
140 {image->width() - 28.f, image->height() + 100.f},
141 {0.f, image->height() - 10.f}};
142 SkMatrix persp;
143 SkAssertResult(persp.setPolyToPoly(src, dst, 4));
144
145 // combined persp * scale
146 SkMatrix perspScale = SkMatrix::Concat(persp, scale);
147
148 auto draw = [&](sk_sp<SkShader> shader, bool applyPerspToCTM) {
149 canvas->save();
150 canvas->clipRect(imgRect);
151 if (applyPerspToCTM) {
152 canvas->concat(persp);
153 }
154 SkPaint imgShaderPaint;
155 imgShaderPaint.setShader(std::move(shader));
156 canvas->drawPaint(imgShaderPaint);
157 canvas->restore();
158
159 canvas->translate(10.f + image->width(), 0.f); // advance
160 };
161
162 // SkImageShader
163 canvas->save();
164 // 4 variants that all attempt to apply sample at persp * scale w/ an image shader
165 // 1. scale provided to SkImage::makeShader(...) but drawn with persp
Mike Reedb612b6c2020-12-08 21:58:35 -0500166 auto s1 = image->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
167 SkSamplingOptions(), &scale);
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400168 draw(s1, true);
169
Brian Salomonf87ffc72022-11-02 15:30:32 -0400170 // 2. scale provided to SkImage::makeShader, then wrapped in persp makeWithLocalMatrix
171 // These post-concat, so it ends up as persp * scale.
Mike Reedb612b6c2020-12-08 21:58:35 -0500172 auto s2 = image->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
Brian Salomonf87ffc72022-11-02 15:30:32 -0400173 SkSamplingOptions(), &scale)
174 ->makeWithLocalMatrix(persp);
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400175 draw(s2, false);
176
177 // 3. Providing pre-computed persp*scale to SkImage::makeShader()
Mike Reedb612b6c2020-12-08 21:58:35 -0500178 auto s3 = image->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat,
179 SkSamplingOptions(), &perspScale);
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400180 draw(s3, false);
181
182 // 4. Providing pre-computed persp*scale to makeWithLocalMatrix
Mike Reedb612b6c2020-12-08 21:58:35 -0500183 auto s4 = image->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat, SkSamplingOptions())
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400184 ->makeWithLocalMatrix(perspScale);
185 draw(s4, false);
186 canvas->restore();
187
188 canvas->translate(0.f, 10.f + image->height()); // advance to next row
189
190 // SkGradientShader
191 const SkColor kGradColors[] = { SK_ColorBLACK, SK_ColorTRANSPARENT };
192 canvas->save();
193 // 1. scale provided to Make, drawn with persp
194 auto g1 = SkGradientShader::MakeRadial({imgRect.centerX(), imgRect.centerY()},
195 imgRect.width() / 2.f, kGradColors, nullptr, 2,
196 SkTileMode::kRepeat, 0, &scale);
197 draw(g1, true);
198
Brian Salomonf87ffc72022-11-02 15:30:32 -0400199 // 2. scale provided to Make, then wrapped with makeWithLocalMatrix (post-concat as before).
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400200 auto g2 = SkGradientShader::MakeRadial({imgRect.centerX(), imgRect.centerY()},
201 imgRect.width() / 2.f, kGradColors, nullptr, 2,
Brian Salomonf87ffc72022-11-02 15:30:32 -0400202 SkTileMode::kRepeat, 0, &scale)
203 ->makeWithLocalMatrix(persp);
Michael Ludwig5c51f5f2020-06-02 12:55:10 -0400204 draw(g2, false);
205
206 // 3. Provide per-computed persp*scale to Make
207 auto g3 = SkGradientShader::MakeRadial({imgRect.centerX(), imgRect.centerY()},
208 imgRect.width() / 2.f, kGradColors, nullptr, 2,
209 SkTileMode::kRepeat, 0, &perspScale);
210 draw(g3, false);
211
212 // 4. Providing pre-computed persp*scale to makeWithLocalMatrix
213 auto g4 = SkGradientShader::MakeRadial({imgRect.centerX(), imgRect.centerY()},
214 imgRect.width() / 2.f, kGradColors, nullptr, 2,
215 SkTileMode::kRepeat)
216 ->makeWithLocalMatrix(perspScale);
217 draw(g4, false);
218 canvas->restore();
219}
Brian Salomonf87ffc72022-11-02 15:30:32 -0400220
221namespace skiagm {
222class LocalMatrixOrder : public GM {
223public:
224 LocalMatrixOrder() {}
225
226protected:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000227 SkString getName() const override { return SkString("localmatrix_order"); }
Brian Salomonf87ffc72022-11-02 15:30:32 -0400228
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000229 SkISize getISize() override { return SkISize::Make(500, 500); }
Brian Salomonf87ffc72022-11-02 15:30:32 -0400230
231 void onOnceBeforeDraw() override {
Kevin Lubick8b741882023-10-06 11:41:38 -0400232 auto mandrill = ToolUtils::GetResourceAsImage("images/mandrill_256.png"); // 256x256
233 auto example5 = ToolUtils::GetResourceAsImage("images/example_5.png"); // 128x128
Brian Salomonf87ffc72022-11-02 15:30:32 -0400234
235 auto mshader = mandrill->makeShader(
236 SkTileMode::kRepeat,
237 SkTileMode::kRepeat,
Michael Ludwig2c49e602023-05-30 13:12:19 -0400238 SkFilterMode::kNearest,
Brian Salomonf87ffc72022-11-02 15:30:32 -0400239 SkMatrix::RotateDeg(45, {128, 128})); // rotate about center
240 auto eshader = example5->makeShader(
241 SkTileMode::kRepeat,
242 SkTileMode::kRepeat,
Michael Ludwig2c49e602023-05-30 13:12:19 -0400243 SkFilterMode::kNearest,
Brian Salomonf87ffc72022-11-02 15:30:32 -0400244 SkMatrix::Scale(2, 2)); // make same size as mandrill and...
245 // ... rotate about center
246 eshader = eshader->makeWithLocalMatrix(SkMatrix::RotateDeg(45, {128, 128}));
247
248 // blend the two rotated and aligned images.
249 fShader = SkShaders::Blend(SkBlendMode::kModulate, mshader, eshader);
250 }
251
252 void onDraw(SkCanvas* canvas) override {
253 // Rotate fShader about the canvas center
254 auto center = SkRect::Make(canvas->imageInfo().bounds()).center();
255
256 // viewer can insert a dpi scaling matrix. Make the animation always rotate about the device
257 // center.
258 if (auto ictm = canvas->getTotalMatrix(); ictm.invert(&ictm)) {
259 center = ictm.mapPoint(center);
260 }
261
262 auto shader = fShader->makeWithLocalMatrix(SkMatrix::RotateDeg(fAngle, center));
263
264 SkPaint paint;
265 paint.setShader(shader);
266 canvas->drawPaint(paint);
267 }
268
269 bool onAnimate(double nanos) override {
270 fAngle = TimeUtils::NanosToSeconds(nanos) * 5.f;
271 return true;
272 }
273
274 sk_sp<SkShader> fShader;
275 float fAngle = 0.f;
276};
277
278DEF_GM(return new LocalMatrixOrder;)
279} // namespace skiagm