blob: 0fe866feb2b60d26d7a7fd467d44e46df5dcf15d [file] [log] [blame]
reed@google.comdb87c962012-11-02 21:11:12 +00001/*
humper@google.coma99a92c2013-02-20 16:42:06 +00002* Copyright 2012 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*/
reed@google.comdb87c962012-11-02 21:11:12 +00007
8#include "gm.h"
humper@google.com7c7292c2013-01-04 20:29:03 +00009#include "SkBlurMask.h"
robertphillips@google.comb7061172013-09-06 14:16:12 +000010#include "SkBlurMaskFilter.h"
reed@google.comdb87c962012-11-02 21:11:12 +000011#include "SkCanvas.h"
12#include "SkPath.h"
13
14#define STROKE_WIDTH SkIntToScalar(10)
15
16typedef void (*Proc)(SkCanvas*, const SkRect&, const SkPaint&);
17
18static void fill_rect(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
19 canvas->drawRect(r, p);
20}
21
reed@google.comdb87c962012-11-02 21:11:12 +000022static void draw_donut(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
23 SkRect rect;
24 SkPath path;
skia.committer@gmail.com34587162012-11-20 02:01:23 +000025
reed@google.comdb87c962012-11-02 21:11:12 +000026 rect = r;
27 rect.outset(STROKE_WIDTH/2, STROKE_WIDTH/2);
28 path.addRect(rect);
29 rect = r;
30 rect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000031
reed@google.comdb87c962012-11-02 21:11:12 +000032 path.addRect(rect);
33 path.setFillType(SkPath::kEvenOdd_FillType);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000034
reed@google.com808b70f2012-11-19 16:14:02 +000035 canvas->drawPath(path, p);
36}
reed@google.comdb87c962012-11-02 21:11:12 +000037
reed@google.com808b70f2012-11-19 16:14:02 +000038static void draw_donut_skewed(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
39 SkRect rect;
40 SkPath path;
skia.committer@gmail.com34587162012-11-20 02:01:23 +000041
reed@google.com808b70f2012-11-19 16:14:02 +000042 rect = r;
43 rect.outset(STROKE_WIDTH/2, STROKE_WIDTH/2);
44 path.addRect(rect);
45 rect = r;
46 rect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000047
reed@google.com808b70f2012-11-19 16:14:02 +000048 rect.offset(7, -7);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000049
reed@google.com808b70f2012-11-19 16:14:02 +000050 path.addRect(rect);
51 path.setFillType(SkPath::kEvenOdd_FillType);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000052
reed@google.comdb87c962012-11-02 21:11:12 +000053 canvas->drawPath(path, p);
54}
55
reed@google.com53007a22012-11-26 14:39:50 +000056#include "SkGradientShader.h"
57
joshualitt341400e2014-12-18 11:54:13 -080058/*
59 * Spits out a dummy gradient to test blur with shader on paint
60 */
reed2ad1aa62016-03-09 09:50:50 -080061static sk_sp<SkShader> MakeRadial() {
joshualitt341400e2014-12-18 11:54:13 -080062 SkPoint pts[2] = {
63 { 0, 0 },
64 { SkIntToScalar(100), SkIntToScalar(100) }
65 };
66 SkShader::TileMode tm = SkShader::kClamp_TileMode;
67 const SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, };
68 const SkScalar pos[] = { SK_Scalar1/4, SK_Scalar1*3/4 };
69 SkMatrix scale;
70 scale.setScale(0.5f, 0.5f);
71 scale.postTranslate(25.f, 25.f);
72 SkPoint center0, center1;
73 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
74 SkScalarAve(pts[0].fY, pts[1].fY));
75 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
76 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
reed2ad1aa62016-03-09 09:50:50 -080077 return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
78 center0, (pts[1].fX - pts[0].fX) / 2,
79 colors, pos, SK_ARRAY_COUNT(colors), tm,
80 0, &scale);
joshualitt341400e2014-12-18 11:54:13 -080081}
82
reed@google.com53007a22012-11-26 14:39:50 +000083typedef void (*PaintProc)(SkPaint*, SkScalar width);
84
reed@google.comdb87c962012-11-02 21:11:12 +000085class BlurRectGM : public skiagm::GM {
commit-bot@chromium.orge3964552014-04-28 16:25:35 +000086 SkAutoTUnref<SkMaskFilter> fMaskFilters[kLastEnum_SkBlurStyle + 1];
humper@google.coma99a92c2013-02-20 16:42:06 +000087 SkString fName;
humper@google.coma99a92c2013-02-20 16:42:06 +000088 SkAlpha fAlpha;
reed@google.comdb87c962012-11-02 21:11:12 +000089public:
commit-bot@chromium.org7cced562014-01-10 23:10:13 +000090 BlurRectGM(const char name[], U8CPU alpha)
91 : fName(name)
92 , fAlpha(SkToU8(alpha)) {
reed@google.com57850b92012-12-17 21:20:53 +000093 }
reed@google.comdb87c962012-11-02 21:11:12 +000094
95protected:
mtklein36352bf2015-03-25 18:17:31 -070096 void onOnceBeforeDraw() override {
commit-bot@chromium.orge3964552014-04-28 16:25:35 +000097 for (int i = 0; i <= kLastEnum_SkBlurStyle; ++i) {
98 fMaskFilters[i].reset(SkBlurMaskFilter::Create((SkBlurStyle)i,
commit-bot@chromium.org7cced562014-01-10 23:10:13 +000099 SkBlurMask::ConvertRadiusToSigma(SkIntToScalar(STROKE_WIDTH/2)),
100 SkBlurMaskFilter::kHighQuality_BlurFlag));
101 }
102 }
103
mtklein36352bf2015-03-25 18:17:31 -0700104 SkString onShortName() override {
reed@google.com53007a22012-11-26 14:39:50 +0000105 return fName;
reed@google.comdb87c962012-11-02 21:11:12 +0000106 }
107
mtklein36352bf2015-03-25 18:17:31 -0700108 SkISize onISize() override {
joshualitt341400e2014-12-18 11:54:13 -0800109 return SkISize::Make(860, 820);
reed@google.comdb87c962012-11-02 21:11:12 +0000110 }
111
mtklein36352bf2015-03-25 18:17:31 -0700112 void onDraw(SkCanvas* canvas) override {
reed@google.comdb87c962012-11-02 21:11:12 +0000113 canvas->translate(STROKE_WIDTH*3/2, STROKE_WIDTH*3/2);
114
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000115 SkRect r = { 0, 0, 100, 50 };
116 SkScalar scales[] = { SK_Scalar1, 0.6f };
skia.committer@gmail.com8ccf5902012-11-27 02:01:19 +0000117
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000118 for (size_t s = 0; s < SK_ARRAY_COUNT(scales); ++s) {
119 canvas->save();
120 for (size_t f = 0; f < SK_ARRAY_COUNT(fMaskFilters); ++f) {
121 SkPaint paint;
122 paint.setMaskFilter(fMaskFilters[f]);
123 paint.setAlpha(fAlpha);
124
joshualitt341400e2014-12-18 11:54:13 -0800125 SkPaint paintWithRadial = paint;
reed2ad1aa62016-03-09 09:50:50 -0800126 paintWithRadial.setShader(MakeRadial());
joshualitt341400e2014-12-18 11:54:13 -0800127
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000128 static const Proc procs[] = {
129 fill_rect, draw_donut, draw_donut_skewed
130 };
131
132 canvas->save();
133 canvas->scale(scales[s], scales[s]);
134 this->drawProcs(canvas, r, paint, false, procs, SK_ARRAY_COUNT(procs));
135 canvas->translate(r.width() * 4/3, 0);
joshualitt341400e2014-12-18 11:54:13 -0800136 this->drawProcs(canvas, r, paintWithRadial, false, procs, SK_ARRAY_COUNT(procs));
137 canvas->translate(r.width() * 4/3, 0);
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000138 this->drawProcs(canvas, r, paint, true, procs, SK_ARRAY_COUNT(procs));
joshualitt341400e2014-12-18 11:54:13 -0800139 canvas->translate(r.width() * 4/3, 0);
140 this->drawProcs(canvas, r, paintWithRadial, true, procs, SK_ARRAY_COUNT(procs));
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000141 canvas->restore();
142
143 canvas->translate(0, SK_ARRAY_COUNT(procs) * r.height() * 4/3 * scales[s]);
144 }
145 canvas->restore();
joshualitt341400e2014-12-18 11:54:13 -0800146 canvas->translate(4 * r.width() * 4/3 * scales[s], 0);
reed@google.com53007a22012-11-26 14:39:50 +0000147 }
reed@google.comdb87c962012-11-02 21:11:12 +0000148 }
149
reed@google.comdb87c962012-11-02 21:11:12 +0000150private:
151 void drawProcs(SkCanvas* canvas, const SkRect& r, const SkPaint& paint,
152 bool doClip, const Proc procs[], size_t procsCount) {
153 SkAutoCanvasRestore acr(canvas, true);
154 for (size_t i = 0; i < procsCount; ++i) {
155 if (doClip) {
156 SkRect clipRect(r);
157 clipRect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
158 canvas->save();
159 canvas->clipRect(r);
160 }
161 procs[i](canvas, r, paint);
162 if (doClip) {
163 canvas->restore();
164 }
165 canvas->translate(0, r.height() * 4/3);
166 }
167 }
humper@google.coma99a92c2013-02-20 16:42:06 +0000168private:
reed@google.comdb87c962012-11-02 21:11:12 +0000169 typedef GM INHERITED;
170};
171
halcanary2a243382015-09-09 08:16:41 -0700172DEF_SIMPLE_GM(blurrect_gallery, canvas, 1200, 1024) {
173 const int fGMWidth = 1200;
174 const int fPadding = 10;
175 const int fMargin = 100;
commit-bot@chromium.org3c1594a2014-05-28 21:52:12 +0000176
commit-bot@chromium.org3c1594a2014-05-28 21:52:12 +0000177 const int widths[] = {25, 5, 5, 100, 150, 25};
178 const int heights[] = {100, 100, 5, 25, 150, 25};
179 const SkBlurStyle styles[] = {kNormal_SkBlurStyle, kInner_SkBlurStyle, kOuter_SkBlurStyle};
180 const float radii[] = {20, 5, 10};
181
182 canvas->translate(50,20);
183
184 int cur_x = 0;
185 int cur_y = 0;
186
187 int max_height = 0;
188
189 for (size_t i = 0 ; i < SK_ARRAY_COUNT(widths) ; i++) {
190 int width = widths[i];
191 int height = heights[i];
192 SkRect r;
193 r.setWH(SkIntToScalar(width), SkIntToScalar(height));
194 SkAutoCanvasRestore autoRestore(canvas, true);
195
196 for (size_t j = 0 ; j < SK_ARRAY_COUNT(radii) ; j++) {
197 float radius = radii[j];
198 for (size_t k = 0 ; k < SK_ARRAY_COUNT(styles) ; k++) {
199 SkBlurStyle style = styles[k];
200
201 SkMask mask;
robertphillipse80eb922015-12-17 11:33:12 -0800202 if (!SkBlurMask::BlurRect(SkBlurMask::ConvertRadiusToSigma(radius),
203 &mask, r, style)) {
204 continue;
205 }
commit-bot@chromium.org3c1594a2014-05-28 21:52:12 +0000206
207 SkAutoMaskFreeImage amfi(mask.fImage);
208
209 SkBitmap bm;
210 bm.installMaskPixels(mask);
211
212 if (cur_x + bm.width() >= fGMWidth - fMargin) {
213 cur_x = 0;
214 cur_y += max_height + fPadding;
215 max_height = 0;
216 }
217
218 canvas->save();
commit-bot@chromium.org793ddd92014-05-28 22:42:31 +0000219 canvas->translate((SkScalar)cur_x, (SkScalar)cur_y);
commit-bot@chromium.org3c1594a2014-05-28 21:52:12 +0000220 canvas->translate(-(bm.width() - r.width())/2, -(bm.height()-r.height())/2);
halcanary96fcdcc2015-08-27 07:41:13 -0700221 canvas->drawBitmap(bm, 0.f, 0.f, nullptr);
commit-bot@chromium.org3c1594a2014-05-28 21:52:12 +0000222 canvas->restore();
223
224 cur_x += bm.width() + fPadding;
225 if (bm.height() > max_height)
226 max_height = bm.height();
227 }
228 }
229 }
halcanary2a243382015-09-09 08:16:41 -0700230}
humper@google.com7c7292c2013-01-04 20:29:03 +0000231
reed@google.comdb87c962012-11-02 21:11:12 +0000232//////////////////////////////////////////////////////////////////////////////
233
commit-bot@chromium.org7cced562014-01-10 23:10:13 +0000234DEF_GM(return new BlurRectGM("blurrects", 0xFF);)