blob: 1fe4dad2a80155d1538a3b523cb50f496d941c69 [file] [log] [blame]
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +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
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +00008#include "SkAlphaThresholdFilter.h"
Brian Osmana4aa1332017-10-19 12:54:28 -04009#include "SkImageSource.h"
Robert Phillips22c57ab2016-12-19 16:51:53 -050010#include "SkOffsetImageFilter.h"
Brian Osmana4aa1332017-10-19 12:54:28 -040011#include "SkRandom.h"
Mike Reed267be7f2017-02-13 09:32:54 -050012#include "SkRegion.h"
senorblanco1ea67a32016-01-19 08:50:18 -080013#include "SkSurface.h"
Mike Kleinea3f0142019-03-20 11:12:10 -050014#include "ToolUtils.h"
15#include "gm.h"
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000016
17#define WIDTH 500
18#define HEIGHT 500
19
Robert Phillips22c57ab2016-12-19 16:51:53 -050020static void draw_rects(SkCanvas* canvas) {
senorblanco1ea67a32016-01-19 08:50:18 -080021 SkPaint rectPaint;
Robert Phillips22c57ab2016-12-19 16:51:53 -050022 rectPaint.setColor(SK_ColorBLUE);
senorblanco1ea67a32016-01-19 08:50:18 -080023 canvas->drawRect(SkRect::MakeXYWH(0, 0, WIDTH / 2, HEIGHT / 2), rectPaint);
24 rectPaint.setColor(0xBFFF0000);
25 canvas->drawRect(SkRect::MakeXYWH(WIDTH / 2, 0, WIDTH / 2, HEIGHT / 2), rectPaint);
26 rectPaint.setColor(0x3F00FF00);
27 canvas->drawRect(SkRect::MakeXYWH(0, HEIGHT / 2, WIDTH / 2, HEIGHT / 2), rectPaint);
Robert Phillips22c57ab2016-12-19 16:51:53 -050028 rectPaint.setColor(SK_ColorTRANSPARENT);
senorblanco1ea67a32016-01-19 08:50:18 -080029 canvas->drawRect(SkRect::MakeXYWH(WIDTH / 2, HEIGHT / 2, WIDTH / 2, HEIGHT / 2), rectPaint);
30}
31
Robert Phillips22c57ab2016-12-19 16:51:53 -050032static SkPaint create_filter_paint(SkImageFilter::CropRect* cropRect = nullptr) {
senorblanco1ea67a32016-01-19 08:50:18 -080033 SkIRect rects[2];
34 rects[0] = SkIRect::MakeXYWH(0, 150, WIDTH, HEIGHT - 300);
35 rects[1] = SkIRect::MakeXYWH(150, 0, WIDTH - 300, HEIGHT);
36 SkRegion region;
37 region.setRects(rects, 2);
38
39 SkPaint paint;
Robert Phillips22c57ab2016-12-19 16:51:53 -050040 sk_sp<SkImageFilter> offset(SkOffsetImageFilter::Make(25, 25, nullptr));
41 paint.setImageFilter(SkAlphaThresholdFilter::Make(region, 0.2f, 0.7f, std::move(offset), cropRect));
senorblanco1ea67a32016-01-19 08:50:18 -080042 return paint;
43}
44
Robert Phillips22c57ab2016-12-19 16:51:53 -050045class ImageAlphaThresholdGM : public skiagm::GM {
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000046public:
Brian Osman01015fb2016-10-03 11:06:45 -040047 ImageAlphaThresholdGM(bool useCropRect) : fUseCropRect(useCropRect) {
Robert Phillips22c57ab2016-12-19 16:51:53 -050048 this->setBGColor(SK_ColorWHITE);
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000049 }
50
51protected:
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000052
mtklein36352bf2015-03-25 18:17:31 -070053 SkString onShortName() override {
robertphillipsaf9b8c82016-04-12 11:02:25 -070054 if (fUseCropRect) {
55 return SkString("imagealphathreshold_crop");
56 }
57
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000058 return SkString("imagealphathreshold");
59 }
60
mtklein36352bf2015-03-25 18:17:31 -070061 SkISize onISize() override {
tfarinaf5393182014-06-09 23:59:03 -070062 return SkISize::Make(WIDTH, HEIGHT);
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000063 }
64
mtklein36352bf2015-03-25 18:17:31 -070065 void onDraw(SkCanvas* canvas) override {
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000066 SkMatrix matrix;
67 matrix.reset();
68 matrix.setTranslate(WIDTH * .1f, HEIGHT * .1f);
commit-bot@chromium.org9109e182014-01-07 16:04:01 +000069 matrix.postScale(.8f, .8f);
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000070
71 canvas->concat(matrix);
72
robertphillipsaf9b8c82016-04-12 11:02:25 -070073 SkRect r = SkRect::MakeLTRB(100, 100, WIDTH - 100, HEIGHT - 100);
74 SkImageFilter::CropRect cropRect(r);
75
76 SkPaint paint = create_filter_paint(fUseCropRect ? &cropRect : nullptr);
halcanary96fcdcc2015-08-27 07:41:13 -070077 canvas->saveLayer(nullptr, &paint);
senorblanco1ea67a32016-01-19 08:50:18 -080078 draw_rects(canvas);
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000079
80 canvas->restore();
81 }
82
83private:
robertphillipsaf9b8c82016-04-12 11:02:25 -070084 bool fUseCropRect;
85
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +000086 typedef GM INHERITED;
87};
88
Robert Phillips22c57ab2016-12-19 16:51:53 -050089// Create a 'width' x 'height' SkSurface that matches the colorType of 'canvas' as
90// best we can
91static sk_sp<SkSurface> make_color_matching_surface(SkCanvas* canvas, int width, int height,
Mike Reed16178992018-02-09 17:35:29 -050092 SkAlphaType at) {
robertphillipsaf9b8c82016-04-12 11:02:25 -070093
Robert Phillips22c57ab2016-12-19 16:51:53 -050094 SkColorType ct = canvas->imageInfo().colorType();
Mike Reed693fdbd2017-01-12 10:13:40 -050095 sk_sp<SkColorSpace> cs(canvas->imageInfo().refColorSpace());
Robert Phillips22c57ab2016-12-19 16:51:53 -050096
97 if (kUnknown_SkColorType == ct) {
98 // For backends that aren't yet color-space aware we just fallback to N32.
99 ct = kN32_SkColorType;
100 cs = nullptr;
Mike Reed16178992018-02-09 17:35:29 -0500101 } else if (SkColorTypeIsAlwaysOpaque(ct)) {
102 at = kOpaque_SkAlphaType;
Robert Phillips22c57ab2016-12-19 16:51:53 -0500103 }
104
Mike Reed16178992018-02-09 17:35:29 -0500105 SkImageInfo info = SkImageInfo::Make(width, height, ct, at, std::move(cs));
Robert Phillips22c57ab2016-12-19 16:51:53 -0500106
Mike Kleinea3f0142019-03-20 11:12:10 -0500107 return ToolUtils::makeSurface(canvas, info);
Robert Phillips22c57ab2016-12-19 16:51:53 -0500108}
109
110class ImageAlphaThresholdSurfaceGM : public skiagm::GM {
senorblanco1ea67a32016-01-19 08:50:18 -0800111public:
112 ImageAlphaThresholdSurfaceGM() {
113 this->setBGColor(0xFFFFFFFF);
114 }
115
116protected:
117 SkString onShortName() override {
118 return SkString("imagealphathreshold_surface");
119 }
120
121 SkISize onISize() override {
122 return SkISize::Make(WIDTH, HEIGHT);
123 }
124
Chris Dalton50e24d72019-02-07 16:20:09 -0700125 DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
Robert Phillips22c57ab2016-12-19 16:51:53 -0500126 SkMatrix matrix;
127 matrix.reset();
128 matrix.setTranslate(WIDTH * .1f, HEIGHT * .1f);
129 matrix.postScale(.8f, .8f);
130
131 canvas->concat(matrix);
132
133 sk_sp<SkSurface> surface(make_color_matching_surface(canvas, WIDTH, HEIGHT,
134 kPremul_SkAlphaType));
Robert Phillips253b4dd2016-12-20 19:05:09 -0500135 if (!surface) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700136 *errorMsg = "make_color_matching_surface failed";
137 return DrawResult::kFail;
Robert Phillips253b4dd2016-12-20 19:05:09 -0500138 }
Robert Phillips22c57ab2016-12-19 16:51:53 -0500139
140 surface->getCanvas()->clear(SK_ColorTRANSPARENT);
senorblanco1ea67a32016-01-19 08:50:18 -0800141 draw_rects(surface->getCanvas());
142
senorblanco1ea67a32016-01-19 08:50:18 -0800143 SkPaint paint = create_filter_paint();
144 canvas->clipRect(SkRect::MakeLTRB(100, 100, WIDTH - 100, HEIGHT - 100));
reed9ce9d672016-03-17 10:51:11 -0700145 canvas->drawImage(surface->makeImageSnapshot().get(), 0, 0, &paint);
Chris Dalton50e24d72019-02-07 16:20:09 -0700146 return DrawResult::kOk;
senorblanco1ea67a32016-01-19 08:50:18 -0800147 }
148
149private:
Robert Phillips22c57ab2016-12-19 16:51:53 -0500150 typedef skiagm::GM INHERITED;
senorblanco1ea67a32016-01-19 08:50:18 -0800151};
152
commit-bot@chromium.org40eb3c12014-01-06 23:41:14 +0000153//////////////////////////////////////////////////////////////////////////////
154
robertphillipsaf9b8c82016-04-12 11:02:25 -0700155DEF_GM(return new ImageAlphaThresholdGM(true);)
156DEF_GM(return new ImageAlphaThresholdGM(false);)
senorblanco1ea67a32016-01-19 08:50:18 -0800157DEF_GM(return new ImageAlphaThresholdSurfaceGM();)
Brian Osmana4aa1332017-10-19 12:54:28 -0400158
159//////////////////////////////////////////////////////////////////////////////
160
161static sk_sp<SkImage> make_img() {
162 SkBitmap bitmap;
163 bitmap.allocPixels(SkImageInfo::MakeS32(WIDTH, HEIGHT, kPremul_SkAlphaType));
164 SkCanvas canvas(bitmap);
165
166 SkPaint paint;
167 SkRect rect = SkRect::MakeWH(WIDTH, HEIGHT);
168 SkRandom rnd;
169
170 while (!rect.isEmpty()) {
171 paint.setColor(rnd.nextU() | (0xFF << 24));
172 canvas.drawRect(rect, paint);
173 rect.inset(25, 25);
174 }
175
176 return SkImage::MakeFromBitmap(bitmap);
177}
178
179DEF_SIMPLE_GM_BG(imagealphathreshold_image, canvas, WIDTH * 2, HEIGHT, SK_ColorBLACK) {
180 sk_sp<SkImage> image(make_img());
181
182 SkIRect rects[2];
183 rects[0] = SkIRect::MakeXYWH(0, 150, WIDTH, HEIGHT - 300);
184 rects[1] = SkIRect::MakeXYWH(150, 0, WIDTH - 300, HEIGHT);
185 SkRegion region;
186 region.setRects(rects, 2);
187
188 SkPaint filterPaint;
189 sk_sp<SkImageFilter> imageSource(SkImageSource::Make(image));
190 filterPaint.setImageFilter(SkAlphaThresholdFilter::Make(region, 0.2f, 0.7f,
191 std::move(imageSource)));
192
193 canvas->saveLayer(nullptr, &filterPaint);
194 canvas->restore();
195 canvas->translate(WIDTH, 0);
196 canvas->drawImage(image, 0, 0);
197}