blob: 7b1a1f737e94b5cd4d5778ac74f14e4b2597765a [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"
13#include "include/core/SkMatrix.h"
14#include "include/core/SkPaint.h"
15#include "include/core/SkPicture.h"
16#include "include/core/SkPictureRecorder.h"
17#include "include/core/SkPoint.h"
18#include "include/core/SkRect.h"
19#include "include/core/SkRefCnt.h"
20#include "include/core/SkScalar.h"
21#include "include/core/SkShader.h"
22#include "include/core/SkSize.h"
23#include "include/core/SkString.h"
24#include "include/core/SkTileMode.h"
25#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050026#include "include/effects/SkGradientShader.h"
reed@android.com42309d42009-06-22 02:06:35 +000027
Ben Wagner7fde8e12019-05-01 17:28:53 -040028#include <math.h>
29
reed@android.com42309d42009-06-22 02:06:35 +000030namespace skiagm {
31
32struct GradData {
brianosmane25d71c2016-09-28 11:27:28 -070033 int fCount;
34 const SkColor* fColors;
35 const SkColor4f* fColors4f;
36 const SkScalar* fPos;
reed@android.com42309d42009-06-22 02:06:35 +000037};
38
mtkleindbfd7ab2016-09-01 11:24:54 -070039constexpr SkColor gColors[] = {
reed@android.com42309d42009-06-22 02:06:35 +000040 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK
41};
brianosmane25d71c2016-09-28 11:27:28 -070042constexpr SkColor4f gColors4f[] ={
43 { 1.0f, 0.0f, 0.0f, 1.0f }, // Red
44 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
45 { 0.0f, 0.0f, 1.0f, 1.0f }, // Blue
46 { 1.0f, 1.0f, 1.0f, 1.0f }, // White
47 { 0.0f, 0.0f, 0.0f, 1.0f } // Black
48};
mtkleindbfd7ab2016-09-01 11:24:54 -070049constexpr SkScalar gPos0[] = { 0, SK_Scalar1 };
50constexpr SkScalar gPos1[] = { SK_Scalar1/4, SK_Scalar1*3/4 };
51constexpr SkScalar gPos2[] = {
reed@android.com42309d42009-06-22 02:06:35 +000052 0, SK_Scalar1/8, SK_Scalar1/2, SK_Scalar1*7/8, SK_Scalar1
53};
54
mtkleindbfd7ab2016-09-01 11:24:54 -070055constexpr SkScalar gPosClamp[] = {0.0f, 0.0f, 1.0f, 1.0f};
56constexpr SkColor gColorClamp[] = {
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +000057 SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorBLUE
58};
brianosmane25d71c2016-09-28 11:27:28 -070059constexpr SkColor4f gColor4fClamp[] ={
60 { 1.0f, 0.0f, 0.0f, 1.0f }, // Red
61 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
62 { 0.0f, 1.0f, 0.0f, 1.0f }, // Green
63 { 0.0f, 0.0f, 1.0f, 1.0f } // Blue
64};
mtkleindbfd7ab2016-09-01 11:24:54 -070065constexpr GradData gGradData[] = {
brianosmane25d71c2016-09-28 11:27:28 -070066 { 2, gColors, gColors4f, nullptr },
67 { 2, gColors, gColors4f, gPos0 },
68 { 2, gColors, gColors4f, gPos1 },
69 { 5, gColors, gColors4f, nullptr },
70 { 5, gColors, gColors4f, gPos2 },
71 { 4, gColorClamp, gColor4fClamp, gPosClamp }
reed@android.com42309d42009-06-22 02:06:35 +000072};
73
reed1a9b9642016-03-13 14:13:58 -070074static sk_sp<SkShader> MakeLinear(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040075 SkTileMode tm, const SkMatrix& localMatrix) {
reed1a9b9642016-03-13 14:13:58 -070076 return SkGradientShader::MakeLinear(pts, data.fColors, data.fPos, data.fCount, tm, 0,
77 &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000078}
reed@google.comf3c1cc92010-12-23 16:45:33 +000079
brianosmane25d71c2016-09-28 11:27:28 -070080static sk_sp<SkShader> MakeLinear4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040081 SkTileMode tm, const SkMatrix& localMatrix) {
Mike Reed2e56f3c2019-03-06 11:21:15 -050082 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -070083 return SkGradientShader::MakeLinear(pts, data.fColors4f, srgb, data.fPos, data.fCount, tm, 0,
84 &localMatrix);
85}
86
reed1a9b9642016-03-13 14:13:58 -070087static sk_sp<SkShader> MakeRadial(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040088 SkTileMode tm, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +000089 SkPoint center;
90 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
91 SkScalarAve(pts[0].fY, pts[1].fY));
reed1a9b9642016-03-13 14:13:58 -070092 return SkGradientShader::MakeRadial(center, center.fX, data.fColors, data.fPos, data.fCount,
93 tm, 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000094}
95
brianosmane25d71c2016-09-28 11:27:28 -070096static sk_sp<SkShader> MakeRadial4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -040097 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -070098 SkPoint center;
99 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
100 SkScalarAve(pts[0].fY, pts[1].fY));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500101 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700102 return SkGradientShader::MakeRadial(center, center.fX, data.fColors4f, srgb, data.fPos,
103 data.fCount, tm, 0, &localMatrix);
104}
105
reed1a9b9642016-03-13 14:13:58 -0700106static sk_sp<SkShader> MakeSweep(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400107 SkTileMode, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +0000108 SkPoint center;
109 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
110 SkScalarAve(pts[0].fY, pts[1].fY));
reed1a9b9642016-03-13 14:13:58 -0700111 return SkGradientShader::MakeSweep(center.fX, center.fY, data.fColors, data.fPos, data.fCount,
112 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +0000113}
114
brianosmane25d71c2016-09-28 11:27:28 -0700115static sk_sp<SkShader> MakeSweep4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400116 SkTileMode, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700117 SkPoint center;
118 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
119 SkScalarAve(pts[0].fY, pts[1].fY));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500120 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700121 return SkGradientShader::MakeSweep(center.fX, center.fY, data.fColors4f, srgb, data.fPos,
122 data.fCount, 0, &localMatrix);
123}
124
reed1a9b9642016-03-13 14:13:58 -0700125static sk_sp<SkShader> Make2Radial(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400126 SkTileMode tm, const SkMatrix& localMatrix) {
reed@google.comf3c1cc92010-12-23 16:45:33 +0000127 SkPoint center0, center1;
128 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
129 SkScalarAve(pts[0].fY, pts[1].fY));
130 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
131 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
reed1a9b9642016-03-13 14:13:58 -0700132 return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
133 center0, (pts[1].fX - pts[0].fX) / 2,
134 data.fColors, data.fPos, data.fCount, tm,
135 0, &localMatrix);
reed@google.comf3c1cc92010-12-23 16:45:33 +0000136}
137
brianosmane25d71c2016-09-28 11:27:28 -0700138static sk_sp<SkShader> Make2Radial4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400139 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700140 SkPoint center0, center1;
141 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
142 SkScalarAve(pts[0].fY, pts[1].fY));
143 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3) / 5),
144 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1) / 4));
Mike Reed2e56f3c2019-03-06 11:21:15 -0500145 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700146 return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
147 center0, (pts[1].fX - pts[0].fX) / 2,
148 data.fColors4f, srgb, data.fPos, data.fCount, tm,
149 0, &localMatrix);
150}
151
reed1a9b9642016-03-13 14:13:58 -0700152static sk_sp<SkShader> Make2Conical(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400153 SkTileMode tm, const SkMatrix& localMatrix) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000154 SkPoint center0, center1;
reed80ea19c2015-05-12 10:37:34 -0700155 SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10;
156 SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000157 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
158 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
reed1a9b9642016-03-13 14:13:58 -0700159 return SkGradientShader::MakeTwoPointConical(center1, radius1, center0, radius0,
160 data.fColors, data.fPos,
161 data.fCount, tm, 0, &localMatrix);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000162}
163
brianosmane25d71c2016-09-28 11:27:28 -0700164static sk_sp<SkShader> Make2Conical4f(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400165 SkTileMode tm, const SkMatrix& localMatrix) {
brianosmane25d71c2016-09-28 11:27:28 -0700166 SkPoint center0, center1;
167 SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10;
168 SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3;
169 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
170 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
Mike Reed2e56f3c2019-03-06 11:21:15 -0500171 auto srgb = SkColorSpace::MakeSRGB();
brianosmane25d71c2016-09-28 11:27:28 -0700172 return SkGradientShader::MakeTwoPointConical(center1, radius1, center0, radius0,
173 data.fColors4f, srgb, data.fPos,
174 data.fCount, tm, 0, &localMatrix);
175}
176
reed1a9b9642016-03-13 14:13:58 -0700177typedef sk_sp<SkShader> (*GradMaker)(const SkPoint pts[2], const GradData& data,
Mike Reedfae8fce2019-04-03 10:27:45 -0400178 SkTileMode tm, const SkMatrix& localMatrix);
mtkleindbfd7ab2016-09-01 11:24:54 -0700179constexpr GradMaker gGradMakers[] = {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000180 MakeLinear, MakeRadial, MakeSweep, Make2Radial, Make2Conical
reed@android.com42309d42009-06-22 02:06:35 +0000181};
brianosmane25d71c2016-09-28 11:27:28 -0700182constexpr GradMaker gGradMakers4f[] ={
183 MakeLinear4f, MakeRadial4f, MakeSweep4f, Make2Radial4f, Make2Conical4f
184};
reed@android.com42309d42009-06-22 02:06:35 +0000185
186///////////////////////////////////////////////////////////////////////////////
187
188class GradientsGM : public GM {
189public:
fmalita063675b2015-10-12 10:41:48 -0700190 GradientsGM(bool dither) : fDither(dither) {
Mike Kleind46dce32018-08-16 10:17:03 -0400191 this->setBGColor(0xFFDDDDDD);
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000192 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000193
reed@android.com42309d42009-06-22 02:06:35 +0000194protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000195
reed@android.com42309d42009-06-22 02:06:35 +0000196 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700197 return SkString(fDither ? "gradients" : "gradients_nodither");
reed@android.com42309d42009-06-22 02:06:35 +0000198 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000199
edisonn@google.com1da3a802013-09-19 17:55:49 +0000200 virtual SkISize onISize() { return SkISize::Make(840, 815); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000201
reed@android.com42309d42009-06-22 02:06:35 +0000202 virtual void onDraw(SkCanvas* canvas) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000203
reed@android.com42309d42009-06-22 02:06:35 +0000204 SkPoint pts[2] = {
205 { 0, 0 },
Matt Saretteebe87f2017-01-26 15:14:56 -0500206 { SkIntToScalar(100), SkIntToScalar(100) }
reed@android.com42309d42009-06-22 02:06:35 +0000207 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400208 SkTileMode tm = SkTileMode::kClamp;
Matt Saretteebe87f2017-01-26 15:14:56 -0500209 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
reed@android.com42309d42009-06-22 02:06:35 +0000210 SkPaint paint;
211 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700212 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000213
reed@android.com42309d42009-06-22 02:06:35 +0000214 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
215 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
216 canvas->save();
217 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers); j++) {
commit-bot@chromium.org9c9005a2014-04-28 14:55:39 +0000218 SkMatrix scale = SkMatrix::I();
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000219
220 if (i == 5) { // if the clamp case
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000221 scale.setScale(0.5f, 0.5f);
222 scale.postTranslate(25.f, 25.f);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000223 }
skia.committer@gmail.comd55e3572013-08-08 07:01:20 +0000224
reed1a9b9642016-03-13 14:13:58 -0700225 paint.setShader(gGradMakers[j](pts, gGradData[i], tm, scale));
reed@android.com42309d42009-06-22 02:06:35 +0000226 canvas->drawRect(r, paint);
reed@android.com42309d42009-06-22 02:06:35 +0000227 canvas->translate(0, SkIntToScalar(120));
228 }
229 canvas->restore();
230 canvas->translate(SkIntToScalar(120), 0);
231 }
232 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000233
fmalita063675b2015-10-12 10:41:48 -0700234protected:
235 bool fDither;
236
reed@android.com42309d42009-06-22 02:06:35 +0000237private:
238 typedef GM INHERITED;
239};
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.
244class Gradients4fGM : public GM {
245public:
246 Gradients4fGM(bool dither) : fDither(dither) {
Mike Kleind46dce32018-08-16 10:17:03 -0400247 this->setBGColor(0xFFDDDDDD);
brianosmane25d71c2016-09-28 11:27:28 -0700248 }
249
250protected:
251
252 SkString onShortName() {
253 return SkString(fDither ? "gradients4f" : "gradients4f_nodither");
254 }
255
256 virtual SkISize onISize() { return SkISize::Make(840, 815); }
257
258 virtual void onDraw(SkCanvas* canvas) {
259
260 SkPoint pts[2] ={
261 { 0, 0 },
262 { SkIntToScalar(100), SkIntToScalar(100) }
263 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400264 SkTileMode tm = SkTileMode::kClamp;
brianosmane25d71c2016-09-28 11:27:28 -0700265 SkRect r ={ 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
266 SkPaint paint;
267 paint.setAntiAlias(true);
268 paint.setDither(fDither);
269
270 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
271 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
272 canvas->save();
273 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers4f); j++) {
274 SkMatrix scale = SkMatrix::I();
275
276 if (i == 5) { // if the clamp case
277 scale.setScale(0.5f, 0.5f);
278 scale.postTranslate(25.f, 25.f);
279 }
280
281 paint.setShader(gGradMakers4f[j](pts, gGradData[i], tm, scale));
282 canvas->drawRect(r, paint);
283 canvas->translate(0, SkIntToScalar(120));
284 }
285 canvas->restore();
286 canvas->translate(SkIntToScalar(120), 0);
287 }
288 }
289
290protected:
291 bool fDither;
292
293private:
294 typedef GM INHERITED;
295};
296DEF_GM(return new Gradients4fGM(true); )
297DEF_GM(return new Gradients4fGM(false); )
298
rileya@google.com5cf2c282012-07-09 14:42:16 +0000299// Based on the original gradient slide, but with perspective applied to the
300// gradient shaders' local matrices
301class GradientsLocalPerspectiveGM : public GM {
302public:
fmalita063675b2015-10-12 10:41:48 -0700303 GradientsLocalPerspectiveGM(bool dither) : fDither(dither) {
Mike Kleind46dce32018-08-16 10:17:03 -0400304 this->setBGColor(0xFFDDDDDD);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000305 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000306
rileya@google.com5cf2c282012-07-09 14:42:16 +0000307protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000308
rileya@google.com5cf2c282012-07-09 14:42:16 +0000309 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700310 return SkString(fDither ? "gradients_local_perspective" :
311 "gradients_local_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000312 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000313
edisonn@google.com1da3a802013-09-19 17:55:49 +0000314 virtual SkISize onISize() { return SkISize::Make(840, 815); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000315
rileya@google.com5cf2c282012-07-09 14:42:16 +0000316 virtual void onDraw(SkCanvas* canvas) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000317
rileya@google.com5cf2c282012-07-09 14:42:16 +0000318 SkPoint pts[2] = {
319 { 0, 0 },
320 { SkIntToScalar(100), SkIntToScalar(100) }
321 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400322 SkTileMode tm = SkTileMode::kClamp;
rileya@google.com5cf2c282012-07-09 14:42:16 +0000323 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
324 SkPaint paint;
325 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700326 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000327
rileya@google.com5cf2c282012-07-09 14:42:16 +0000328 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
329 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
330 canvas->save();
331 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers); j++) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000332 // apply an increasing y perspective as we move to the right
333 SkMatrix perspective;
334 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700335 perspective.setPerspY(SkIntToScalar(i+1) / 500);
336 perspective.setSkewX(SkIntToScalar(i+1) / 10);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000337
reed1a9b9642016-03-13 14:13:58 -0700338 paint.setShader(gGradMakers[j](pts, gGradData[i], tm, perspective));
rileya@google.com5cf2c282012-07-09 14:42:16 +0000339 canvas->drawRect(r, paint);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000340 canvas->translate(0, SkIntToScalar(120));
341 }
342 canvas->restore();
343 canvas->translate(SkIntToScalar(120), 0);
344 }
345 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000346
rileya@google.com5cf2c282012-07-09 14:42:16 +0000347private:
fmalita063675b2015-10-12 10:41:48 -0700348 bool fDither;
349
rileya@google.com5cf2c282012-07-09 14:42:16 +0000350 typedef GM INHERITED;
351};
fmalita063675b2015-10-12 10:41:48 -0700352DEF_GM( return new GradientsLocalPerspectiveGM(true); )
353DEF_GM( return new GradientsLocalPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000354
355// Based on the original gradient slide, but with perspective applied to
356// the view matrix
357class GradientsViewPerspectiveGM : public GradientsGM {
fmalita063675b2015-10-12 10:41:48 -0700358public:
359 GradientsViewPerspectiveGM(bool dither) : INHERITED(dither) { }
360
rileya@google.com5cf2c282012-07-09 14:42:16 +0000361protected:
362 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700363 return SkString(fDither ? "gradients_view_perspective" :
364 "gradients_view_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000365 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000366
edisonn@google.com1da3a802013-09-19 17:55:49 +0000367 virtual SkISize onISize() { return SkISize::Make(840, 500); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000368
rileya@google.com5cf2c282012-07-09 14:42:16 +0000369 virtual void onDraw(SkCanvas* canvas) {
370 SkMatrix perspective;
371 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700372 perspective.setPerspY(0.001f);
373 perspective.setSkewX(SkIntToScalar(8) / 25);
scroggo@google.com837d31a2012-08-15 18:42:45 +0000374 canvas->concat(perspective);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000375 INHERITED::onDraw(canvas);
376 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000377
rileya@google.com5cf2c282012-07-09 14:42:16 +0000378private:
379 typedef GradientsGM INHERITED;
380};
fmalita063675b2015-10-12 10:41:48 -0700381DEF_GM( return new GradientsViewPerspectiveGM(true); )
382DEF_GM( return new GradientsViewPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000383
reed@google.comac864a92011-06-27 18:11:17 +0000384/*
385 Inspired by this <canvas> javascript, where we need to detect that we are not
386 solving a quadratic equation, but must instead solve a linear (since our X^2
387 coefficient is 0)
388
389 ctx.fillStyle = '#f00';
390 ctx.fillRect(0, 0, 100, 50);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000391
reed@google.comac864a92011-06-27 18:11:17 +0000392 var g = ctx.createRadialGradient(-80, 25, 70, 0, 25, 150);
393 g.addColorStop(0, '#f00');
394 g.addColorStop(0.01, '#0f0');
395 g.addColorStop(0.99, '#0f0');
396 g.addColorStop(1, '#f00');
397 ctx.fillStyle = g;
398 ctx.fillRect(0, 0, 100, 50);
399 */
400class GradientsDegenrate2PointGM : public GM {
401public:
fmalita063675b2015-10-12 10:41:48 -0700402 GradientsDegenrate2PointGM(bool dither) : fDither(dither) {}
rmistry@google.comd6176b02012-08-23 18:14:13 +0000403
reed@google.comac864a92011-06-27 18:11:17 +0000404protected:
405 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700406 return SkString(fDither ? "gradients_degenerate_2pt" : "gradients_degenerate_2pt_nodither");
reed@google.comac864a92011-06-27 18:11:17 +0000407 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000408
edisonn@google.com1da3a802013-09-19 17:55:49 +0000409 virtual SkISize onISize() { return SkISize::Make(320, 320); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000410
reed@google.comac864a92011-06-27 18:11:17 +0000411 void drawBG(SkCanvas* canvas) {
412 canvas->drawColor(SK_ColorBLUE);
413 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000414
reed@google.comac864a92011-06-27 18:11:17 +0000415 virtual void onDraw(SkCanvas* canvas) {
416 this->drawBG(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000417
reed@google.comac864a92011-06-27 18:11:17 +0000418 SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorRED };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000419 SkScalar pos[] = { 0, 0.01f, 0.99f, SK_Scalar1 };
epoger@google.com59f3abf2011-07-21 15:50:33 +0000420 SkPoint c0;
421 c0.iset(-80, 25);
422 SkScalar r0 = SkIntToScalar(70);
423 SkPoint c1;
424 c1.iset(0, 25);
425 SkScalar r1 = SkIntToScalar(150);
reed9283d202016-03-13 13:01:57 -0700426 SkPaint paint;
reed1a9b9642016-03-13 14:13:58 -0700427 paint.setShader(SkGradientShader::MakeTwoPointConical(c0, r0, c1, r1, colors,
428 pos, SK_ARRAY_COUNT(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400429 SkTileMode::kClamp));
fmalita063675b2015-10-12 10:41:48 -0700430 paint.setDither(fDither);
reed@google.comac864a92011-06-27 18:11:17 +0000431 canvas->drawPaint(paint);
432 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000433
reed@google.comac864a92011-06-27 18:11:17 +0000434private:
fmalita063675b2015-10-12 10:41:48 -0700435 bool fDither;
436
reed@google.comac864a92011-06-27 18:11:17 +0000437 typedef GM INHERITED;
438};
fmalita063675b2015-10-12 10:41:48 -0700439DEF_GM( return new GradientsDegenrate2PointGM(true); )
440DEF_GM( return new GradientsDegenrate2PointGM(false); )
reed@google.comac864a92011-06-27 18:11:17 +0000441
caryclarkcb100712016-02-26 05:59:40 -0800442/* bug.skia.org/517
443<canvas id="canvas"></canvas>
444<script>
445var c = document.getElementById("canvas");
446var ctx = c.getContext("2d");
447ctx.fillStyle = '#ff0';
448ctx.fillRect(0, 0, 100, 50);
449
450var g = ctx.createRadialGradient(200, 25, 20, 200, 25, 10);
451g.addColorStop(0, '#0f0');
452g.addColorStop(0.003, '#f00'); // 0.004 makes this work
453g.addColorStop(1, '#ff0');
454ctx.fillStyle = g;
455ctx.fillRect(0, 0, 100, 50);
456</script>
457*/
458
459// should draw only green
460DEF_SIMPLE_GM(small_color_stop, canvas, 100, 150) {
461 SkColor colors[] = { SK_ColorGREEN, SK_ColorRED, SK_ColorYELLOW };
462 SkScalar pos[] = { 0, 0.003f, SK_Scalar1 }; // 0.004f makes this work
463 SkPoint c0 = { 200, 25 };
464 SkScalar r0 = 20;
465 SkPoint c1 = { 200, 25 };
466 SkScalar r1 = 10;
reed1a9b9642016-03-13 14:13:58 -0700467
caryclarkcb100712016-02-26 05:59:40 -0800468 SkPaint paint;
469 paint.setColor(SK_ColorYELLOW);
470 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
reed1a9b9642016-03-13 14:13:58 -0700471 paint.setShader(SkGradientShader::MakeTwoPointConical(c0, r0, c1, r1, colors, pos,
472 SK_ARRAY_COUNT(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400473 SkTileMode::kClamp));
caryclarkcb100712016-02-26 05:59:40 -0800474 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
475}
476
477
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000478/// Tests correctness of *optimized* codepaths in gradients.
479
480class ClampedGradientsGM : public GM {
481public:
fmalita063675b2015-10-12 10:41:48 -0700482 ClampedGradientsGM(bool dither) : fDither(dither) {}
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000483
484protected:
fmalita063675b2015-10-12 10:41:48 -0700485 SkString onShortName() {
486 return SkString(fDither ? "clamped_gradients" : "clamped_gradients_nodither");
487 }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000488
edisonn@google.com1da3a802013-09-19 17:55:49 +0000489 virtual SkISize onISize() { return SkISize::Make(640, 510); }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000490
491 void drawBG(SkCanvas* canvas) {
Mike Kleind46dce32018-08-16 10:17:03 -0400492 canvas->drawColor(0xFFDDDDDD);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000493 }
494
495 virtual void onDraw(SkCanvas* canvas) {
496 this->drawBG(canvas);
497
498 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(300) };
499 SkPaint paint;
fmalita063675b2015-10-12 10:41:48 -0700500 paint.setDither(fDither);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000501 paint.setAntiAlias(true);
502
503 SkPoint center;
504 center.iset(0, 300);
505 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
reed1a9b9642016-03-13 14:13:58 -0700506 paint.setShader(SkGradientShader::MakeRadial(
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000507 SkPoint(center),
halcanary96fcdcc2015-08-27 07:41:13 -0700508 SkIntToScalar(200), gColors, nullptr, 5,
Mike Reedfae8fce2019-04-03 10:27:45 -0400509 SkTileMode::kClamp));
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000510 canvas->drawRect(r, paint);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000511 }
512
513private:
fmalita063675b2015-10-12 10:41:48 -0700514 bool fDither;
515
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000516 typedef GM INHERITED;
517};
fmalita063675b2015-10-12 10:41:48 -0700518DEF_GM( return new ClampedGradientsGM(true); )
519DEF_GM( return new ClampedGradientsGM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000520
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000521/// Checks quality of large radial gradients, which may display
522/// some banding.
523
524class RadialGradientGM : public GM {
525public:
526 RadialGradientGM() {}
527
528protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000529
mtklein36352bf2015-03-25 18:17:31 -0700530 SkString onShortName() override { return SkString("radial_gradient"); }
531 SkISize onISize() override { return SkISize::Make(1280, 1280); }
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000532 void drawBG(SkCanvas* canvas) {
533 canvas->drawColor(0xFF000000);
534 }
mtklein36352bf2015-03-25 18:17:31 -0700535 void onDraw(SkCanvas* canvas) override {
reed@google.combb0948f2012-01-31 14:44:13 +0000536 const SkISize dim = this->getISize();
537
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000538 this->drawBG(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000539
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000540 SkPaint paint;
541 paint.setDither(true);
542 SkPoint center;
reed@google.combb0948f2012-01-31 14:44:13 +0000543 center.set(SkIntToScalar(dim.width())/2, SkIntToScalar(dim.height())/2);
544 SkScalar radius = SkIntToScalar(dim.width())/2;
545 const SkColor colors[] = { 0x7f7f7f7f, 0x7f7f7f7f, 0xb2000000 };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000546 const SkScalar pos[] = { 0.0f,
547 0.35f,
548 1.0f };
reed1a9b9642016-03-13 14:13:58 -0700549 paint.setShader(SkGradientShader::MakeRadial(center, radius, colors, pos,
550 SK_ARRAY_COUNT(pos),
Mike Reedfae8fce2019-04-03 10:27:45 -0400551 SkTileMode::kClamp));
reed@google.combb0948f2012-01-31 14:44:13 +0000552 SkRect r = {
553 0, 0, SkIntToScalar(dim.width()), SkIntToScalar(dim.height())
554 };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000555 canvas->drawRect(r, paint);
556 }
557private:
558 typedef GM INHERITED;
559};
reed3d9005c2015-04-23 10:30:27 -0700560DEF_GM( return new RadialGradientGM; )
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000561
mtklein@google.com361a72f2013-08-19 18:43:34 +0000562class RadialGradient2GM : public GM {
563public:
fmalita063675b2015-10-12 10:41:48 -0700564 RadialGradient2GM(bool dither) : fDither(dither) {}
mtklein@google.com361a72f2013-08-19 18:43:34 +0000565
566protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000567
fmalita063675b2015-10-12 10:41:48 -0700568 SkString onShortName() override {
569 return SkString(fDither ? "radial_gradient2" : "radial_gradient2_nodither");
570 }
571
mtklein36352bf2015-03-25 18:17:31 -0700572 SkISize onISize() override { return SkISize::Make(800, 400); }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000573 void drawBG(SkCanvas* canvas) {
574 canvas->drawColor(0xFF000000);
575 }
576
577 // Reproduces the example given in bug 7671058.
mtklein36352bf2015-03-25 18:17:31 -0700578 void onDraw(SkCanvas* canvas) override {
mtklein@google.com361a72f2013-08-19 18:43:34 +0000579 SkPaint paint1, paint2, paint3;
580 paint1.setStyle(SkPaint::kFill_Style);
581 paint2.setStyle(SkPaint::kFill_Style);
582 paint3.setStyle(SkPaint::kFill_Style);
583
584 const SkColor sweep_colors[] =
585 { 0xFFFF0000, 0xFFFFFF00, 0xFF00FF00, 0xFF00FFFF, 0xFF0000FF, 0xFFFF00FF, 0xFFFF0000 };
586 const SkColor colors1[] = { 0xFFFFFFFF, 0x00000000 };
587 const SkColor colors2[] = { 0xFF000000, 0x00000000 };
588
589 const SkScalar cx = 200, cy = 200, radius = 150;
590 SkPoint center;
591 center.set(cx, cy);
592
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000593 // We can either interpolate endpoints and premultiply each point (default, more precision),
594 // or premultiply the endpoints first, avoiding the need to premultiply each point (cheap).
595 const uint32_t flags[] = { 0, SkGradientShader::kInterpolateColorsInPremul_Flag };
mtklein@google.com361a72f2013-08-19 18:43:34 +0000596
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000597 for (size_t i = 0; i < SK_ARRAY_COUNT(flags); i++) {
reed1a9b9642016-03-13 14:13:58 -0700598 paint1.setShader(SkGradientShader::MakeSweep(cx, cy, sweep_colors,
599 nullptr, SK_ARRAY_COUNT(sweep_colors),
600 flags[i], nullptr));
601 paint2.setShader(SkGradientShader::MakeRadial(center, radius, colors1,
602 nullptr, SK_ARRAY_COUNT(colors1),
Mike Reedfae8fce2019-04-03 10:27:45 -0400603 SkTileMode::kClamp,
reed1a9b9642016-03-13 14:13:58 -0700604 flags[i], nullptr));
605 paint3.setShader(SkGradientShader::MakeRadial(center, radius, colors2,
606 nullptr, SK_ARRAY_COUNT(colors2),
Mike Reedfae8fce2019-04-03 10:27:45 -0400607 SkTileMode::kClamp,
reed1a9b9642016-03-13 14:13:58 -0700608 flags[i], nullptr));
fmalita063675b2015-10-12 10:41:48 -0700609 paint1.setDither(fDither);
fmalita063675b2015-10-12 10:41:48 -0700610 paint2.setDither(fDither);
fmalita063675b2015-10-12 10:41:48 -0700611 paint3.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000612
613 canvas->drawCircle(cx, cy, radius, paint1);
614 canvas->drawCircle(cx, cy, radius, paint3);
615 canvas->drawCircle(cx, cy, radius, paint2);
616
617 canvas->translate(400, 0);
618 }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000619 }
620
621private:
fmalita063675b2015-10-12 10:41:48 -0700622 bool fDither;
623
mtklein@google.com361a72f2013-08-19 18:43:34 +0000624 typedef GM INHERITED;
625};
fmalita063675b2015-10-12 10:41:48 -0700626DEF_GM( return new RadialGradient2GM(true); )
627DEF_GM( return new RadialGradient2GM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000628
reed3d9005c2015-04-23 10:30:27 -0700629// Shallow radial (shows banding on raster)
630class RadialGradient3GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700631public:
632 RadialGradient3GM(bool dither) : fDither(dither) { }
reed@android.com42309d42009-06-22 02:06:35 +0000633
reed3d9005c2015-04-23 10:30:27 -0700634protected:
fmalita063675b2015-10-12 10:41:48 -0700635 SkString onShortName() override {
636 return SkString(fDither ? "radial_gradient3" : "radial_gradient3_nodither");
637 }
reed@android.com42309d42009-06-22 02:06:35 +0000638
reed3d9005c2015-04-23 10:30:27 -0700639 SkISize onISize() override { return SkISize::Make(500, 500); }
reed@google.comac864a92011-06-27 18:11:17 +0000640
reed3d9005c2015-04-23 10:30:27 -0700641 bool runAsBench() const override { return true; }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000642
reed3d9005c2015-04-23 10:30:27 -0700643 void onOnceBeforeDraw() override {
644 const SkPoint center = { 0, 0 };
645 const SkScalar kRadius = 3000;
646 const SkColor gColors[] = { 0xFFFFFFFF, 0xFF000000 };
reed1a9b9642016-03-13 14:13:58 -0700647 fShader = SkGradientShader::MakeRadial(center, kRadius, gColors, nullptr, 2,
Mike Reedfae8fce2019-04-03 10:27:45 -0400648 SkTileMode::kClamp);
reed3d9005c2015-04-23 10:30:27 -0700649 }
rileya@google.com5cf2c282012-07-09 14:42:16 +0000650
reed3d9005c2015-04-23 10:30:27 -0700651 void onDraw(SkCanvas* canvas) override {
652 SkPaint paint;
653 paint.setShader(fShader);
fmalita063675b2015-10-12 10:41:48 -0700654 paint.setDither(fDither);
reed3d9005c2015-04-23 10:30:27 -0700655 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
656 }
halcanary9d524f22016-03-29 09:03:52 -0700657
reed3d9005c2015-04-23 10:30:27 -0700658private:
reed1a9b9642016-03-13 14:13:58 -0700659 sk_sp<SkShader> fShader;
fmalita063675b2015-10-12 10:41:48 -0700660 bool fDither;
661
reed3d9005c2015-04-23 10:30:27 -0700662 typedef GM INHERITED;
663};
fmalita063675b2015-10-12 10:41:48 -0700664DEF_GM( return new RadialGradient3GM(true); )
665DEF_GM( return new RadialGradient3GM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000666
caryclark1864bfa2015-07-30 06:41:39 -0700667class RadialGradient4GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700668public:
669 RadialGradient4GM(bool dither) : fDither(dither) { }
caryclark1864bfa2015-07-30 06:41:39 -0700670
671protected:
fmalita063675b2015-10-12 10:41:48 -0700672 SkString onShortName() override {
673 return SkString(fDither ? "radial_gradient4" : "radial_gradient4_nodither");
674 }
caryclark1864bfa2015-07-30 06:41:39 -0700675
676 SkISize onISize() override { return SkISize::Make(500, 500); }
677
678 void onOnceBeforeDraw() override {
679 const SkPoint center = { 250, 250 };
680 const SkScalar kRadius = 250;
681 const SkColor colors[] = { SK_ColorRED, SK_ColorRED, SK_ColorWHITE, SK_ColorWHITE,
682 SK_ColorRED };
683 const SkScalar pos[] = { 0, .4f, .4f, .8f, .8f, 1 };
reed1a9b9642016-03-13 14:13:58 -0700684 fShader = SkGradientShader::MakeRadial(center, kRadius, colors, pos,
Mike Reedfae8fce2019-04-03 10:27:45 -0400685 SK_ARRAY_COUNT(gColors), SkTileMode::kClamp);
caryclark1864bfa2015-07-30 06:41:39 -0700686 }
687
688 void onDraw(SkCanvas* canvas) override {
689 SkPaint paint;
690 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700691 paint.setDither(fDither);
caryclark1864bfa2015-07-30 06:41:39 -0700692 paint.setShader(fShader);
693 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
694 }
halcanary9d524f22016-03-29 09:03:52 -0700695
caryclark1864bfa2015-07-30 06:41:39 -0700696private:
reed1a9b9642016-03-13 14:13:58 -0700697 sk_sp<SkShader> fShader;
fmalita063675b2015-10-12 10:41:48 -0700698 bool fDither;
699
caryclark1864bfa2015-07-30 06:41:39 -0700700 typedef GM INHERITED;
701};
fmalita063675b2015-10-12 10:41:48 -0700702DEF_GM( return new RadialGradient4GM(true); )
703DEF_GM( return new RadialGradient4GM(false); )
caryclark1864bfa2015-07-30 06:41:39 -0700704
caryclark159fa572015-07-30 12:35:48 -0700705class LinearGradientGM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700706public:
707 LinearGradientGM(bool dither) : fDither(dither) { }
caryclark159fa572015-07-30 12:35:48 -0700708
709protected:
fmalita063675b2015-10-12 10:41:48 -0700710 SkString onShortName() override {
711 return SkString(fDither ? "linear_gradient" : "linear_gradient_nodither");
712 }
713
caryclark159fa572015-07-30 12:35:48 -0700714 const SkScalar kWidthBump = 30.f;
715 const SkScalar kHeight = 5.f;
716 const SkScalar kMinWidth = 540.f;
717
718 SkISize onISize() override { return SkISize::Make(500, 500); }
719
720 void onOnceBeforeDraw() override {
721 SkPoint pts[2] = { {0, 0}, {0, 0} };
722 const SkColor colors[] = { SK_ColorWHITE, SK_ColorWHITE, 0xFF008200, 0xFF008200,
723 SK_ColorWHITE, SK_ColorWHITE };
724 const SkScalar unitPos[] = { 0, 50, 70, 500, 540 };
725 SkScalar pos[6];
726 pos[5] = 1;
727 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
728 pts[1].fX = 500.f + index * kWidthBump;
729 for (int inner = 0; inner < (int) SK_ARRAY_COUNT(unitPos); ++inner) {
730 pos[inner] = unitPos[inner] / (kMinWidth + index * kWidthBump);
731 }
reed1a9b9642016-03-13 14:13:58 -0700732 fShader[index] = SkGradientShader::MakeLinear(pts, colors, pos,
Mike Reedfae8fce2019-04-03 10:27:45 -0400733 SK_ARRAY_COUNT(gColors), SkTileMode::kClamp);
caryclark159fa572015-07-30 12:35:48 -0700734 }
735 }
736
737 void onDraw(SkCanvas* canvas) override {
738 SkPaint paint;
739 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700740 paint.setDither(fDither);
caryclark159fa572015-07-30 12:35:48 -0700741 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
742 paint.setShader(fShader[index]);
743 canvas->drawRect(SkRect::MakeLTRB(0, index * kHeight, kMinWidth + index * kWidthBump,
744 (index + 1) * kHeight), paint);
745 }
746 }
halcanary9d524f22016-03-29 09:03:52 -0700747
caryclark159fa572015-07-30 12:35:48 -0700748private:
reed1a9b9642016-03-13 14:13:58 -0700749 sk_sp<SkShader> fShader[100];
fmalita063675b2015-10-12 10:41:48 -0700750 bool fDither;
751
caryclark159fa572015-07-30 12:35:48 -0700752 typedef GM INHERITED;
753};
fmalita063675b2015-10-12 10:41:48 -0700754DEF_GM( return new LinearGradientGM(true); )
755DEF_GM( return new LinearGradientGM(false); )
caryclark159fa572015-07-30 12:35:48 -0700756
fmalita8b78bd62015-11-20 13:58:24 -0800757class LinearGradientTinyGM : public GM {
fmalitabc590c02016-02-22 09:12:33 -0800758public:
759 LinearGradientTinyGM(uint32_t flags, const char* suffix = nullptr)
760 : fName("linear_gradient_tiny")
761 , fFlags(flags) {
762 fName.append(suffix);
763 }
764
fmalita8b78bd62015-11-20 13:58:24 -0800765protected:
766 SkString onShortName() override {
fmalitabc590c02016-02-22 09:12:33 -0800767 return fName;
fmalita8b78bd62015-11-20 13:58:24 -0800768 }
769
770 SkISize onISize() override {
771 return SkISize::Make(600, 500);
772 }
773
774 void onDraw(SkCanvas* canvas) override {
775 const SkScalar kRectSize = 100;
776 const unsigned kStopCount = 3;
777 const SkColor colors[kStopCount] = { SK_ColorGREEN, SK_ColorRED, SK_ColorGREEN };
778 const struct {
779 SkPoint pts[2];
780 SkScalar pos[kStopCount];
781 } configs[] = {
reedde3aac82015-11-22 13:00:04 -0800782 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999f, 1 }},
783 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000001f, 1 }},
784 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999999f, 1 }},
785 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000000001f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800786
reedde3aac82015-11-22 13:00:04 -0800787 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999f, 1 }},
788 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000001f, 1 }},
789 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999999f, 1 }},
790 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000000001f, 1 }},
791
792 { { SkPoint::Make(0, 0), SkPoint::Make(0.00001f, 0) }, { 0, 0.5f, 1 }},
793 { { SkPoint::Make(9.99999f, 0), SkPoint::Make(10, 0) }, { 0, 0.5f, 1 }},
794 { { SkPoint::Make(0, 0), SkPoint::Make(0, 0.00001f) }, { 0, 0.5f, 1 }},
795 { { SkPoint::Make(0, 9.99999f), SkPoint::Make(0, 10) }, { 0, 0.5f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800796 };
797
798 SkPaint paint;
799 for (unsigned i = 0; i < SK_ARRAY_COUNT(configs); ++i) {
800 SkAutoCanvasRestore acr(canvas, true);
reed1a9b9642016-03-13 14:13:58 -0700801 paint.setShader(SkGradientShader::MakeLinear(configs[i].pts, colors, configs[i].pos,
Mike Reedfae8fce2019-04-03 10:27:45 -0400802 kStopCount, SkTileMode::kClamp,
reed1a9b9642016-03-13 14:13:58 -0700803 fFlags, nullptr));
fmalita8b78bd62015-11-20 13:58:24 -0800804 canvas->translate(kRectSize * ((i % 4) * 1.5f + 0.25f),
805 kRectSize * ((i / 4) * 1.5f + 0.25f));
806
fmalita8b78bd62015-11-20 13:58:24 -0800807 canvas->drawRect(SkRect::MakeWH(kRectSize, kRectSize), paint);
808 }
809 }
810
811private:
812 typedef GM INHERITED;
fmalita8b78bd62015-11-20 13:58:24 -0800813
fmalitabc590c02016-02-22 09:12:33 -0800814 SkString fName;
815 uint32_t fFlags;
816};
817DEF_GM( return new LinearGradientTinyGM(0); )
reed@android.com42309d42009-06-22 02:06:35 +0000818}
reedd4eaa252016-01-22 10:35:26 -0800819
820///////////////////////////////////////////////////////////////////////////////////////////////////
821
822struct GradRun {
823 SkColor fColors[4];
824 SkScalar fPos[4];
825 int fCount;
826};
827
828#define SIZE 121
829
Mike Reedfae8fce2019-04-03 10:27:45 -0400830static sk_sp<SkShader> make_linear(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800831 const SkPoint pts[] { { 30, 30 }, { SIZE - 30, SIZE - 30 } };
reed1a9b9642016-03-13 14:13:58 -0700832 return SkGradientShader::MakeLinear(pts, run.fColors, run.fPos, run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800833}
834
Mike Reedfae8fce2019-04-03 10:27:45 -0400835static sk_sp<SkShader> make_radial(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800836 const SkScalar half = SIZE * 0.5f;
reed1a9b9642016-03-13 14:13:58 -0700837 return SkGradientShader::MakeRadial({half,half}, half - 10, run.fColors, run.fPos,
838 run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800839}
840
Mike Reedfae8fce2019-04-03 10:27:45 -0400841static sk_sp<SkShader> make_conical(const GradRun& run, SkTileMode mode) {
reedd4eaa252016-01-22 10:35:26 -0800842 const SkScalar half = SIZE * 0.5f;
843 const SkPoint center { half, half };
reed1a9b9642016-03-13 14:13:58 -0700844 return SkGradientShader::MakeTwoPointConical(center, 20, center, half - 10,
845 run.fColors, run.fPos, run.fCount, mode);
reedd4eaa252016-01-22 10:35:26 -0800846}
847
Mike Reedfae8fce2019-04-03 10:27:45 -0400848static sk_sp<SkShader> make_sweep(const GradRun& run, SkTileMode) {
reedd4eaa252016-01-22 10:35:26 -0800849 const SkScalar half = SIZE * 0.5f;
reed1a9b9642016-03-13 14:13:58 -0700850 return SkGradientShader::MakeSweep(half, half, run.fColors, run.fPos, run.fCount);
reedd4eaa252016-01-22 10:35:26 -0800851}
852
853/*
854 * Exercise duplicate color-stops, at the ends, and in the middle
855 *
856 * At the time of this writing, only Linear correctly deals with duplicates at the ends,
857 * and then only correctly on CPU backend.
858 */
859DEF_SIMPLE_GM(gradients_dup_color_stops, canvas, 704, 564) {
860 const SkColor preColor = 0xFFFF0000; // clamp color before start
861 const SkColor postColor = 0xFF0000FF; // clamp color after end
862 const SkColor color0 = 0xFF000000;
863 const SkColor color1 = 0xFF00FF00;
864 const SkColor badColor = 0xFF3388BB; // should never be seen, fills out fixed-size array
865
866 const GradRun runs[] = {
867 { { color0, color1, badColor, badColor },
868 { 0, 1, -1, -1 },
869 2,
870 },
871 { { preColor, color0, color1, badColor },
872 { 0, 0, 1, -1 },
873 3,
874 },
875 { { color0, color1, postColor, badColor },
876 { 0, 1, 1, -1 },
877 3,
878 },
879 { { preColor, color0, color1, postColor },
880 { 0, 0, 1, 1 },
881 4,
882 },
883 { { color0, color0, color1, color1 },
884 { 0, 0.5f, 0.5f, 1 },
885 4,
886 },
887 };
Mike Reedfae8fce2019-04-03 10:27:45 -0400888 sk_sp<SkShader> (*factories[])(const GradRun&, SkTileMode) {
reedd4eaa252016-01-22 10:35:26 -0800889 make_linear, make_radial, make_conical, make_sweep
890 };
891
892 const SkRect rect = SkRect::MakeWH(SIZE, SIZE);
893 const SkScalar dx = SIZE + 20;
894 const SkScalar dy = SIZE + 20;
Mike Reedfae8fce2019-04-03 10:27:45 -0400895 const SkTileMode mode = SkTileMode::kClamp;
reedd4eaa252016-01-22 10:35:26 -0800896
897 SkPaint paint;
898 canvas->translate(10, 10 - dy);
899 for (auto factory : factories) {
900 canvas->translate(0, dy);
901 SkAutoCanvasRestore acr(canvas, true);
902 for (const auto& run : runs) {
reed1a9b9642016-03-13 14:13:58 -0700903 paint.setShader(factory(run, mode));
reedd4eaa252016-01-22 10:35:26 -0800904 canvas->drawRect(rect, paint);
905 canvas->translate(dx, 0);
906 }
907 }
908}
fmalitabc590c02016-02-22 09:12:33 -0800909
Florin Malitad1aedde2017-06-07 15:03:38 -0400910static void draw_many_stops(SkCanvas* canvas) {
fmalitabc590c02016-02-22 09:12:33 -0800911 const unsigned kStopCount = 200;
912 const SkPoint pts[] = { {50, 50}, {450, 465}};
913
914 SkColor colors[kStopCount];
915 for (unsigned i = 0; i < kStopCount; i++) {
916 switch (i % 5) {
917 case 0: colors[i] = SK_ColorRED; break;
918 case 1: colors[i] = SK_ColorGREEN; break;
919 case 2: colors[i] = SK_ColorGREEN; break;
920 case 3: colors[i] = SK_ColorBLUE; break;
921 case 4: colors[i] = SK_ColorRED; break;
922 }
923 }
924
reed9283d202016-03-13 13:01:57 -0700925 SkPaint p;
reed1a9b9642016-03-13 14:13:58 -0700926 p.setShader(SkGradientShader::MakeLinear(
Mike Reedfae8fce2019-04-03 10:27:45 -0400927 pts, colors, nullptr, SK_ARRAY_COUNT(colors), SkTileMode::kClamp));
reed9283d202016-03-13 13:01:57 -0700928
fmalitabc590c02016-02-22 09:12:33 -0800929 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
930}
931
932DEF_SIMPLE_GM(gradient_many_stops, canvas, 500, 500) {
Florin Malitad1aedde2017-06-07 15:03:38 -0400933 draw_many_stops(canvas);
fmalitabc590c02016-02-22 09:12:33 -0800934}
935
Florin Malitaa0ac9632017-05-03 13:07:28 -0400936static void draw_circle_shader(SkCanvas* canvas, SkScalar cx, SkScalar cy, SkScalar r,
937 sk_sp<SkShader> (*shaderFunc)()) {
938 SkPaint p;
939 p.setAntiAlias(true);
940 p.setShader(shaderFunc());
941 canvas->drawCircle(cx, cy, r, p);
942
943 p.setShader(nullptr);
944 p.setColor(SK_ColorGRAY);
945 p.setStyle(SkPaint::kStroke_Style);
946 p.setStrokeWidth(2);
947 canvas->drawCircle(cx, cy, r, p);
948}
949
950DEF_SIMPLE_GM(fancy_gradients, canvas, 800, 300) {
951 draw_circle_shader(canvas, 150, 150, 100, []() -> sk_sp<SkShader> {
952 // Checkerboard using two linear gradients + picture shader.
953 SkScalar kTileSize = 80 / sqrtf(2);
954 SkColor colors1[] = { 0xff000000, 0xff000000,
955 0xffffffff, 0xffffffff,
956 0xff000000, 0xff000000 };
957 SkColor colors2[] = { 0xff000000, 0xff000000,
958 0x00000000, 0x00000000,
959 0xff000000, 0xff000000 };
960 SkScalar pos[] = { 0, .25f, .25f, .75f, .75f, 1 };
961 static_assert(SK_ARRAY_COUNT(colors1) == SK_ARRAY_COUNT(pos), "color/pos size mismatch");
962 static_assert(SK_ARRAY_COUNT(colors2) == SK_ARRAY_COUNT(pos), "color/pos size mismatch");
963
964 SkPictureRecorder recorder;
965 recorder.beginRecording(SkRect::MakeWH(kTileSize, kTileSize));
966
967 SkPaint p;
968
969 SkPoint pts1[] = { { 0, 0 }, { kTileSize, kTileSize }};
970 p.setShader(SkGradientShader::MakeLinear(pts1, colors1, pos, SK_ARRAY_COUNT(colors1),
Mike Reedfae8fce2019-04-03 10:27:45 -0400971 SkTileMode::kClamp, 0, nullptr));
Florin Malitaa0ac9632017-05-03 13:07:28 -0400972 recorder.getRecordingCanvas()->drawPaint(p);
973
974 SkPoint pts2[] = { { 0, kTileSize }, { kTileSize, 0 }};
975 p.setShader(SkGradientShader::MakeLinear(pts2, colors2, pos, SK_ARRAY_COUNT(colors2),
Mike Reedfae8fce2019-04-03 10:27:45 -0400976 SkTileMode::kClamp, 0, nullptr));
Florin Malitaa0ac9632017-05-03 13:07:28 -0400977 recorder.getRecordingCanvas()->drawPaint(p);
978
979 SkMatrix m = SkMatrix::I();
980 m.preRotate(45);
Mike Reedfae8fce2019-04-03 10:27:45 -0400981 return recorder.finishRecordingAsPicture()->makeShader(
982 SkTileMode::kRepeat,
983 SkTileMode::kRepeat, &m, nullptr);
Florin Malitaa0ac9632017-05-03 13:07:28 -0400984 });
985
986 draw_circle_shader(canvas, 400, 150, 100, []() -> sk_sp<SkShader> {
987 // Checkerboard using a sweep gradient + picture shader.
988 SkScalar kTileSize = 80;
989 SkColor colors[] = { 0xff000000, 0xff000000,
990 0xffffffff, 0xffffffff,
991 0xff000000, 0xff000000,
992 0xffffffff, 0xffffffff };
993 SkScalar pos[] = { 0, .25f, .25f, .5f, .5f, .75f, .75f, 1 };
994 static_assert(SK_ARRAY_COUNT(colors) == SK_ARRAY_COUNT(pos), "color/pos size mismatch");
995
996 SkPaint p;
997 p.setShader(SkGradientShader::MakeSweep(kTileSize / 2, kTileSize / 2,
998 colors, pos, SK_ARRAY_COUNT(colors), 0, nullptr));
999 SkPictureRecorder recorder;
1000 recorder.beginRecording(SkRect::MakeWH(kTileSize, kTileSize))->drawPaint(p);
1001
Mike Reedfae8fce2019-04-03 10:27:45 -04001002 return recorder.finishRecordingAsPicture()->makeShader(
1003 SkTileMode::kRepeat,
1004 SkTileMode::kRepeat);
Florin Malitaa0ac9632017-05-03 13:07:28 -04001005 });
1006
1007 draw_circle_shader(canvas, 650, 150, 100, []() -> sk_sp<SkShader> {
1008 // Dartboard using sweep + radial.
1009 const SkColor a = 0xffffffff;
1010 const SkColor b = 0xff000000;
1011 SkColor colors[] = { a, a, b, b, a, a, b, b, a, a, b, b, a, a, b, b};
1012 SkScalar pos[] = { 0, .125f, .125f, .25f, .25f, .375f, .375f, .5f, .5f,
1013 .625f, .625f, .75f, .75f, .875f, .875f, 1};
1014 static_assert(SK_ARRAY_COUNT(colors) == SK_ARRAY_COUNT(pos), "color/pos size mismatch");
1015
1016 SkPoint center = { 650, 150 };
1017 sk_sp<SkShader> sweep1 = SkGradientShader::MakeSweep(center.x(), center.y(), colors, pos,
1018 SK_ARRAY_COUNT(colors), 0, nullptr);
1019 SkMatrix m = SkMatrix::I();
1020 m.preRotate(22.5f, center.x(), center.y());
1021 sk_sp<SkShader> sweep2 = SkGradientShader::MakeSweep(center.x(), center.y(), colors, pos,
1022 SK_ARRAY_COUNT(colors), 0, &m);
1023
Mike Reedc8bea7d2019-04-09 13:55:36 -04001024 sk_sp<SkShader> sweep(SkShaders::Blend(SkBlendMode::kExclusion, sweep1, sweep2));
Florin Malitaa0ac9632017-05-03 13:07:28 -04001025
1026 SkScalar radialPos[] = { 0, .02f, .02f, .04f, .04f, .08f, .08f, .16f, .16f, .31f, .31f,
1027 .62f, .62f, 1, 1, 1 };
1028 static_assert(SK_ARRAY_COUNT(colors) == SK_ARRAY_COUNT(radialPos),
1029 "color/pos size mismatch");
1030
Mike Reedc8bea7d2019-04-09 13:55:36 -04001031 return SkShaders::Blend(SkBlendMode::kExclusion, sweep,
1032 SkGradientShader::MakeRadial(center, 100, colors,
1033 radialPos,
1034 SK_ARRAY_COUNT(radialPos),
1035 SkTileMode::kClamp));
Florin Malitaa0ac9632017-05-03 13:07:28 -04001036 });
1037}
Florin Malita5a9a9812017-08-01 16:38:08 -04001038
Florin Malita4d274292017-08-02 10:04:48 -04001039DEF_SIMPLE_GM(sweep_tiling, canvas, 690, 512) {
Florin Malita5a9a9812017-08-01 16:38:08 -04001040 static constexpr SkScalar size = 160;
1041 static constexpr SkColor colors[] = { SK_ColorBLUE, SK_ColorYELLOW, SK_ColorGREEN };
1042 static constexpr SkScalar pos[] = { 0, .25f, .50f };
1043 static_assert(SK_ARRAY_COUNT(colors) == SK_ARRAY_COUNT(pos), "size mismatch");
1044
Mike Reedfae8fce2019-04-03 10:27:45 -04001045 static constexpr SkTileMode modes[] = { SkTileMode::kClamp,
1046 SkTileMode::kRepeat,
1047 SkTileMode::kMirror };
Florin Malita5a9a9812017-08-01 16:38:08 -04001048
1049 static const struct {
1050 SkScalar start, end;
1051 } angles[] = {
1052 { -330, -270 },
1053 { 30, 90 },
1054 { 390, 450 },
Florin Malita50f7a1e2017-08-02 09:40:26 -04001055 { -30, 800 },
Florin Malita5a9a9812017-08-01 16:38:08 -04001056 };
1057
1058 SkPaint p;
1059 const SkRect r = SkRect::MakeWH(size, size);
1060
1061 for (auto mode : modes) {
1062 {
1063 SkAutoCanvasRestore acr(canvas, true);
1064
1065 for (auto angle : angles) {
1066 p.setShader(SkGradientShader::MakeSweep(size / 2, size / 2, colors, pos,
1067 SK_ARRAY_COUNT(colors), mode,
1068 angle.start, angle.end, 0, nullptr));
1069
1070 canvas->drawRect(r, p);
1071 canvas->translate(size * 1.1f, 0);
1072 }
1073 }
1074 canvas->translate(0, size * 1.1f);
1075 }
1076}
Florin Malita36f054a2017-08-03 12:55:41 -04001077
1078// Exercises the special-case Ganesh gradient effects.
Brian Osmana8e57442017-09-11 17:21:35 -04001079DEF_SIMPLE_GM(gradients_interesting, canvas, 640, 1300) {
Florin Malita36f054a2017-08-03 12:55:41 -04001080 static const SkColor colors2[] = { SK_ColorRED, SK_ColorBLUE };
1081 static const SkColor colors3[] = { SK_ColorRED, SK_ColorYELLOW, SK_ColorBLUE };
1082 static const SkColor colors4[] = { SK_ColorRED, SK_ColorYELLOW, SK_ColorYELLOW, SK_ColorBLUE };
1083
Brian Osmana8e57442017-09-11 17:21:35 -04001084 static const SkScalar softRight[] = { 0, .999f, 1 }; // Based on Android launcher "clipping"
1085 static const SkScalar hardLeft[] = { 0, 0, 1 };
1086 static const SkScalar hardRight[] = { 0, 1, 1 };
1087 static const SkScalar hardCenter[] = { 0, .5f, .5f, 1 };
Florin Malita36f054a2017-08-03 12:55:41 -04001088
1089 static const struct {
1090 const SkColor* colors;
1091 const SkScalar* pos;
1092 int count;
1093 } configs[] = {
1094 { colors2, nullptr, 2 }, // kTwo_ColorType
Brian Osman2ab4b2b2017-09-12 11:20:56 -04001095 { colors3, nullptr, 3 }, // kThree_ColorType (simple)
1096 { colors3, softRight, 3 }, // kThree_ColorType (tricky)
Florin Malita36f054a2017-08-03 12:55:41 -04001097 { colors3, hardLeft, 3 }, // kHardStopLeftEdged_ColorType
1098 { colors3, hardRight, 3 }, // kHardStopRightEdged_ColorType
1099 { colors4, hardCenter, 4 }, // kSingleHardStop_ColorType
1100 };
1101
Mike Reedfae8fce2019-04-03 10:27:45 -04001102 static const SkTileMode modes[] = {
1103 SkTileMode::kClamp,
1104 SkTileMode::kRepeat,
1105 SkTileMode::kMirror,
Florin Malita36f054a2017-08-03 12:55:41 -04001106 };
1107
1108 static constexpr SkScalar size = 200;
1109 static const SkPoint pts[] = { { size / 3, size / 3 }, { size * 2 / 3, size * 2 / 3} };
1110
1111 SkPaint p;
1112 for (const auto& cfg : configs) {
1113 {
1114 SkAutoCanvasRestore acr(canvas, true);
1115 for (auto mode : modes) {
1116 p.setShader(SkGradientShader::MakeLinear(pts, cfg.colors, cfg.pos, cfg.count,
1117 mode));
1118 canvas->drawRect(SkRect::MakeWH(size, size), p);
1119 canvas->translate(size * 1.1f, 0);
1120 }
1121 }
1122 canvas->translate(0, size * 1.1f);
1123 }
1124}