• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2011 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/gm.h"
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/SkColorFilter.h"
14 #include "include/core/SkFlattenable.h"
15 #include "include/core/SkFont.h"
16 #include "include/core/SkImageFilter.h"
17 #include "include/core/SkPaint.h"
18 #include "include/core/SkPoint.h"
19 #include "include/core/SkRect.h"
20 #include "include/core/SkRefCnt.h"
21 #include "include/core/SkScalar.h"
22 #include "include/core/SkSize.h"
23 #include "include/core/SkString.h"
24 #include "include/core/SkTypeface.h"
25 #include "include/core/SkTypes.h"
26 #include "include/effects/SkImageFilters.h"
27 #include "include/utils/SkTextUtils.h"
28 #include "src/core/SkImageFilter_Base.h"
29 #include "src/core/SkSpecialImage.h"
30 #include "tools/ToolUtils.h"
31 
32 #include <utility>
33 
34 class SkReadBuffer;
35 
36 class FailImageFilter : public SkImageFilter_Base {
37 public:
Make()38     static sk_sp<SkImageFilter> Make() {
39         return sk_sp<SkImageFilter>(new FailImageFilter);
40     }
41 
42     SK_FLATTENABLE_HOOKS(FailImageFilter)
43 protected:
FailImageFilter()44     FailImageFilter() : INHERITED(nullptr, 0, nullptr) {}
45 
onFilterImage(const Context &,SkIPoint * offset) const46     sk_sp<SkSpecialImage> onFilterImage(const Context&, SkIPoint* offset) const override {
47         return nullptr;
48     }
49 
50 private:
51 
52     typedef SkImageFilter_Base INHERITED;
53 };
54 
CreateProc(SkReadBuffer & buffer)55 sk_sp<SkFlattenable> FailImageFilter::CreateProc(SkReadBuffer& buffer) {
56     SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 0);
57     return FailImageFilter::Make();
58 }
59 
60 class IdentityImageFilter : public SkImageFilter_Base {
61 public:
Make(sk_sp<SkImageFilter> input)62     static sk_sp<SkImageFilter> Make(sk_sp<SkImageFilter> input) {
63         return sk_sp<SkImageFilter>(new IdentityImageFilter(std::move(input)));
64     }
65 
66 
67     SK_FLATTENABLE_HOOKS(IdentityImageFilter)
68 protected:
onFilterImage(const Context & ctx,SkIPoint * offset) const69     sk_sp<SkSpecialImage> onFilterImage(const Context& ctx, SkIPoint* offset) const override {
70         offset->set(0, 0);
71         return sk_ref_sp<SkSpecialImage>(ctx.sourceImage());
72     }
73 
74 private:
IdentityImageFilter(sk_sp<SkImageFilter> input)75     IdentityImageFilter(sk_sp<SkImageFilter> input) : INHERITED(&input, 1, nullptr) {}
76 
77     typedef SkImageFilter_Base INHERITED;
78 };
79 
80 // Register these image filters as deserializable before main().
81 namespace {
82     static struct Initializer {
Initializer__anon70fa63b80111::Initializer83         Initializer() {
84             SK_REGISTER_FLATTENABLE(IdentityImageFilter);
85             SK_REGISTER_FLATTENABLE(FailImageFilter);
86         }
87     } initializer;
88 }
89 
CreateProc(SkReadBuffer & buffer)90 sk_sp<SkFlattenable> IdentityImageFilter::CreateProc(SkReadBuffer& buffer) {
91     SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
92     return IdentityImageFilter::Make(common.getInput(0));
93 }
94 
95 ///////////////////////////////////////////////////////////////////////////////
96 
draw_paint(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)97 static void draw_paint(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
98     SkPaint paint;
99     paint.setImageFilter(std::move(imf));
100     paint.setColor(SK_ColorGREEN);
101     canvas->save();
102     canvas->clipRect(r);
103     canvas->drawPaint(paint);
104     canvas->restore();
105 }
106 
draw_line(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)107 static void draw_line(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
108     SkPaint paint;
109     paint.setColor(SK_ColorBLUE);
110     paint.setImageFilter(imf);
111     paint.setStrokeWidth(r.width()/10);
112     canvas->drawLine(r.fLeft, r.fTop, r.fRight, r.fBottom, paint);
113 }
114 
draw_rect(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)115 static void draw_rect(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
116     SkPaint paint;
117     paint.setColor(SK_ColorYELLOW);
118     paint.setImageFilter(imf);
119     SkRect rr(r);
120     rr.inset(r.width()/10, r.height()/10);
121     canvas->drawRect(rr, paint);
122 }
123 
draw_path(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)124 static void draw_path(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
125     SkPaint paint;
126     paint.setColor(SK_ColorMAGENTA);
127     paint.setImageFilter(imf);
128     paint.setAntiAlias(true);
129     canvas->drawCircle(r.centerX(), r.centerY(), r.width()*2/5, paint);
130 }
131 
draw_text(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)132 static void draw_text(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
133     SkPaint paint;
134     paint.setImageFilter(imf);
135     paint.setColor(SK_ColorCYAN);
136     SkFont font(ToolUtils::create_portable_typeface(), r.height() / 2);
137     SkTextUtils::DrawString(canvas, "Text", r.centerX(), r.centerY(), font, paint,
138                             SkTextUtils::kCenter_Align);
139 }
140 
draw_bitmap(SkCanvas * canvas,const SkRect & r,sk_sp<SkImageFilter> imf)141 static void draw_bitmap(SkCanvas* canvas, const SkRect& r, sk_sp<SkImageFilter> imf) {
142     SkPaint paint;
143     paint.setImageFilter(std::move(imf));
144 
145     SkIRect bounds;
146     r.roundOut(&bounds);
147 
148     SkBitmap bm;
149     bm.allocN32Pixels(bounds.width(), bounds.height());
150     bm.eraseColor(SK_ColorTRANSPARENT);
151     SkCanvas c(bm);
152     draw_path(&c, r, nullptr);
153 
154     canvas->drawBitmap(bm, 0, 0, &paint);
155 }
156 
157 ///////////////////////////////////////////////////////////////////////////////
158 
159 class ImageFiltersBaseGM : public skiagm::GM {
160 public:
ImageFiltersBaseGM()161     ImageFiltersBaseGM () {}
162 
163 protected:
onShortName()164     SkString onShortName() override {
165         return SkString("imagefiltersbase");
166     }
167 
onISize()168     SkISize onISize() override { return SkISize::Make(700, 500); }
169 
draw_frame(SkCanvas * canvas,const SkRect & r)170     void draw_frame(SkCanvas* canvas, const SkRect& r) {
171         SkPaint paint;
172         paint.setStyle(SkPaint::kStroke_Style);
173         paint.setColor(SK_ColorRED);
174         canvas->drawRect(r, paint);
175     }
176 
onDraw(SkCanvas * canvas)177     void onDraw(SkCanvas* canvas) override {
178         void (*drawProc[])(SkCanvas*, const SkRect&, sk_sp<SkImageFilter>) = {
179             draw_paint,
180             draw_line, draw_rect, draw_path, draw_text,
181             draw_bitmap,
182         };
183 
184         auto cf = SkColorFilters::Blend(SK_ColorRED, SkBlendMode::kSrcIn);
185         sk_sp<SkImageFilter> filters[] = {
186             nullptr,
187             IdentityImageFilter::Make(nullptr),
188             FailImageFilter::Make(),
189             SkImageFilters::ColorFilter(std::move(cf), nullptr),
190             // The strange 0.29 value tickles an edge case where crop rect calculates
191             // a small border, but the blur really needs no border. This tickles
192             // an msan uninitialized value bug.
193             SkImageFilters::Blur(12.0f, 0.29f, nullptr),
194             SkImageFilters::DropShadow(10.0f, 5.0f, 3.0f, 3.0f, SK_ColorBLUE, nullptr),
195         };
196 
197         SkRect r = SkRect::MakeWH(SkIntToScalar(64), SkIntToScalar(64));
198         SkScalar MARGIN = SkIntToScalar(16);
199         SkScalar DX = r.width() + MARGIN;
200         SkScalar DY = r.height() + MARGIN;
201 
202         canvas->translate(MARGIN, MARGIN);
203         for (size_t i = 0; i < SK_ARRAY_COUNT(drawProc); ++i) {
204             canvas->save();
205             for (size_t j = 0; j < SK_ARRAY_COUNT(filters); ++j) {
206                 drawProc[i](canvas, r, filters[j]);
207 
208                 draw_frame(canvas, r);
209                 canvas->translate(0, DY);
210             }
211             canvas->restore();
212             canvas->translate(DX, 0);
213         }
214     }
215 
216 private:
217     typedef GM INHERITED;
218 };
219 DEF_GM( return new ImageFiltersBaseGM; )
220 
221 ///////////////////////////////////////////////////////////////////////////////
222 
223 /*
224  *  Want to test combos of filter and LCD text, to be sure we disable LCD in the presence of
225  *  a filter.
226  */
227 class ImageFiltersTextBaseGM : public skiagm::GM {
228     SkString fSuffix;
229 public:
ImageFiltersTextBaseGM(const char suffix[])230     ImageFiltersTextBaseGM(const char suffix[]) : fSuffix(suffix) {}
231 
232 protected:
onShortName()233     SkString onShortName() override {
234         SkString name;
235         name.printf("%s_%s", "textfilter", fSuffix.c_str());
236         return name;
237     }
238 
onISize()239     SkISize onISize() override { return SkISize::Make(512, 342); }
240 
drawWaterfall(SkCanvas * canvas,const SkPaint & paint)241     void drawWaterfall(SkCanvas* canvas, const SkPaint& paint) {
242         static const SkFont::Edging kEdgings[3] = {
243             SkFont::Edging::kAlias,
244             SkFont::Edging::kAntiAlias,
245             SkFont::Edging::kSubpixelAntiAlias,
246         };
247         SkFont font(ToolUtils::create_portable_typeface(), 30);
248 
249         SkAutoCanvasRestore acr(canvas, true);
250         for (SkFont::Edging edging : kEdgings) {
251             font.setEdging(edging);
252             canvas->drawString("Hamburgefon", 0, 0, font, paint);
253             canvas->translate(0, 40);
254         }
255     }
256 
257     virtual void installFilter(SkPaint* paint) = 0;
258 
onDraw(SkCanvas * canvas)259     void onDraw(SkCanvas* canvas) override {
260         SkPaint paint;
261 
262         canvas->translate(20, 40);
263 
264         for (int doSaveLayer = 0; doSaveLayer <= 1; ++doSaveLayer) {
265             SkAutoCanvasRestore acr(canvas, true);
266             for (int useFilter = 0; useFilter <= 1; ++useFilter) {
267                 SkAutoCanvasRestore acr2(canvas, true);
268 
269                 SkPaint paint;
270                 if (useFilter) {
271                     this->installFilter(&paint);
272                 }
273                 if (doSaveLayer) {
274                     canvas->saveLayer(nullptr, &paint);
275                     paint.setImageFilter(nullptr);
276                 }
277                 this->drawWaterfall(canvas, paint);
278 
279                 acr2.restore();
280                 canvas->translate(250, 0);
281             }
282             acr.restore();
283             canvas->translate(0, 200);
284         }
285     }
286 
287 private:
288     typedef GM INHERITED;
289 };
290 
291 class ImageFiltersText_IF : public ImageFiltersTextBaseGM {
292 public:
ImageFiltersText_IF()293     ImageFiltersText_IF() : ImageFiltersTextBaseGM("image") {}
294 
installFilter(SkPaint * paint)295     void installFilter(SkPaint* paint) override {
296         paint->setImageFilter(SkImageFilters::Blur(1.5f, 1.5f, nullptr));
297     }
298 };
299 DEF_GM( return new ImageFiltersText_IF; )
300 
301 class ImageFiltersText_CF : public ImageFiltersTextBaseGM {
302 public:
ImageFiltersText_CF()303     ImageFiltersText_CF() : ImageFiltersTextBaseGM("color") {}
304 
installFilter(SkPaint * paint)305     void installFilter(SkPaint* paint) override {
306         paint->setColorFilter(SkColorFilters::Blend(SK_ColorBLUE, SkBlendMode::kSrcIn));
307     }
308 };
309 DEF_GM( return new ImageFiltersText_CF; )
310