1 /*
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"
9 #include "sk_tool_utils.h"
10 #include "SkArithmeticImageFilter.h"
11 #include "SkImage.h"
12 #include "SkImageSource.h"
13 #include "SkOffsetImageFilter.h"
14 #include "SkXfermodeImageFilter.h"
15 
16 #define WIDTH 600
17 #define HEIGHT 700
18 #define MARGIN 12
19 
20 namespace skiagm {
21 
22 class XfermodeImageFilterGM : public GM {
23 public:
XfermodeImageFilterGM()24     XfermodeImageFilterGM(){
25         this->setBGColor(0xFF000000);
26     }
27 
28 protected:
onShortName()29     SkString onShortName() override {
30         return SkString("xfermodeimagefilter");
31     }
32 
onISize()33     SkISize onISize() override {
34         return SkISize::Make(WIDTH, HEIGHT);
35     }
36 
onOnceBeforeDraw()37     void onOnceBeforeDraw() override {
38         fBitmap = sk_tool_utils::create_string_bitmap(80, 80, 0xD000D000, 15, 65, 96, "e");
39 
40         fCheckerboard = SkImage::MakeFromBitmap(
41             sk_tool_utils::create_checkerboard_bitmap(80, 80,
42                                                       0xFFA0A0A0,
43                                                       0xFF404040,
44                                                       8));
45     }
46 
onDraw(SkCanvas * canvas)47     void onDraw(SkCanvas* canvas) override {
48         canvas->clear(SK_ColorBLACK);
49         SkPaint paint;
50 
51         const SkBlendMode gModes[] = {
52             SkBlendMode::kClear,
53             SkBlendMode::kSrc,
54             SkBlendMode::kDst,
55             SkBlendMode::kSrcOver,
56             SkBlendMode::kDstOver,
57             SkBlendMode::kSrcIn,
58             SkBlendMode::kDstIn,
59             SkBlendMode::kSrcOut,
60             SkBlendMode::kDstOut,
61             SkBlendMode::kSrcATop,
62             SkBlendMode::kDstATop,
63             SkBlendMode::kXor,
64 
65             SkBlendMode::kPlus,
66             SkBlendMode::kModulate,
67             SkBlendMode::kScreen,
68             SkBlendMode::kOverlay,
69             SkBlendMode::kDarken,
70             SkBlendMode::kLighten,
71             SkBlendMode::kColorDodge,
72             SkBlendMode::kColorBurn,
73             SkBlendMode::kHardLight,
74             SkBlendMode::kSoftLight,
75             SkBlendMode::kDifference,
76             SkBlendMode::kExclusion,
77             SkBlendMode::kMultiply,
78             SkBlendMode::kHue,
79             SkBlendMode::kSaturation,
80             SkBlendMode::kColor,
81             SkBlendMode::kLuminosity,
82         };
83 
84         int x = 0, y = 0;
85         sk_sp<SkImageFilter> background(SkImageSource::Make(fCheckerboard));
86         for (size_t i = 0; i < SK_ARRAY_COUNT(gModes); i++) {
87             paint.setImageFilter(SkXfermodeImageFilter::Make(gModes[i], background));
88             DrawClippedBitmap(canvas, fBitmap, paint, x, y);
89             x += fBitmap.width() + MARGIN;
90             if (x + fBitmap.width() > WIDTH) {
91                 x = 0;
92                 y += fBitmap.height() + MARGIN;
93             }
94         }
95         // Test arithmetic mode as image filter
96         paint.setImageFilter(SkArithmeticImageFilter::Make(0, 1, 1, 0, true, background,
97                                                            nullptr, nullptr));
98         DrawClippedBitmap(canvas, fBitmap, paint, x, y);
99         x += fBitmap.width() + MARGIN;
100         if (x + fBitmap.width() > WIDTH) {
101             x = 0;
102             y += fBitmap.height() + MARGIN;
103         }
104         // Test nullptr mode
105         paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kSrcOver, background));
106         DrawClippedBitmap(canvas, fBitmap, paint, x, y);
107         x += fBitmap.width() + MARGIN;
108         if (x + fBitmap.width() > WIDTH) {
109             x = 0;
110             y += fBitmap.height() + MARGIN;
111         }
112         SkRect clipRect = SkRect::MakeWH(SkIntToScalar(fBitmap.width() + 4),
113                                          SkIntToScalar(fBitmap.height() + 4));
114         // Test offsets on SrcMode (uses fixed-function blend)
115         sk_sp<SkImage> bitmapImage(SkImage::MakeFromBitmap(fBitmap));
116         sk_sp<SkImageFilter> foreground(SkImageSource::Make(std::move(bitmapImage)));
117         sk_sp<SkImageFilter> offsetForeground(SkOffsetImageFilter::Make(SkIntToScalar(4),
118                                                                         SkIntToScalar(-4),
119                                                                         foreground));
120         sk_sp<SkImageFilter> offsetBackground(SkOffsetImageFilter::Make(SkIntToScalar(4),
121                                                                         SkIntToScalar(4),
122                                                                         background));
123         paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kSrcOver,
124                                                          offsetBackground,
125                                                          offsetForeground,
126                                                          nullptr));
127         DrawClippedPaint(canvas, clipRect, paint, x, y);
128         x += fBitmap.width() + MARGIN;
129         if (x + fBitmap.width() > WIDTH) {
130             x = 0;
131             y += fBitmap.height() + MARGIN;
132         }
133         // Test offsets on Darken (uses shader blend)
134         paint.setImageFilter(SkXfermodeImageFilter::Make(SkBlendMode::kDarken,
135                                                          offsetBackground,
136                                                          offsetForeground,
137                                                          nullptr));
138         DrawClippedPaint(canvas, clipRect, paint, x, y);
139         x += fBitmap.width() + MARGIN;
140         if (x + fBitmap.width() > WIDTH) {
141             x = 0;
142             y += fBitmap.height() + MARGIN;
143         }
144         // Test cropping
145         constexpr size_t nbSamples = 3;
146         const SkBlendMode sampledModes[nbSamples] = {
147             SkBlendMode::kOverlay, SkBlendMode::kSrcOver, SkBlendMode::kPlus
148         };
149         int offsets[nbSamples][4] = {{ 10,  10, -16, -16},
150                                      { 10,  10,  10,  10},
151                                      {-10, -10,  -6,  -6}};
152         for (size_t i = 0; i < nbSamples; ++i) {
153             SkIRect cropRect = SkIRect::MakeXYWH(offsets[i][0],
154                                                  offsets[i][1],
155                                                  fBitmap.width()  + offsets[i][2],
156                                                  fBitmap.height() + offsets[i][3]);
157             SkImageFilter::CropRect rect(SkRect::Make(cropRect));
158             paint.setImageFilter(SkXfermodeImageFilter::Make(sampledModes[i],
159                                                              offsetBackground,
160                                                              offsetForeground,
161                                                              &rect));
162             DrawClippedPaint(canvas, clipRect, paint, x, y);
163             x += fBitmap.width() + MARGIN;
164             if (x + fBitmap.width() > WIDTH) {
165                 x = 0;
166                 y += fBitmap.height() + MARGIN;
167             }
168         }
169         // Test small bg, large fg with Screen (uses shader blend)
170         SkBlendMode mode = SkBlendMode::kScreen;
171         SkImageFilter::CropRect cropRect(SkRect::MakeXYWH(10, 10, 60, 60));
172         sk_sp<SkImageFilter> cropped(SkOffsetImageFilter::Make(0, 0, foreground, &cropRect));
173         paint.setImageFilter(SkXfermodeImageFilter::Make(mode, cropped, background, nullptr));
174         DrawClippedPaint(canvas, clipRect, paint, x, y);
175         x += fBitmap.width() + MARGIN;
176         if (x + fBitmap.width() > WIDTH) {
177             x = 0;
178             y += fBitmap.height() + MARGIN;
179         }
180         // Test small fg, large bg with Screen (uses shader blend)
181         paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background, cropped, nullptr));
182         DrawClippedPaint(canvas, clipRect, paint, x, y);
183         x += fBitmap.width() + MARGIN;
184         if (x + fBitmap.width() > WIDTH) {
185             x = 0;
186             y += fBitmap.height() + MARGIN;
187         }
188         // Test small fg, large bg with SrcIn with a crop that forces it to full size.
189         // This tests that SkXfermodeImageFilter correctly applies the compositing mode to
190         // the region outside the foreground.
191         mode = SkBlendMode::kSrcIn;
192         SkImageFilter::CropRect cropRectFull(SkRect::MakeXYWH(0, 0, 80, 80));
193         paint.setImageFilter(SkXfermodeImageFilter::Make(mode, background,
194                                                          cropped, &cropRectFull));
195         DrawClippedPaint(canvas, clipRect, paint, x, y);
196         x += fBitmap.width() + MARGIN;
197         if (x + fBitmap.width() > WIDTH) {
198             x = 0;
199             y += fBitmap.height() + MARGIN;
200         }
201     }
202 
203 private:
DrawClippedBitmap(SkCanvas * canvas,const SkBitmap & bitmap,const SkPaint & paint,int x,int y)204     static void DrawClippedBitmap(SkCanvas* canvas, const SkBitmap& bitmap, const SkPaint& paint,
205                                   int x, int y) {
206         canvas->save();
207         canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
208         canvas->clipRect(SkRect::MakeIWH(bitmap.width(), bitmap.height()));
209         canvas->drawBitmap(bitmap, 0, 0, &paint);
210         canvas->restore();
211     }
212 
DrawClippedPaint(SkCanvas * canvas,const SkRect & rect,const SkPaint & paint,int x,int y)213     static void DrawClippedPaint(SkCanvas* canvas, const SkRect& rect, const SkPaint& paint,
214                                  int x, int y) {
215         canvas->save();
216         canvas->translate(SkIntToScalar(x), SkIntToScalar(y));
217         canvas->clipRect(rect);
218         canvas->drawPaint(paint);
219         canvas->restore();
220     }
221 
222     SkBitmap        fBitmap;
223     sk_sp<SkImage>  fCheckerboard;
224 
225     typedef GM INHERITED;
226 };
227 
228 //////////////////////////////////////////////////////////////////////////////
229 
230 DEF_GM( return new XfermodeImageFilterGM; );
231 
232 }
233