blob: 2c3eb4c11e74525c21a9a4b108908ef0de1320a8 [file] [log] [blame]
reedafdbedb2015-01-14 13:16:26 -08001/*
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
8#include "gm.h"
9
10#include "Resources.h"
11#include "SampleCode.h"
reed339cdbf2015-02-05 22:02:37 -080012#include "SkAnimTimer.h"
reedafdbedb2015-01-14 13:16:26 -080013#include "SkCanvas.h"
14#include "SkInterpolator.h"
reed73f6cfc2015-11-09 08:36:05 -080015#include "SkGradientShader.h"
16#include "SkData.h"
bungemand3ebb482015-08-05 13:57:49 -070017#include "SkPath.h"
reedafdbedb2015-01-14 13:16:26 -080018#include "SkSurface.h"
19#include "SkRandom.h"
20#include "SkTime.h"
21
reede8f30622016-03-23 18:59:25 -070022static sk_sp<SkSurface> make_surface(SkCanvas* canvas, const SkImageInfo& info) {
23 auto surface = canvas->makeSurface(info);
reed09519db2015-01-17 19:36:08 -080024 if (!surface) {
reede8f30622016-03-23 18:59:25 -070025 surface = SkSurface::MakeRaster(info);
reed09519db2015-01-17 19:36:08 -080026 }
27 return surface;
28}
29
reed5671c5b2016-03-09 14:47:34 -080030static sk_sp<SkShader> make_shader(const SkRect& bounds) {
reed9ce9d672016-03-17 10:51:11 -070031 sk_sp<SkImage> image(GetResourceAsImage("mandrill_128.png"));
32 if (!image) {
bungemanfbd87d62015-11-10 13:29:43 -080033 return nullptr;
34 }
reed5671c5b2016-03-09 14:47:34 -080035 return image->makeShader(SkShader::kClamp_TileMode, SkShader::kClamp_TileMode);
reed73f6cfc2015-11-09 08:36:05 -080036}
37
reedafdbedb2015-01-14 13:16:26 -080038#define N 128
reed09519db2015-01-17 19:36:08 -080039#define ANGLE_DELTA 3
40#define SCALE_DELTA (SK_Scalar1 / 32)
reedafdbedb2015-01-14 13:16:26 -080041
reed9ce9d672016-03-17 10:51:11 -070042static sk_sp<SkImage> make_image() {
reed09519db2015-01-17 19:36:08 -080043 SkImageInfo info = SkImageInfo::MakeN32(N, N, kOpaque_SkAlphaType);
reede8f30622016-03-23 18:59:25 -070044 auto surface(SkSurface::MakeRaster(info));
reedafdbedb2015-01-14 13:16:26 -080045 SkCanvas* canvas = surface->getCanvas();
reed09519db2015-01-17 19:36:08 -080046 canvas->drawColor(SK_ColorWHITE);
reedafdbedb2015-01-14 13:16:26 -080047
48 SkPath path;
49 path.setFillType(SkPath::kEvenOdd_FillType);
50
51 path.addRect(SkRect::MakeWH(N/2, N));
52 path.addRect(SkRect::MakeWH(N, N/2));
53 path.moveTo(0, 0); path.lineTo(N, 0); path.lineTo(0, N); path.close();
54
reed73f6cfc2015-11-09 08:36:05 -080055 SkPaint paint;
reed5671c5b2016-03-09 14:47:34 -080056 paint.setShader(make_shader(SkRect::MakeWH(N, N)));
halcanary9d524f22016-03-29 09:03:52 -070057
reed73f6cfc2015-11-09 08:36:05 -080058 canvas->drawPath(path, paint);
reed9ce9d672016-03-17 10:51:11 -070059 return surface->makeImageSnapshot();
reedafdbedb2015-01-14 13:16:26 -080060}
61
reed9ce9d672016-03-17 10:51:11 -070062static sk_sp<SkImage> zoom_up(SkSurface* origSurf, SkImage* orig) {
reed73f6cfc2015-11-09 08:36:05 -080063 const SkScalar S = 16; // amount to scale up
reed09519db2015-01-17 19:36:08 -080064 const int D = 2; // dimension scaling for the offscreen
65 // since we only view the center, don't need to produce the entire thing
halcanary9d524f22016-03-29 09:03:52 -070066
reed09519db2015-01-17 19:36:08 -080067 SkImageInfo info = SkImageInfo::MakeN32(orig->width() * D, orig->height() * D,
68 kOpaque_SkAlphaType);
reede8f30622016-03-23 18:59:25 -070069 auto surface(origSurf->makeSurface(info));
reed09519db2015-01-17 19:36:08 -080070 SkCanvas* canvas = surface->getCanvas();
71 canvas->drawColor(SK_ColorWHITE);
72 canvas->scale(S, S);
73 canvas->translate(-SkScalarHalf(orig->width()) * (S - D) / S,
74 -SkScalarHalf(orig->height()) * (S - D) / S);
halcanary96fcdcc2015-08-27 07:41:13 -070075 canvas->drawImage(orig, 0, 0, nullptr);
halcanary9d524f22016-03-29 09:03:52 -070076
reed09519db2015-01-17 19:36:08 -080077 if (S > 3) {
78 SkPaint paint;
79 paint.setColor(SK_ColorWHITE);
80 for (int i = 1; i < orig->height(); ++i) {
81 SkScalar y = SkIntToScalar(i);
82 canvas->drawLine(0, y, SkIntToScalar(orig->width()), y, paint);
83 }
84 for (int i = 1; i < orig->width(); ++i) {
85 SkScalar x = SkIntToScalar(i);
86 canvas->drawLine(x, 0, x, SkIntToScalar(orig->height()), paint);
87 }
88 }
reed9ce9d672016-03-17 10:51:11 -070089 return surface->makeImageSnapshot();
reed09519db2015-01-17 19:36:08 -080090}
91
reedafdbedb2015-01-14 13:16:26 -080092struct AnimValue {
93 SkScalar fValue;
94 SkScalar fMin;
95 SkScalar fMax;
96 SkScalar fMod;
97
98 operator SkScalar() const { return fValue; }
99
100 void set(SkScalar value, SkScalar min, SkScalar max) {
101 fValue = value;
102 fMin = min;
103 fMax = max;
104 fMod = 0;
105 }
106
107 void setMod(SkScalar value, SkScalar mod) {
108 fValue = value;
109 fMin = 0;
110 fMax = 0;
111 fMod = mod;
112 }
113
114 SkScalar inc(SkScalar delta) {
115 fValue += delta;
116 return this->fixUp();
117 }
118
119 SkScalar fixUp() {
120 if (fMod) {
121 fValue = SkScalarMod(fValue, fMod);
122 } else {
123 if (fValue > fMax) {
124 fValue = fMax;
125 } else if (fValue < fMin) {
126 fValue = fMin;
127 }
128 }
129 return fValue;
130 }
131};
132
reed09519db2015-01-17 19:36:08 -0800133static void draw_box_frame(SkCanvas* canvas, int width, int height) {
134 SkPaint p;
135 p.setStyle(SkPaint::kStroke_Style);
136 p.setColor(SK_ColorRED);
137 SkRect r = SkRect::MakeIWH(width, height);
138 r.inset(0.5f, 0.5f);
139 canvas->drawRect(r, p);
140 canvas->drawLine(r.left(), r.top(), r.right(), r.bottom(), p);
141 canvas->drawLine(r.left(), r.bottom(), r.right(), r.top(), p);
142}
reedafdbedb2015-01-14 13:16:26 -0800143
144class FilterQualityView : public SampleView {
reed9ce9d672016-03-17 10:51:11 -0700145 sk_sp<SkImage> fImage;
146 AnimValue fScale, fAngle;
147 SkSize fCell;
148 SkInterpolator fTrans;
149 SkMSec fCurrTime;
150 bool fShowFatBits;
reedafdbedb2015-01-14 13:16:26 -0800151
152public:
reed09519db2015-01-17 19:36:08 -0800153 FilterQualityView() : fImage(make_image()), fTrans(2, 2), fShowFatBits(true) {
154 fCell.set(256, 256);
155
reedafdbedb2015-01-14 13:16:26 -0800156 fScale.set(1, SK_Scalar1 / 8, 1);
157 fAngle.setMod(0, 360);
158
159 SkScalar values[2];
160 fTrans.setMirror(true);
161 fTrans.setReset(true);
162
reed339cdbf2015-02-05 22:02:37 -0800163 fCurrTime = 0;
164
reedafdbedb2015-01-14 13:16:26 -0800165 fTrans.setRepeatCount(999);
166 values[0] = values[1] = 0;
reed339cdbf2015-02-05 22:02:37 -0800167 fTrans.setKeyFrame(0, fCurrTime, values);
reedafdbedb2015-01-14 13:16:26 -0800168 values[0] = values[1] = 1;
reed339cdbf2015-02-05 22:02:37 -0800169 fTrans.setKeyFrame(1, fCurrTime + 2000, values);
reedafdbedb2015-01-14 13:16:26 -0800170 }
171
172protected:
mtklein36352bf2015-03-25 18:17:31 -0700173 bool onQuery(SkEvent* evt) override {
reedafdbedb2015-01-14 13:16:26 -0800174 if (SampleCode::TitleQ(*evt)) {
175 SampleCode::TitleR(evt, "FilterQuality");
176 return true;
177 }
178 SkUnichar uni;
179 if (SampleCode::CharQ(*evt, &uni)) {
180 switch (uni) {
halcanary96fcdcc2015-08-27 07:41:13 -0700181 case '1': fAngle.inc(-ANGLE_DELTA); this->inval(nullptr); return true;
182 case '2': fAngle.inc( ANGLE_DELTA); this->inval(nullptr); return true;
183 case '3': fScale.inc(-SCALE_DELTA); this->inval(nullptr); return true;
184 case '4': fScale.inc( SCALE_DELTA); this->inval(nullptr); return true;
185 case '5': fShowFatBits = !fShowFatBits; this->inval(nullptr); return true;
reedafdbedb2015-01-14 13:16:26 -0800186 default: break;
187 }
188 }
189 return this->INHERITED::onQuery(evt);
190 }
191
reed93a12152015-03-16 10:08:34 -0700192 void drawTheImage(SkCanvas* canvas, const SkISize& size, SkFilterQuality filter,
reed09519db2015-01-17 19:36:08 -0800193 SkScalar dx, SkScalar dy) {
reedafdbedb2015-01-14 13:16:26 -0800194 SkPaint paint;
195 paint.setAntiAlias(true);
reed93a12152015-03-16 10:08:34 -0700196 paint.setFilterQuality(filter);
reedafdbedb2015-01-14 13:16:26 -0800197
reed09519db2015-01-17 19:36:08 -0800198 SkAutoCanvasRestore acr(canvas, true);
199
200 canvas->translate(dx, dy);
201
202 canvas->translate(SkScalarHalf(size.width()), SkScalarHalf(size.height()));
reedafdbedb2015-01-14 13:16:26 -0800203 canvas->scale(fScale, fScale);
204 canvas->rotate(fAngle);
reed9ce9d672016-03-17 10:51:11 -0700205 canvas->drawImage(fImage.get(), -SkScalarHalf(fImage->width()), -SkScalarHalf(fImage->height()),
reedafdbedb2015-01-14 13:16:26 -0800206 &paint);
reed09519db2015-01-17 19:36:08 -0800207
208 if (false) {
209 acr.restore();
210 draw_box_frame(canvas, size.width(), size.height());
211 }
212 }
213
reed93a12152015-03-16 10:08:34 -0700214 void drawHere(SkCanvas* canvas, SkFilterQuality filter, SkScalar dx, SkScalar dy) {
reed09519db2015-01-17 19:36:08 -0800215 SkCanvas* origCanvas = canvas;
216 SkAutoCanvasRestore acr(canvas, true);
217
218 SkISize size = SkISize::Make(fImage->width(), fImage->height());
219
reede8f30622016-03-23 18:59:25 -0700220 sk_sp<SkSurface> surface;
reed09519db2015-01-17 19:36:08 -0800221 if (fShowFatBits) {
222 // scale up so we don't clip rotations
223 SkImageInfo info = SkImageInfo::MakeN32(fImage->width() * 2, fImage->height() * 2,
224 kOpaque_SkAlphaType);
reede8f30622016-03-23 18:59:25 -0700225 surface = make_surface(canvas, info);
reed09519db2015-01-17 19:36:08 -0800226 canvas = surface->getCanvas();
227 canvas->drawColor(SK_ColorWHITE);
228 size.set(info.width(), info.height());
229 } else {
230 canvas->translate(SkScalarHalf(fCell.width() - fImage->width()),
231 SkScalarHalf(fCell.height() - fImage->height()));
232 }
233 this->drawTheImage(canvas, size, filter, dx, dy);
234
235 if (surface) {
reed9ce9d672016-03-17 10:51:11 -0700236 sk_sp<SkImage> orig(surface->makeImageSnapshot());
reede8f30622016-03-23 18:59:25 -0700237 sk_sp<SkImage> zoomed(zoom_up(surface.get(), orig.get()));
reed9ce9d672016-03-17 10:51:11 -0700238 origCanvas->drawImage(zoomed.get(),
reed09519db2015-01-17 19:36:08 -0800239 SkScalarHalf(fCell.width() - zoomed->width()),
240 SkScalarHalf(fCell.height() - zoomed->height()));
241 }
242 }
243
244 void drawBorders(SkCanvas* canvas) {
245 SkPaint p;
246 p.setStyle(SkPaint::kStroke_Style);
247 p.setColor(SK_ColorBLUE);
248
249 SkRect r = SkRect::MakeWH(fCell.width() * 2, fCell.height() * 2);
250 r.inset(SK_ScalarHalf, SK_ScalarHalf);
251 canvas->drawRect(r, p);
252 canvas->drawLine(r.left(), r.centerY(), r.right(), r.centerY(), p);
253 canvas->drawLine(r.centerX(), r.top(), r.centerX(), r.bottom(), p);
reedafdbedb2015-01-14 13:16:26 -0800254 }
255
mtklein36352bf2015-03-25 18:17:31 -0700256 void onDrawContent(SkCanvas* canvas) override {
reed09519db2015-01-17 19:36:08 -0800257 fCell.set(this->height() / 2, this->height() / 2);
258
reedafdbedb2015-01-14 13:16:26 -0800259 SkScalar trans[2];
reed339cdbf2015-02-05 22:02:37 -0800260 fTrans.timeToValues(fCurrTime, trans);
reedafdbedb2015-01-14 13:16:26 -0800261
reed09519db2015-01-17 19:36:08 -0800262 for (int y = 0; y < 2; ++y) {
263 for (int x = 0; x < 2; ++x) {
264 int index = y * 2 + x;
265 SkAutoCanvasRestore acr(canvas, true);
266 canvas->translate(fCell.width() * x, fCell.height() * y);
267 SkRect r = SkRect::MakeWH(fCell.width(), fCell.height());
268 r.inset(4, 4);
269 canvas->clipRect(r);
reed93a12152015-03-16 10:08:34 -0700270 this->drawHere(canvas, SkFilterQuality(index), trans[0], trans[1]);
reed09519db2015-01-17 19:36:08 -0800271 }
reedafdbedb2015-01-14 13:16:26 -0800272 }
273
reed09519db2015-01-17 19:36:08 -0800274 this->drawBorders(canvas);
275
276 const SkScalar textX = fCell.width() * 2 + 30;
277
reedafdbedb2015-01-14 13:16:26 -0800278 SkPaint paint;
279 paint.setAntiAlias(true);
reed09519db2015-01-17 19:36:08 -0800280 paint.setTextSize(36);
reedafdbedb2015-01-14 13:16:26 -0800281 SkString str;
282 str.appendScalar(fScale);
reed09519db2015-01-17 19:36:08 -0800283 canvas->drawText(str.c_str(), str.size(), textX, 100, paint);
reedafdbedb2015-01-14 13:16:26 -0800284 str.reset(); str.appendScalar(fAngle);
reed09519db2015-01-17 19:36:08 -0800285 canvas->drawText(str.c_str(), str.size(), textX, 150, paint);
reedafdbedb2015-01-14 13:16:26 -0800286
287 str.reset(); str.appendScalar(trans[0]);
reed09519db2015-01-17 19:36:08 -0800288 canvas->drawText(str.c_str(), str.size(), textX, 200, paint);
reedafdbedb2015-01-14 13:16:26 -0800289 str.reset(); str.appendScalar(trans[1]);
reed09519db2015-01-17 19:36:08 -0800290 canvas->drawText(str.c_str(), str.size(), textX, 250, paint);
reed339cdbf2015-02-05 22:02:37 -0800291 }
reed09519db2015-01-17 19:36:08 -0800292
mtklein36352bf2015-03-25 18:17:31 -0700293 bool onAnimate(const SkAnimTimer& timer) override {
reed339cdbf2015-02-05 22:02:37 -0800294 fCurrTime = timer.msec();
295 return true;
reed09519db2015-01-17 19:36:08 -0800296 }
reedafdbedb2015-01-14 13:16:26 -0800297
298 virtual bool handleKey(SkKey key) {
halcanary96fcdcc2015-08-27 07:41:13 -0700299 this->inval(nullptr);
reedafdbedb2015-01-14 13:16:26 -0800300 return true;
301 }
302
303private:
304 typedef SampleView INHERITED;
305};
306
307//////////////////////////////////////////////////////////////////////////////
308
309static SkView* MyFactory() { return new FilterQualityView; }
310static SkViewRegister reg(MyFactory);