blob: bc91411c6b1df83ea0ae05644746c3f7192d4490 [file] [log] [blame]
senorblancod4bb9912015-03-20 08:54:32 -07001/*
2 * Copyright 2015 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"
11#include "include/core/SkCanvas.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkColor.h"
13#include "include/core/SkImage.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -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"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050019#include "include/core/SkScalar.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040020#include "include/core/SkShader.h"
21#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 Wagner7fde8e12019-05-01 17:28:53 -040024#include "include/core/SkTileMode.h"
25#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050026#include "include/effects/SkGradientShader.h"
Michael Ludwig898bbfa2019-08-02 15:21:23 -040027#include "include/effects/SkImageFilters.h"
Michael Ludwig1c07aa72020-06-11 10:57:37 -040028#include "tools/Resources.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050029#include "tools/ToolUtils.h"
Michael Ludwig1c07aa72020-06-11 10:57:37 -040030#include "tools/timer/TimeUtils.h"
senorblancod4bb9912015-03-20 08:54:32 -070031
Ben Wagner7fde8e12019-05-01 17:28:53 -040032#include <utility>
33
senorblancod4bb9912015-03-20 08:54:32 -070034namespace skiagm {
35
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.
39
reed9ce9d672016-03-17 10:51:11 -070040static sk_sp<SkImage> make_gradient_circle(int width, int height) {
fmalita2f5891e2015-09-25 09:15:55 -070041 SkScalar x = SkIntToScalar(width / 2);
42 SkScalar y = SkIntToScalar(height / 2);
Brian Osman116b33e2020-02-05 13:34:09 -050043 SkScalar radius = std::min(x, y) * 0.8f;
fmalita2f5891e2015-09-25 09:15:55 -070044
reede8f30622016-03-23 18:59:25 -070045 auto surface(SkSurface::MakeRasterN32Premul(width, height));
fmalita2f5891e2015-09-25 09:15:55 -070046 SkCanvas* canvas = surface->getCanvas();
47
48 canvas->clear(0x00000000);
49 SkColor colors[2];
50 colors[0] = SK_ColorWHITE;
51 colors[1] = SK_ColorBLACK;
fmalita2f5891e2015-09-25 09:15:55 -070052 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -070053 paint.setShader(SkGradientShader::MakeRadial(SkPoint::Make(x, y), radius, colors, nullptr, 2,
Mike Reedfae8fce2019-04-03 10:27:45 -040054 SkTileMode::kClamp));
fmalita2f5891e2015-09-25 09:15:55 -070055 canvas->drawCircle(x, y, radius, paint);
56
reed9ce9d672016-03-17 10:51:11 -070057 return surface->makeImageSnapshot();
fmalita2f5891e2015-09-25 09:15:55 -070058}
59
senorblancod4bb9912015-03-20 08:54:32 -070060class ImageFiltersTransformedGM : public GM {
61public:
62 ImageFiltersTransformedGM() {
63 this->setBGColor(SK_ColorBLACK);
64 }
65
66protected:
67
mtklein36352bf2015-03-25 18:17:31 -070068 SkString onShortName() override { return SkString("imagefilterstransformed"); }
senorblancod4bb9912015-03-20 08:54:32 -070069
mtklein36352bf2015-03-25 18:17:31 -070070 SkISize onISize() override { return SkISize::Make(420, 240); }
senorblancod4bb9912015-03-20 08:54:32 -070071
mtklein36352bf2015-03-25 18:17:31 -070072 void onOnceBeforeDraw() override {
reed9ce9d672016-03-17 10:51:11 -070073 fCheckerboard = SkImage::MakeFromBitmap(
Mike Kleinea3f0142019-03-20 11:12:10 -050074 ToolUtils::create_checkerboard_bitmap(64, 64, 0xFFA0A0A0, 0xFF404040, 8));
reed9ce9d672016-03-17 10:51:11 -070075 fGradientCircle = make_gradient_circle(64, 64);
senorblancod4bb9912015-03-20 08:54:32 -070076 }
77
mtklein36352bf2015-03-25 18:17:31 -070078 void onDraw(SkCanvas* canvas) override {
Michael Ludwig898bbfa2019-08-02 15:21:23 -040079 sk_sp<SkImageFilter> gradient(SkImageFilters::Image(fGradientCircle));
80 sk_sp<SkImageFilter> checkerboard(SkImageFilters::Image(fCheckerboard));
robertphillipsc4169122016-04-06 08:40:59 -070081 sk_sp<SkImageFilter> filters[] = {
Michael Ludwig898bbfa2019-08-02 15:21:23 -040082 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),
senorblancod4bb9912015-03-20 08:54:32 -070088 };
89
90 const SkScalar margin = SkIntToScalar(20);
91 const SkScalar size = SkIntToScalar(60);
92
93 for (size_t j = 0; j < 3; j++) {
94 canvas->save();
95 canvas->translate(margin, 0);
96 for (size_t i = 0; i < SK_ARRAY_COUNT(filters); ++i) {
97 SkPaint paint;
98 paint.setColor(SK_ColorWHITE);
99 paint.setImageFilter(filters[i]);
100 paint.setAntiAlias(true);
101 canvas->save();
102 canvas->translate(size * SK_ScalarHalf, size * SK_ScalarHalf);
103 canvas->scale(SkDoubleToScalar(0.8), SkDoubleToScalar(0.8));
104 if (j == 1) {
105 canvas->rotate(SkIntToScalar(45));
106 } else if (j == 2) {
107 canvas->skew(SkDoubleToScalar(0.5), SkDoubleToScalar(0.2));
108 }
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);
112 canvas->restore();
113 canvas->translate(size + margin, 0);
114 }
115 canvas->restore();
116 canvas->translate(0, size + margin);
117 }
senorblancod4bb9912015-03-20 08:54:32 -0700118 }
119
120private:
reed9ce9d672016-03-17 10:51:11 -0700121 sk_sp<SkImage> fCheckerboard;
122 sk_sp<SkImage> fGradientCircle;
senorblancod4bb9912015-03-20 08:54:32 -0700123 typedef GM INHERITED;
124};
reed51dece32016-04-19 14:02:52 -0700125DEF_GM( return new ImageFiltersTransformedGM; )
126}
senorblancod4bb9912015-03-20 08:54:32 -0700127
128//////////////////////////////////////////////////////////////////////////////
129
reed51dece32016-04-19 14:02:52 -0700130DEF_SIMPLE_GM(rotate_imagefilter, canvas, 500, 500) {
131 SkPaint paint;
senorblancod4bb9912015-03-20 08:54:32 -0700132
reed51dece32016-04-19 14:02:52 -0700133 const SkRect r = SkRect::MakeXYWH(50, 50, 100, 100);
134
135 sk_sp<SkImageFilter> filters[] = {
136 nullptr,
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400137 SkImageFilters::Blur(6, 0, nullptr),
138 SkImageFilters::Xfermode(SkBlendMode::kSrcOver, nullptr),
reed51dece32016-04-19 14:02:52 -0700139 };
140
141 for (auto& filter : filters) {
142 paint.setAntiAlias(false);
143 paint.setImageFilter(filter);
144
145 canvas->save();
146
147 canvas->drawRect(r, paint);
148
149 canvas->translate(150, 0);
150 canvas->save();
bungeman7438bfc2016-07-12 15:01:19 -0700151 canvas->rotate(30, 100, 100);
reed51dece32016-04-19 14:02:52 -0700152 canvas->drawRect(r, paint);
153 canvas->restore();
154
155 paint.setAntiAlias(true);
156 canvas->translate(150, 0);
157 canvas->save();
bungeman7438bfc2016-07-12 15:01:19 -0700158 canvas->rotate(30, 100, 100);
reed51dece32016-04-19 14:02:52 -0700159 canvas->drawRect(r, paint);
160 canvas->restore();
161
162 canvas->restore();
163 canvas->translate(0, 150);
164 }
senorblancod4bb9912015-03-20 08:54:32 -0700165}
Michael Ludwig1c07aa72020-06-11 10:57:37 -0400166
167class ImageFilterMatrixWLocalMatrix : public skiagm::GM {
168public:
169
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) {}
173
174protected:
175 SkString onShortName() override {
176 return SkString("imagefilter_matrix_localmatrix");
177 }
178
179 SkISize onISize() override {
180 return SkISize::Make(512, 512);
181 }
182
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);
187 return true;
188 }
189
190 void onOnceBeforeDraw() override {
191 fImage = GetResourceAsImage("images/mandrill_256.png");
192 }
193
194 void onDraw(SkCanvas* canvas) override {
195 SkMatrix localMatrix;
196 localMatrix.preTranslate(128, 128);
197 localMatrix.preScale(2.0f, 2.0f);
198
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);
203
204 sk_sp<SkImageFilter> filter =
205 SkImageFilter::MakeMatrixFilter(filterMatrix, kLow_SkFilterQuality, nullptr)
206 ->makeWithLocalMatrix(localMatrix);
207
208 SkPaint p;
209 p.setImageFilter(filter);
210 canvas->drawImage(fImage.get(), 128, 128, &p);
211 }
212
213private:
214 SkScalar fDegrees;
215 sk_sp<SkImage> fImage;
216};
217
218DEF_GM(return new ImageFilterMatrixWLocalMatrix();)