blob: 77c350235bdacb7412a566620686672b44d095f5 [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"
Brian Salomon89cb8212017-01-09 10:48:23 -050010#include "SkArithmeticImageFilter.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 {
reed374772b2016-10-05 17:33:02 -070052 SkBlendMode fMode;
53 const char* fLabel;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000054 } gModes[] = {
reed374772b2016-10-05 17:33:02 -070055 { SkBlendMode::kClear, "Clear" },
56 { SkBlendMode::kSrc, "Src" },
57 { SkBlendMode::kDst, "Dst" },
58 { SkBlendMode::kSrcOver, "SrcOver" },
59 { SkBlendMode::kDstOver, "DstOver" },
60 { SkBlendMode::kSrcIn, "SrcIn" },
61 { SkBlendMode::kDstIn, "DstIn" },
62 { SkBlendMode::kSrcOut, "SrcOut" },
63 { SkBlendMode::kDstOut, "DstOut" },
64 { SkBlendMode::kSrcATop, "SrcATop" },
65 { SkBlendMode::kDstATop, "DstATop" },
66 { SkBlendMode::kXor, "Xor" },
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000067
reed374772b2016-10-05 17:33:02 -070068 { SkBlendMode::kPlus, "Plus" },
69 { SkBlendMode::kModulate, "Modulate" },
70 { SkBlendMode::kScreen, "Screen" },
71 { SkBlendMode::kOverlay, "Overlay" },
72 { SkBlendMode::kDarken, "Darken" },
73 { SkBlendMode::kLighten, "Lighten" },
74 { SkBlendMode::kColorDodge, "ColorDodge" },
75 { SkBlendMode::kColorBurn, "ColorBurn" },
76 { SkBlendMode::kHardLight, "HardLight" },
77 { SkBlendMode::kSoftLight, "SoftLight" },
78 { SkBlendMode::kDifference, "Difference" },
79 { SkBlendMode::kExclusion, "Exclusion" },
80 { SkBlendMode::kMultiply, "Multiply" },
81 { SkBlendMode::kHue, "Hue" },
82 { SkBlendMode::kSaturation, "Saturation" },
83 { SkBlendMode::kColor, "Color" },
84 { SkBlendMode::kLuminosity, "Luminosity" },
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000085 };
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++) {
reed374772b2016-10-05 17:33:02 -070090 paint.setImageFilter(SkXfermodeImageFilter::Make(gModes[i].fMode, background));
fmalita5598b632015-09-15 11:26:13 -070091 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +000092 x += fBitmap.width() + MARGIN;
93 if (x + fBitmap.width() > WIDTH) {
94 x = 0;
95 y += fBitmap.height() + MARGIN;
96 }
97 }
98 // Test arithmetic mode as image filter
Brian Salomon89cb8212017-01-09 10:48:23 -050099 paint.setImageFilter(SkArithmeticImageFilter::Make(0, 1, 1, 0, true, background));
fmalita5598b632015-09-15 11:26:13 -0700100 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000101 x += fBitmap.width() + MARGIN;
102 if (x + fBitmap.width() > WIDTH) {
103 x = 0;
104 y += fBitmap.height() + MARGIN;
105 }
halcanary96fcdcc2015-08-27 07:41:13 -0700106 // Test nullptr mode
reed374772b2016-10-05 17:33:02 -0700107 paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kSrcOver, background));
fmalita5598b632015-09-15 11:26:13 -0700108 DrawClippedBitmap(canvas, fBitmap, paint, x, y);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000109 x += fBitmap.width() + MARGIN;
110 if (x + fBitmap.width() > WIDTH) {
111 x = 0;
112 y += fBitmap.height() + MARGIN;
113 }
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000114 SkRect clipRect = SkRect::MakeWH(SkIntToScalar(fBitmap.width() + 4),
115 SkIntToScalar(fBitmap.height() + 4));
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000116 // Test offsets on SrcMode (uses fixed-function blend)
reed9ce9d672016-03-17 10:51:11 -0700117 sk_sp<SkImage> bitmapImage(SkImage::MakeFromBitmap(fBitmap));
robertphillips549c8992016-04-01 09:28:51 -0700118 sk_sp<SkImageFilter> foreground(SkImageSource::Make(std::move(bitmapImage)));
robertphillips51a315e2016-03-31 09:05:49 -0700119 sk_sp<SkImageFilter> offsetForeground(SkOffsetImageFilter::Make(SkIntToScalar(4),
120 SkIntToScalar(-4),
121 foreground));
122 sk_sp<SkImageFilter> offsetBackground(SkOffsetImageFilter::Make(SkIntToScalar(4),
123 SkIntToScalar(4),
124 background));
reed374772b2016-10-05 17:33:02 -0700125 paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kSrcOver,
126 offsetBackground,
127 offsetForeground,
128 nullptr));
fmalita5598b632015-09-15 11:26:13 -0700129 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org7b320702013-07-10 21:22:18 +0000130 x += fBitmap.width() + MARGIN;
131 if (x + fBitmap.width() > WIDTH) {
132 x = 0;
133 y += fBitmap.height() + MARGIN;
134 }
135 // Test offsets on Darken (uses shader blend)
reed374772b2016-10-05 17:33:02 -0700136 paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kDarken,
robertphillips8c0326d2016-04-05 12:48:34 -0700137 offsetBackground,
138 offsetForeground,
robertphillips51a315e2016-03-31 09:05:49 -0700139 nullptr));
fmalita5598b632015-09-15 11:26:13 -0700140 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000141 x += fBitmap.width() + MARGIN;
142 if (x + fBitmap.width() > WIDTH) {
143 x = 0;
144 y += fBitmap.height() + MARGIN;
145 }
146 // Test cropping
mtkleindbfd7ab2016-09-01 11:24:54 -0700147 constexpr size_t nbSamples = 3;
reed374772b2016-10-05 17:33:02 -0700148 const SkBlendMode sampledModes[nbSamples] = {
149 SkBlendMode::kOverlay, SkBlendMode::kSrcOver, SkBlendMode::kPlus
150 };
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000151 int offsets[nbSamples][4] = {{ 10, 10, -16, -16},
152 { 10, 10, 10, 10},
153 {-10, -10, -6, -6}};
154 for (size_t i = 0; i < nbSamples; ++i) {
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000155 SkIRect cropRect = SkIRect::MakeXYWH(offsets[i][0],
156 offsets[i][1],
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000157 fBitmap.width() + offsets[i][2],
158 fBitmap.height() + offsets[i][3]);
senorblanco@chromium.orgb295fb62013-10-10 13:51:19 +0000159 SkImageFilter::CropRect rect(SkRect::Make(cropRect));
reed374772b2016-10-05 17:33:02 -0700160 paint.setImageFilter(SkXfermodeImageFilter::Make(sampledModes[i],
robertphillips8c0326d2016-04-05 12:48:34 -0700161 offsetBackground,
162 offsetForeground,
robertphillips51a315e2016-03-31 09:05:49 -0700163 &rect));
fmalita5598b632015-09-15 11:26:13 -0700164 DrawClippedPaint(canvas, clipRect, paint, x, y);
commit-bot@chromium.org1a4fb702013-09-26 16:09:28 +0000165 x += fBitmap.width() + MARGIN;
166 if (x + fBitmap.width() > WIDTH) {
167 x = 0;
168 y += fBitmap.height() + MARGIN;
169 }
170 }
senorblanco06d54ad2016-01-13 13:48:54 -0800171 // Test small bg, large fg with Screen (uses shader blend)
reed374772b2016-10-05 17:33:02 -0700172 SkBlendMode mode = SkBlendMode::kScreen;
senorblanco06d54ad2016-01-13 13:48:54 -0800173 SkImageFilter::CropRect cropRect(SkRect::MakeXYWH(10, 10, 60, 60));
robertphillips51a315e2016-03-31 09:05:49 -0700174 sk_sp<SkImageFilter> cropped(SkOffsetImageFilter::Make(0, 0, foreground, &cropRect));
robertphillips8c0326d2016-04-05 12:48:34 -0700175 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, cropped, background, nullptr));
senorblanco06d54ad2016-01-13 13:48:54 -0800176 DrawClippedPaint(canvas, clipRect, paint, x, y);
177 x += fBitmap.width() + MARGIN;
178 if (x + fBitmap.width() > WIDTH) {
179 x = 0;
180 y += fBitmap.height() + MARGIN;
181 }
182 // Test small fg, large bg with Screen (uses shader blend)
robertphillips8c0326d2016-04-05 12:48:34 -0700183 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background, cropped, nullptr));
senorblanco06d54ad2016-01-13 13:48:54 -0800184 DrawClippedPaint(canvas, clipRect, paint, x, y);
185 x += fBitmap.width() + MARGIN;
186 if (x + fBitmap.width() > WIDTH) {
187 x = 0;
188 y += fBitmap.height() + MARGIN;
189 }
senorblanco9db04272016-03-31 08:24:29 -0700190 // Test small fg, large bg with SrcIn with a crop that forces it to full size.
191 // This tests that SkXfermodeImageFilter correctly applies the compositing mode to
192 // the region outside the foreground.
reed374772b2016-10-05 17:33:02 -0700193 mode = SkBlendMode::kSrcIn;
senorblanco9db04272016-03-31 08:24:29 -0700194 SkImageFilter::CropRect cropRectFull(SkRect::MakeXYWH(0, 0, 80, 80));
robertphillips8c0326d2016-04-05 12:48:34 -0700195 paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background,
196 cropped, &cropRectFull));
senorblanco9db04272016-03-31 08:24:29 -0700197 DrawClippedPaint(canvas, clipRect, paint, x, y);
198 x += fBitmap.width() + MARGIN;
199 if (x + fBitmap.width() > WIDTH) {
200 x = 0;
201 y += fBitmap.height() + MARGIN;
202 }
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000203 }
fmalita5598b632015-09-15 11:26:13 -0700204
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000205private:
fmalita5598b632015-09-15 11:26:13 -0700206 static void DrawClippedBitmap(SkCanvas* canvas, const SkBitmap& bitmap, const SkPaint& paint,
207 int x, int y) {
208 canvas->save();
209 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
210 canvas->clipRect(SkRect::MakeWH(
211 SkIntToScalar(bitmap.width()), SkIntToScalar(bitmap.height())));
212 canvas->drawBitmap(bitmap, 0, 0, &paint);
213 canvas->restore();
214 }
215
216 static void DrawClippedPaint(SkCanvas* canvas, const SkRect& rect, const SkPaint& paint,
217 int x, int y) {
218 canvas->save();
219 canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
220 canvas->clipRect(rect);
221 canvas->drawPaint(paint);
222 canvas->restore();
223 }
224
reed9ce9d672016-03-17 10:51:11 -0700225 SkBitmap fBitmap;
226 sk_sp<SkImage> fCheckerboard;
fmalita5598b632015-09-15 11:26:13 -0700227
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000228 typedef GM INHERITED;
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000229};
230
231//////////////////////////////////////////////////////////////////////////////
232
senorblanco@chromium.orge93e1db2013-12-09 18:31:42 +0000233DEF_GM( return new XfermodeImageFilterGM; );
senorblanco@chromium.org86fc2662013-05-31 17:49:12 +0000234
235}