blob: 6aac09fbce76d0e2e0ab4677349bf181fee7f846 [file] [log] [blame]
reed@google.com12fa9ba2013-01-16 18:54:15 +00001/*
2 * Copyright 2013 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#include "SkCanvas.h"
10#include "SkBlurImageFilter.h"
reed71c3c762015-06-24 10:29:17 -070011#include "SkRSXform.h"
12#include "SkSurface.h"
reed@google.com12fa9ba2013-01-16 18:54:15 +000013
14static void make_bm(SkBitmap* bm) {
reed@google.comeb9a46c2014-01-25 16:46:20 +000015 bm->allocN32Pixels(100, 100);
reed@google.com12fa9ba2013-01-16 18:54:15 +000016 bm->eraseColor(SK_ColorBLUE);
17
18 SkCanvas canvas(*bm);
19 SkPaint paint;
20 paint.setAntiAlias(true);
21 paint.setColor(SK_ColorRED);
22 canvas.drawCircle(50, 50, 50, paint);
23}
24
25static void draw_2_bitmaps(SkCanvas* canvas, const SkBitmap& bm, bool doClip,
halcanary96fcdcc2015-08-27 07:41:13 -070026 int dx, int dy, SkImageFilter* filter = nullptr) {
reed@google.com12fa9ba2013-01-16 18:54:15 +000027 SkAutoCanvasRestore acr(canvas, true);
28 SkPaint paint;
29
skia.committer@gmail.com4d28d982013-01-17 07:06:06 +000030 SkRect clipR = SkRect::MakeXYWH(SkIntToScalar(dx),
robertphillips@google.com914a2f22013-01-16 19:57:02 +000031 SkIntToScalar(dy),
reed@google.com1c711ca2013-01-16 19:24:15 +000032 SkIntToScalar(bm.width()),
33 SkIntToScalar(bm.height()));
reed@google.com12fa9ba2013-01-16 18:54:15 +000034
35 paint.setImageFilter(filter);
36 clipR.inset(5, 5);
37
38 if (doClip) {
39 canvas->save();
40 canvas->clipRect(clipR);
41 }
42 canvas->drawSprite(bm, dx, dy, &paint);
43 if (doClip) {
44 canvas->restore();
45 }
46
reed@google.com1c711ca2013-01-16 19:24:15 +000047 canvas->translate(SkIntToScalar(bm.width() + 20), 0);
reed@google.com12fa9ba2013-01-16 18:54:15 +000048
49 if (doClip) {
50 canvas->save();
51 canvas->clipRect(clipR);
52 }
reed@google.com1c711ca2013-01-16 19:24:15 +000053 canvas->drawBitmap(bm, SkIntToScalar(dx), SkIntToScalar(dy), &paint);
reed@google.com12fa9ba2013-01-16 18:54:15 +000054 if (doClip) {
55 canvas->restore();
56 }
57}
58
59/**
60 * Compare output of drawSprite and drawBitmap (esp. clipping and imagefilters)
61 */
62class SpriteBitmapGM : public skiagm::GM {
63public:
64 SpriteBitmapGM() {}
65
66protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +000067
mtklein36352bf2015-03-25 18:17:31 -070068 SkString onShortName() override {
reed@google.com12fa9ba2013-01-16 18:54:15 +000069 return SkString("spritebitmap");
70 }
71
mtklein36352bf2015-03-25 18:17:31 -070072 SkISize onISize() override {
reed@google.com12fa9ba2013-01-16 18:54:15 +000073 return SkISize::Make(640, 480);
74 }
75
mtklein36352bf2015-03-25 18:17:31 -070076 void onDraw(SkCanvas* canvas) override {
reed@google.com12fa9ba2013-01-16 18:54:15 +000077 SkBitmap bm;
78 make_bm(&bm);
79
80 int dx = 10;
81 int dy = 10;
82
83 SkScalar sigma = 8;
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000084 SkAutoTUnref<SkImageFilter> filter(SkBlurImageFilter::Create(sigma, sigma));
reed@google.com12fa9ba2013-01-16 18:54:15 +000085
86 draw_2_bitmaps(canvas, bm, false, dx, dy);
87 dy += bm.height() + 20;
88 draw_2_bitmaps(canvas, bm, false, dx, dy, filter);
89 dy += bm.height() + 20;
90 draw_2_bitmaps(canvas, bm, true, dx, dy);
91 dy += bm.height() + 20;
92 draw_2_bitmaps(canvas, bm, true, dx, dy, filter);
93 }
94
95private:
96 typedef GM INHERITED;
97};
reed@google.com12fa9ba2013-01-16 18:54:15 +000098DEF_GM( return new SpriteBitmapGM; )
reed71c3c762015-06-24 10:29:17 -070099
reed88d064d2015-10-12 11:30:02 -0700100///////////////////////////////////////////////////////////////////////////////////////////////////
101
102#include "SkColorFilterImageFilter.h"
103#include "SkModeColorFilter.h"
104#include "SkMorphologyImageFilter.h"
105#include "SkOffsetImageFilter.h"
106
107static SkImage* make_image(SkCanvas* rootCanvas) {
108 SkImageInfo info = SkImageInfo::MakeN32Premul(100, 100);
109 SkAutoTUnref<SkSurface> surface(rootCanvas->newSurface(info));
110 if (!surface) {
111 surface.reset(SkSurface::NewRaster(info));
112 }
113
114 SkPaint paint;
115 paint.setAntiAlias(true);
116 paint.setColor(SK_ColorRED);
117 surface->getCanvas()->drawCircle(50, 50, 50, paint);
118 return surface->newImageSnapshot();
119}
120
121static void show_image(SkCanvas* canvas, SkImage* image, const SkIPoint& offset) {
122 SkScalar x = SkIntToScalar(offset.x());
123 SkScalar y = SkIntToScalar(offset.y());
124
125 SkPaint paint;
126 paint.setStyle(SkPaint::kStroke_Style);
127
128 SkRect r = SkRect::MakeIWH(image->width(), image->height());
129 r.offset(x, y);
130 // get on pixel-centers to make the hairline land on a numerical stable boundary
131 r.outset(SK_ScalarHalf, SK_ScalarHalf);
132 canvas->drawRect(r, paint);
133
134 canvas->drawImage(image, x, y, nullptr);
135}
136
137typedef SkImageFilter* (*ImageFilterFactory)();
138
139// +[]{...} did not work on windows (VS)
140// (ImageFilterFactory)[]{...} did not work on linux (gcc)
141// hence this cast function
142template <typename T> ImageFilterFactory IFCCast(T arg) { return arg; }
143
reed59dc0d22015-10-19 08:24:21 -0700144// We expect that applying the filter will keep us in the same domain (raster or gpu)
145static void check_same_domain(SkImage* a, SkImage* b) {
146 SkASSERT(a->isTextureBacked() == b->isTextureBacked());
147}
148
reed88d064d2015-10-12 11:30:02 -0700149/**
150 * Compare output of drawSprite and drawBitmap (esp. clipping and imagefilters)
151 */
152class ApplyFilterGM : public skiagm::GM {
153public:
154 ApplyFilterGM() {}
155
156protected:
157 SkString onShortName() override {
158 return SkString("apply-filter");
159 }
160
161 SkISize onISize() override {
reedc837d8f2015-10-12 14:38:34 -0700162 return SkISize::Make(780, 780);
reed88d064d2015-10-12 11:30:02 -0700163 }
164
165 void onDraw(SkCanvas* canvas) override {
166 SkAutoTUnref<SkImage> image0(make_image(canvas));
167
168 const ImageFilterFactory factories[] = {
169 IFCCast([]{ return SkBlurImageFilter::Create(8, 8); }),
170 IFCCast([]{ SkAutoTUnref<SkColorFilter> cf(SkModeColorFilter::Create(SK_ColorBLUE,
171 SkXfermode::kSrcIn_Mode));
172 return SkColorFilterImageFilter::Create(cf);
173 }),
174 IFCCast([]{ return SkDilateImageFilter::Create(8, 8); }),
175 IFCCast([]{ return SkErodeImageFilter::Create(8, 8); }),
176 IFCCast([]{ return SkOffsetImageFilter::Create(8, 8); }),
177 };
178
179 const SkScalar spacer = image0->width() * 3.0f / 2;
180
181 for (auto&& factory : factories) {
182 SkAutoTUnref<SkImageFilter> filter(factory());
183
184 SkIPoint offset1, offset2;
185 SkAutoTUnref<SkImage> image1(image0->applyFilter(filter, &offset1, true));
186 SkAutoTUnref<SkImage> image2(image0->applyFilter(filter, &offset2, false));
187
reed59dc0d22015-10-19 08:24:21 -0700188 check_same_domain(image0, image1);
189 check_same_domain(image0, image2);
190
reed88d064d2015-10-12 11:30:02 -0700191 canvas->save();
192 canvas->translate(30, 30);
193 show_image(canvas, image0, SkIPoint::Make(0, 0)); // original
194 canvas->translate(spacer, 0);
195 show_image(canvas, image1, offset1); // snug
196 canvas->translate(spacer, 0);
197 show_image(canvas, image2, offset2); // not snug
198
199 // Try drawing the original w/ the filter, to see that it "draws" the same as
200 // when we have manually applied the filter (above).
201 {
202 SkPaint paint;
203 paint.setImageFilter(filter);
204
205 SkBitmap bm;
206 image0->asLegacyBitmap(&bm, SkImage::kRO_LegacyBitmapMode);
207 SkPoint loc = { 0, 0 };
208 canvas->translate(spacer, 0);
209 canvas->getTotalMatrix().mapPoints(&loc, 1);
210 canvas->drawSprite(bm, (int)loc.x(), (int)loc.y(), &paint); // like snug
211
212 canvas->translate(spacer, 0);
213 canvas->drawImage(image0, 0, 0, &paint); // like not snug
214 }
215 canvas->restore();
216
217 canvas->translate(0, spacer);
218 }
219 }
220
221private:
222 typedef GM INHERITED;
223};
224DEF_GM( return new ApplyFilterGM; )
225