blob: b7402eea1b422b71f32cda239d8dc92e8686d1a1 [file] [log] [blame]
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +00001/*
2 * 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 */
7
8#include "gm.h"
9
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000010#include "SkArithmeticMode.h"
commit-bot@chromium.orgc3bd8af2014-02-13 17:14:46 +000011#include "SkDevice.h"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000012#include "SkBlurImageFilter.h"
senorblanco@chromium.org44888c62012-08-20 19:23:24 +000013#include "SkColorFilter.h"
14#include "SkColorFilterImageFilter.h"
commit-bot@chromium.org6c4e71a2013-11-20 21:32:10 +000015#include "SkColorMatrixFilter.h"
fmalita5598b632015-09-15 11:26:13 -070016#include "SkImage.h"
17#include "SkImageSource.h"
senorblancoeae84c22016-01-26 08:41:02 -080018#include "SkMatrixConvolutionImageFilter.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000019#include "SkReadBuffer.h"
20#include "SkWriteBuffer.h"
senorblanco@chromium.org4a9a6122012-12-04 14:18:50 +000021#include "SkMergeImageFilter.h"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000022#include "SkMorphologyImageFilter.h"
commit-bot@chromium.org6c4e71a2013-11-20 21:32:10 +000023#include "SkTestImageFilters.h"
senorblanco@chromium.org350b4d52013-08-01 14:59:05 +000024#include "SkXfermodeImageFilter.h"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000025
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000026// More closely models how Blink's OffsetFilter works as of 10/23/13. SkOffsetImageFilter doesn't
27// perform a draw and this one does.
28class SimpleOffsetFilter : public SkImageFilter {
29public:
reed7daaaa42014-08-21 10:53:34 -070030 class Registrar {
31 public:
32 Registrar() {
33 SkFlattenable::Register("SimpleOffsetFilter",
reed7daaaa42014-08-21 10:53:34 -070034 SimpleOffsetFilter::CreateProc,
reed7daaaa42014-08-21 10:53:34 -070035 SimpleOffsetFilter::GetFlattenableType());
36 }
37 };
commit-bot@chromium.org20774272014-03-18 10:28:27 +000038 static SkImageFilter* Create(SkScalar dx, SkScalar dy, SkImageFilter* input) {
halcanary385fe4d2015-08-26 13:07:48 -070039 return new SimpleOffsetFilter(dx, dy, input);
commit-bot@chromium.org20774272014-03-18 10:28:27 +000040 }
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000041
robertphillips48e78462016-02-17 13:57:16 -080042 bool onFilterImageDeprecated(Proxy* proxy, const SkBitmap& src, const Context& ctx,
43 SkBitmap* dst, SkIPoint* offset) const override {
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000044 SkBitmap source = src;
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000045 SkIPoint srcOffset = SkIPoint::Make(0, 0);
robertphillips48e78462016-02-17 13:57:16 -080046 if (!this->filterInputDeprecated(0, proxy, src, ctx, &source, &srcOffset)) {
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000047 return false;
48 }
49
50 SkIRect bounds;
robertphillips48e78462016-02-17 13:57:16 -080051 if (!this->applyCropRectDeprecated(ctx, proxy, source, &srcOffset, &bounds, &source)) {
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000052 return false;
53 }
54
55 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
56 SkCanvas canvas(device);
57 SkPaint paint;
58 paint.setXfermodeMode(SkXfermode::kSrc_Mode);
59 canvas.drawBitmap(source, fDX - bounds.left(), fDY - bounds.top(), &paint);
60 *dst = device->accessBitmap(false);
61 offset->fX += bounds.left();
62 offset->fY += bounds.top();
63 return true;
64 }
65
robertphillipsf3f5bad2014-12-19 13:49:15 -080066 SK_TO_STRING_OVERRIDE()
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000067 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SimpleOffsetFilter);
68
69protected:
mtklein36352bf2015-03-25 18:17:31 -070070 void flatten(SkWriteBuffer& buffer) const override {
reed9fa60da2014-08-21 07:59:51 -070071 this->INHERITED::flatten(buffer);
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000072 buffer.writeScalar(fDX);
73 buffer.writeScalar(fDY);
74 }
75
76private:
commit-bot@chromium.org20774272014-03-18 10:28:27 +000077 SimpleOffsetFilter(SkScalar dx, SkScalar dy, SkImageFilter* input)
reed7daaaa42014-08-21 10:53:34 -070078 : SkImageFilter(1, &input), fDX(dx), fDY(dy) {}
commit-bot@chromium.org20774272014-03-18 10:28:27 +000079
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000080 SkScalar fDX, fDY;
reed9fa60da2014-08-21 07:59:51 -070081
82 typedef SkImageFilter INHERITED;
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000083};
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000084
reed7daaaa42014-08-21 10:53:34 -070085static SimpleOffsetFilter::Registrar gReg;
86
reed60c9b582016-04-03 09:11:13 -070087sk_sp<SkFlattenable> SimpleOffsetFilter::CreateProc(SkReadBuffer& buffer) {
reed9fa60da2014-08-21 07:59:51 -070088 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
89 SkScalar dx = buffer.readScalar();
90 SkScalar dy = buffer.readScalar();
reed60c9b582016-04-03 09:11:13 -070091 return sk_sp<SkFlattenable>(Create(dx, dy, common.getInput(0).get()));
reed9fa60da2014-08-21 07:59:51 -070092}
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000093
robertphillipsf3f5bad2014-12-19 13:49:15 -080094#ifndef SK_IGNORE_TO_STRING
95void SimpleOffsetFilter::toString(SkString* str) const {
96 str->appendf("SimpleOffsetFilter: (");
97 str->append(")");
98}
99#endif
100
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000101class ImageFiltersGraphGM : public skiagm::GM {
102public:
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +0000103 ImageFiltersGraphGM() {}
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000104
105protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000106
robertphillips943a4622015-09-03 13:32:33 -0700107 SkString onShortName() override {
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000108 return SkString("imagefiltersgraph");
109 }
110
senorblancoeae84c22016-01-26 08:41:02 -0800111 SkISize onISize() override { return SkISize::Make(600, 150); }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000112
robertphillips943a4622015-09-03 13:32:33 -0700113 void onOnceBeforeDraw() override {
reed9ce9d672016-03-17 10:51:11 -0700114 fImage = SkImage::MakeFromBitmap(
115 sk_tool_utils::create_string_bitmap(100, 100, SK_ColorWHITE, 20, 70, 96, "e"));
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +0000116 }
117
robertphillips943a4622015-09-03 13:32:33 -0700118 void onDraw(SkCanvas* canvas) override {
mtklein871ad7a2015-03-27 12:33:46 -0700119 canvas->clear(SK_ColorBLACK);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000120 {
robertphillips549c8992016-04-01 09:28:51 -0700121 sk_sp<SkImageFilter> bitmapSource(SkImageSource::Make(fImage));
robertphillips2238c9d2016-03-30 13:34:16 -0700122 sk_sp<SkColorFilter> cf(SkColorFilter::MakeModeFilter(SK_ColorRED,
123 SkXfermode::kSrcIn_Mode));
robertphillips6e7025a2016-04-04 04:31:25 -0700124 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(4.0f, 4.0f, std::move(bitmapSource)));
robertphillips2238c9d2016-03-30 13:34:16 -0700125 sk_sp<SkImageFilter> erode(SkErodeImageFilter::Create(4, 4, blur.get()));
126 sk_sp<SkImageFilter> color(SkColorFilterImageFilter::Create(cf.get(), erode.get()));
127 sk_sp<SkImageFilter> merge(SkMergeImageFilter::Make(blur, color));
rmistry@google.comae933ce2012-08-23 18:19:56 +0000128
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000129 SkPaint paint;
robertphillips2238c9d2016-03-30 13:34:16 -0700130 paint.setImageFilter(std::move(merge));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000131 canvas->drawPaint(paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000132 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000133 }
134 {
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000135 SkAutoTUnref<SkImageFilter> morph(SkDilateImageFilter::Create(5, 5));
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000136
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000137 SkScalar matrix[20] = { SK_Scalar1, 0, 0, 0, 0,
138 0, SK_Scalar1, 0, 0, 0,
139 0, 0, SK_Scalar1, 0, 0,
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000140 0, 0, 0, 0.5f, 0 };
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000141
reedd053ce92016-03-22 10:17:23 -0700142 auto matrixFilter(SkColorFilter::MakeMatrixFilterRowMajor255(matrix));
143 SkAutoTUnref<SkImageFilter> colorMorph(SkColorFilterImageFilter::Create(matrixFilter.get(), morph));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000144 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -0700145 paint.setImageFilter(SkXfermodeImageFilter::Make(
146 SkXfermode::Make(SkXfermode::kSrcOver_Mode), colorMorph));
147
reed9ce9d672016-03-17 10:51:11 -0700148 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000149 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +0000150 }
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000151 {
152 SkScalar matrix[20] = { SK_Scalar1, 0, 0, 0, 0,
153 0, SK_Scalar1, 0, 0, 0,
154 0, 0, SK_Scalar1, 0, 0,
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000155 0, 0, 0, 0.5f, 0 };
reedd053ce92016-03-22 10:17:23 -0700156 auto matrixCF(SkColorFilter::MakeMatrixFilterRowMajor255(matrix));
157 SkAutoTUnref<SkImageFilter> matrixFilter(SkColorFilterImageFilter::Create(matrixCF.get()));
commit-bot@chromium.org20774272014-03-18 10:28:27 +0000158 SkAutoTUnref<SkImageFilter> offsetFilter(
159 SimpleOffsetFilter::Create(10.0f, 10.f, matrixFilter));
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000160
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000161 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -0700162 paint.setImageFilter(
163 SkXfermodeImageFilter::Make(SkArithmeticMode::Make(0, SK_Scalar1, SK_Scalar1, 0),
164 matrixFilter, offsetFilter, nullptr));
165
reed9ce9d672016-03-17 10:51:11 -0700166 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000167 canvas->translate(SkIntToScalar(100), 0);
168 }
169 {
robertphillips6e7025a2016-04-04 04:31:25 -0700170 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(SkIntToScalar(10),
171 SkIntToScalar(10),
172 nullptr));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000173
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000174 SkImageFilter::CropRect cropRect(SkRect::MakeWH(SkIntToScalar(95), SkIntToScalar(100)));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000175 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -0700176 paint.setImageFilter(
robertphillips6e7025a2016-04-04 04:31:25 -0700177 SkXfermodeImageFilter::Make(SkXfermode::Make(SkXfermode::kSrcIn_Mode), blur.get(),
reedcfb6bdf2016-03-29 11:32:50 -0700178 nullptr, &cropRect));
reed9ce9d672016-03-17 10:51:11 -0700179 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000180 canvas->translate(SkIntToScalar(100), 0);
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000181 }
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000182 {
senorblancoeae84c22016-01-26 08:41:02 -0800183 // Dilate -> matrix convolution.
184 // This tests that a filter using asFragmentProcessor (matrix
185 // convolution) correctly handles a non-zero source offset
186 // (supplied by the dilate).
187 SkAutoTUnref<SkImageFilter> dilate(SkDilateImageFilter::Create(5, 5));
188
senorblancoeae84c22016-01-26 08:41:02 -0800189 SkScalar kernel[9] = {
190 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
191 SkIntToScalar(-1), SkIntToScalar( 7 ), SkIntToScalar(-1),
192 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
193 };
194 SkISize kernelSize = SkISize::Make(3, 3);
195 SkScalar gain = 1.0f, bias = SkIntToScalar(0);
196 SkIPoint kernelOffset = SkIPoint::Make(1, 1);
197 auto tileMode = SkMatrixConvolutionImageFilter::kClamp_TileMode;
198 bool convolveAlpha = false;
199 SkAutoTUnref<SkImageFilter> convolve(
200 SkMatrixConvolutionImageFilter::Create(kernelSize,
201 kernel,
202 gain,
203 bias,
204 kernelOffset,
205 tileMode,
206 convolveAlpha,
207 dilate));
208
209 SkPaint paint;
210 paint.setImageFilter(convolve);
reed9ce9d672016-03-17 10:51:11 -0700211 DrawClippedImage(canvas, fImage.get(), paint);
senorblancoeae84c22016-01-26 08:41:02 -0800212 canvas->translate(SkIntToScalar(100), 0);
213 }
214 {
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000215 // Test that crop offsets are absolute, not relative to the parent's crop rect.
reedd053ce92016-03-22 10:17:23 -0700216 auto cf1(SkColorFilter::MakeModeFilter(SK_ColorBLUE, SkXfermode::kSrcIn_Mode));
217 auto cf2(SkColorFilter::MakeModeFilter(SK_ColorGREEN, SkXfermode::kSrcIn_Mode));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000218 SkImageFilter::CropRect outerRect(SkRect::MakeXYWH(SkIntToScalar(10), SkIntToScalar(10),
219 SkIntToScalar(80), SkIntToScalar(80)));
220 SkImageFilter::CropRect innerRect(SkRect::MakeXYWH(SkIntToScalar(20), SkIntToScalar(20),
221 SkIntToScalar(60), SkIntToScalar(60)));
reedd053ce92016-03-22 10:17:23 -0700222 SkAutoTUnref<SkImageFilter> color1(SkColorFilterImageFilter::Create(cf1.get(), nullptr, &outerRect));
223 SkAutoTUnref<SkImageFilter> color2(SkColorFilterImageFilter::Create(cf2.get(), color1, &innerRect));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000224
225 SkPaint paint;
226 paint.setImageFilter(color2);
mtklein871ad7a2015-03-27 12:33:46 -0700227 paint.setColor(SK_ColorRED);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000228 canvas->drawRect(SkRect::MakeXYWH(0, 0, 100, 100), paint);
229 canvas->translate(SkIntToScalar(100), 0);
230 }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000231 }
232
233private:
fmalita5598b632015-09-15 11:26:13 -0700234 static void DrawClippedImage(SkCanvas* canvas, const SkImage* image, const SkPaint& paint) {
235 canvas->save();
236 canvas->clipRect(SkRect::MakeIWH(image->width(), image->height()));
237 canvas->drawImage(image, 0, 0, &paint);
238 canvas->restore();
239 }
240
reed9ce9d672016-03-17 10:51:11 -0700241 sk_sp<SkImage> fImage;
fmalita5598b632015-09-15 11:26:13 -0700242
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000243 typedef GM INHERITED;
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000244};
245
246///////////////////////////////////////////////////////////////////////////////
247
robertphillips943a4622015-09-03 13:32:33 -0700248DEF_GM(return new ImageFiltersGraphGM;)