2 * Copyright 2015 Google Inc.
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
9 #include "include/core/SkBitmap.h"
10 #include "include/core/SkBlendMode.h"
11 #include "include/core/SkCanvas.h"
12 #include "include/core/SkColor.h"
13 #include "include/core/SkImage.h"
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"
20 #include "include/core/SkShader.h"
21 #include "include/core/SkSize.h"
22 #include "include/core/SkString.h"
23 #include "include/core/SkSurface.h"
24 #include "include/core/SkTileMode.h"
25 #include "include/core/SkTypes.h"
26 #include "include/effects/SkGradientShader.h"
27 #include "include/effects/SkImageFilters.h"
28 #include "tools/Resources.h"
29 #include "tools/ToolUtils.h"
30 #include "tools/timer/TimeUtils.h"
36 // This GM draws image filters with a CTM containing shearing / rotation.
37 // It checks that the scale portion of the CTM is correctly extracted
38 // and applied to the image inputs separately from the non-scale portion.
40 static sk_sp<SkImage> make_gradient_circle(int width, int height) {
41 SkScalar x = SkIntToScalar(width / 2);
42 SkScalar y = SkIntToScalar(height / 2);
43 SkScalar radius = std::min(x, y) * 0.8f;
45 auto surface(SkSurface::MakeRasterN32Premul(width, height));
46 SkCanvas* canvas = surface->getCanvas();
48 canvas->clear(0x00000000);
50 colors[0] = SK_ColorWHITE;
51 colors[1] = SK_ColorBLACK;
53 paint.setShader(SkGradientShader::MakeRadial(SkPoint::Make(x, y), radius, colors, nullptr, 2,
55 canvas->drawCircle(x, y, radius, paint);
57 return surface->makeImageSnapshot();
60 class ImageFiltersTransformedGM : public GM {
62 ImageFiltersTransformedGM() {
63 this->setBGColor(SK_ColorBLACK);
68 SkString onShortName() override { return SkString("imagefilterstransformed"); }
70 SkISize onISize() override { return SkISize::Make(420, 240); }
72 void onOnceBeforeDraw() override {
74 ToolUtils::create_checkerboard_image(64, 64, 0xFFA0A0A0, 0xFF404040, 8);
75 fGradientCircle = make_gradient_circle(64, 64);
78 void onDraw(SkCanvas* canvas) override {
79 sk_sp<SkImageFilter> gradient(SkImageFilters::Image(fGradientCircle));
80 sk_sp<SkImageFilter> checkerboard(SkImageFilters::Image(fCheckerboard));
81 sk_sp<SkImageFilter> filters[] = {
82 SkImageFilters::Blur(12, 0, nullptr),
83 SkImageFilters::DropShadow(0, 15, 8, 0, SK_ColorGREEN, nullptr),
84 SkImageFilters::DisplacementMap(SkColorChannel::kR, SkColorChannel::kR, 12,
85 std::move(gradient), checkerboard),
86 SkImageFilters::Dilate(2, 2, checkerboard),
87 SkImageFilters::Erode(2, 2, checkerboard),
90 const SkScalar margin = SkIntToScalar(20);
91 const SkScalar size = SkIntToScalar(60);
93 for (size_t j = 0; j < 3; j++) {
95 canvas->translate(margin, 0);
96 for (size_t i = 0; i < SK_ARRAY_COUNT(filters); ++i) {
98 paint.setColor(SK_ColorWHITE);
99 paint.setImageFilter(filters[i]);
100 paint.setAntiAlias(true);
102 canvas->translate(size * SK_ScalarHalf, size * SK_ScalarHalf);
103 canvas->scale(SkDoubleToScalar(0.8), SkDoubleToScalar(0.8));
105 canvas->rotate(SkIntToScalar(45));
107 canvas->skew(SkDoubleToScalar(0.5), SkDoubleToScalar(0.2));
109 canvas->translate(-size * SK_ScalarHalf, -size * SK_ScalarHalf);
110 canvas->drawOval(SkRect::MakeXYWH(0, size * SkDoubleToScalar(0.1),
111 size, size * SkDoubleToScalar(0.6)), paint);
113 canvas->translate(size + margin, 0);
116 canvas->translate(0, size + margin);
121 sk_sp<SkImage> fCheckerboard;
122 sk_sp<SkImage> fGradientCircle;
123 using INHERITED = GM;
125 DEF_GM( return new ImageFiltersTransformedGM; )
126 } // namespace skiagm
128 //////////////////////////////////////////////////////////////////////////////
130 DEF_SIMPLE_GM(rotate_imagefilter, canvas, 500, 500) {
133 const SkRect r = SkRect::MakeXYWH(50, 50, 100, 100);
135 sk_sp<SkImageFilter> filters[] = {
137 SkImageFilters::Blur(6, 0, nullptr),
138 SkImageFilters::Blend(SkBlendMode::kSrcOver, nullptr),
141 for (auto& filter : filters) {
142 paint.setAntiAlias(false);
143 paint.setImageFilter(filter);
147 canvas->drawRect(r, paint);
149 canvas->translate(150, 0);
151 canvas->rotate(30, 100, 100);
152 canvas->drawRect(r, paint);
155 paint.setAntiAlias(true);
156 canvas->translate(150, 0);
158 canvas->rotate(30, 100, 100);
159 canvas->drawRect(r, paint);
163 canvas->translate(0, 150);
167 class ImageFilterMatrixWLocalMatrix : public skiagm::GM {
170 // Start at 132 degrees, since that resulted in a skipped draw before the fix to
171 // SkLocalMatrixImageFilter's computeFastBounds() function.
172 ImageFilterMatrixWLocalMatrix() : fDegrees(132.f) {}
175 SkString onShortName() override {
176 return SkString("imagefilter_matrix_localmatrix");
179 SkISize onISize() override {
180 return SkISize::Make(512, 512);
183 bool onAnimate(double nanos) override {
184 // Animate the rotation angle to ensure the local matrix bounds modifications work
185 // for a variety of transformations.
186 fDegrees = TimeUtils::Scaled(1e-9f * nanos, 360.f);
190 void onOnceBeforeDraw() override {
191 fImage = GetResourceAsImage("images/mandrill_256.png");
194 void onDraw(SkCanvas* canvas) override {
195 SkMatrix localMatrix;
196 localMatrix.preTranslate(128, 128);
197 localMatrix.preScale(2.0f, 2.0f);
199 // This matrix applies a rotate around the center of the image (prior to the simulated
200 // hi-dpi 2x device scale).
201 SkMatrix filterMatrix;
202 filterMatrix.setRotate(fDegrees, 64, 64);
204 sk_sp<SkImageFilter> filter =
205 SkImageFilters::MatrixTransform(filterMatrix,
206 SkSamplingOptions(SkFilterMode::kLinear), nullptr)
207 ->makeWithLocalMatrix(localMatrix);
210 p.setImageFilter(filter);
211 canvas->drawImage(fImage.get(), 128, 128, SkSamplingOptions(), &p);
216 sk_sp<SkImage> fImage;
219 DEF_GM(return new ImageFilterMatrixWLocalMatrix();)
221 class ImageFilterComposedTransform : public skiagm::GM {
224 // Start at 70 degrees since that highlighted the issue in skbug.com/10888
225 ImageFilterComposedTransform() : fDegrees(70.f) {}
228 SkString onShortName() override {
229 return SkString("imagefilter_composed_transform");
232 SkISize onISize() override {
233 return SkISize::Make(512, 512);
236 bool onAnimate(double nanos) override {
237 // Animate the rotation angle to test a variety of transformations
238 fDegrees = TimeUtils::Scaled(1e-9f * nanos, 360.f);
242 void onOnceBeforeDraw() override {
243 fImage = GetResourceAsImage("images/mandrill_256.png");
246 void onDraw(SkCanvas* canvas) override {
247 SkMatrix matrix = SkMatrix::RotateDeg(fDegrees);
248 // All four quadrants should render the same
249 this->drawFilter(canvas, 0.f, 0.f, this->makeDirectFilter(matrix));
250 this->drawFilter(canvas, 256.f, 0.f, this->makeEarlyComposeFilter(matrix));
251 this->drawFilter(canvas, 0.f, 256.f, this->makeLateComposeFilter(matrix));
252 this->drawFilter(canvas, 256.f, 256.f, this->makeFullComposeFilter(matrix));
257 sk_sp<SkImage> fImage;
259 void drawFilter(SkCanvas* canvas, SkScalar tx, SkScalar ty, sk_sp<SkImageFilter> filter) const {
261 p.setImageFilter(std::move(filter));
264 canvas->translate(tx, ty);
265 canvas->clipRect(SkRect::MakeWH(256, 256));
266 canvas->scale(0.5f, 0.5f);
267 canvas->translate(128, 128);
268 canvas->drawImage(fImage, 0, 0, SkSamplingOptions(), &p);
272 // offset(matrix(offset))
273 sk_sp<SkImageFilter> makeDirectFilter(const SkMatrix& matrix) const {
274 SkPoint v = {fImage->width() / 2.f, fImage->height() / 2.f};
275 sk_sp<SkImageFilter> filter = SkImageFilters::Offset(-v.fX, -v.fY, nullptr);
276 filter = SkImageFilters::MatrixTransform(matrix, SkSamplingOptions(SkFilterMode::kLinear),
278 filter = SkImageFilters::Offset(v.fX, v.fY, std::move(filter));
282 // offset(compose(matrix, offset))
283 sk_sp<SkImageFilter> makeEarlyComposeFilter(const SkMatrix& matrix) const {
284 SkPoint v = {fImage->width() / 2.f, fImage->height() / 2.f};
285 sk_sp<SkImageFilter> offset = SkImageFilters::Offset(-v.fX, -v.fY, nullptr);
286 sk_sp<SkImageFilter> filter = SkImageFilters::MatrixTransform(
287 matrix, SkSamplingOptions(SkFilterMode::kLinear), nullptr);
288 filter = SkImageFilters::Compose(std::move(filter), std::move(offset));
289 filter = SkImageFilters::Offset(v.fX, v.fY, std::move(filter));
293 // compose(offset, matrix(offset))
294 sk_sp<SkImageFilter> makeLateComposeFilter(const SkMatrix& matrix) const {
295 SkPoint v = {fImage->width() / 2.f, fImage->height() / 2.f};
296 sk_sp<SkImageFilter> filter = SkImageFilters::Offset(-v.fX, -v.fY, nullptr);
297 filter = SkImageFilters::MatrixTransform(matrix, SkSamplingOptions(SkFilterMode::kLinear),
299 sk_sp<SkImageFilter> offset = SkImageFilters::Offset(v.fX, v.fY, nullptr);
300 filter = SkImageFilters::Compose(std::move(offset), std::move(filter));
304 // compose(offset, compose(matrix, offset))
305 sk_sp<SkImageFilter> makeFullComposeFilter(const SkMatrix& matrix) const {
306 SkPoint v = {fImage->width() / 2.f, fImage->height() / 2.f};
307 sk_sp<SkImageFilter> offset = SkImageFilters::Offset(-v.fX, -v.fY, nullptr);
308 sk_sp<SkImageFilter> filter = SkImageFilters::MatrixTransform(
309 matrix, SkSamplingOptions(SkFilterMode::kLinear), nullptr);
310 filter = SkImageFilters::Compose(std::move(filter), std::move(offset));
311 offset = SkImageFilters::Offset(v.fX, v.fY, nullptr);
312 filter = SkImageFilters::Compose(std::move(offset), std::move(filter));
317 DEF_GM(return new ImageFilterComposedTransform();)
319 // Tests SkImageFilters::Image under tricky matrices (mirrors and perspective)
320 DEF_SIMPLE_GM(imagefilter_transformed_image, canvas, 256, 256) {
321 sk_sp<SkImage> image = GetResourceAsImage("images/color_wheel.png");
322 sk_sp<SkImageFilter> imageFilter = SkImageFilters::Image(image);
324 const SkRect imageRect = SkRect::MakeIWH(image->width(), image->height());
326 SkM44 m1 = SkM44::Translate(0.9f * image->width(), 0.1f * image->height()) *
327 SkM44::Scale(-.8f, .8f);
329 SkM44 m2 = SkM44::RectToRect({-1.f, -1.f, 1.f, 1.f}, imageRect) *
330 SkM44::Perspective(0.01f, 100.f, SK_ScalarPI / 3.f) *
331 SkM44::Translate(0.f, 0.f, -2.f) *
332 SkM44::Rotate({0.f, 1.f, 0.f}, SK_ScalarPI / 6.f) *
333 SkM44::RectToRect(imageRect, {-1.f, -1.f, 1.f, 1.f});
335 SkFont font(ToolUtils::create_portable_typeface());
336 canvas->drawString("Columns should match", 5.f, 15.f, font, SkPaint());
337 canvas->translate(0.f, 10.f);
339 SkSamplingOptions sampling(SkFilterMode::kLinear);
340 for (auto m : {m1, m2}) {
342 for (bool canvasTransform : {false, true}) {
344 canvas->clipRect(imageRect);
346 sk_sp<SkImageFilter> finalFilter;
347 if (canvasTransform) {
349 finalFilter = imageFilter;
351 finalFilter = SkImageFilters::MatrixTransform(m.asM33(), sampling, imageFilter);
355 paint.setImageFilter(std::move(finalFilter));
356 canvas->drawPaint(paint);
359 canvas->translate(image->width(), 0.f);
363 canvas->translate(0.f, image->height());