blob: ecc4dd8160de28536d08f8e72729496d571dd3e5 [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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -04009#include "include/core/SkBitmap.h"
10#include "include/core/SkBlendMode.h"
11#include "include/core/SkCanvas.h"
12#include "include/core/SkColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "include/core/SkColorFilter.h"
14#include "include/core/SkImage.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040015#include "include/core/SkImageFilter.h"
16#include "include/core/SkPaint.h"
17#include "include/core/SkPoint.h"
18#include "include/core/SkRect.h"
19#include "include/core/SkRefCnt.h"
20#include "include/core/SkScalar.h"
21#include "include/core/SkSize.h"
22#include "include/core/SkString.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "include/effects/SkArithmeticImageFilter.h"
24#include "include/effects/SkBlurImageFilter.h"
25#include "include/effects/SkColorFilterImageFilter.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050026#include "include/effects/SkImageSource.h"
27#include "include/effects/SkMatrixConvolutionImageFilter.h"
28#include "include/effects/SkMergeImageFilter.h"
29#include "include/effects/SkMorphologyImageFilter.h"
30#include "include/effects/SkOffsetImageFilter.h"
31#include "include/effects/SkXfermodeImageFilter.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040032#include "tools/ToolUtils.h"
33
34#include <utility>
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000035
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000036class ImageFiltersGraphGM : public skiagm::GM {
37public:
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +000038 ImageFiltersGraphGM() {}
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000039
40protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +000041
robertphillips943a4622015-09-03 13:32:33 -070042 SkString onShortName() override {
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000043 return SkString("imagefiltersgraph");
44 }
45
senorblancoeae84c22016-01-26 08:41:02 -080046 SkISize onISize() override { return SkISize::Make(600, 150); }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000047
robertphillips943a4622015-09-03 13:32:33 -070048 void onOnceBeforeDraw() override {
reed9ce9d672016-03-17 10:51:11 -070049 fImage = SkImage::MakeFromBitmap(
Mike Kleinea3f0142019-03-20 11:12:10 -050050 ToolUtils::create_string_bitmap(100, 100, SK_ColorWHITE, 20, 70, 96, "e"));
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +000051 }
52
robertphillips943a4622015-09-03 13:32:33 -070053 void onDraw(SkCanvas* canvas) override {
mtklein871ad7a2015-03-27 12:33:46 -070054 canvas->clear(SK_ColorBLACK);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000055 {
robertphillips549c8992016-04-01 09:28:51 -070056 sk_sp<SkImageFilter> bitmapSource(SkImageSource::Make(fImage));
Mike Reedb286bc22019-04-08 16:23:20 -040057 sk_sp<SkColorFilter> cf(SkColorFilters::Blend(SK_ColorRED,
Mike Reed7d954ad2016-10-28 15:42:34 -040058 SkBlendMode::kSrcIn));
robertphillips6e7025a2016-04-04 04:31:25 -070059 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(4.0f, 4.0f, std::move(bitmapSource)));
robertphillipsfc11b0a2016-04-05 09:09:36 -070060 sk_sp<SkImageFilter> erode(SkErodeImageFilter::Make(4, 4, blur));
robertphillips5605b562016-04-05 11:50:42 -070061 sk_sp<SkImageFilter> color(SkColorFilterImageFilter::Make(std::move(cf),
62 std::move(erode)));
Mike Reed0bdaf052017-06-18 23:35:57 -040063 sk_sp<SkImageFilter> merge(SkMergeImageFilter::Make(blur, color));
rmistry@google.comae933ce2012-08-23 18:19:56 +000064
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000065 SkPaint paint;
robertphillips2238c9d2016-03-30 13:34:16 -070066 paint.setImageFilter(std::move(merge));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000067 canvas->drawPaint(paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +000068 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000069 }
70 {
robertphillipsfc11b0a2016-04-05 09:09:36 -070071 sk_sp<SkImageFilter> morph(SkDilateImageFilter::Make(5, 5, nullptr));
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000072
Mike Reede869a1e2019-04-30 12:18:54 -040073 float matrix[20] = { 1, 0, 0, 0, 0,
74 0, 1, 0, 0, 0,
75 0, 0, 1, 0, 0,
76 0, 0, 0, 0.5f, 0 };
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000077
Mike Reede869a1e2019-04-30 12:18:54 -040078 sk_sp<SkColorFilter> matrixFilter(SkColorFilters::Matrix(matrix));
robertphillips5605b562016-04-05 11:50:42 -070079 sk_sp<SkImageFilter> colorMorph(SkColorFilterImageFilter::Make(std::move(matrixFilter),
80 std::move(morph)));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000081 SkPaint paint;
reed374772b2016-10-05 17:33:02 -070082 paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kSrcOver,
83 std::move(colorMorph)));
reedcfb6bdf2016-03-29 11:32:50 -070084
reed9ce9d672016-03-17 10:51:11 -070085 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +000086 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000087 }
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000088 {
Mike Reede869a1e2019-04-30 12:18:54 -040089 float matrix[20] = { 1, 0, 0, 0, 0,
90 0, 1, 0, 0, 0,
91 0, 0, 1, 0, 0,
92 0, 0, 0, 0.5f, 0 };
93 sk_sp<SkColorFilter> matrixCF(SkColorFilters::Matrix(matrix));
robertphillips5605b562016-04-05 11:50:42 -070094 sk_sp<SkImageFilter> matrixFilter(SkColorFilterImageFilter::Make(std::move(matrixCF),
95 nullptr));
robertphillips134ff5c2016-04-12 12:41:00 -070096 sk_sp<SkImageFilter> offsetFilter(SkOffsetImageFilter::Make(10.0f, 10.f,
97 matrixFilter));
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000098
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000099 SkPaint paint;
Brian Salomon89cb8212017-01-09 10:48:23 -0500100 paint.setImageFilter(SkArithmeticImageFilter::Make(
101 0, 1, 1, 0, true, std::move(matrixFilter), std::move(offsetFilter), nullptr));
reedcfb6bdf2016-03-29 11:32:50 -0700102
reed9ce9d672016-03-17 10:51:11 -0700103 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000104 canvas->translate(SkIntToScalar(100), 0);
105 }
106 {
robertphillips6e7025a2016-04-04 04:31:25 -0700107 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(SkIntToScalar(10),
108 SkIntToScalar(10),
109 nullptr));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000110
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000111 SkImageFilter::CropRect cropRect(SkRect::MakeWH(SkIntToScalar(95), SkIntToScalar(100)));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000112 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -0700113 paint.setImageFilter(
reed374772b2016-10-05 17:33:02 -0700114 SkXfermodeImageFilter::Make(SkBlendMode::kSrcIn, std::move(blur), nullptr,
115 &cropRect));
reed9ce9d672016-03-17 10:51:11 -0700116 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000117 canvas->translate(SkIntToScalar(100), 0);
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000118 }
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000119 {
senorblancoeae84c22016-01-26 08:41:02 -0800120 // Dilate -> matrix convolution.
121 // This tests that a filter using asFragmentProcessor (matrix
122 // convolution) correctly handles a non-zero source offset
123 // (supplied by the dilate).
robertphillipsfc11b0a2016-04-05 09:09:36 -0700124 sk_sp<SkImageFilter> dilate(SkDilateImageFilter::Make(5, 5, nullptr));
senorblancoeae84c22016-01-26 08:41:02 -0800125
senorblancoeae84c22016-01-26 08:41:02 -0800126 SkScalar kernel[9] = {
127 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
128 SkIntToScalar(-1), SkIntToScalar( 7 ), SkIntToScalar(-1),
129 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
130 };
131 SkISize kernelSize = SkISize::Make(3, 3);
132 SkScalar gain = 1.0f, bias = SkIntToScalar(0);
133 SkIPoint kernelOffset = SkIPoint::Make(1, 1);
134 auto tileMode = SkMatrixConvolutionImageFilter::kClamp_TileMode;
135 bool convolveAlpha = false;
robertphillipsef6a47b2016-04-08 08:01:20 -0700136 sk_sp<SkImageFilter> convolve(SkMatrixConvolutionImageFilter::Make(kernelSize,
137 kernel,
138 gain,
139 bias,
140 kernelOffset,
141 tileMode,
142 convolveAlpha,
143 std::move(dilate)));
senorblancoeae84c22016-01-26 08:41:02 -0800144
145 SkPaint paint;
robertphillipsfc11b0a2016-04-05 09:09:36 -0700146 paint.setImageFilter(std::move(convolve));
reed9ce9d672016-03-17 10:51:11 -0700147 DrawClippedImage(canvas, fImage.get(), paint);
senorblancoeae84c22016-01-26 08:41:02 -0800148 canvas->translate(SkIntToScalar(100), 0);
149 }
150 {
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000151 // Test that crop offsets are absolute, not relative to the parent's crop rect.
Mike Reedb286bc22019-04-08 16:23:20 -0400152 sk_sp<SkColorFilter> cf1(SkColorFilters::Blend(SK_ColorBLUE, SkBlendMode::kSrcIn));
153 sk_sp<SkColorFilter> cf2(SkColorFilters::Blend(SK_ColorGREEN, SkBlendMode::kSrcIn));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000154 SkImageFilter::CropRect outerRect(SkRect::MakeXYWH(SkIntToScalar(10), SkIntToScalar(10),
155 SkIntToScalar(80), SkIntToScalar(80)));
156 SkImageFilter::CropRect innerRect(SkRect::MakeXYWH(SkIntToScalar(20), SkIntToScalar(20),
157 SkIntToScalar(60), SkIntToScalar(60)));
robertphillips5605b562016-04-05 11:50:42 -0700158 sk_sp<SkImageFilter> color1(SkColorFilterImageFilter::Make(std::move(cf1),
159 nullptr,
160 &outerRect));
161 sk_sp<SkImageFilter> color2(SkColorFilterImageFilter::Make(std::move(cf2),
162 std::move(color1),
163 &innerRect));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000164
165 SkPaint paint;
robertphillipsfc11b0a2016-04-05 09:09:36 -0700166 paint.setImageFilter(std::move(color2));
mtklein871ad7a2015-03-27 12:33:46 -0700167 paint.setColor(SK_ColorRED);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000168 canvas->drawRect(SkRect::MakeXYWH(0, 0, 100, 100), paint);
169 canvas->translate(SkIntToScalar(100), 0);
170 }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000171 }
172
173private:
fmalita5598b632015-09-15 11:26:13 -0700174 static void DrawClippedImage(SkCanvas* canvas, const SkImage* image, const SkPaint& paint) {
175 canvas->save();
176 canvas->clipRect(SkRect::MakeIWH(image->width(), image->height()));
177 canvas->drawImage(image, 0, 0, &paint);
178 canvas->restore();
179 }
180
reed9ce9d672016-03-17 10:51:11 -0700181 sk_sp<SkImage> fImage;
fmalita5598b632015-09-15 11:26:13 -0700182
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000183 typedef GM INHERITED;
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000184};
185
186///////////////////////////////////////////////////////////////////////////////
187
robertphillips943a4622015-09-03 13:32:33 -0700188DEF_GM(return new ImageFiltersGraphGM;)