blob: bd70eb2dedd4f53adae3e9c0499c06900d04c020 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26: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 */
reed3d9005c2015-04-23 10:30:27 -07007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -04009#include "include/core/SkBlendMode.h"
10#include "include/core/SkCanvas.h"
11#include "include/core/SkColor.h"
12#include "include/core/SkColorSpace.h"
Brian Osmanf2d20dd2022-10-04 16:51:57 -040013#include "include/core/SkFont.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040014#include "include/core/SkMatrix.h"
15#include "include/core/SkPaint.h"
16#include "include/core/SkPicture.h"
17#include "include/core/SkPictureRecorder.h"
18#include "include/core/SkPoint.h"
19#include "include/core/SkRect.h"
20#include "include/core/SkRefCnt.h"
21#include "include/core/SkScalar.h"
22#include "include/core/SkShader.h"
23#include "include/core/SkSize.h"
24#include "include/core/SkString.h"
25#include "include/core/SkTileMode.h"
26#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "include/effects/SkGradientShader.h"
Brian Osmanf7866332023-10-11 13:56:58 -040028#include "include/private/base/SkAssert.h"
29#include "tools/ToolUtils.h"
Kevin Lubickbca43ec2023-10-30 10:11:22 -040030#include "tools/fonts/FontToolUtils.h"
reed@android.com42309d42009-06-22 02:06:35 +000031
Brian Osmanf7866332023-10-11 13:56:58 -040032#include <initializer_list>
Ben Wagner7fde8e12019-05-01 17:28:53 -040033#include <math.h>
34
Hal Canaryfa3305a2019-07-18 12:36:54 -040035namespace {
reed@android.com42309d42009-06-22 02:06:35 +000036
37struct GradData {
brianosmane25d71c2016-09-28 11:27:28 -070038 int fCount;
39 const SkColor* fColors;
40 const SkColor4f* fColors4f;
41 const SkScalar* fPos;
reed@android.com42309d42009-06-22 02:06:35 +000042};
43
mtkleindbfd7ab2016-09-01 11:24:54 -070044constexpr SkColor gColors[] = {
reed@android.com42309d42009-06-22 02:06:35 +000045 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK
46};
brianosmane25d71c2016-09-28 11:27:28 -070047constexpr SkColor4f gColors4f[] ={
48 { 1.0f, 0.0f, 0.0f, 1.0f }, // Red
49 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
50 { 0.0f, 0.0f, 1.0f, 1.0f }, // Blue
51 { 1.0f, 1.0f, 1.0f, 1.0f }, // White
52 { 0.0f, 0.0f, 0.0f, 1.0f } // Black
53};
mtkleindbfd7ab2016-09-01 11:24:54 -070054constexpr SkScalar gPos0[] = { 0, SK_Scalar1 };
55constexpr SkScalar gPos1[] = { SK_Scalar1/4, SK_Scalar1*3/4 };
56constexpr SkScalar gPos2[] = {
reed@android.com42309d42009-06-22 02:06:35 +000057 0, SK_Scalar1/8, SK_Scalar1/2, SK_Scalar1*7/8, SK_Scalar1
58};
59
mtkleindbfd7ab2016-09-01 11:24:54 -070060constexpr SkScalar gPosClamp[] = {0.0f, 0.0f, 1.0f, 1.0f};
61constexpr SkColor gColorClamp[] = {
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +000062 SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorBLUE
63};
brianosmane25d71c2016-09-28 11:27:28 -070064constexpr SkColor4f gColor4fClamp[] ={
65 { 1.0f, 0.0f, 0.0f, 1.0f }, // Red
66 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
67 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
68 { 0.0f, 0.0f, 1.0f, 1.0f } // Blue
69};
mtkleindbfd7ab2016-09-01 11:24:54 -070070constexpr GradData gGradData[] = {
brianosmane25d71c2016-09-28 11:27:28 -070071 { 2, gColors, gColors4f, nullptr },
72 { 2, gColors, gColors4f, gPos0 },
73 { 2, gColors, gColors4f, gPos1 },
74 { 5, gColors, gColors4f, nullptr },
75 { 5, gColors, gColors4f, gPos2 },
76 { 4, gColorClamp, gColor4fClamp, gPosClamp }
reed@android.com42309d42009-06-22 02:06:35 +000077};
78
reed1a9b9642016-03-13 14:13:58 -070079static sk_sp<SkShader> MakeLinear(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040080 SkTileMode tm, const SkMatrix& localMatrix) {
reed1a9b9642016-03-13 14:13:58 -070081 return SkGradientShader::MakeLinear(pts, data.fColors, data.fPos, data.fCount, tm, 0,
82 &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000083}
reed@google.comf3c1cc92010-12-23 16:45:33 +000084
brianosmane25d71c2016-09-28 11:27:28 -070085static sk_sp<SkShader> MakeLinear4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040086 SkTileMode tm, const SkMatrix& localMatrix) {
Mike Reed2e56f3c2019-03-06 11:21:15 -050087 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -070088 return SkGradientShader::MakeLinear(pts, data.fColors4f, srgb, data.fPos, data.fCount, tm, 0,
89 &localMatrix);
90}
91
reed1a9b9642016-03-13 14:13:58 -070092static sk_sp<SkShader> MakeRadial(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040093 SkTileMode tm, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +000094 SkPoint center;
95 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
96 SkScalarAve(pts[0].fY, pts[1].fY));
reed1a9b9642016-03-13 14:13:58 -070097 return SkGradientShader::MakeRadial(center, center.fX, data.fColors, data.fPos, data.fCount,
98 tm, 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000099}
100
brianosmane25d71c2016-09-28 11:27:28 -0700101static sk_sp<SkShader> MakeRadial4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400102 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700103 SkPoint center;
104 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
105 SkScalarAve(pts[0].fY, pts[1].fY));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500106 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700107 return SkGradientShader::MakeRadial(center, center.fX, data.fColors4f, srgb, data.fPos,
108 data.fCount, tm, 0, &localMatrix);
109}
110
reed1a9b9642016-03-13 14:13:58 -0700111static sk_sp<SkShader> MakeSweep(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400112 SkTileMode, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +0000113 SkPoint center;
114 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
115 SkScalarAve(pts[0].fY, pts[1].fY));
reed1a9b9642016-03-13 14:13:58 -0700116 return SkGradientShader::MakeSweep(center.fX, center.fY, data.fColors, data.fPos, data.fCount,
117 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +0000118}
119
brianosmane25d71c2016-09-28 11:27:28 -0700120static sk_sp<SkShader> MakeSweep4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400121 SkTileMode, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700122 SkPoint center;
123 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
124 SkScalarAve(pts[0].fY, pts[1].fY));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500125 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700126 return SkGradientShader::MakeSweep(center.fX, center.fY, data.fColors4f, srgb, data.fPos,
127 data.fCount, 0, &localMatrix);
128}
129
reed1a9b9642016-03-13 14:13:58 -0700130static sk_sp<SkShader> Make2Radial(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400131 SkTileMode tm, const SkMatrix& localMatrix) {
reed@google.comf3c1cc92010-12-23 16:45:33 +0000132 SkPoint center0, center1;
133 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
134 SkScalarAve(pts[0].fY, pts[1].fY));
135 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
136 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
reed1a9b9642016-03-13 14:13:58 -0700137 return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
138 center0, (pts[1].fX - pts[0].fX) / 2,
139 data.fColors, data.fPos, data.fCount, tm,
140 0, &localMatrix);
reed@google.comf3c1cc92010-12-23 16:45:33 +0000141}
142
brianosmane25d71c2016-09-28 11:27:28 -0700143static sk_sp<SkShader> Make2Radial4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400144 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700145 SkPoint center0, center1;
146 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
147 SkScalarAve(pts[0].fY, pts[1].fY));
148 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3) / 5),
149 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1) / 4));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500150 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700151 return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
152 center0, (pts[1].fX - pts[0].fX) / 2,
153 data.fColors4f, srgb, data.fPos, data.fCount, tm,
154 0, &localMatrix);
155}
156
reed1a9b9642016-03-13 14:13:58 -0700157static sk_sp<SkShader> Make2Conical(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400158 SkTileMode tm, const SkMatrix& localMatrix) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000159 SkPoint center0, center1;
reed80ea19c2015-05-12 10:37:34 -0700160 SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10;
161 SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000162 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
163 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
reed1a9b9642016-03-13 14:13:58 -0700164 return SkGradientShader::MakeTwoPointConical(center1, radius1, center0, radius0,
165 data.fColors, data.fPos,
166 data.fCount, tm, 0, &localMatrix);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000167}
168
brianosmane25d71c2016-09-28 11:27:28 -0700169static sk_sp<SkShader> Make2Conical4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400170 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700171 SkPoint center0, center1;
172 SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10;
173 SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3;
174 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
175 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
Mike Reed2e56f3c2019-03-06 11:21:15 -0500176 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700177 return SkGradientShader::MakeTwoPointConical(center1, radius1, center0, radius0,
178 data.fColors4f, srgb, data.fPos,
179 data.fCount, tm, 0, &localMatrix);
180}
181
reed1a9b9642016-03-13 14:13:58 -0700182typedef sk_sp<SkShader> (*GradMaker)(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400183 SkTileMode tm, const SkMatrix& localMatrix);
mtkleindbfd7ab2016-09-01 11:24:54 -0700184constexpr GradMaker gGradMakers[] = {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000185 MakeLinear, MakeRadial, MakeSweep, Make2Radial, Make2Conical
reed@android.com42309d42009-06-22 02:06:35 +0000186};
brianosmane25d71c2016-09-28 11:27:28 -0700187constexpr GradMaker gGradMakers4f[] ={
188 MakeLinear4f, MakeRadial4f, MakeSweep4f, Make2Radial4f, Make2Conical4f
189};
reed@android.com42309d42009-06-22 02:06:35 +0000190
191///////////////////////////////////////////////////////////////////////////////
192
Hal Canaryfa3305a2019-07-18 12:36:54 -0400193class GradientsGM : public skiagm::GM {
reed@android.com42309d42009-06-22 02:06:35 +0000194public:
Hal Canaryfa3305a2019-07-18 12:36:54 -0400195 GradientsGM(bool dither) : fDither(dither) {}
rmistry@google.comd6176b02012-08-23 18:14:13 +0000196
reed@android.com42309d42009-06-22 02:06:35 +0000197protected:
Hal Canaryfa3305a2019-07-18 12:36:54 -0400198 const bool fDither;
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000199
Hal Canaryfa3305a2019-07-18 12:36:54 -0400200 void onDraw(SkCanvas* canvas) override {
reed@android.com42309d42009-06-22 02:06:35 +0000201 SkPoint pts[2] = {
202 { 0, 0 },
Matt Saretteebe87f2017-01-26 15:14:56 -0500203 { SkIntToScalar(100), SkIntToScalar(100) }
reed@android.com42309d42009-06-22 02:06:35 +0000204 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400205 SkTileMode tm = SkTileMode::kClamp;
Matt Saretteebe87f2017-01-26 15:14:56 -0500206 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
reed@android.com42309d42009-06-22 02:06:35 +0000207 SkPaint paint;
208 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700209 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000210
reed@android.com42309d42009-06-22 02:06:35 +0000211 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
Herb Derbyc37b3862022-06-21 09:49:17 -0400212 for (size_t i = 0; i < std::size(gGradData); i++) {
reed@android.com42309d42009-06-22 02:06:35 +0000213 canvas->save();
Herb Derbyc37b3862022-06-21 09:49:17 -0400214 for (size_t j = 0; j < std::size(gGradMakers); j++) {
commit-bot@chromium.org9c9005a2014-04-28 14:55:39 +0000215 SkMatrix scale = SkMatrix::I();
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000216
217 if (i == 5) { // if the clamp case
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000218 scale.setScale(0.5f, 0.5f);
219 scale.postTranslate(25.f, 25.f);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000220 }
skia.committer@gmail.comd55e3572013-08-08 07:01:20 +0000221
reed1a9b9642016-03-13 14:13:58 -0700222 paint.setShader(gGradMakers[j](pts, gGradData[i], tm, scale));
reed@android.com42309d42009-06-22 02:06:35 +0000223 canvas->drawRect(r, paint);
reed@android.com42309d42009-06-22 02:06:35 +0000224 canvas->translate(0, SkIntToScalar(120));
225 }
226 canvas->restore();
227 canvas->translate(SkIntToScalar(120), 0);
228 }
229 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000230
reed@android.com42309d42009-06-22 02:06:35 +0000231private:
Hal Canaryfa3305a2019-07-18 12:36:54 -0400232 void onOnceBeforeDraw() override { this->setBGColor(0xFFDDDDDD); }
233
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000234 SkString getName() const override {
Hal Canaryfa3305a2019-07-18 12:36:54 -0400235 return SkString(fDither ? "gradients" : "gradients_nodither");
236 }
237
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000238 SkISize getISize() override { return {840, 815}; }
reed@android.com42309d42009-06-22 02:06:35 +0000239};
fmalita063675b2015-10-12 10:41:48 -0700240DEF_GM( return new GradientsGM(true); )
241DEF_GM( return new GradientsGM(false); )
reed@android.com42309d42009-06-22 02:06:35 +0000242
brianosmane25d71c2016-09-28 11:27:28 -0700243// Like the original gradients GM, but using the SkColor4f shader factories. Should be identical.
Hal Canaryfa3305a2019-07-18 12:36:54 -0400244class Gradients4fGM : public skiagm::GM {
brianosmane25d71c2016-09-28 11:27:28 -0700245public:
Hal Canaryfa3305a2019-07-18 12:36:54 -0400246 Gradients4fGM(bool dither) : fDither(dither) {}
brianosmane25d71c2016-09-28 11:27:28 -0700247
Hal Canaryfa3305a2019-07-18 12:36:54 -0400248private:
249 void onOnceBeforeDraw() override { this->setBGColor(0xFFDDDDDD); }
brianosmane25d71c2016-09-28 11:27:28 -0700250
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000251 SkString getName() const override {
brianosmane25d71c2016-09-28 11:27:28 -0700252 return SkString(fDither ? "gradients4f" : "gradients4f_nodither");
253 }
254
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000255 SkISize getISize() override { return {840, 815}; }
brianosmane25d71c2016-09-28 11:27:28 -0700256
Hal Canaryfa3305a2019-07-18 12:36:54 -0400257 void onDraw(SkCanvas* canvas) override {
brianosmane25d71c2016-09-28 11:27:28 -0700258 SkPoint pts[2] ={
259 { 0, 0 },
260 { SkIntToScalar(100), SkIntToScalar(100) }
261 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400262 SkTileMode tm = SkTileMode::kClamp;
brianosmane25d71c2016-09-28 11:27:28 -0700263 SkRect r ={ 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
264 SkPaint paint;
265 paint.setAntiAlias(true);
266 paint.setDither(fDither);
267
268 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
Herb Derbyc37b3862022-06-21 09:49:17 -0400269 for (size_t i = 0; i < std::size(gGradData); i++) {
brianosmane25d71c2016-09-28 11:27:28 -0700270 canvas->save();
Herb Derbyc37b3862022-06-21 09:49:17 -0400271 for (size_t j = 0; j < std::size(gGradMakers4f); j++) {
brianosmane25d71c2016-09-28 11:27:28 -0700272 SkMatrix scale = SkMatrix::I();
273
274 if (i == 5) { // if the clamp case
275 scale.setScale(0.5f, 0.5f);
276 scale.postTranslate(25.f, 25.f);
277 }
278
279 paint.setShader(gGradMakers4f[j](pts, gGradData[i], tm, scale));
280 canvas->drawRect(r, paint);
281 canvas->translate(0, SkIntToScalar(120));
282 }
283 canvas->restore();
284 canvas->translate(SkIntToScalar(120), 0);
285 }
286 }
287
brianosmane25d71c2016-09-28 11:27:28 -0700288 bool fDither;
brianosmane25d71c2016-09-28 11:27:28 -0700289};
290DEF_GM(return new Gradients4fGM(true); )
291DEF_GM(return new Gradients4fGM(false); )
292
rileya@google.com5cf2c282012-07-09 14:42:16 +0000293// Based on the original gradient slide, but with perspective applied to the
294// gradient shaders' local matrices
Hal Canaryfa3305a2019-07-18 12:36:54 -0400295class GradientsLocalPerspectiveGM : public skiagm::GM {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000296public:
fmalita063675b2015-10-12 10:41:48 -0700297 GradientsLocalPerspectiveGM(bool dither) : fDither(dither) {
Mike Kleind46dce32018-08-16 10:17:03 -0400298 this->setBGColor(0xFFDDDDDD);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000299 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000300
Hal Canaryfa3305a2019-07-18 12:36:54 -0400301private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000302 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700303 return SkString(fDither ? "gradients_local_perspective" :
304 "gradients_local_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000305 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000306
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000307 SkISize getISize() override { return {840, 815}; }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000308
Hal Canaryfa3305a2019-07-18 12:36:54 -0400309 void onDraw(SkCanvas* canvas) override {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000310 SkPoint pts[2] = {
311 { 0, 0 },
312 { SkIntToScalar(100), SkIntToScalar(100) }
313 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400314 SkTileMode tm = SkTileMode::kClamp;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000315 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
316 SkPaint paint;
317 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700318 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000319
rileya@google.com5cf2c282012-07-09 14:42:16 +0000320 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
Herb Derbyc37b3862022-06-21 09:49:17 -0400321 for (size_t i = 0; i < std::size(gGradData); i++) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000322 canvas->save();
Herb Derbyc37b3862022-06-21 09:49:17 -0400323 for (size_t j = 0; j < std::size(gGradMakers); j++) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000324 // apply an increasing y perspective as we move to the right
325 SkMatrix perspective;
326 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700327 perspective.setPerspY(SkIntToScalar(i+1) / 500);
328 perspective.setSkewX(SkIntToScalar(i+1) / 10);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000329
reed1a9b9642016-03-13 14:13:58 -0700330 paint.setShader(gGradMakers[j](pts, gGradData[i], tm, perspective));
rileya@google.com5cf2c282012-07-09 14:42:16 +0000331 canvas->drawRect(r, paint);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000332 canvas->translate(0, SkIntToScalar(120));
333 }
334 canvas->restore();
335 canvas->translate(SkIntToScalar(120), 0);
336 }
337 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000338
fmalita063675b2015-10-12 10:41:48 -0700339 bool fDither;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000340};
fmalita063675b2015-10-12 10:41:48 -0700341DEF_GM( return new GradientsLocalPerspectiveGM(true); )
342DEF_GM( return new GradientsLocalPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000343
344// Based on the original gradient slide, but with perspective applied to
345// the view matrix
346class GradientsViewPerspectiveGM : public GradientsGM {
fmalita063675b2015-10-12 10:41:48 -0700347public:
348 GradientsViewPerspectiveGM(bool dither) : INHERITED(dither) { }
349
Hal Canaryfa3305a2019-07-18 12:36:54 -0400350private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000351 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700352 return SkString(fDither ? "gradients_view_perspective" :
353 "gradients_view_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000354 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000355
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000356 SkISize getISize() override { return {840, 500}; }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000357
Hal Canaryfa3305a2019-07-18 12:36:54 -0400358 void onDraw(SkCanvas* canvas) override {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000359 SkMatrix perspective;
360 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700361 perspective.setPerspY(0.001f);
362 perspective.setSkewX(SkIntToScalar(8) / 25);
scroggo@google.com837d31a2012-08-15 18:42:45 +0000363 canvas->concat(perspective);
Hal Canaryfa3305a2019-07-18 12:36:54 -0400364 this->INHERITED::onDraw(canvas);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000365 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000366
rileya@google.com5cf2c282012-07-09 14:42:16 +0000367private:
John Stiles7571f9e2020-09-02 22:42:33 -0400368 using INHERITED = GradientsGM;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000369};
fmalita063675b2015-10-12 10:41:48 -0700370DEF_GM( return new GradientsViewPerspectiveGM(true); )
371DEF_GM( return new GradientsViewPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000372
reed@google.comac864a92011-06-27 18:11:17 +0000373/*
374 Inspired by this <canvas> javascript, where we need to detect that we are not
375 solving a quadratic equation, but must instead solve a linear (since our X^2
376 coefficient is 0)
377
378 ctx.fillStyle = '#f00';
379 ctx.fillRect(0, 0, 100, 50);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000380
reed@google.comac864a92011-06-27 18:11:17 +0000381 var g = ctx.createRadialGradient(-80, 25, 70, 0, 25, 150);
382 g.addColorStop(0, '#f00');
383 g.addColorStop(0.01, '#0f0');
384 g.addColorStop(0.99, '#0f0');
385 g.addColorStop(1, '#f00');
386 ctx.fillStyle = g;
387 ctx.fillRect(0, 0, 100, 50);
388 */
Hal Canaryfa3305a2019-07-18 12:36:54 -0400389class GradientsDegenrate2PointGM : public skiagm::GM {
reed@google.comac864a92011-06-27 18:11:17 +0000390public:
fmalita063675b2015-10-12 10:41:48 -0700391 GradientsDegenrate2PointGM(bool dither) : fDither(dither) {}
rmistry@google.comd6176b02012-08-23 18:14:13 +0000392
Hal Canaryfa3305a2019-07-18 12:36:54 -0400393private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000394 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700395 return SkString(fDither ? "gradients_degenerate_2pt" : "gradients_degenerate_2pt_nodither");
reed@google.comac864a92011-06-27 18:11:17 +0000396 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000397
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000398 SkISize getISize() override { return {320, 320}; }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000399
Hal Canaryfa3305a2019-07-18 12:36:54 -0400400 void onDraw(SkCanvas* canvas) override {
reed@google.comac864a92011-06-27 18:11:17 +0000401 canvas->drawColor(SK_ColorBLUE);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000402
reed@google.comac864a92011-06-27 18:11:17 +0000403 SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorRED };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000404 SkScalar pos[] = { 0, 0.01f, 0.99f, SK_Scalar1 };
epoger@google.com59f3abf2011-07-21 15:50:33 +0000405 SkPoint c0;
406 c0.iset(-80, 25);
407 SkScalar r0 = SkIntToScalar(70);
408 SkPoint c1;
409 c1.iset(0, 25);
410 SkScalar r1 = SkIntToScalar(150);
reed9283d202016-03-13 13:01:57 -0700411 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -0700412 paint.setShader(SkGradientShader::MakeTwoPointConical(c0, r0, c1, r1, colors,
Herb Derbyc37b3862022-06-21 09:49:17 -0400413 pos, std::size(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400414 SkTileMode::kClamp));
fmalita063675b2015-10-12 10:41:48 -0700415 paint.setDither(fDither);
reed@google.comac864a92011-06-27 18:11:17 +0000416 canvas->drawPaint(paint);
417 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000418
fmalita063675b2015-10-12 10:41:48 -0700419 bool fDither;
reed@google.comac864a92011-06-27 18:11:17 +0000420};
fmalita063675b2015-10-12 10:41:48 -0700421DEF_GM( return new GradientsDegenrate2PointGM(true); )
422DEF_GM( return new GradientsDegenrate2PointGM(false); )
reed@google.comac864a92011-06-27 18:11:17 +0000423
caryclarkcb100712016-02-26 05:59:40 -0800424/* bug.skia.org/517
425<canvas id="canvas"></canvas>
426<script>
427var c = document.getElementById("canvas");
428var ctx = c.getContext("2d");
429ctx.fillStyle = '#ff0';
430ctx.fillRect(0, 0, 100, 50);
431
432var g = ctx.createRadialGradient(200, 25, 20, 200, 25, 10);
433g.addColorStop(0, '#0f0');
434g.addColorStop(0.003, '#f00'); // 0.004 makes this work
435g.addColorStop(1, '#ff0');
436ctx.fillStyle = g;
437ctx.fillRect(0, 0, 100, 50);
438</script>
439*/
440
441// should draw only green
442DEF_SIMPLE_GM(small_color_stop, canvas, 100, 150) {
443 SkColor colors[] = { SK_ColorGREEN, SK_ColorRED, SK_ColorYELLOW };
444 SkScalar pos[] = { 0, 0.003f, SK_Scalar1 }; // 0.004f makes this work
445 SkPoint c0 = { 200, 25 };
446 SkScalar r0 = 20;
447 SkPoint c1 = { 200, 25 };
448 SkScalar r1 = 10;
reed1a9b9642016-03-13 14:13:58 -0700449
caryclarkcb100712016-02-26 05:59:40 -0800450 SkPaint paint;
451 paint.setColor(SK_ColorYELLOW);
452 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
reed1a9b9642016-03-13 14:13:58 -0700453 paint.setShader(SkGradientShader::MakeTwoPointConical(c0, r0, c1, r1, colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400454 std::size(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400455 SkTileMode::kClamp));
caryclarkcb100712016-02-26 05:59:40 -0800456 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
457}
458
459
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000460/// Tests correctness of *optimized* codepaths in gradients.
461
Hal Canaryfa3305a2019-07-18 12:36:54 -0400462class ClampedGradientsGM : public skiagm::GM {
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000463public:
fmalita063675b2015-10-12 10:41:48 -0700464 ClampedGradientsGM(bool dither) : fDither(dither) {}
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000465
Hal Canaryfa3305a2019-07-18 12:36:54 -0400466private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000467 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700468 return SkString(fDither ? "clamped_gradients" : "clamped_gradients_nodither");
469 }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000470
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000471 SkISize getISize() override { return {640, 510}; }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000472
Hal Canaryfa3305a2019-07-18 12:36:54 -0400473 void onDraw(SkCanvas* canvas) override {
Mike Kleind46dce32018-08-16 10:17:03 -0400474 canvas->drawColor(0xFFDDDDDD);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000475
476 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(300) };
477 SkPaint paint;
fmalita063675b2015-10-12 10:41:48 -0700478 paint.setDither(fDither);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000479 paint.setAntiAlias(true);
480
481 SkPoint center;
482 center.iset(0, 300);
483 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
reed1a9b9642016-03-13 14:13:58 -0700484 paint.setShader(SkGradientShader::MakeRadial(
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000485 SkPoint(center),
halcanary96fcdcc2015-08-27 07:41:13 -0700486 SkIntToScalar(200), gColors, nullptr, 5,
Mike Reedfae8fce2019-04-03 10:27:45 -0400487 SkTileMode::kClamp));
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000488 canvas->drawRect(r, paint);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000489 }
490
fmalita063675b2015-10-12 10:41:48 -0700491 bool fDither;
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000492};
fmalita063675b2015-10-12 10:41:48 -0700493DEF_GM( return new ClampedGradientsGM(true); )
494DEF_GM( return new ClampedGradientsGM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000495
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000496/// Checks quality of large radial gradients, which may display
497/// some banding.
498
Hal Canaryfa3305a2019-07-18 12:36:54 -0400499class RadialGradientGM : public skiagm::GM {
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000500 SkString getName() const override { return SkString("radial_gradient"); }
Hal Canaryfa3305a2019-07-18 12:36:54 -0400501
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000502 SkISize getISize() override { return {1280, 1280}; }
Hal Canaryfa3305a2019-07-18 12:36:54 -0400503
mtklein36352bf2015-03-25 18:17:31 -0700504 void onDraw(SkCanvas* canvas) override {
reed@google.combb0948f2012-01-31 14:44:13 +0000505 const SkISize dim = this->getISize();
506
Hal Canaryfa3305a2019-07-18 12:36:54 -0400507 canvas->drawColor(0xFF000000);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000508
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000509 SkPaint paint;
510 paint.setDither(true);
511 SkPoint center;
reed@google.combb0948f2012-01-31 14:44:13 +0000512 center.set(SkIntToScalar(dim.width())/2, SkIntToScalar(dim.height())/2);
513 SkScalar radius = SkIntToScalar(dim.width())/2;
514 const SkColor colors[] = { 0x7f7f7f7f, 0x7f7f7f7f, 0xb2000000 };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000515 const SkScalar pos[] = { 0.0f,
516 0.35f,
517 1.0f };
reed1a9b9642016-03-13 14:13:58 -0700518 paint.setShader(SkGradientShader::MakeRadial(center, radius, colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400519 std::size(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400520 SkTileMode::kClamp));
reed@google.combb0948f2012-01-31 14:44:13 +0000521 SkRect r = {
522 0, 0, SkIntToScalar(dim.width()), SkIntToScalar(dim.height())
523 };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000524 canvas->drawRect(r, paint);
525 }
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000526};
reed3d9005c2015-04-23 10:30:27 -0700527DEF_GM( return new RadialGradientGM; )
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000528
Hal Canaryfa3305a2019-07-18 12:36:54 -0400529class RadialGradient2GM : public skiagm::GM {
mtklein@google.com361a72f2013-08-19 18:43:34 +0000530public:
fmalita063675b2015-10-12 10:41:48 -0700531 RadialGradient2GM(bool dither) : fDither(dither) {}
mtklein@google.com361a72f2013-08-19 18:43:34 +0000532
Hal Canaryfa3305a2019-07-18 12:36:54 -0400533private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000534 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700535 return SkString(fDither ? "radial_gradient2" : "radial_gradient2_nodither");
536 }
537
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000538 SkISize getISize() override { return {800, 400}; }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000539
540 // Reproduces the example given in bug 7671058.
mtklein36352bf2015-03-25 18:17:31 -0700541 void onDraw(SkCanvas* canvas) override {
mtklein@google.com361a72f2013-08-19 18:43:34 +0000542 SkPaint paint1, paint2, paint3;
543 paint1.setStyle(SkPaint::kFill_Style);
544 paint2.setStyle(SkPaint::kFill_Style);
545 paint3.setStyle(SkPaint::kFill_Style);
546
547 const SkColor sweep_colors[] =
548 { 0xFFFF0000, 0xFFFFFF00, 0xFF00FF00, 0xFF00FFFF, 0xFF0000FF, 0xFFFF00FF, 0xFFFF0000 };
549 const SkColor colors1[] = { 0xFFFFFFFF, 0x00000000 };
550 const SkColor colors2[] = { 0xFF000000, 0x00000000 };
551
552 const SkScalar cx = 200, cy = 200, radius = 150;
553 SkPoint center;
554 center.set(cx, cy);
555
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000556 // We can either interpolate endpoints and premultiply each point (default, more precision),
557 // or premultiply the endpoints first, avoiding the need to premultiply each point (cheap).
558 const uint32_t flags[] = { 0, SkGradientShader::kInterpolateColorsInPremul_Flag };
mtklein@google.com361a72f2013-08-19 18:43:34 +0000559
Herb Derbyc37b3862022-06-21 09:49:17 -0400560 for (size_t i = 0; i < std::size(flags); i++) {
reed1a9b9642016-03-13 14:13:58 -0700561 paint1.setShader(SkGradientShader::MakeSweep(cx, cy, sweep_colors,
Herb Derbyc37b3862022-06-21 09:49:17 -0400562 nullptr, std::size(sweep_colors),
reed1a9b9642016-03-13 14:13:58 -0700563 flags[i], nullptr));
564 paint2.setShader(SkGradientShader::MakeRadial(center, radius, colors1,
Herb Derbyc37b3862022-06-21 09:49:17 -0400565 nullptr, std::size(colors1),
Mike Reedfae8fce2019-04-03 10:27:45 -0400566 SkTileMode::kClamp,
reed1a9b9642016-03-13 14:13:58 -0700567 flags[i], nullptr));
568 paint3.setShader(SkGradientShader::MakeRadial(center, radius, colors2,
Herb Derbyc37b3862022-06-21 09:49:17 -0400569 nullptr, std::size(colors2),
Mike Reedfae8fce2019-04-03 10:27:45 -0400570 SkTileMode::kClamp,
reed1a9b9642016-03-13 14:13:58 -0700571 flags[i], nullptr));
fmalita063675b2015-10-12 10:41:48 -0700572 paint1.setDither(fDither);
fmalita063675b2015-10-12 10:41:48 -0700573 paint2.setDither(fDither);
fmalita063675b2015-10-12 10:41:48 -0700574 paint3.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000575
576 canvas->drawCircle(cx, cy, radius, paint1);
577 canvas->drawCircle(cx, cy, radius, paint3);
578 canvas->drawCircle(cx, cy, radius, paint2);
579
580 canvas->translate(400, 0);
581 }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000582 }
583
584private:
fmalita063675b2015-10-12 10:41:48 -0700585 bool fDither;
586
John Stiles7571f9e2020-09-02 22:42:33 -0400587 using INHERITED = GM;
mtklein@google.com361a72f2013-08-19 18:43:34 +0000588};
fmalita063675b2015-10-12 10:41:48 -0700589DEF_GM( return new RadialGradient2GM(true); )
590DEF_GM( return new RadialGradient2GM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000591
reed3d9005c2015-04-23 10:30:27 -0700592// Shallow radial (shows banding on raster)
Hal Canaryfa3305a2019-07-18 12:36:54 -0400593class RadialGradient3GM : public skiagm::GM {
fmalita063675b2015-10-12 10:41:48 -0700594public:
595 RadialGradient3GM(bool dither) : fDither(dither) { }
reed@android.com42309d42009-06-22 02:06:35 +0000596
Hal Canaryfa3305a2019-07-18 12:36:54 -0400597private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000598 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700599 return SkString(fDither ? "radial_gradient3" : "radial_gradient3_nodither");
600 }
reed@android.com42309d42009-06-22 02:06:35 +0000601
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000602 SkISize getISize() override { return {500, 500}; }
reed@google.comac864a92011-06-27 18:11:17 +0000603
reed3d9005c2015-04-23 10:30:27 -0700604 bool runAsBench() const override { return true; }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000605
reed3d9005c2015-04-23 10:30:27 -0700606 void onOnceBeforeDraw() override {
607 const SkPoint center = { 0, 0 };
608 const SkScalar kRadius = 3000;
John Stilesacf71642021-08-12 22:33:57 -0400609 const SkColor kColors[] = { 0xFFFFFFFF, 0xFF000000 };
610 fShader = SkGradientShader::MakeRadial(center, kRadius, kColors, nullptr, 2,
Mike Reedfae8fce2019-04-03 10:27:45 -0400611 SkTileMode::kClamp);
reed3d9005c2015-04-23 10:30:27 -0700612 }
rileya@google.com5cf2c282012-07-09 14:42:16 +0000613
reed3d9005c2015-04-23 10:30:27 -0700614 void onDraw(SkCanvas* canvas) override {
615 SkPaint paint;
616 paint.setShader(fShader);
fmalita063675b2015-10-12 10:41:48 -0700617 paint.setDither(fDither);
reed3d9005c2015-04-23 10:30:27 -0700618 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
619 }
halcanary9d524f22016-03-29 09:03:52 -0700620
reed3d9005c2015-04-23 10:30:27 -0700621private:
reed1a9b9642016-03-13 14:13:58 -0700622 sk_sp<SkShader> fShader;
fmalita063675b2015-10-12 10:41:48 -0700623 bool fDither;
624
John Stiles7571f9e2020-09-02 22:42:33 -0400625 using INHERITED = GM;
reed3d9005c2015-04-23 10:30:27 -0700626};
fmalita063675b2015-10-12 10:41:48 -0700627DEF_GM( return new RadialGradient3GM(true); )
628DEF_GM( return new RadialGradient3GM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000629
Hal Canaryfa3305a2019-07-18 12:36:54 -0400630class RadialGradient4GM : public skiagm::GM {
fmalita063675b2015-10-12 10:41:48 -0700631public:
632 RadialGradient4GM(bool dither) : fDither(dither) { }
caryclark1864bfa2015-07-30 06:41:39 -0700633
Hal Canaryfa3305a2019-07-18 12:36:54 -0400634private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000635 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700636 return SkString(fDither ? "radial_gradient4" : "radial_gradient4_nodither");
637 }
caryclark1864bfa2015-07-30 06:41:39 -0700638
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000639 SkISize getISize() override { return {500, 500}; }
caryclark1864bfa2015-07-30 06:41:39 -0700640
641 void onOnceBeforeDraw() override {
642 const SkPoint center = { 250, 250 };
643 const SkScalar kRadius = 250;
644 const SkColor colors[] = { SK_ColorRED, SK_ColorRED, SK_ColorWHITE, SK_ColorWHITE,
645 SK_ColorRED };
646 const SkScalar pos[] = { 0, .4f, .4f, .8f, .8f, 1 };
reed1a9b9642016-03-13 14:13:58 -0700647 fShader = SkGradientShader::MakeRadial(center, kRadius, colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400648 std::size(gColors), SkTileMode::kClamp);
caryclark1864bfa2015-07-30 06:41:39 -0700649 }
650
651 void onDraw(SkCanvas* canvas) override {
652 SkPaint paint;
653 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700654 paint.setDither(fDither);
caryclark1864bfa2015-07-30 06:41:39 -0700655 paint.setShader(fShader);
656 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
657 }
halcanary9d524f22016-03-29 09:03:52 -0700658
caryclark1864bfa2015-07-30 06:41:39 -0700659private:
reed1a9b9642016-03-13 14:13:58 -0700660 sk_sp<SkShader> fShader;
fmalita063675b2015-10-12 10:41:48 -0700661 bool fDither;
662
John Stiles7571f9e2020-09-02 22:42:33 -0400663 using INHERITED = GM;
caryclark1864bfa2015-07-30 06:41:39 -0700664};
fmalita063675b2015-10-12 10:41:48 -0700665DEF_GM( return new RadialGradient4GM(true); )
666DEF_GM( return new RadialGradient4GM(false); )
caryclark1864bfa2015-07-30 06:41:39 -0700667
Hal Canaryfa3305a2019-07-18 12:36:54 -0400668class LinearGradientGM : public skiagm::GM {
fmalita063675b2015-10-12 10:41:48 -0700669public:
670 LinearGradientGM(bool dither) : fDither(dither) { }
caryclark159fa572015-07-30 12:35:48 -0700671
Hal Canaryfa3305a2019-07-18 12:36:54 -0400672private:
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000673 SkString getName() const override {
fmalita063675b2015-10-12 10:41:48 -0700674 return SkString(fDither ? "linear_gradient" : "linear_gradient_nodither");
675 }
676
caryclark159fa572015-07-30 12:35:48 -0700677 const SkScalar kWidthBump = 30.f;
678 const SkScalar kHeight = 5.f;
679 const SkScalar kMinWidth = 540.f;
680
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000681 SkISize getISize() override { return {500, 500}; }
caryclark159fa572015-07-30 12:35:48 -0700682
683 void onOnceBeforeDraw() override {
684 SkPoint pts[2] = { {0, 0}, {0, 0} };
685 const SkColor colors[] = { SK_ColorWHITE, SK_ColorWHITE, 0xFF008200, 0xFF008200,
686 SK_ColorWHITE, SK_ColorWHITE };
687 const SkScalar unitPos[] = { 0, 50, 70, 500, 540 };
688 SkScalar pos[6];
689 pos[5] = 1;
Herb Derbyc37b3862022-06-21 09:49:17 -0400690 for (int index = 0; index < (int) std::size(fShader); ++index) {
caryclark159fa572015-07-30 12:35:48 -0700691 pts[1].fX = 500.f + index * kWidthBump;
Herb Derbyc37b3862022-06-21 09:49:17 -0400692 for (int inner = 0; inner < (int) std::size(unitPos); ++inner) {
caryclark159fa572015-07-30 12:35:48 -0700693 pos[inner] = unitPos[inner] / (kMinWidth + index * kWidthBump);
694 }
reed1a9b9642016-03-13 14:13:58 -0700695 fShader[index] = SkGradientShader::MakeLinear(pts, colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400696 std::size(gColors), SkTileMode::kClamp);
caryclark159fa572015-07-30 12:35:48 -0700697 }
698 }
699
700 void onDraw(SkCanvas* canvas) override {
701 SkPaint paint;
702 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700703 paint.setDither(fDither);
Herb Derbyc37b3862022-06-21 09:49:17 -0400704 for (int index = 0; index < (int) std::size(fShader); ++index) {
caryclark159fa572015-07-30 12:35:48 -0700705 paint.setShader(fShader[index]);
706 canvas->drawRect(SkRect::MakeLTRB(0, index * kHeight, kMinWidth + index * kWidthBump,
707 (index + 1) * kHeight), paint);
708 }
709 }
halcanary9d524f22016-03-29 09:03:52 -0700710
caryclark159fa572015-07-30 12:35:48 -0700711private:
reed1a9b9642016-03-13 14:13:58 -0700712 sk_sp<SkShader> fShader[100];
fmalita063675b2015-10-12 10:41:48 -0700713 bool fDither;
714
John Stiles7571f9e2020-09-02 22:42:33 -0400715 using INHERITED = GM;
caryclark159fa572015-07-30 12:35:48 -0700716};
fmalita063675b2015-10-12 10:41:48 -0700717DEF_GM( return new LinearGradientGM(true); )
718DEF_GM( return new LinearGradientGM(false); )
caryclark159fa572015-07-30 12:35:48 -0700719
Hal Canaryfa3305a2019-07-18 12:36:54 -0400720class LinearGradientTinyGM : public skiagm::GM {
Brian Salomon9fa47cc2021-10-08 18:48:26 -0400721 inline static constexpr uint32_t kFlags = 0;
fmalitabc590c02016-02-22 09:12:33 -0800722
Leandro Lovisolo24fa2112023-08-15 19:05:17 +0000723 SkString getName() const override { return SkString("linear_gradient_tiny"); }
fmalita8b78bd62015-11-20 13:58:24 -0800724
Leandro Lovisolo8f023882023-08-15 21:13:52 +0000725 SkISize getISize() override { return {600, 500}; }
fmalita8b78bd62015-11-20 13:58:24 -0800726
727 void onDraw(SkCanvas* canvas) override {
728 const SkScalar kRectSize = 100;
729 const unsigned kStopCount = 3;
730 const SkColor colors[kStopCount] = { SK_ColorGREEN, SK_ColorRED, SK_ColorGREEN };
731 const struct {
732 SkPoint pts[2];
733 SkScalar pos[kStopCount];
734 } configs[] = {
reedde3aac82015-11-22 13:00:04 -0800735 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999f, 1 }},
736 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000001f, 1 }},
737 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999999f, 1 }},
738 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000000001f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800739
reedde3aac82015-11-22 13:00:04 -0800740 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999f, 1 }},
741 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000001f, 1 }},
742 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999999f, 1 }},
743 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000000001f, 1 }},
744
745 { { SkPoint::Make(0, 0), SkPoint::Make(0.00001f, 0) }, { 0, 0.5f, 1 }},
746 { { SkPoint::Make(9.99999f, 0), SkPoint::Make(10, 0) }, { 0, 0.5f, 1 }},
747 { { SkPoint::Make(0, 0), SkPoint::Make(0, 0.00001f) }, { 0, 0.5f, 1 }},
748 { { SkPoint::Make(0, 9.99999f), SkPoint::Make(0, 10) }, { 0, 0.5f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800749 };
750
751 SkPaint paint;
Herb Derbyc37b3862022-06-21 09:49:17 -0400752 for (unsigned i = 0; i < std::size(configs); ++i) {
fmalita8b78bd62015-11-20 13:58:24 -0800753 SkAutoCanvasRestore acr(canvas, true);
reed1a9b9642016-03-13 14:13:58 -0700754 paint.setShader(SkGradientShader::MakeLinear(configs[i].pts, colors, configs[i].pos,
Mike Reedfae8fce2019-04-03 10:27:45 -0400755 kStopCount, SkTileMode::kClamp,
Hal Canaryfa3305a2019-07-18 12:36:54 -0400756 kFlags, nullptr));
fmalita8b78bd62015-11-20 13:58:24 -0800757 canvas->translate(kRectSize * ((i % 4) * 1.5f + 0.25f),
758 kRectSize * ((i / 4) * 1.5f + 0.25f));
759
fmalita8b78bd62015-11-20 13:58:24 -0800760 canvas->drawRect(SkRect::MakeWH(kRectSize, kRectSize), paint);
761 }
762 }
fmalitabc590c02016-02-22 09:12:33 -0800763};
Hal Canaryfa3305a2019-07-18 12:36:54 -0400764
765DEF_GM( return new LinearGradientTinyGM; )
766} // namespace
reedd4eaa252016-01-22 10:35:26 -0800767
768///////////////////////////////////////////////////////////////////////////////////////////////////
769
770struct GradRun {
771 SkColor fColors[4];
772 SkScalar fPos[4];
773 int fCount;
774};
775
776#define SIZE 121
777
Mike Reedfae8fce2019-04-03 10:27:45 -0400778static sk_sp<SkShader> make_linear(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800779 const SkPoint pts[] { { 30, 30 }, { SIZE - 30, SIZE - 30 } };
reed1a9b9642016-03-13 14:13:58 -0700780 return SkGradientShader::MakeLinear(pts, run.fColors, run.fPos, run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800781}
782
Mike Reedfae8fce2019-04-03 10:27:45 -0400783static sk_sp<SkShader> make_radial(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800784 const SkScalar half = SIZE * 0.5f;
reed1a9b9642016-03-13 14:13:58 -0700785 return SkGradientShader::MakeRadial({half,half}, half - 10, run.fColors, run.fPos,
786 run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800787}
788
Mike Reedfae8fce2019-04-03 10:27:45 -0400789static sk_sp<SkShader> make_conical(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800790 const SkScalar half = SIZE * 0.5f;
791 const SkPoint center { half, half };
reed1a9b9642016-03-13 14:13:58 -0700792 return SkGradientShader::MakeTwoPointConical(center, 20, center, half - 10,
793 run.fColors, run.fPos, run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800794}
795
Mike Reedfae8fce2019-04-03 10:27:45 -0400796static sk_sp<SkShader> make_sweep(const GradRun& run, SkTileMode) {
reedd4eaa252016-01-22 10:35:26 -0800797 const SkScalar half = SIZE * 0.5f;
reed1a9b9642016-03-13 14:13:58 -0700798 return SkGradientShader::MakeSweep(half, half, run.fColors, run.fPos, run.fCount);
reedd4eaa252016-01-22 10:35:26 -0800799}
800
801/*
802 * Exercise duplicate color-stops, at the ends, and in the middle
803 *
804 * At the time of this writing, only Linear correctly deals with duplicates at the ends,
805 * and then only correctly on CPU backend.
806 */
807DEF_SIMPLE_GM(gradients_dup_color_stops, canvas, 704, 564) {
808 const SkColor preColor = 0xFFFF0000; // clamp color before start
809 const SkColor postColor = 0xFF0000FF; // clamp color after end
810 const SkColor color0 = 0xFF000000;
811 const SkColor color1 = 0xFF00FF00;
812 const SkColor badColor = 0xFF3388BB; // should never be seen, fills out fixed-size array
813
814 const GradRun runs[] = {
815 { { color0, color1, badColor, badColor },
816 { 0, 1, -1, -1 },
817 2,
818 },
819 { { preColor, color0, color1, badColor },
820 { 0, 0, 1, -1 },
821 3,
822 },
823 { { color0, color1, postColor, badColor },
824 { 0, 1, 1, -1 },
825 3,
826 },
827 { { preColor, color0, color1, postColor },
828 { 0, 0, 1, 1 },
829 4,
830 },
831 { { color0, color0, color1, color1 },
832 { 0, 0.5f, 0.5f, 1 },
833 4,
834 },
835 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400836 sk_sp<SkShader> (*factories[])(const GradRun&, SkTileMode) {
reedd4eaa252016-01-22 10:35:26 -0800837 make_linear, make_radial, make_conical, make_sweep
838 };
839
840 const SkRect rect = SkRect::MakeWH(SIZE, SIZE);
841 const SkScalar dx = SIZE + 20;
842 const SkScalar dy = SIZE + 20;
Mike Reedfae8fce2019-04-03 10:27:45 -0400843 const SkTileMode mode = SkTileMode::kClamp;
reedd4eaa252016-01-22 10:35:26 -0800844
845 SkPaint paint;
846 canvas->translate(10, 10 - dy);
847 for (auto factory : factories) {
848 canvas->translate(0, dy);
849 SkAutoCanvasRestore acr(canvas, true);
850 for (const auto& run : runs) {
reed1a9b9642016-03-13 14:13:58 -0700851 paint.setShader(factory(run, mode));
reedd4eaa252016-01-22 10:35:26 -0800852 canvas->drawRect(rect, paint);
853 canvas->translate(dx, 0);
854 }
855 }
856}
fmalitabc590c02016-02-22 09:12:33 -0800857
Florin Malitad1aedde2017-06-07 15:03:38 -0400858static void draw_many_stops(SkCanvas* canvas) {
fmalitabc590c02016-02-22 09:12:33 -0800859 const unsigned kStopCount = 200;
Robert Phillips3b2b0552023-03-17 10:17:34 -0400860 const SkPoint pts[] = { {50, 50}, {450, 450}};
fmalitabc590c02016-02-22 09:12:33 -0800861
862 SkColor colors[kStopCount];
863 for (unsigned i = 0; i < kStopCount; i++) {
864 switch (i % 5) {
Robert Phillips3b2b0552023-03-17 10:17:34 -0400865 case 0: colors[i] = SK_ColorRED; break;
fmalitabc590c02016-02-22 09:12:33 -0800866 case 1: colors[i] = SK_ColorGREEN; break;
867 case 2: colors[i] = SK_ColorGREEN; break;
Robert Phillips3b2b0552023-03-17 10:17:34 -0400868 case 3: colors[i] = SK_ColorBLUE; break;
869 case 4: colors[i] = SK_ColorRED; break;
fmalitabc590c02016-02-22 09:12:33 -0800870 }
871 }
872
reed9283d202016-03-13 13:01:57 -0700873 SkPaint p;
Robert Phillips3b2b0552023-03-17 10:17:34 -0400874 p.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, std::size(colors),
875 SkTileMode::kClamp));
reed9283d202016-03-13 13:01:57 -0700876
fmalitabc590c02016-02-22 09:12:33 -0800877 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
878}
879
880DEF_SIMPLE_GM(gradient_many_stops, canvas, 500, 500) {
Florin Malitad1aedde2017-06-07 15:03:38 -0400881 draw_many_stops(canvas);
fmalitabc590c02016-02-22 09:12:33 -0800882}
883
Robert Phillips3b2b0552023-03-17 10:17:34 -0400884static void draw_many_hard_stops(SkCanvas* canvas) {
885 const unsigned kStopCount = 300;
886 const SkPoint pts[] = {{50, 50}, {450, 450}};
887
888 SkColor colors[kStopCount];
889 SkScalar pos[kStopCount];
890 for (unsigned i = 0; i < kStopCount; i++) {
891 switch (i % 6) {
892 case 0: colors[i] = SK_ColorRED; break;
893 case 1: colors[i] = SK_ColorGREEN; break;
894 case 2: colors[i] = SK_ColorGREEN; break;
895 case 3: colors[i] = SK_ColorBLUE; break;
896 case 4: colors[i] = SK_ColorBLUE; break;
897 case 5: colors[i] = SK_ColorRED; break;
898 }
899 pos[i] = (2.0f * (i / 2)) / kStopCount;
900 }
901
902 SkPaint p;
903 p.setShader(SkGradientShader::MakeLinear(pts, colors, pos, std::size(colors),
904 SkTileMode::kClamp));
905
906 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
907}
908
909DEF_SIMPLE_GM(gradient_many_hard_stops, canvas, 500, 500) {
910 draw_many_hard_stops(canvas);
911}
912
Florin Malitaa0ac9632017-05-03 13:07:28 -0400913static void draw_circle_shader(SkCanvas* canvas, SkScalar cx, SkScalar cy, SkScalar r,
914 sk_sp<SkShader> (*shaderFunc)()) {
915 SkPaint p;
916 p.setAntiAlias(true);
917 p.setShader(shaderFunc());
918 canvas->drawCircle(cx, cy, r, p);
919
920 p.setShader(nullptr);
921 p.setColor(SK_ColorGRAY);
922 p.setStyle(SkPaint::kStroke_Style);
923 p.setStrokeWidth(2);
924 canvas->drawCircle(cx, cy, r, p);
925}
926
927DEF_SIMPLE_GM(fancy_gradients, canvas, 800, 300) {
928 draw_circle_shader(canvas, 150, 150, 100, []() -> sk_sp<SkShader> {
929 // Checkerboard using two linear gradients + picture shader.
930 SkScalar kTileSize = 80 / sqrtf(2);
931 SkColor colors1[] = { 0xff000000, 0xff000000,
932 0xffffffff, 0xffffffff,
933 0xff000000, 0xff000000 };
934 SkColor colors2[] = { 0xff000000, 0xff000000,
935 0x00000000, 0x00000000,
936 0xff000000, 0xff000000 };
937 SkScalar pos[] = { 0, .25f, .25f, .75f, .75f, 1 };
Herb Derbyc37b3862022-06-21 09:49:17 -0400938 static_assert(std::size(colors1) == std::size(pos), "color/pos size mismatch");
939 static_assert(std::size(colors2) == std::size(pos), "color/pos size mismatch");
Florin Malitaa0ac9632017-05-03 13:07:28 -0400940
941 SkPictureRecorder recorder;
942 recorder.beginRecording(SkRect::MakeWH(kTileSize, kTileSize));
943
944 SkPaint p;
945
946 SkPoint pts1[] = { { 0, 0 }, { kTileSize, kTileSize }};
Herb Derbyc37b3862022-06-21 09:49:17 -0400947 p.setShader(SkGradientShader::MakeLinear(pts1, colors1, pos, std::size(colors1),
Mike Reedfae8fce2019-04-03 10:27:45 -0400948 SkTileMode::kClamp, 0, nullptr));
Florin Malitaa0ac9632017-05-03 13:07:28 -0400949 recorder.getRecordingCanvas()->drawPaint(p);
950
951 SkPoint pts2[] = { { 0, kTileSize }, { kTileSize, 0 }};
Herb Derbyc37b3862022-06-21 09:49:17 -0400952 p.setShader(SkGradientShader::MakeLinear(pts2, colors2, pos, std::size(colors2),
Mike Reedfae8fce2019-04-03 10:27:45 -0400953 SkTileMode::kClamp, 0, nullptr));
Florin Malitaa0ac9632017-05-03 13:07:28 -0400954 recorder.getRecordingCanvas()->drawPaint(p);
955
956 SkMatrix m = SkMatrix::I();
957 m.preRotate(45);
Mike Reedfae8fce2019-04-03 10:27:45 -0400958 return recorder.finishRecordingAsPicture()->makeShader(
Mike Reed10a5ff22021-03-18 17:18:58 -0400959 SkTileMode::kRepeat, SkTileMode::kRepeat,
960 SkFilterMode::kNearest, &m, nullptr);
Florin Malitaa0ac9632017-05-03 13:07:28 -0400961 });
962
963 draw_circle_shader(canvas, 400, 150, 100, []() -> sk_sp<SkShader> {
964 // Checkerboard using a sweep gradient + picture shader.
965 SkScalar kTileSize = 80;
966 SkColor colors[] = { 0xff000000, 0xff000000,
967 0xffffffff, 0xffffffff,
968 0xff000000, 0xff000000,
969 0xffffffff, 0xffffffff };
970 SkScalar pos[] = { 0, .25f, .25f, .5f, .5f, .75f, .75f, 1 };
Herb Derbyc37b3862022-06-21 09:49:17 -0400971 static_assert(std::size(colors) == std::size(pos), "color/pos size mismatch");
Florin Malitaa0ac9632017-05-03 13:07:28 -0400972
973 SkPaint p;
974 p.setShader(SkGradientShader::MakeSweep(kTileSize / 2, kTileSize / 2,
Herb Derbyc37b3862022-06-21 09:49:17 -0400975 colors, pos, std::size(colors), 0, nullptr));
Florin Malitaa0ac9632017-05-03 13:07:28 -0400976 SkPictureRecorder recorder;
977 recorder.beginRecording(SkRect::MakeWH(kTileSize, kTileSize))->drawPaint(p);
978
Mike Reedfae8fce2019-04-03 10:27:45 -0400979 return recorder.finishRecordingAsPicture()->makeShader(
980 SkTileMode::kRepeat,
Mike Reed10a5ff22021-03-18 17:18:58 -0400981 SkTileMode::kRepeat, SkFilterMode::kNearest);
Florin Malitaa0ac9632017-05-03 13:07:28 -0400982 });
983
984 draw_circle_shader(canvas, 650, 150, 100, []() -> sk_sp<SkShader> {
985 // Dartboard using sweep + radial.
986 const SkColor a = 0xffffffff;
987 const SkColor b = 0xff000000;
988 SkColor colors[] = { a, a, b, b, a, a, b, b, a, a, b, b, a, a, b, b};
989 SkScalar pos[] = { 0, .125f, .125f, .25f, .25f, .375f, .375f, .5f, .5f,
990 .625f, .625f, .75f, .75f, .875f, .875f, 1};
Herb Derbyc37b3862022-06-21 09:49:17 -0400991 static_assert(std::size(colors) == std::size(pos), "color/pos size mismatch");
Florin Malitaa0ac9632017-05-03 13:07:28 -0400992
993 SkPoint center = { 650, 150 };
994 sk_sp<SkShader> sweep1 = SkGradientShader::MakeSweep(center.x(), center.y(), colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400995 std::size(colors), 0, nullptr);
Florin Malitaa0ac9632017-05-03 13:07:28 -0400996 SkMatrix m = SkMatrix::I();
997 m.preRotate(22.5f, center.x(), center.y());
998 sk_sp<SkShader> sweep2 = SkGradientShader::MakeSweep(center.x(), center.y(), colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -0400999 std::size(colors), 0, &m);
Florin Malitaa0ac9632017-05-03 13:07:28 -04001000
Mike Reedc8bea7d2019-04-09 13:55:36 -04001001 sk_sp<SkShader> sweep(SkShaders::Blend(SkBlendMode::kExclusion, sweep1, sweep2));
Florin Malitaa0ac9632017-05-03 13:07:28 -04001002
1003 SkScalar radialPos[] = { 0, .02f, .02f, .04f, .04f, .08f, .08f, .16f, .16f, .31f, .31f,
1004 .62f, .62f, 1, 1, 1 };
Herb Derbyc37b3862022-06-21 09:49:17 -04001005 static_assert(std::size(colors) == std::size(radialPos),
Florin Malitaa0ac9632017-05-03 13:07:28 -04001006 "color/pos size mismatch");
1007
Mike Reedc8bea7d2019-04-09 13:55:36 -04001008 return SkShaders::Blend(SkBlendMode::kExclusion, sweep,
1009 SkGradientShader::MakeRadial(center, 100, colors,
1010 radialPos,
Herb Derbyc37b3862022-06-21 09:49:17 -04001011 std::size(radialPos),
Mike Reedc8bea7d2019-04-09 13:55:36 -04001012 SkTileMode::kClamp));
Florin Malitaa0ac9632017-05-03 13:07:28 -04001013 });
1014}
Florin Malita5a9a9812017-08-01 16:38:08 -04001015
Florin Malita4d274292017-08-02 10:04:48 -04001016DEF_SIMPLE_GM(sweep_tiling, canvas, 690, 512) {
Florin Malita5a9a9812017-08-01 16:38:08 -04001017 static constexpr SkScalar size = 160;
1018 static constexpr SkColor colors[] = { SK_ColorBLUE, SK_ColorYELLOW, SK_ColorGREEN };
1019 static constexpr SkScalar pos[] = { 0, .25f, .50f };
Herb Derbyc37b3862022-06-21 09:49:17 -04001020 static_assert(std::size(colors) == std::size(pos), "size mismatch");
Florin Malita5a9a9812017-08-01 16:38:08 -04001021
Mike Reedfae8fce2019-04-03 10:27:45 -04001022 static constexpr SkTileMode modes[] = { SkTileMode::kClamp,
1023 SkTileMode::kRepeat,
1024 SkTileMode::kMirror };
Florin Malita5a9a9812017-08-01 16:38:08 -04001025
1026 static const struct {
1027 SkScalar start, end;
1028 } angles[] = {
1029 { -330, -270 },
1030 { 30, 90 },
1031 { 390, 450 },
Florin Malita50f7a1e2017-08-02 09:40:26 -04001032 { -30, 800 },
Florin Malita5a9a9812017-08-01 16:38:08 -04001033 };
1034
1035 SkPaint p;
1036 const SkRect r = SkRect::MakeWH(size, size);
1037
1038 for (auto mode : modes) {
1039 {
1040 SkAutoCanvasRestore acr(canvas, true);
1041
1042 for (auto angle : angles) {
1043 p.setShader(SkGradientShader::MakeSweep(size / 2, size / 2, colors, pos,
Herb Derbyc37b3862022-06-21 09:49:17 -04001044 std::size(colors), mode,
Florin Malita5a9a9812017-08-01 16:38:08 -04001045 angle.start, angle.end, 0, nullptr));
1046
1047 canvas->drawRect(r, p);
1048 canvas->translate(size * 1.1f, 0);
1049 }
1050 }
1051 canvas->translate(0, size * 1.1f);
1052 }
1053}
Florin Malita36f054a2017-08-03 12:55:41 -04001054
John Stiles2c0683e2022-09-21 16:16:30 -04001055DEF_SIMPLE_GM(rgbw_sweep_gradient, canvas, 100, 100) {
1056 static constexpr SkScalar size = 100;
1057 static constexpr SkColor colors[] = {SK_ColorWHITE, SK_ColorWHITE,
1058 SK_ColorBLUE, SK_ColorBLUE,
1059 SK_ColorRED, SK_ColorRED,
1060 SK_ColorGREEN, SK_ColorGREEN};
1061 static constexpr SkScalar pos[] = { 0, .25f, .25f, .50f, .50f, .75, .75, 1 };
1062 static_assert(std::size(colors) == std::size(pos), "size mismatch");
1063
1064 SkPaint p;
1065 p.setShader(SkGradientShader::MakeSweep(size / 2, size / 2, colors, pos, std::size(colors)));
1066 canvas->drawRect(SkRect::MakeWH(size, size), p);
1067}
1068
Florin Malita36f054a2017-08-03 12:55:41 -04001069// Exercises the special-case Ganesh gradient effects.
Brian Osmana8e57442017-09-11 17:21:35 -04001070DEF_SIMPLE_GM(gradients_interesting, canvas, 640, 1300) {
Florin Malita36f054a2017-08-03 12:55:41 -04001071 static const SkColor colors2[] = { SK_ColorRED, SK_ColorBLUE };
1072 static const SkColor colors3[] = { SK_ColorRED, SK_ColorYELLOW, SK_ColorBLUE };
1073 static const SkColor colors4[] = { SK_ColorRED, SK_ColorYELLOW, SK_ColorYELLOW, SK_ColorBLUE };
1074
Brian Osmana8e57442017-09-11 17:21:35 -04001075 static const SkScalar softRight[] = { 0, .999f, 1 }; // Based on Android launcher "clipping"
1076 static const SkScalar hardLeft[] = { 0, 0, 1 };
1077 static const SkScalar hardRight[] = { 0, 1, 1 };
1078 static const SkScalar hardCenter[] = { 0, .5f, .5f, 1 };
Florin Malita36f054a2017-08-03 12:55:41 -04001079
1080 static const struct {
1081 const SkColor* colors;
1082 const SkScalar* pos;
1083 int count;
1084 } configs[] = {
1085 { colors2, nullptr, 2 }, // kTwo_ColorType
Brian Osman2ab4b2b2017-09-12 11:20:56 -04001086 { colors3, nullptr, 3 }, // kThree_ColorType (simple)
1087 { colors3, softRight, 3 }, // kThree_ColorType (tricky)
Florin Malita36f054a2017-08-03 12:55:41 -04001088 { colors3, hardLeft, 3 }, // kHardStopLeftEdged_ColorType
1089 { colors3, hardRight, 3 }, // kHardStopRightEdged_ColorType
1090 { colors4, hardCenter, 4 }, // kSingleHardStop_ColorType
1091 };
1092
Mike Reedfae8fce2019-04-03 10:27:45 -04001093 static const SkTileMode modes[] = {
1094 SkTileMode::kClamp,
1095 SkTileMode::kRepeat,
1096 SkTileMode::kMirror,
Florin Malita36f054a2017-08-03 12:55:41 -04001097 };
1098
1099 static constexpr SkScalar size = 200;
1100 static const SkPoint pts[] = { { size / 3, size / 3 }, { size * 2 / 3, size * 2 / 3} };
1101
1102 SkPaint p;
1103 for (const auto& cfg : configs) {
1104 {
1105 SkAutoCanvasRestore acr(canvas, true);
1106 for (auto mode : modes) {
1107 p.setShader(SkGradientShader::MakeLinear(pts, cfg.colors, cfg.pos, cfg.count,
1108 mode));
1109 canvas->drawRect(SkRect::MakeWH(size, size), p);
1110 canvas->translate(size * 1.1f, 0);
1111 }
1112 }
1113 canvas->translate(0, size * 1.1f);
1114 }
1115}
Brian Osmanf2d20dd2022-10-04 16:51:57 -04001116
Brian Osmana1d325e2022-11-16 14:15:22 -05001117// TODO(skia:13774): Still need to test degenerate gradients in strange color spaces
Christopher Cameron611c4572023-12-28 16:27:17 +01001118DEF_SIMPLE_GM_BG(gradients_color_space, canvas, 265, 255, SK_ColorGRAY) {
Brian Osmanf2d20dd2022-10-04 16:51:57 -04001119 using CS = SkGradientShader::Interpolation::ColorSpace;
1120
1121 struct Config {
1122 CS fColorSpace;
1123 const char* fLabel;
1124 };
1125 static const Config kConfigs[] = {
Christopher Cameron611c4572023-12-28 16:27:17 +01001126 { CS::kSRGB, "sRGB" },
1127 { CS::kSRGBLinear, "Linear" },
1128 { CS::kLab, "Lab" },
1129 { CS::kOKLab, "OKLab" },
1130 { CS::kOKLabGamutMap, "OKLabGamutMap" },
1131 { CS::kLCH, "LCH" },
1132 { CS::kOKLCH, "OKLCH" },
1133 { CS::kOKLCHGamutMap, "OKLCHGamutMap" },
1134 { CS::kHSL, "HSL" },
1135 { CS::kHWB, "HWB" },
Brian Osmanf2d20dd2022-10-04 16:51:57 -04001136 };
1137
1138 SkPoint pts[] = {{0, 0}, {200, 0}};
1139 SkColor4f colors[] = {SkColors::kBlue, SkColors::kYellow};
1140
1141 SkPaint labelPaint;
1142 SkPaint p;
1143 SkGradientShader::Interpolation interpolation;
1144 canvas->translate(5, 5);
Kevin Lubickbca43ec2023-10-30 10:11:22 -04001145 SkFont font = ToolUtils::DefaultPortableFont();
Brian Osmanf2d20dd2022-10-04 16:51:57 -04001146
1147 for (const Config& config : kConfigs) {
1148 interpolation.fColorSpace = config.fColorSpace;
1149 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), nullptr, 2,
1150 SkTileMode::kClamp, interpolation, nullptr));
1151 canvas->drawRect({0, 0, 200, 20}, p);
1152 canvas->drawSimpleText(config.fLabel, strlen(config.fLabel), SkTextEncoding::kUTF8, 210, 15,
Kevin Lubickbca43ec2023-10-30 10:11:22 -04001153 font, labelPaint);
Brian Osmanf2d20dd2022-10-04 16:51:57 -04001154 canvas->translate(0, 25);
1155 }
1156}
Brian Osmana9255bc2022-10-19 10:25:22 -04001157
Brian Osmanfbeb5d12022-11-21 15:37:32 -05001158DEF_SIMPLE_GM_BG(gradients_hue_method, canvas, 285, 155, SK_ColorGRAY) {
Brian Osmana9255bc2022-10-19 10:25:22 -04001159 using HM = SkGradientShader::Interpolation::HueMethod;
1160
1161 struct Config {
1162 HM fHueMethod;
1163 const char* fLabel;
1164 };
1165 static const Config kConfigs[] = {
1166 { HM::kShorter, "Shorter" },
1167 { HM::kLonger, "Longer" },
1168 { HM::kIncreasing, "Increasing" },
1169 { HM::kDecreasing, "Decreasing" },
1170 };
1171
1172 SkPoint pts[] = {{0, 0}, {200, 0}};
1173 SkColor4f colors[] = {SkColors::kRed, SkColors::kGreen, SkColors::kRed, SkColors::kRed };
1174
1175 SkPaint labelPaint;
1176 SkPaint p;
1177 SkGradientShader::Interpolation interpolation;
1178 interpolation.fColorSpace = SkGradientShader::Interpolation::ColorSpace::kHSL;
1179 canvas->translate(5, 5);
Kevin Lubickbca43ec2023-10-30 10:11:22 -04001180 SkFont font = ToolUtils::DefaultPortableFont();
Brian Osmana9255bc2022-10-19 10:25:22 -04001181
1182 for (const Config& config : kConfigs) {
1183 interpolation.fHueMethod = config.fHueMethod;
1184 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), nullptr, 4,
1185 SkTileMode::kClamp, interpolation, nullptr));
1186 canvas->drawRect({0, 0, 200, 20}, p);
1187 canvas->drawSimpleText(config.fLabel, strlen(config.fLabel), SkTextEncoding::kUTF8, 210, 15,
Kevin Lubickbca43ec2023-10-30 10:11:22 -04001188 font, labelPaint);
Brian Osmana9255bc2022-10-19 10:25:22 -04001189 canvas->translate(0, 25);
1190 }
Brian Osmanfbeb5d12022-11-21 15:37:32 -05001191
1192 // Test a bug (skia:13941) with how gradient shaders handle explicit positions.
1193 // If there are no explicit positions at 0 or 1, those are automatically added, with copies of
1194 // the first/last color. When using kLonger, this can produce extra gradient that should
1195 // actually be solid. This gradient *should* be:
1196 // |- solid red -|- red to green, the long way -|- solid green -|
1197 interpolation.fHueMethod = HM::kLonger;
1198 SkScalar middlePos[] = { 0.3f, 0.7f };
1199 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), middlePos, 2,
1200 SkTileMode::kClamp, interpolation, nullptr));
1201 canvas->drawRect({0, 0, 200, 20}, p);
1202 canvas->translate(0, 25);
1203
1204 // However... if the user explicitly includes those duplicate color stops in kLonger mode,
1205 // we expect the gradient to do a full rotation in those regions:
1206 // |- full circle, red to red -|- red to green -|- full circle, green to green -|
1207 colors[0] = colors[1] = SkColors::kRed;
1208 colors[2] = colors[3] = SkColors::kGreen;
1209 SkScalar allPos[] = { 0.0f, 0.3f, 0.7f, 1.0f };
1210 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), allPos, 4,
1211 SkTileMode::kClamp, interpolation, nullptr));
1212 canvas->drawRect({0, 0, 200, 20}, p);
1213 canvas->translate(0, 25);
Brian Osmana9255bc2022-10-19 10:25:22 -04001214}
Brian Osmana1d325e2022-11-16 14:15:22 -05001215
1216DEF_SIMPLE_GM_BG(gradients_color_space_tilemode, canvas, 360, 105, SK_ColorGRAY) {
1217 // Test exotic (CSS) gradient color spaces in conjunction with tile modes. Rather than test
1218 // every combination, we pick one color space that has a sufficiently strange interpolated
1219 // representation (OKLCH) and just use that. We're mostly interested in making sure that things
1220 // like decal mode are implemented at the correct time in the pipeline, relative to hue
1221 // conversion, re-premultiplication, etc.
1222 SkPoint pts[] = {{20, 0}, {120, 0}};
1223 SkColor4f colors[] = {SkColors::kBlue, SkColors::kYellow};
1224
1225 SkPaint p;
1226 SkGradientShader::Interpolation interpolation;
1227 interpolation.fColorSpace = SkGradientShader::Interpolation::ColorSpace::kOKLCH;
1228
1229 canvas->translate(5, 5);
1230
1231 for (int tm = 0; tm < kSkTileModeCount; ++tm) {
1232 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), nullptr, 2,
1233 static_cast<SkTileMode>(tm), interpolation,
1234 nullptr));
1235 canvas->drawRect({0, 0, 350, 20}, p);
1236 canvas->translate(0, 25);
1237 }
1238}
1239
1240DEF_SIMPLE_GM_BG(gradients_color_space_many_stops, canvas, 500, 500, SK_ColorGRAY) {
1241 // Test exotic (CSS) gradient color spaces with many stops. Rather than test every combination,
1242 // we pick one color space that has a sufficiently strange interpolated representation (OKLCH)
1243 // and just use that. We're mostly interested in making sure that the texture fallback on GPU
1244 // works correctly.
1245 const SkPoint pts[] = { {50, 50}, {450, 465}};
1246
1247 const unsigned kStopCount = 200;
1248 SkColor4f colors[kStopCount];
1249 for (unsigned i = 0; i < kStopCount; i++) {
1250 switch (i % 5) {
1251 case 0: colors[i] = SkColors::kRed; break;
1252 case 1: colors[i] = SkColors::kGreen; break;
1253 case 2: colors[i] = SkColors::kGreen; break;
1254 case 3: colors[i] = SkColors::kBlue; break;
1255 case 4: colors[i] = SkColors::kRed; break;
1256 }
1257 }
1258
1259 SkPaint p;
1260
1261 SkGradientShader::Interpolation interpolation;
1262 interpolation.fColorSpace = SkGradientShader::Interpolation::ColorSpace::kOKLCH;
1263 p.setShader(SkGradientShader::MakeLinear(pts, colors, SkColorSpace::MakeSRGB(), nullptr,
1264 std::size(colors), SkTileMode::kClamp, interpolation,
1265 nullptr));
1266
1267 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
1268}
Brian Osmanf7866332023-10-11 13:56:58 -04001269
Brian Osman77abd512023-10-27 14:00:13 -04001270static void draw_powerless_hue_gradients(SkCanvas* canvas,
1271 SkGradientShader::Interpolation::ColorSpace colorSpace) {
Brian Osmanf7866332023-10-11 13:56:58 -04001272 ToolUtils::draw_checkerboard(canvas);
1273
1274 auto nextRow = [=]() {
1275 canvas->restore();
1276 canvas->translate(0, 25);
1277 canvas->save();
1278 };
1279
1280 auto gradient = [&](std::initializer_list<SkColor4f> colors,
1281 std::initializer_list<float> pos,
1282 bool inPremul = false) {
1283 using Interpolation = SkGradientShader::Interpolation;
1284 SkASSERT(pos.size() == 0 || pos.size() == colors.size());
1285 SkPaint paint;
1286 SkPoint pts[] = {{0, 0}, {200, 0}};
1287 Interpolation interpolation;
Brian Osman77abd512023-10-27 14:00:13 -04001288 interpolation.fColorSpace = colorSpace;
Brian Osmanf7866332023-10-11 13:56:58 -04001289 interpolation.fInPremul = static_cast<Interpolation::InPremul>(inPremul);
1290 paint.setShader(SkGradientShader::MakeLinear(pts,
1291 colors.begin(),
1292 SkColorSpace::MakeSRGB(),
1293 pos.size() == 0 ? nullptr : pos.begin(),
1294 colors.size(),
1295 SkTileMode::kClamp,
1296 interpolation,
1297 nullptr));
1298 canvas->drawRect({0, 0, 200, 20}, paint);
1299 canvas->translate(205, 0); // next column
1300 };
1301
1302 canvas->translate(5, 5);
1303 canvas->save();
1304
1305 // For each test case, the first gradient (first column) has an under-specified result due to a
1306 // powerless component after conversion to LCH. The second gradient (second column) "hints" the
1307 // correct result, by slightly tinting the otherwise powerless color.
1308
1309 gradient({SkColors::kWhite, SkColors::kBlue}, {});
1310 gradient({{0.99f, 0.99f, 1.00f, 1.0f}, SkColors::kBlue}, {}); // white, with blue hue
1311 nextRow();
1312
1313 gradient({SkColors::kBlack, SkColors::kBlue}, {});
1314 gradient({{0.00f, 0.00f, 0.01f, 1.0f}, SkColors::kBlue}, {}); // black, with blue hue
1315 nextRow();
1316
1317 // Transparent cases are done in both premul and unpremul interpolation:
1318
1319 gradient({SkColors::kTransparent, SkColors::kBlue}, {}, /*inPremul=*/false);
1320 gradient({{0.00f, 0.00f, 0.01f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/false);
1321 nextRow();
1322
1323 gradient({SkColors::kTransparent, SkColors::kBlue}, {}, /*inPremul=*/true);
1324 gradient({{0.00f, 0.00f, 0.01f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/true);
1325 nextRow();
1326
1327 gradient({{1.00f, 1.00f, 1.00f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/false);
1328 gradient({{0.99f, 0.99f, 1.00f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/false);
1329 nextRow();
1330
1331 gradient({{1.00f, 1.00f, 1.00f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/true);
1332 gradient({{0.99f, 0.99f, 1.00f, 0.0f}, SkColors::kBlue}, {}, /*inPremul=*/true);
1333 nextRow();
1334
1335 // Now we test three-stop gradients, where the middle stop needs to be "split" to handle the
1336 // different hues on either side. Again, the second column explicitly injects those to produce
1337 // a reference result. See: https://github.com/w3c/csswg-drafts/issues/9295
1338
1339 gradient({SkColors::kRed, SkColors::kWhite, SkColors::kBlue}, {});
1340 gradient({SkColors::kRed,
1341 {1.00f, 0.99f, 0.99f, 1.0f},
1342 {0.99f, 0.99f, 1.00f, 1.0f},
1343 SkColors::kBlue},
1344 {0.0f, 0.5f, 0.5f, 1.0f});
1345 nextRow();
1346
1347 gradient({SkColors::kRed, SkColors::kBlack, SkColors::kBlue}, {});
1348 gradient({SkColors::kRed,
1349 {0.01f, 0.00f, 0.00f, 1.0f},
1350 {0.00f, 0.00f, 0.01f, 1.0f},
1351 SkColors::kBlue},
1352 {0.0f, 0.5f, 0.5f, 1.0f});
1353 nextRow();
1354
1355 gradient({SkColors::kRed, SkColors::kTransparent, SkColors::kBlue}, {});
1356 gradient({SkColors::kRed,
1357 {0.01f, 0.00f, 0.00f, 0.0f},
1358 {0.00f, 0.00f, 0.01f, 0.0f},
1359 SkColors::kBlue},
1360 {0.0f, 0.5f, 0.5f, 1.0f});
1361 nextRow();
Brian Osman77abd512023-10-27 14:00:13 -04001362
1363 // Now do a few black-white tests, to ensure that the hue propagation works correctly, even
1364 // when there isn't any hue in the adjacent stops.
1365 using HueMethod = SkGradientShader::Interpolation::HueMethod;
1366 auto blackWhiteGradient = [&](HueMethod hm) {
1367 using Interpolation = SkGradientShader::Interpolation;
1368 SkPaint paint;
1369 SkPoint pts[] = {{0, 0}, {405, 0}};
1370 Interpolation interpolation;
1371 interpolation.fColorSpace = colorSpace;
1372 interpolation.fHueMethod = hm;
1373 const SkColor4f colors[] = {SkColors::kWhite, SkColors::kGray,
1374 SkColors::kWhite, SkColors::kDkGray,
1375 SkColors::kWhite, SkColors::kBlack};
1376 paint.setShader(SkGradientShader::MakeLinear(pts,
1377 colors,
1378 SkColorSpace::MakeSRGB(),
1379 nullptr,
1380 std::size(colors),
1381 SkTileMode::kClamp,
1382 interpolation,
1383 nullptr));
1384 canvas->drawRect({0, 0, 405, 20}, paint);
1385 nextRow();
1386 };
1387
1388 blackWhiteGradient(HueMethod::kShorter);
1389 blackWhiteGradient(HueMethod::kIncreasing);
1390 blackWhiteGradient(HueMethod::kDecreasing);
1391 blackWhiteGradient(HueMethod::kLonger);
Brian Osmanf7866332023-10-11 13:56:58 -04001392}
Brian Osman77abd512023-10-27 14:00:13 -04001393
1394#define DEF_POWERLESS_HUE_GM(colorSpace) \
1395 DEF_SIMPLE_GM(gradients_powerless_hue_##colorSpace, canvas, 415, 330) { \
1396 draw_powerless_hue_gradients(canvas, \
1397 SkGradientShader::Interpolation::ColorSpace::k##colorSpace); \
1398 }
1399
1400DEF_POWERLESS_HUE_GM(LCH)
1401DEF_POWERLESS_HUE_GM(OKLCH)
1402DEF_POWERLESS_HUE_GM(HSL)
1403DEF_POWERLESS_HUE_GM(HWB)