blob: 3293c81ba4ba49c12edea15e754436426215fed4 [file] [log] [blame]
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +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"
halcanaryb0cce2c2015-01-26 12:49:00 -08009#include "sk_tool_utils.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000010#include "SkArithmeticMode.h"
fmalita5598b632015-09-15 11:26:13 -070011#include "SkImage.h"
12#include "SkImageSource.h"
commit-bot@chromium.org7b320702013-07-10 21:22:18 +000013#include "SkOffsetImageFilter.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000014#include "SkXfermodeImageFilter.h"
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000015
16#define WIDTH 600
senorblanco06d54ad2016-01-13 13:48:54 -080017#define HEIGHT 700
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000018#define MARGIN 12
19
20namespace skiagm {
21
22class XfermodeImageFilterGM : public GM {
23public:
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +000024 XfermodeImageFilterGM(){
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000025 this->setBGColor(0xFF000000);
26 }
27
28protected:
mtklein36352bf2015-03-25 18:17:31 -070029 SkString onShortName() override {
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000030 return SkString("xfermodeimagefilter");
31 }
32
mtklein36352bf2015-03-25 18:17:31 -070033 SkISize onISize() override {
tfarinaf5393182014-06-09 23:59:03 -070034 return SkISize::Make(WIDTH, HEIGHT);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000035 }
36
mtklein36352bf2015-03-25 18:17:31 -070037 void onOnceBeforeDraw() override {
robertphillips943a4622015-09-03 13:32:33 -070038 fBitmap = sk_tool_utils::create_string_bitmap(80, 80, 0xD000D000, 15, 65, 96, "e");
halcanary878fa022015-01-26 11:24:32 -080039
reed9ce9d672016-03-17 10:51:11 -070040 fCheckerboard = SkImage::MakeFromBitmap(
fmalita5598b632015-09-15 11:26:13 -070041 sk_tool_utils::create_checkerboard_bitmap(80, 80,
42 sk_tool_utils::color_to_565(0xFFA0A0A0),
43 sk_tool_utils::color_to_565(0xFF404040),
reed9ce9d672016-03-17 10:51:11 -070044 8));
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +000045 }
46
mtklein36352bf2015-03-25 18:17:31 -070047 void onDraw(SkCanvas* canvas) override {
senorblanco16b254a2015-04-09 11:13:24 -070048 canvas->clear(SK_ColorBLACK);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000049 SkPaint paint;
50
51 const struct {
52 SkXfermode::Mode fMode;
53 const char* fLabel;
54 } gModes[] = {
55 { SkXfermode::kClear_Mode, "Clear" },
56 { SkXfermode::kSrc_Mode, "Src" },
57 { SkXfermode::kDst_Mode, "Dst" },
58 { SkXfermode::kSrcOver_Mode, "SrcOver" },
59 { SkXfermode::kDstOver_Mode, "DstOver" },
60 { SkXfermode::kSrcIn_Mode, "SrcIn" },
61 { SkXfermode::kDstIn_Mode, "DstIn" },
62 { SkXfermode::kSrcOut_Mode, "SrcOut" },
63 { SkXfermode::kDstOut_Mode, "DstOut" },
64 { SkXfermode::kSrcATop_Mode, "SrcATop" },
65 { SkXfermode::kDstATop_Mode, "DstATop" },
66 { SkXfermode::kXor_Mode, "Xor" },
67
68 { SkXfermode::kPlus_Mode, "Plus" },
69 { SkXfermode::kModulate_Mode, "Modulate" },
70 { SkXfermode::kScreen_Mode, "Screen" },
71 { SkXfermode::kOverlay_Mode, "Overlay" },
72 { SkXfermode::kDarken_Mode, "Darken" },
73 { SkXfermode::kLighten_Mode, "Lighten" },
74 { SkXfermode::kColorDodge_Mode, "ColorDodge" },
75 { SkXfermode::kColorBurn_Mode, "ColorBurn" },
76 { SkXfermode::kHardLight_Mode, "HardLight" },
77 { SkXfermode::kSoftLight_Mode, "SoftLight" },
78 { SkXfermode::kDifference_Mode, "Difference" },
79 { SkXfermode::kExclusion_Mode, "Exclusion" },
80 { SkXfermode::kMultiply_Mode, "Multiply" },
81 { SkXfermode::kHue_Mode, "Hue" },
82 { SkXfermode::kSaturation_Mode, "Saturation" },
83 { SkXfermode::kColor_Mode, "Color" },
84 { SkXfermode::kLuminosity_Mode, "Luminosity" },
85 };
86
87 int x = 0, y = 0;
robertphillips549c8992016-04-01 09:28:51 -070088 sk_sp<SkImageFilter> background(SkImageSource::Make(fCheckerboard));
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000089 for (size_t i = 0; i < SK_ARRAY_COUNT(gModes); i++) {
reedcfb6bdf2016-03-29 11:32:50 -070090 paint.setImageFilter(SkXfermodeImageFilter::Make(SkXfermode::Make(gModes[i].fMode),
robertphillips51a315e2016-03-31 09:05:49 -070091 background.get()));
fmalita5598b632015-09-15 11:26:13 -070092 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000093 x += fBitmap.width() + MARGIN;
94 if (x + fBitmap.width() > WIDTH) {
95 x = 0;
96 y += fBitmap.height() + MARGIN;
97 }
98 }
99 // Test arithmetic mode as image filter
reedcfb6bdf2016-03-29 11:32:50 -0700100 paint.setImageFilter(SkXfermodeImageFilter::Make(
robertphillips549c8992016-04-01 09:28:51 -0700101 SkArithmeticMode::Make(0, SK_Scalar1, SK_Scalar1, 0), background.get()));
fmalita5598b632015-09-15 11:26:13 -0700102 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000103 x += fBitmap.width() + MARGIN;
104 if (x + fBitmap.width() > WIDTH) {
105 x = 0;
106 y += fBitmap.height() + MARGIN;
107 }
halcanary96fcdcc2015-08-27 07:41:13 -0700108 // Test nullptr mode
robertphillips51a315e2016-03-31 09:05:49 -0700109 paint.setImageFilter(SkXfermodeImageFilter::Make(nullptr, background.get()));
fmalita5598b632015-09-15 11:26:13 -0700110 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000111 x += fBitmap.width() + MARGIN;
112 if (x + fBitmap.width() > WIDTH) {
113 x = 0;
114 y += fBitmap.height() + MARGIN;
115 }
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000116 SkRect clipRect = SkRect::MakeWH(SkIntToScalar(fBitmap.width() + 4),
117 SkIntToScalar(fBitmap.height() + 4));
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000118 // Test offsets on SrcMode (uses fixed-function blend)
reed9ce9d672016-03-17 10:51:11 -0700119 sk_sp<SkImage> bitmapImage(SkImage::MakeFromBitmap(fBitmap));
robertphillips549c8992016-04-01 09:28:51 -0700120 sk_sp<SkImageFilter> foreground(SkImageSource::Make(std::move(bitmapImage)));
robertphillips51a315e2016-03-31 09:05:49 -0700121 sk_sp<SkImageFilter> offsetForeground(SkOffsetImageFilter::Make(SkIntToScalar(4),
122 SkIntToScalar(-4),
123 foreground));
124 sk_sp<SkImageFilter> offsetBackground(SkOffsetImageFilter::Make(SkIntToScalar(4),
125 SkIntToScalar(4),
126 background));
reedcfb6bdf2016-03-29 11:32:50 -0700127 paint.setImageFilter(SkXfermodeImageFilter::Make(
robertphillips51a315e2016-03-31 09:05:49 -0700128 SkXfermode::Make(SkXfermode::kSrcOver_Mode),
129 offsetBackground.get(),
130 offsetForeground.get(),
131 nullptr));
fmalita5598b632015-09-15 11:26:13 -0700132 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000133 x += fBitmap.width() + MARGIN;
134 if (x + fBitmap.width() > WIDTH) {
135 x = 0;
136 y += fBitmap.height() + MARGIN;
137 }
138 // Test offsets on Darken (uses shader blend)
reedcfb6bdf2016-03-29 11:32:50 -0700139 paint.setImageFilter(SkXfermodeImageFilter::Make(SkXfermode::Make(SkXfermode::kDarken_Mode),
robertphillips51a315e2016-03-31 09:05:49 -0700140 offsetBackground.get(),
141 offsetForeground.get(),
142 nullptr));
fmalita5598b632015-09-15 11:26:13 -0700143 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000144 x += fBitmap.width() + MARGIN;
145 if (x + fBitmap.width() > WIDTH) {
146 x = 0;
147 y += fBitmap.height() + MARGIN;
148 }
149 // Test cropping
150 static const size_t nbSamples = 3;
151 SkXfermode::Mode sampledModes[nbSamples] = {SkXfermode::kOverlay_Mode,
152 SkXfermode::kSrcOver_Mode,
153 SkXfermode::kPlus_Mode};
154 int offsets[nbSamples][4] = {{ 10, 10, -16, -16},
155 { 10, 10, 10, 10},
156 {-10, -10, -6, -6}};
157 for (size_t i = 0; i < nbSamples; ++i) {
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000158 SkIRect cropRect = SkIRect::MakeXYWH(offsets[i][0],
159 offsets[i][1],
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000160 fBitmap.width() + offsets[i][2],
161 fBitmap.height() + offsets[i][3]);
senorblanco@chromium.orgb295fb62013-10-10 13:51:19 +0000162 SkImageFilter::CropRect rect(SkRect::Make(cropRect));
robertphillips51a315e2016-03-31 09:05:49 -0700163 paint.setImageFilter(SkXfermodeImageFilter::Make(SkXfermode::Make(sampledModes[i]),
164 offsetBackground.get(),
165 offsetForeground.get(),
166 &rect));
fmalita5598b632015-09-15 11:26:13 -0700167 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000168 x += fBitmap.width() + MARGIN;
169 if (x + fBitmap.width() > WIDTH) {
170 x = 0;
171 y += fBitmap.height() + MARGIN;
172 }
173 }
senorblanco06d54ad2016-01-13 13:48:54 -0800174 // Test small bg, large fg with Screen (uses shader blend)
reedcfb6bdf2016-03-29 11:32:50 -0700175 auto mode = SkXfermode::Make(SkXfermode::kScreen_Mode);
senorblanco06d54ad2016-01-13 13:48:54 -0800176 SkImageFilter::CropRect cropRect(SkRect::MakeXYWH(10, 10, 60, 60));
robertphillips51a315e2016-03-31 09:05:49 -0700177 sk_sp<SkImageFilter> cropped(SkOffsetImageFilter::Make(0, 0, foreground, &cropRect));
178 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, cropped.get(),
179 background.get(), nullptr));
senorblanco06d54ad2016-01-13 13:48:54 -0800180 DrawClippedPaint(canvas, clipRect, paint, x, y);
181 x += fBitmap.width() + MARGIN;
182 if (x + fBitmap.width() > WIDTH) {
183 x = 0;
184 y += fBitmap.height() + MARGIN;
185 }
186 // Test small fg, large bg with Screen (uses shader blend)
robertphillips51a315e2016-03-31 09:05:49 -0700187 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background.get(),
188 cropped.get(), nullptr));
senorblanco06d54ad2016-01-13 13:48:54 -0800189 DrawClippedPaint(canvas, clipRect, paint, x, y);
190 x += fBitmap.width() + MARGIN;
191 if (x + fBitmap.width() > WIDTH) {
192 x = 0;
193 y += fBitmap.height() + MARGIN;
194 }
senorblanco9db04272016-03-31 08:24:29 -0700195 // Test small fg, large bg with SrcIn with a crop that forces it to full size.
196 // This tests that SkXfermodeImageFilter correctly applies the compositing mode to
197 // the region outside the foreground.
198 mode = SkXfermode::Make(SkXfermode::kSrcIn_Mode);
199 SkImageFilter::CropRect cropRectFull(SkRect::MakeXYWH(0, 0, 80, 80));
robertphillips51a315e2016-03-31 09:05:49 -0700200 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background.get(),
201 cropped.get(), &cropRectFull));
senorblanco9db04272016-03-31 08:24:29 -0700202 DrawClippedPaint(canvas, clipRect, paint, x, y);
203 x += fBitmap.width() + MARGIN;
204 if (x + fBitmap.width() > WIDTH) {
205 x = 0;
206 y += fBitmap.height() + MARGIN;
207 }
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000208 }
fmalita5598b632015-09-15 11:26:13 -0700209
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000210private:
fmalita5598b632015-09-15 11:26:13 -0700211 static void DrawClippedBitmap(SkCanvas* canvas, const SkBitmap& bitmap, const SkPaint& paint,
212 int x, int y) {
213 canvas->save();
214 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
215 canvas->clipRect(SkRect::MakeWH(
216 SkIntToScalar(bitmap.width()), SkIntToScalar(bitmap.height())));
217 canvas->drawBitmap(bitmap, 0, 0, &paint);
218 canvas->restore();
219 }
220
221 static void DrawClippedPaint(SkCanvas* canvas, const SkRect& rect, const SkPaint& paint,
222 int x, int y) {
223 canvas->save();
224 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
225 canvas->clipRect(rect);
226 canvas->drawPaint(paint);
227 canvas->restore();
228 }
229
reed9ce9d672016-03-17 10:51:11 -0700230 SkBitmap fBitmap;
231 sk_sp<SkImage> fCheckerboard;
fmalita5598b632015-09-15 11:26:13 -0700232
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000233 typedef GM INHERITED;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000234};
235
236//////////////////////////////////////////////////////////////////////////////
237
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000238DEF_GM( return new XfermodeImageFilterGM; );
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000239
240}