blob: d9ab4d025ed3ba2db30f793248c5f2acd8663f7a [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"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000011#include "SkBlurImageFilter.h"
senorblanco@chromium.org44888c62012-08-20 19:23:24 +000012#include "SkColorFilter.h"
13#include "SkColorFilterImageFilter.h"
commit-bot@chromium.org6c4e71a2013-11-20 21:32:10 +000014#include "SkColorMatrixFilter.h"
fmalita5598b632015-09-15 11:26:13 -070015#include "SkImage.h"
16#include "SkImageSource.h"
senorblancoeae84c22016-01-26 08:41:02 -080017#include "SkMatrixConvolutionImageFilter.h"
robertphillips134ff5c2016-04-12 12:41:00 -070018#include "SkOffsetImageFilter.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000019#include "SkReadBuffer.h"
robertphillips4ba94e22016-04-04 12:07:47 -070020#include "SkSpecialImage.h"
21#include "SkSpecialSurface.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000022#include "SkWriteBuffer.h"
senorblanco@chromium.org4a9a6122012-12-04 14:18:50 +000023#include "SkMergeImageFilter.h"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000024#include "SkMorphologyImageFilter.h"
commit-bot@chromium.org6c4e71a2013-11-20 21:32:10 +000025#include "SkTestImageFilters.h"
senorblanco@chromium.org350b4d52013-08-01 14:59:05 +000026#include "SkXfermodeImageFilter.h"
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000027
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000028class ImageFiltersGraphGM : public skiagm::GM {
29public:
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +000030 ImageFiltersGraphGM() {}
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000031
32protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +000033
robertphillips943a4622015-09-03 13:32:33 -070034 SkString onShortName() override {
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000035 return SkString("imagefiltersgraph");
36 }
37
senorblancoeae84c22016-01-26 08:41:02 -080038 SkISize onISize() override { return SkISize::Make(600, 150); }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000039
robertphillips943a4622015-09-03 13:32:33 -070040 void onOnceBeforeDraw() override {
reed9ce9d672016-03-17 10:51:11 -070041 fImage = SkImage::MakeFromBitmap(
42 sk_tool_utils::create_string_bitmap(100, 100, SK_ColorWHITE, 20, 70, 96, "e"));
commit-bot@chromium.org60c8d242014-05-27 16:28:43 +000043 }
44
robertphillips943a4622015-09-03 13:32:33 -070045 void onDraw(SkCanvas* canvas) override {
mtklein871ad7a2015-03-27 12:33:46 -070046 canvas->clear(SK_ColorBLACK);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000047 {
robertphillips549c8992016-04-01 09:28:51 -070048 sk_sp<SkImageFilter> bitmapSource(SkImageSource::Make(fImage));
robertphillips2238c9d2016-03-30 13:34:16 -070049 sk_sp<SkColorFilter> cf(SkColorFilter::MakeModeFilter(SK_ColorRED,
50 SkXfermode::kSrcIn_Mode));
robertphillips6e7025a2016-04-04 04:31:25 -070051 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(4.0f, 4.0f, std::move(bitmapSource)));
robertphillipsfc11b0a2016-04-05 09:09:36 -070052 sk_sp<SkImageFilter> erode(SkErodeImageFilter::Make(4, 4, blur));
robertphillips5605b562016-04-05 11:50:42 -070053 sk_sp<SkImageFilter> color(SkColorFilterImageFilter::Make(std::move(cf),
54 std::move(erode)));
robertphillips2238c9d2016-03-30 13:34:16 -070055 sk_sp<SkImageFilter> merge(SkMergeImageFilter::Make(blur, color));
rmistry@google.comae933ce2012-08-23 18:19:56 +000056
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000057 SkPaint paint;
robertphillips2238c9d2016-03-30 13:34:16 -070058 paint.setImageFilter(std::move(merge));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000059 canvas->drawPaint(paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +000060 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000061 }
62 {
robertphillipsfc11b0a2016-04-05 09:09:36 -070063 sk_sp<SkImageFilter> morph(SkDilateImageFilter::Make(5, 5, nullptr));
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000064
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000065 SkScalar matrix[20] = { SK_Scalar1, 0, 0, 0, 0,
66 0, SK_Scalar1, 0, 0, 0,
67 0, 0, SK_Scalar1, 0, 0,
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +000068 0, 0, 0, 0.5f, 0 };
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +000069
robertphillips43c2ad42016-04-04 05:05:11 -070070 sk_sp<SkColorFilter> matrixFilter(SkColorFilter::MakeMatrixFilterRowMajor255(matrix));
robertphillips5605b562016-04-05 11:50:42 -070071 sk_sp<SkImageFilter> colorMorph(SkColorFilterImageFilter::Make(std::move(matrixFilter),
72 std::move(morph)));
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000073 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -070074 paint.setImageFilter(SkXfermodeImageFilter::Make(
robertphillips43c2ad42016-04-04 05:05:11 -070075 SkXfermode::Make(SkXfermode::kSrcOver_Mode),
robertphillips8c0326d2016-04-05 12:48:34 -070076 std::move(colorMorph)));
reedcfb6bdf2016-03-29 11:32:50 -070077
reed9ce9d672016-03-17 10:51:11 -070078 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +000079 canvas->translate(SkIntToScalar(100), 0);
senorblanco@chromium.org985fa792012-10-24 15:14:26 +000080 }
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000081 {
82 SkScalar matrix[20] = { SK_Scalar1, 0, 0, 0, 0,
83 0, SK_Scalar1, 0, 0, 0,
84 0, 0, SK_Scalar1, 0, 0,
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +000085 0, 0, 0, 0.5f, 0 };
robertphillips43c2ad42016-04-04 05:05:11 -070086 sk_sp<SkColorFilter> matrixCF(SkColorFilter::MakeMatrixFilterRowMajor255(matrix));
robertphillips5605b562016-04-05 11:50:42 -070087 sk_sp<SkImageFilter> matrixFilter(SkColorFilterImageFilter::Make(std::move(matrixCF),
88 nullptr));
robertphillips134ff5c2016-04-12 12:41:00 -070089 sk_sp<SkImageFilter> offsetFilter(SkOffsetImageFilter::Make(10.0f, 10.f,
90 matrixFilter));
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000091
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +000092 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -070093 paint.setImageFilter(
94 SkXfermodeImageFilter::Make(SkArithmeticMode::Make(0, SK_Scalar1, SK_Scalar1, 0),
robertphillips8c0326d2016-04-05 12:48:34 -070095 std::move(matrixFilter),
96 std::move(offsetFilter),
97 nullptr));
reedcfb6bdf2016-03-29 11:32:50 -070098
reed9ce9d672016-03-17 10:51:11 -070099 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000100 canvas->translate(SkIntToScalar(100), 0);
101 }
102 {
robertphillips6e7025a2016-04-04 04:31:25 -0700103 sk_sp<SkImageFilter> blur(SkBlurImageFilter::Make(SkIntToScalar(10),
104 SkIntToScalar(10),
105 nullptr));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000106
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000107 SkImageFilter::CropRect cropRect(SkRect::MakeWH(SkIntToScalar(95), SkIntToScalar(100)));
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000108 SkPaint paint;
reedcfb6bdf2016-03-29 11:32:50 -0700109 paint.setImageFilter(
robertphillips8c0326d2016-04-05 12:48:34 -0700110 SkXfermodeImageFilter::Make(SkXfermode::Make(SkXfermode::kSrcIn_Mode),
111 std::move(blur),
reedcfb6bdf2016-03-29 11:32:50 -0700112 nullptr, &cropRect));
reed9ce9d672016-03-17 10:51:11 -0700113 DrawClippedImage(canvas, fImage.get(), paint);
senorblanco@chromium.org34a849d2013-10-24 15:59:31 +0000114 canvas->translate(SkIntToScalar(100), 0);
commit-bot@chromium.orgbbfe4542013-10-24 01:46:11 +0000115 }
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000116 {
senorblancoeae84c22016-01-26 08:41:02 -0800117 // Dilate -> matrix convolution.
118 // This tests that a filter using asFragmentProcessor (matrix
119 // convolution) correctly handles a non-zero source offset
120 // (supplied by the dilate).
robertphillipsfc11b0a2016-04-05 09:09:36 -0700121 sk_sp<SkImageFilter> dilate(SkDilateImageFilter::Make(5, 5, nullptr));
senorblancoeae84c22016-01-26 08:41:02 -0800122
senorblancoeae84c22016-01-26 08:41:02 -0800123 SkScalar kernel[9] = {
124 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
125 SkIntToScalar(-1), SkIntToScalar( 7 ), SkIntToScalar(-1),
126 SkIntToScalar(-1), SkIntToScalar( -1 ), SkIntToScalar(-1),
127 };
128 SkISize kernelSize = SkISize::Make(3, 3);
129 SkScalar gain = 1.0f, bias = SkIntToScalar(0);
130 SkIPoint kernelOffset = SkIPoint::Make(1, 1);
131 auto tileMode = SkMatrixConvolutionImageFilter::kClamp_TileMode;
132 bool convolveAlpha = false;
robertphillipsef6a47b2016-04-08 08:01:20 -0700133 sk_sp<SkImageFilter> convolve(SkMatrixConvolutionImageFilter::Make(kernelSize,
134 kernel,
135 gain,
136 bias,
137 kernelOffset,
138 tileMode,
139 convolveAlpha,
140 std::move(dilate)));
senorblancoeae84c22016-01-26 08:41:02 -0800141
142 SkPaint paint;
robertphillipsfc11b0a2016-04-05 09:09:36 -0700143 paint.setImageFilter(std::move(convolve));
reed9ce9d672016-03-17 10:51:11 -0700144 DrawClippedImage(canvas, fImage.get(), paint);
senorblancoeae84c22016-01-26 08:41:02 -0800145 canvas->translate(SkIntToScalar(100), 0);
146 }
147 {
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000148 // Test that crop offsets are absolute, not relative to the parent's crop rect.
robertphillips5605b562016-04-05 11:50:42 -0700149 sk_sp<SkColorFilter> cf1(SkColorFilter::MakeModeFilter(SK_ColorBLUE,
150 SkXfermode::kSrcIn_Mode));
151 sk_sp<SkColorFilter> cf2(SkColorFilter::MakeModeFilter(SK_ColorGREEN,
152 SkXfermode::kSrcIn_Mode));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000153 SkImageFilter::CropRect outerRect(SkRect::MakeXYWH(SkIntToScalar(10), SkIntToScalar(10),
154 SkIntToScalar(80), SkIntToScalar(80)));
155 SkImageFilter::CropRect innerRect(SkRect::MakeXYWH(SkIntToScalar(20), SkIntToScalar(20),
156 SkIntToScalar(60), SkIntToScalar(60)));
robertphillips5605b562016-04-05 11:50:42 -0700157 sk_sp<SkImageFilter> color1(SkColorFilterImageFilter::Make(std::move(cf1),
158 nullptr,
159 &outerRect));
160 sk_sp<SkImageFilter> color2(SkColorFilterImageFilter::Make(std::move(cf2),
161 std::move(color1),
162 &innerRect));
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000163
164 SkPaint paint;
robertphillipsfc11b0a2016-04-05 09:09:36 -0700165 paint.setImageFilter(std::move(color2));
mtklein871ad7a2015-03-27 12:33:46 -0700166 paint.setColor(SK_ColorRED);
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000167 canvas->drawRect(SkRect::MakeXYWH(0, 0, 100, 100), paint);
168 canvas->translate(SkIntToScalar(100), 0);
169 }
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000170 }
171
172private:
fmalita5598b632015-09-15 11:26:13 -0700173 static void DrawClippedImage(SkCanvas* canvas, const SkImage* image, const SkPaint& paint) {
174 canvas->save();
175 canvas->clipRect(SkRect::MakeIWH(image->width(), image->height()));
176 canvas->drawImage(image, 0, 0, &paint);
177 canvas->restore();
178 }
179
reed9ce9d672016-03-17 10:51:11 -0700180 sk_sp<SkImage> fImage;
fmalita5598b632015-09-15 11:26:13 -0700181
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000182 typedef GM INHERITED;
senorblanco@chromium.orgf1369ce2012-08-20 14:53:21 +0000183};
184
185///////////////////////////////////////////////////////////////////////////////
186
robertphillips943a4622015-09-03 13:32:33 -0700187DEF_GM(return new ImageFiltersGraphGM;)