blob: b4bbd12d5ef8fffe8b0f102292222f07b7367cb7 [file] [log] [blame]
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +00001/*
2 * Copyright 2011 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"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040012#include "include/core/SkColor.h"
13#include "include/core/SkColorFilter.h"
14#include "include/core/SkImageFilter.h"
15#include "include/core/SkPaint.h"
16#include "include/core/SkPoint.h"
17#include "include/core/SkRect.h"
18#include "include/core/SkRefCnt.h"
19#include "include/core/SkScalar.h"
20#include "include/core/SkShader.h"
21#include "include/core/SkSize.h"
22#include "include/core/SkString.h"
23#include "include/core/SkTileMode.h"
24#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050025#include "include/effects/SkGradientShader.h"
Michael Ludwig898bbfa2019-08-02 15:21:23 -040026#include "include/effects/SkImageFilters.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "include/effects/SkTableColorFilter.h"
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000028
Ben Wagner7fde8e12019-05-01 17:28:53 -040029#include <math.h>
30#include <utility>
31
reed1a9b9642016-03-13 14:13:58 -070032static sk_sp<SkShader> make_shader0(int w, int h) {
reeddb873d82015-03-01 19:53:47 -080033 SkPoint pts[] = { {0, 0}, {SkIntToScalar(w), SkIntToScalar(h)} };
34 SkColor colors[] = {
35 SK_ColorBLACK, SK_ColorGREEN, SK_ColorCYAN,
36 SK_ColorRED, 0, SK_ColorBLUE, SK_ColorWHITE
37 };
reed1a9b9642016-03-13 14:13:58 -070038 return SkGradientShader::MakeLinear(pts, colors, nullptr, SK_ARRAY_COUNT(colors),
Mike Reedfae8fce2019-04-03 10:27:45 -040039 SkTileMode::kClamp);
reeddb873d82015-03-01 19:53:47 -080040}
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000041static void make_bm0(SkBitmap* bm) {
42 int W = 120;
43 int H = 120;
reed@google.comeb9a46c2014-01-25 16:46:20 +000044 bm->allocN32Pixels(W, H);
junov@google.comdbfac8a2012-12-06 21:47:40 +000045 bm->eraseColor(SK_ColorTRANSPARENT);
rmistry@google.comd6176b02012-08-23 18:14:13 +000046
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000047 SkCanvas canvas(*bm);
48 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -070049 paint.setShader(make_shader0(W, H));
reed5bd055c2015-03-01 19:16:38 -080050 canvas.drawPaint(paint);
reedb675a732015-03-01 18:00:47 -080051}
reed1a9b9642016-03-13 14:13:58 -070052static sk_sp<SkShader> make_shader1(int w, int h) {
reeddb873d82015-03-01 19:53:47 -080053 SkScalar cx = SkIntToScalar(w)/2;
54 SkScalar cy = SkIntToScalar(h)/2;
55 SkColor colors[] = {
56 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE,
57 };
reed1a9b9642016-03-13 14:13:58 -070058 return SkGradientShader::MakeRadial(SkPoint::Make(cx, cy), cx, colors, nullptr,
Mike Reedfae8fce2019-04-03 10:27:45 -040059 SK_ARRAY_COUNT(colors), SkTileMode::kClamp);
reeddb873d82015-03-01 19:53:47 -080060}
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000061static void make_bm1(SkBitmap* bm) {
62 int W = 120;
63 int H = 120;
reeddb873d82015-03-01 19:53:47 -080064 SkScalar cx = SkIntToScalar(W)/2;
65 SkScalar cy = SkIntToScalar(H)/2;
reed@google.comeb9a46c2014-01-25 16:46:20 +000066 bm->allocN32Pixels(W, H);
junov@google.comdbfac8a2012-12-06 21:47:40 +000067 bm->eraseColor(SK_ColorTRANSPARENT);
rmistry@google.comd6176b02012-08-23 18:14:13 +000068
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000069 SkCanvas canvas(*bm);
70 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -070071 paint.setShader(make_shader1(W, H));
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000072 paint.setAntiAlias(true);
73 canvas.drawCircle(cx, cy, cx, paint);
74}
75
76static void make_table0(uint8_t table[]) {
77 for (int i = 0; i < 256; ++i) {
78 int n = i >> 5;
79 table[i] = (n << 5) | (n << 2) | (n >> 1);
80 }
81}
82static void make_table1(uint8_t table[]) {
83 for (int i = 0; i < 256; ++i) {
84 table[i] = i * i / 255;
85 }
86}
87static void make_table2(uint8_t table[]) {
88 for (int i = 0; i < 256; ++i) {
89 float fi = i / 255.0f;
robertphillips@google.com6853e802012-04-16 15:50:18 +000090 table[i] = static_cast<uint8_t>(sqrtf(fi) * 255);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000091 }
92}
93
reedd053ce92016-03-22 10:17:23 -070094static sk_sp<SkColorFilter> make_null_cf() {
halcanary96fcdcc2015-08-27 07:41:13 -070095 return nullptr;
cwallezc12b74d2015-01-26 07:45:53 -080096}
97
reedd053ce92016-03-22 10:17:23 -070098static sk_sp<SkColorFilter> make_cf0() {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +000099 uint8_t table[256]; make_table0(table);
reedd053ce92016-03-22 10:17:23 -0700100 return SkTableColorFilter::Make(table);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000101}
reedd053ce92016-03-22 10:17:23 -0700102static sk_sp<SkColorFilter> make_cf1() {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000103 uint8_t table[256]; make_table1(table);
reedd053ce92016-03-22 10:17:23 -0700104 return SkTableColorFilter::Make(table);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000105}
reedd053ce92016-03-22 10:17:23 -0700106static sk_sp<SkColorFilter> make_cf2() {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000107 uint8_t table[256]; make_table2(table);
reedd053ce92016-03-22 10:17:23 -0700108 return SkTableColorFilter::Make(table);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000109}
reedd053ce92016-03-22 10:17:23 -0700110static sk_sp<SkColorFilter> make_cf3() {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000111 uint8_t table0[256]; make_table0(table0);
112 uint8_t table1[256]; make_table1(table1);
113 uint8_t table2[256]; make_table2(table2);
reedd053ce92016-03-22 10:17:23 -0700114 return SkTableColorFilter::MakeARGB(nullptr, table0, table1, table2);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000115}
116
117class TableColorFilterGM : public skiagm::GM {
118public:
119 TableColorFilterGM() {}
rmistry@google.comd6176b02012-08-23 18:14:13 +0000120
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000121protected:
Hal Canarybd865e22019-07-18 11:51:19 -0400122 SkString onShortName() override {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000123 return SkString("tablecolorfilter");
124 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000125
Hal Canarybd865e22019-07-18 11:51:19 -0400126 SkISize onISize() override {
127 return {700, 1650};
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000128 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000129
Hal Canarybd865e22019-07-18 11:51:19 -0400130 void onDraw(SkCanvas* canvas) override {
Mike Kleind46dce32018-08-16 10:17:03 -0400131 canvas->drawColor(0xFFDDDDDD);
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000132 canvas->translate(20, 20);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000133
cwallezc12b74d2015-01-26 07:45:53 -0800134
reedd053ce92016-03-22 10:17:23 -0700135 static sk_sp<SkColorFilter> (*gColorFilterMakers[])() = {
136 make_null_cf, make_cf0, make_cf1, make_cf2, make_cf3
137 };
cwallezc12b74d2015-01-26 07:45:53 -0800138 static void (*gBitmapMakers[])(SkBitmap*) = { make_bm0, make_bm1 };
139
140 // This test will be done once for each bitmap with the results stacked vertically.
141 // For a single bitmap the resulting image will be the following:
142 // - A first line with the original bitmap, followed by the image drawn once
143 // with each of the N color filters
144 // - N lines of the bitmap drawn N times, this will cover all N*N combinations of
145 // pair of color filters in order to test the collpsing of consecutive table
146 // color filters.
147 //
148 // Here is a graphical representation of the result for 2 bitmaps and 2 filters
149 // with the number corresponding to the number of filters the bitmap goes through:
150 //
151 // --bitmap1
152 // 011
153 // 22
154 // 22
155 // --bitmap2
156 // 011
157 // 22
158 // 22
159
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000160 SkScalar x = 0, y = 0;
cwallezc12b74d2015-01-26 07:45:53 -0800161 for (size_t bitmapMaker = 0; bitmapMaker < SK_ARRAY_COUNT(gBitmapMakers); ++bitmapMaker) {
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000162 SkBitmap bm;
cwallezc12b74d2015-01-26 07:45:53 -0800163 gBitmapMakers[bitmapMaker](&bm);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000164
cwallezc12b74d2015-01-26 07:45:53 -0800165 SkScalar xOffset = SkScalar(bm.width() * 9 / 8);
166 SkScalar yOffset = SkScalar(bm.height() * 9 / 8);
167
168 // Draw the first element of the first line
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000169 x = 0;
cwallezc12b74d2015-01-26 07:45:53 -0800170 SkPaint paint;
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000171 canvas->drawBitmap(bm, x, y, &paint);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000172
cwallezc12b74d2015-01-26 07:45:53 -0800173 // Draws the rest of the first line for this bitmap
174 // each draw being at xOffset of the previous one
175 for (unsigned i = 1; i < SK_ARRAY_COUNT(gColorFilterMakers); ++i) {
176 x += xOffset;
reedd053ce92016-03-22 10:17:23 -0700177 paint.setColorFilter(gColorFilterMakers[i]());
cwallezc12b74d2015-01-26 07:45:53 -0800178 canvas->drawBitmap(bm, x, y, &paint);
179 }
180
halcanary96fcdcc2015-08-27 07:41:13 -0700181 paint.setColorFilter(nullptr);
cwallezc12b74d2015-01-26 07:45:53 -0800182
183 for (unsigned i = 0; i < SK_ARRAY_COUNT(gColorFilterMakers); ++i) {
robertphillips5605b562016-04-05 11:50:42 -0700184 sk_sp<SkColorFilter> colorFilter1(gColorFilterMakers[i]());
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400185 sk_sp<SkImageFilter> imageFilter1(SkImageFilters::ColorFilter(
186 std::move(colorFilter1), nullptr));
cwallezc12b74d2015-01-26 07:45:53 -0800187
188 // Move down to the next line and draw it
189 // each draw being at xOffset of the previous one
190 y += yOffset;
191 x = 0;
192 for (unsigned j = 1; j < SK_ARRAY_COUNT(gColorFilterMakers); ++j) {
robertphillips5605b562016-04-05 11:50:42 -0700193 sk_sp<SkColorFilter> colorFilter2(gColorFilterMakers[j]());
Michael Ludwig898bbfa2019-08-02 15:21:23 -0400194 sk_sp<SkImageFilter> imageFilter2(SkImageFilters::ColorFilter(
195 std::move(colorFilter2), imageFilter1, nullptr));
robertphillips5605b562016-04-05 11:50:42 -0700196 paint.setImageFilter(std::move(imageFilter2));
cwallezc12b74d2015-01-26 07:45:53 -0800197 canvas->drawBitmap(bm, x, y, &paint);
198 x += xOffset;
199 }
200 }
201
202 // Move down one line to the beginning of the block for next bitmap
203 y += yOffset;
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000204 }
205 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000206
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000207private:
208 typedef GM INHERITED;
209};
reeddb873d82015-03-01 19:53:47 -0800210DEF_GM( return new TableColorFilterGM; )
mike@reedtribe.orga69b48c2011-12-28 20:31:00 +0000211
212//////////////////////////////////////////////////////////////////////////////
213
reeddb873d82015-03-01 19:53:47 -0800214class ComposeColorFilterGM : public skiagm::GM {
reedcff10b22015-03-03 06:41:45 -0800215 enum {
216 COLOR_COUNT = 3,
217 MODE_COUNT = 4,
218 };
Mike Reed7d954ad2016-10-28 15:42:34 -0400219 const SkColor* fColors;
220 const SkBlendMode* fModes;
Hal Canarybd865e22019-07-18 11:51:19 -0400221 const char* fName;
reedcff10b22015-03-03 06:41:45 -0800222
reeddb873d82015-03-01 19:53:47 -0800223public:
Hal Canarybd865e22019-07-18 11:51:19 -0400224 ComposeColorFilterGM(const SkColor colors[], const SkBlendMode modes[], const char* name)
225 : fColors(colors), fModes(modes), fName(name) {}
halcanary9d524f22016-03-29 09:03:52 -0700226
Hal Canarybd865e22019-07-18 11:51:19 -0400227private:
228 SkString onShortName() override { return SkString(fName); }
halcanary9d524f22016-03-29 09:03:52 -0700229
Hal Canarybd865e22019-07-18 11:51:19 -0400230 SkISize onISize() override { return {790, 790}; }
reeddb873d82015-03-01 19:53:47 -0800231
Hal Canarybd865e22019-07-18 11:51:19 -0400232 void onDraw(SkCanvas* canvas) override {
reeddb873d82015-03-01 19:53:47 -0800233 SkBitmap bm;
234 make_bm1(&bm);
235
Mike Kleind46dce32018-08-16 10:17:03 -0400236 canvas->drawColor(0xFFDDDDDD);
reeddb873d82015-03-01 19:53:47 -0800237
reedcff10b22015-03-03 06:41:45 -0800238 const int MODES = MODE_COUNT * COLOR_COUNT;
reedd053ce92016-03-22 10:17:23 -0700239 sk_sp<SkColorFilter> filters[MODES];
reeddb873d82015-03-01 19:53:47 -0800240 int index = 0;
reedcff10b22015-03-03 06:41:45 -0800241 for (int i = 0; i < MODE_COUNT; ++i) {
242 for (int j = 0; j < COLOR_COUNT; ++j) {
Mike Reedb286bc22019-04-08 16:23:20 -0400243 filters[index++] = SkColorFilters::Blend(fColors[j], fModes[i]);
reeddb873d82015-03-01 19:53:47 -0800244 }
245 }
246
247 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -0700248 paint.setShader(make_shader1(50, 50));
reeddb873d82015-03-01 19:53:47 -0800249 SkRect r = SkRect::MakeWH(50, 50);
250 const SkScalar spacer = 10;
251
252 canvas->translate(spacer, spacer);
253
reedcff10b22015-03-03 06:41:45 -0800254 canvas->drawRect(r, paint); // orig
255
256 for (int i = 0; i < MODES; ++i) {
257 paint.setColorFilter(filters[i]);
258
reeddb873d82015-03-01 19:53:47 -0800259 canvas->save();
reedcff10b22015-03-03 06:41:45 -0800260 canvas->translate((i + 1) * (r.width() + spacer), 0);
261 canvas->drawRect(r, paint);
262 canvas->restore();
263
264 canvas->save();
265 canvas->translate(0, (i + 1) * (r.width() + spacer));
266 canvas->drawRect(r, paint);
267 canvas->restore();
268 }
269
270 canvas->translate(r.width() + spacer, r.width() + spacer);
271
272 for (int y = 0; y < MODES; ++y) {
273 canvas->save();
274 for (int x = 0; x < MODES; ++x) {
Mike Reed19d7bd62018-02-19 14:10:57 -0500275 paint.setColorFilter(filters[y]->makeComposed(filters[x]));
reeddb873d82015-03-01 19:53:47 -0800276 canvas->drawRect(r, paint);
277 canvas->translate(r.width() + spacer, 0);
278 }
279 canvas->restore();
280 canvas->translate(0, r.height() + spacer);
281 }
282 }
reeddb873d82015-03-01 19:53:47 -0800283};
reeddb873d82015-03-01 19:53:47 -0800284
reedcff10b22015-03-03 06:41:45 -0800285const SkColor gColors0[] = { SK_ColorCYAN, SK_ColorMAGENTA, SK_ColorYELLOW };
Mike Reed7d954ad2016-10-28 15:42:34 -0400286const SkBlendMode gModes0[] = {
287 SkBlendMode::kOverlay,
288 SkBlendMode::kDarken,
289 SkBlendMode::kColorBurn,
290 SkBlendMode::kExclusion,
reedcff10b22015-03-03 06:41:45 -0800291};
Hal Canarybd865e22019-07-18 11:51:19 -0400292DEF_GM( return new ComposeColorFilterGM(gColors0, gModes0, "colorcomposefilter_wacky"); )
reedcff10b22015-03-03 06:41:45 -0800293
294const SkColor gColors1[] = { 0x80FF0000, 0x8000FF00, 0x800000FF };
Mike Reed7d954ad2016-10-28 15:42:34 -0400295const SkBlendMode gModes1[] = {
296 SkBlendMode::kSrcOver,
297 SkBlendMode::kXor,
298 SkBlendMode::kDstOut,
299 SkBlendMode::kSrcATop,
reedcff10b22015-03-03 06:41:45 -0800300};
Hal Canarybd865e22019-07-18 11:51:19 -0400301DEF_GM( return new ComposeColorFilterGM(gColors1, gModes1, "colorcomposefilter_alpha"); )