blob: 5fb3465913bf345326473a8baf89364f7b5cc72b [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
reed@android.com42309d42009-06-22 02:06:35 +00008#include "gm.h"
9#include "SkGradientShader.h"
fmalitabc590c02016-02-22 09:12:33 -080010#include "SkLinearGradient.h"
reed@android.com42309d42009-06-22 02:06:35 +000011
12namespace skiagm {
13
14struct GradData {
15 int fCount;
16 const SkColor* fColors;
17 const SkScalar* fPos;
18};
19
20static const SkColor gColors[] = {
21 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK
22};
23static const SkScalar gPos0[] = { 0, SK_Scalar1 };
24static const SkScalar gPos1[] = { SK_Scalar1/4, SK_Scalar1*3/4 };
25static const SkScalar gPos2[] = {
26 0, SK_Scalar1/8, SK_Scalar1/2, SK_Scalar1*7/8, SK_Scalar1
27};
28
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +000029static const SkScalar gPosClamp[] = {0.0f, 0.0f, 1.0f, 1.0f};
30static const SkColor gColorClamp[] = {
31 SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorBLUE
32};
33
reed@android.com42309d42009-06-22 02:06:35 +000034static const GradData gGradData[] = {
halcanary96fcdcc2015-08-27 07:41:13 -070035 { 2, gColors, nullptr },
reed@android.com42309d42009-06-22 02:06:35 +000036 { 2, gColors, gPos0 },
37 { 2, gColors, gPos1 },
halcanary96fcdcc2015-08-27 07:41:13 -070038 { 5, gColors, nullptr },
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +000039 { 5, gColors, gPos2 },
40 { 4, gColorClamp, gPosClamp }
reed@android.com42309d42009-06-22 02:06:35 +000041};
42
43static SkShader* MakeLinear(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000044 SkShader::TileMode tm, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +000045 return SkGradientShader::CreateLinear(pts, data.fColors, data.fPos,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000046 data.fCount, tm, 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000047}
reed@google.comf3c1cc92010-12-23 16:45:33 +000048
reed@android.com42309d42009-06-22 02:06:35 +000049static SkShader* MakeRadial(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000050 SkShader::TileMode tm, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +000051 SkPoint center;
52 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
53 SkScalarAve(pts[0].fY, pts[1].fY));
54 return SkGradientShader::CreateRadial(center, center.fX, data.fColors,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000055 data.fPos, data.fCount, tm, 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000056}
57
58static SkShader* MakeSweep(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000059 SkShader::TileMode, const SkMatrix& localMatrix) {
reed@android.com42309d42009-06-22 02:06:35 +000060 SkPoint center;
61 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
62 SkScalarAve(pts[0].fY, pts[1].fY));
63 return SkGradientShader::CreateSweep(center.fX, center.fY, data.fColors,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000064 data.fPos, data.fCount, 0, &localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000065}
66
reed@google.comf3c1cc92010-12-23 16:45:33 +000067static SkShader* Make2Radial(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000068 SkShader::TileMode tm, const SkMatrix& localMatrix) {
reed@google.comf3c1cc92010-12-23 16:45:33 +000069 SkPoint center0, center1;
70 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
71 SkScalarAve(pts[0].fY, pts[1].fY));
72 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
73 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
reed71a6cbf2015-05-04 08:32:51 -070074 return SkGradientShader::CreateTwoPointConical(
75 center1, (pts[1].fX - pts[0].fX) / 7,
76 center0, (pts[1].fX - pts[0].fX) / 2,
77 data.fColors, data.fPos, data.fCount, tm,
78 0, &localMatrix);
reed@google.comf3c1cc92010-12-23 16:45:33 +000079}
80
rileya@google.com5cf2c282012-07-09 14:42:16 +000081static SkShader* Make2Conical(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000082 SkShader::TileMode tm, const SkMatrix& localMatrix) {
rileya@google.com5cf2c282012-07-09 14:42:16 +000083 SkPoint center0, center1;
reed80ea19c2015-05-12 10:37:34 -070084 SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10;
85 SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3;
rileya@google.com5cf2c282012-07-09 14:42:16 +000086 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
87 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
rmistry@google.comd6176b02012-08-23 18:14:13 +000088 return SkGradientShader::CreateTwoPointConical(center1, radius1,
89 center0, radius0,
90 data.fColors, data.fPos,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000091 data.fCount, tm, 0, &localMatrix);
rileya@google.com5cf2c282012-07-09 14:42:16 +000092}
93
reed@android.com42309d42009-06-22 02:06:35 +000094typedef SkShader* (*GradMaker)(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +000095 SkShader::TileMode tm, const SkMatrix& localMatrix);
reed@android.com42309d42009-06-22 02:06:35 +000096static const GradMaker gGradMakers[] = {
rileya@google.com5cf2c282012-07-09 14:42:16 +000097 MakeLinear, MakeRadial, MakeSweep, Make2Radial, Make2Conical
reed@android.com42309d42009-06-22 02:06:35 +000098};
99
100///////////////////////////////////////////////////////////////////////////////
101
102class GradientsGM : public GM {
103public:
fmalita063675b2015-10-12 10:41:48 -0700104 GradientsGM(bool dither) : fDither(dither) {
caryclark65cdba62015-06-15 06:51:08 -0700105 this->setBGColor(sk_tool_utils::color_to_565(0xFFDDDDDD));
bsalomon@google.com48dd1a22011-10-31 14:18:20 +0000106 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000107
reed@android.com42309d42009-06-22 02:06:35 +0000108protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000109
reed@android.com42309d42009-06-22 02:06:35 +0000110 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700111 return SkString(fDither ? "gradients" : "gradients_nodither");
reed@android.com42309d42009-06-22 02:06:35 +0000112 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000113
edisonn@google.com1da3a802013-09-19 17:55:49 +0000114 virtual SkISize onISize() { return SkISize::Make(840, 815); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000115
reed@android.com42309d42009-06-22 02:06:35 +0000116 virtual void onDraw(SkCanvas* canvas) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000117
reed@android.com42309d42009-06-22 02:06:35 +0000118 SkPoint pts[2] = {
119 { 0, 0 },
120 { SkIntToScalar(100), SkIntToScalar(100) }
121 };
122 SkShader::TileMode tm = SkShader::kClamp_TileMode;
123 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
124 SkPaint paint;
125 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700126 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000127
reed@android.com42309d42009-06-22 02:06:35 +0000128 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
129 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
130 canvas->save();
131 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers); j++) {
commit-bot@chromium.org9c9005a2014-04-28 14:55:39 +0000132 SkMatrix scale = SkMatrix::I();
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000133
134 if (i == 5) { // if the clamp case
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000135 scale.setScale(0.5f, 0.5f);
136 scale.postTranslate(25.f, 25.f);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000137 }
skia.committer@gmail.comd55e3572013-08-08 07:01:20 +0000138
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +0000139 SkShader* shader = gGradMakers[j](pts, gGradData[i], tm, scale);
commit-bot@chromium.org9c9005a2014-04-28 14:55:39 +0000140
reed@android.com42309d42009-06-22 02:06:35 +0000141 paint.setShader(shader);
142 canvas->drawRect(r, paint);
143 shader->unref();
144 canvas->translate(0, SkIntToScalar(120));
145 }
146 canvas->restore();
147 canvas->translate(SkIntToScalar(120), 0);
148 }
149 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000150
fmalita063675b2015-10-12 10:41:48 -0700151protected:
152 bool fDither;
153
reed@android.com42309d42009-06-22 02:06:35 +0000154private:
155 typedef GM INHERITED;
156};
fmalita063675b2015-10-12 10:41:48 -0700157DEF_GM( return new GradientsGM(true); )
158DEF_GM( return new GradientsGM(false); )
reed@android.com42309d42009-06-22 02:06:35 +0000159
rileya@google.com5cf2c282012-07-09 14:42:16 +0000160// Based on the original gradient slide, but with perspective applied to the
161// gradient shaders' local matrices
162class GradientsLocalPerspectiveGM : public GM {
163public:
fmalita063675b2015-10-12 10:41:48 -0700164 GradientsLocalPerspectiveGM(bool dither) : fDither(dither) {
caryclark65cdba62015-06-15 06:51:08 -0700165 this->setBGColor(sk_tool_utils::color_to_565(0xFFDDDDDD));
rileya@google.com5cf2c282012-07-09 14:42:16 +0000166 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000167
rileya@google.com5cf2c282012-07-09 14:42:16 +0000168protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000169
rileya@google.com5cf2c282012-07-09 14:42:16 +0000170 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700171 return SkString(fDither ? "gradients_local_perspective" :
172 "gradients_local_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000173 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000174
edisonn@google.com1da3a802013-09-19 17:55:49 +0000175 virtual SkISize onISize() { return SkISize::Make(840, 815); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000176
rileya@google.com5cf2c282012-07-09 14:42:16 +0000177 virtual void onDraw(SkCanvas* canvas) {
rmistry@google.comd6176b02012-08-23 18:14:13 +0000178
rileya@google.com5cf2c282012-07-09 14:42:16 +0000179 SkPoint pts[2] = {
180 { 0, 0 },
181 { SkIntToScalar(100), SkIntToScalar(100) }
182 };
183 SkShader::TileMode tm = SkShader::kClamp_TileMode;
184 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
185 SkPaint paint;
186 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700187 paint.setDither(fDither);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000188
rileya@google.com5cf2c282012-07-09 14:42:16 +0000189 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
190 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
191 canvas->save();
192 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers); j++) {
rileya@google.com5cf2c282012-07-09 14:42:16 +0000193 // apply an increasing y perspective as we move to the right
194 SkMatrix perspective;
195 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700196 perspective.setPerspY(SkIntToScalar(i+1) / 500);
197 perspective.setSkewX(SkIntToScalar(i+1) / 10);
commit-bot@chromium.org8ba1ad32013-08-07 15:22:13 +0000198
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +0000199 SkShader* shader = gGradMakers[j](pts, gGradData[i], tm, perspective);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000200
201 paint.setShader(shader);
202 canvas->drawRect(r, paint);
203 shader->unref();
204 canvas->translate(0, SkIntToScalar(120));
205 }
206 canvas->restore();
207 canvas->translate(SkIntToScalar(120), 0);
208 }
209 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000210
rileya@google.com5cf2c282012-07-09 14:42:16 +0000211private:
fmalita063675b2015-10-12 10:41:48 -0700212 bool fDither;
213
rileya@google.com5cf2c282012-07-09 14:42:16 +0000214 typedef GM INHERITED;
215};
fmalita063675b2015-10-12 10:41:48 -0700216DEF_GM( return new GradientsLocalPerspectiveGM(true); )
217DEF_GM( return new GradientsLocalPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000218
219// Based on the original gradient slide, but with perspective applied to
220// the view matrix
221class GradientsViewPerspectiveGM : public GradientsGM {
fmalita063675b2015-10-12 10:41:48 -0700222public:
223 GradientsViewPerspectiveGM(bool dither) : INHERITED(dither) { }
224
rileya@google.com5cf2c282012-07-09 14:42:16 +0000225protected:
226 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700227 return SkString(fDither ? "gradients_view_perspective" :
228 "gradients_view_perspective_nodither");
rileya@google.com5cf2c282012-07-09 14:42:16 +0000229 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000230
edisonn@google.com1da3a802013-09-19 17:55:49 +0000231 virtual SkISize onISize() { return SkISize::Make(840, 500); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000232
rileya@google.com5cf2c282012-07-09 14:42:16 +0000233 virtual void onDraw(SkCanvas* canvas) {
234 SkMatrix perspective;
235 perspective.setIdentity();
reed80ea19c2015-05-12 10:37:34 -0700236 perspective.setPerspY(0.001f);
237 perspective.setSkewX(SkIntToScalar(8) / 25);
scroggo@google.com837d31a2012-08-15 18:42:45 +0000238 canvas->concat(perspective);
rileya@google.com5cf2c282012-07-09 14:42:16 +0000239 INHERITED::onDraw(canvas);
240 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000241
rileya@google.com5cf2c282012-07-09 14:42:16 +0000242private:
243 typedef GradientsGM INHERITED;
244};
fmalita063675b2015-10-12 10:41:48 -0700245DEF_GM( return new GradientsViewPerspectiveGM(true); )
246DEF_GM( return new GradientsViewPerspectiveGM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000247
reed@google.comac864a92011-06-27 18:11:17 +0000248/*
249 Inspired by this <canvas> javascript, where we need to detect that we are not
250 solving a quadratic equation, but must instead solve a linear (since our X^2
251 coefficient is 0)
252
253 ctx.fillStyle = '#f00';
254 ctx.fillRect(0, 0, 100, 50);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000255
reed@google.comac864a92011-06-27 18:11:17 +0000256 var g = ctx.createRadialGradient(-80, 25, 70, 0, 25, 150);
257 g.addColorStop(0, '#f00');
258 g.addColorStop(0.01, '#0f0');
259 g.addColorStop(0.99, '#0f0');
260 g.addColorStop(1, '#f00');
261 ctx.fillStyle = g;
262 ctx.fillRect(0, 0, 100, 50);
263 */
264class GradientsDegenrate2PointGM : public GM {
265public:
fmalita063675b2015-10-12 10:41:48 -0700266 GradientsDegenrate2PointGM(bool dither) : fDither(dither) {}
rmistry@google.comd6176b02012-08-23 18:14:13 +0000267
reed@google.comac864a92011-06-27 18:11:17 +0000268protected:
269 SkString onShortName() {
fmalita063675b2015-10-12 10:41:48 -0700270 return SkString(fDither ? "gradients_degenerate_2pt" : "gradients_degenerate_2pt_nodither");
reed@google.comac864a92011-06-27 18:11:17 +0000271 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000272
edisonn@google.com1da3a802013-09-19 17:55:49 +0000273 virtual SkISize onISize() { return SkISize::Make(320, 320); }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000274
reed@google.comac864a92011-06-27 18:11:17 +0000275 void drawBG(SkCanvas* canvas) {
276 canvas->drawColor(SK_ColorBLUE);
277 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000278
reed@google.comac864a92011-06-27 18:11:17 +0000279 virtual void onDraw(SkCanvas* canvas) {
280 this->drawBG(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000281
reed@google.comac864a92011-06-27 18:11:17 +0000282 SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorRED };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000283 SkScalar pos[] = { 0, 0.01f, 0.99f, SK_Scalar1 };
epoger@google.com59f3abf2011-07-21 15:50:33 +0000284 SkPoint c0;
285 c0.iset(-80, 25);
286 SkScalar r0 = SkIntToScalar(70);
287 SkPoint c1;
288 c1.iset(0, 25);
289 SkScalar r1 = SkIntToScalar(150);
reed71a6cbf2015-05-04 08:32:51 -0700290 SkShader* s = SkGradientShader::CreateTwoPointConical(c0, r0, c1, r1, colors,
291 pos, SK_ARRAY_COUNT(pos),
292 SkShader::kClamp_TileMode);
reed@google.comac864a92011-06-27 18:11:17 +0000293 SkPaint paint;
fmalita063675b2015-10-12 10:41:48 -0700294 paint.setDither(fDither);
reed@google.comac864a92011-06-27 18:11:17 +0000295 paint.setShader(s)->unref();
296 canvas->drawPaint(paint);
297 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000298
reed@google.comac864a92011-06-27 18:11:17 +0000299private:
fmalita063675b2015-10-12 10:41:48 -0700300 bool fDither;
301
reed@google.comac864a92011-06-27 18:11:17 +0000302 typedef GM INHERITED;
303};
fmalita063675b2015-10-12 10:41:48 -0700304DEF_GM( return new GradientsDegenrate2PointGM(true); )
305DEF_GM( return new GradientsDegenrate2PointGM(false); )
reed@google.comac864a92011-06-27 18:11:17 +0000306
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000307/// Tests correctness of *optimized* codepaths in gradients.
308
309class ClampedGradientsGM : public GM {
310public:
fmalita063675b2015-10-12 10:41:48 -0700311 ClampedGradientsGM(bool dither) : fDither(dither) {}
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000312
313protected:
fmalita063675b2015-10-12 10:41:48 -0700314 SkString onShortName() {
315 return SkString(fDither ? "clamped_gradients" : "clamped_gradients_nodither");
316 }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000317
edisonn@google.com1da3a802013-09-19 17:55:49 +0000318 virtual SkISize onISize() { return SkISize::Make(640, 510); }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000319
320 void drawBG(SkCanvas* canvas) {
caryclark12596012015-07-29 05:27:47 -0700321 canvas->drawColor(sk_tool_utils::color_to_565(0xFFDDDDDD));
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000322 }
323
324 virtual void onDraw(SkCanvas* canvas) {
325 this->drawBG(canvas);
326
327 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(300) };
328 SkPaint paint;
fmalita063675b2015-10-12 10:41:48 -0700329 paint.setDither(fDither);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000330 paint.setAntiAlias(true);
331
332 SkPoint center;
333 center.iset(0, 300);
334 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
335 SkShader* shader = SkGradientShader::CreateRadial(
336 SkPoint(center),
halcanary96fcdcc2015-08-27 07:41:13 -0700337 SkIntToScalar(200), gColors, nullptr, 5,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +0000338 SkShader::kClamp_TileMode);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000339 paint.setShader(shader);
340 canvas->drawRect(r, paint);
341 shader->unref();
342 }
343
344private:
fmalita063675b2015-10-12 10:41:48 -0700345 bool fDither;
346
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000347 typedef GM INHERITED;
348};
fmalita063675b2015-10-12 10:41:48 -0700349DEF_GM( return new ClampedGradientsGM(true); )
350DEF_GM( return new ClampedGradientsGM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000351
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000352/// Checks quality of large radial gradients, which may display
353/// some banding.
354
355class RadialGradientGM : public GM {
356public:
357 RadialGradientGM() {}
358
359protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000360
mtklein36352bf2015-03-25 18:17:31 -0700361 SkString onShortName() override { return SkString("radial_gradient"); }
362 SkISize onISize() override { return SkISize::Make(1280, 1280); }
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000363 void drawBG(SkCanvas* canvas) {
364 canvas->drawColor(0xFF000000);
365 }
mtklein36352bf2015-03-25 18:17:31 -0700366 void onDraw(SkCanvas* canvas) override {
reed@google.combb0948f2012-01-31 14:44:13 +0000367 const SkISize dim = this->getISize();
368
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000369 this->drawBG(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000370
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000371 SkPaint paint;
372 paint.setDither(true);
373 SkPoint center;
reed@google.combb0948f2012-01-31 14:44:13 +0000374 center.set(SkIntToScalar(dim.width())/2, SkIntToScalar(dim.height())/2);
375 SkScalar radius = SkIntToScalar(dim.width())/2;
376 const SkColor colors[] = { 0x7f7f7f7f, 0x7f7f7f7f, 0xb2000000 };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000377 const SkScalar pos[] = { 0.0f,
378 0.35f,
379 1.0f };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000380 SkShader* shader =
381 SkGradientShader::CreateRadial(center, radius, colors,
reed@google.combb0948f2012-01-31 14:44:13 +0000382 pos, SK_ARRAY_COUNT(pos),
383 SkShader::kClamp_TileMode);
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000384 paint.setShader(shader)->unref();
reed@google.combb0948f2012-01-31 14:44:13 +0000385 SkRect r = {
386 0, 0, SkIntToScalar(dim.width()), SkIntToScalar(dim.height())
387 };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000388 canvas->drawRect(r, paint);
389 }
390private:
391 typedef GM INHERITED;
392};
reed3d9005c2015-04-23 10:30:27 -0700393DEF_GM( return new RadialGradientGM; )
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000394
mtklein@google.com361a72f2013-08-19 18:43:34 +0000395class RadialGradient2GM : public GM {
396public:
fmalita063675b2015-10-12 10:41:48 -0700397 RadialGradient2GM(bool dither) : fDither(dither) {}
mtklein@google.com361a72f2013-08-19 18:43:34 +0000398
399protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000400
fmalita063675b2015-10-12 10:41:48 -0700401 SkString onShortName() override {
402 return SkString(fDither ? "radial_gradient2" : "radial_gradient2_nodither");
403 }
404
mtklein36352bf2015-03-25 18:17:31 -0700405 SkISize onISize() override { return SkISize::Make(800, 400); }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000406 void drawBG(SkCanvas* canvas) {
407 canvas->drawColor(0xFF000000);
408 }
409
410 // Reproduces the example given in bug 7671058.
mtklein36352bf2015-03-25 18:17:31 -0700411 void onDraw(SkCanvas* canvas) override {
mtklein@google.com361a72f2013-08-19 18:43:34 +0000412 SkPaint paint1, paint2, paint3;
413 paint1.setStyle(SkPaint::kFill_Style);
414 paint2.setStyle(SkPaint::kFill_Style);
415 paint3.setStyle(SkPaint::kFill_Style);
416
417 const SkColor sweep_colors[] =
418 { 0xFFFF0000, 0xFFFFFF00, 0xFF00FF00, 0xFF00FFFF, 0xFF0000FF, 0xFFFF00FF, 0xFFFF0000 };
419 const SkColor colors1[] = { 0xFFFFFFFF, 0x00000000 };
420 const SkColor colors2[] = { 0xFF000000, 0x00000000 };
421
422 const SkScalar cx = 200, cy = 200, radius = 150;
423 SkPoint center;
424 center.set(cx, cy);
425
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000426 // We can either interpolate endpoints and premultiply each point (default, more precision),
427 // or premultiply the endpoints first, avoiding the need to premultiply each point (cheap).
428 const uint32_t flags[] = { 0, SkGradientShader::kInterpolateColorsInPremul_Flag };
mtklein@google.com361a72f2013-08-19 18:43:34 +0000429
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000430 for (size_t i = 0; i < SK_ARRAY_COUNT(flags); i++) {
431 SkAutoTUnref<SkShader> sweep(
432 SkGradientShader::CreateSweep(cx, cy, sweep_colors,
halcanary96fcdcc2015-08-27 07:41:13 -0700433 nullptr, SK_ARRAY_COUNT(sweep_colors),
434 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000435 SkAutoTUnref<SkShader> radial1(
436 SkGradientShader::CreateRadial(center, radius, colors1,
halcanary96fcdcc2015-08-27 07:41:13 -0700437 nullptr, SK_ARRAY_COUNT(colors1),
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000438 SkShader::kClamp_TileMode,
halcanary96fcdcc2015-08-27 07:41:13 -0700439 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000440 SkAutoTUnref<SkShader> radial2(
441 SkGradientShader::CreateRadial(center, radius, colors2,
halcanary96fcdcc2015-08-27 07:41:13 -0700442 nullptr, SK_ARRAY_COUNT(colors2),
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000443 SkShader::kClamp_TileMode,
halcanary96fcdcc2015-08-27 07:41:13 -0700444 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000445 paint1.setShader(sweep);
fmalita063675b2015-10-12 10:41:48 -0700446 paint1.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000447 paint2.setShader(radial1);
fmalita063675b2015-10-12 10:41:48 -0700448 paint2.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000449 paint3.setShader(radial2);
fmalita063675b2015-10-12 10:41:48 -0700450 paint3.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000451
452 canvas->drawCircle(cx, cy, radius, paint1);
453 canvas->drawCircle(cx, cy, radius, paint3);
454 canvas->drawCircle(cx, cy, radius, paint2);
455
456 canvas->translate(400, 0);
457 }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000458 }
459
460private:
fmalita063675b2015-10-12 10:41:48 -0700461 bool fDither;
462
mtklein@google.com361a72f2013-08-19 18:43:34 +0000463 typedef GM INHERITED;
464};
fmalita063675b2015-10-12 10:41:48 -0700465DEF_GM( return new RadialGradient2GM(true); )
466DEF_GM( return new RadialGradient2GM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000467
reed3d9005c2015-04-23 10:30:27 -0700468// Shallow radial (shows banding on raster)
469class RadialGradient3GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700470public:
471 RadialGradient3GM(bool dither) : fDither(dither) { }
reed@android.com42309d42009-06-22 02:06:35 +0000472
reed3d9005c2015-04-23 10:30:27 -0700473protected:
fmalita063675b2015-10-12 10:41:48 -0700474 SkString onShortName() override {
475 return SkString(fDither ? "radial_gradient3" : "radial_gradient3_nodither");
476 }
reed@android.com42309d42009-06-22 02:06:35 +0000477
reed3d9005c2015-04-23 10:30:27 -0700478 SkISize onISize() override { return SkISize::Make(500, 500); }
reed@google.comac864a92011-06-27 18:11:17 +0000479
reed3d9005c2015-04-23 10:30:27 -0700480 bool runAsBench() const override { return true; }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000481
reed3d9005c2015-04-23 10:30:27 -0700482 void onOnceBeforeDraw() override {
483 const SkPoint center = { 0, 0 };
484 const SkScalar kRadius = 3000;
485 const SkColor gColors[] = { 0xFFFFFFFF, 0xFF000000 };
halcanary96fcdcc2015-08-27 07:41:13 -0700486 fShader.reset(SkGradientShader::CreateRadial(center, kRadius, gColors, nullptr, 2,
mtklein1113da72015-04-27 12:08:01 -0700487 SkShader::kClamp_TileMode));
reed3d9005c2015-04-23 10:30:27 -0700488 }
rileya@google.com5cf2c282012-07-09 14:42:16 +0000489
reed3d9005c2015-04-23 10:30:27 -0700490 void onDraw(SkCanvas* canvas) override {
491 SkPaint paint;
492 paint.setShader(fShader);
fmalita063675b2015-10-12 10:41:48 -0700493 paint.setDither(fDither);
reed3d9005c2015-04-23 10:30:27 -0700494 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
495 }
496
497private:
fmalita063675b2015-10-12 10:41:48 -0700498 SkAutoTUnref<SkShader> fShader;
499 bool fDither;
500
reed3d9005c2015-04-23 10:30:27 -0700501 typedef GM INHERITED;
502};
fmalita063675b2015-10-12 10:41:48 -0700503DEF_GM( return new RadialGradient3GM(true); )
504DEF_GM( return new RadialGradient3GM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000505
caryclark1864bfa2015-07-30 06:41:39 -0700506class RadialGradient4GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700507public:
508 RadialGradient4GM(bool dither) : fDither(dither) { }
caryclark1864bfa2015-07-30 06:41:39 -0700509
510protected:
fmalita063675b2015-10-12 10:41:48 -0700511 SkString onShortName() override {
512 return SkString(fDither ? "radial_gradient4" : "radial_gradient4_nodither");
513 }
caryclark1864bfa2015-07-30 06:41:39 -0700514
515 SkISize onISize() override { return SkISize::Make(500, 500); }
516
517 void onOnceBeforeDraw() override {
518 const SkPoint center = { 250, 250 };
519 const SkScalar kRadius = 250;
520 const SkColor colors[] = { SK_ColorRED, SK_ColorRED, SK_ColorWHITE, SK_ColorWHITE,
521 SK_ColorRED };
522 const SkScalar pos[] = { 0, .4f, .4f, .8f, .8f, 1 };
523 fShader.reset(SkGradientShader::CreateRadial(center, kRadius, colors, pos,
524 SK_ARRAY_COUNT(gColors), SkShader::kClamp_TileMode));
525 }
526
527 void onDraw(SkCanvas* canvas) override {
528 SkPaint paint;
529 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700530 paint.setDither(fDither);
caryclark1864bfa2015-07-30 06:41:39 -0700531 paint.setShader(fShader);
532 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
533 }
534
535private:
fmalita063675b2015-10-12 10:41:48 -0700536 SkAutoTUnref<SkShader> fShader;
537 bool fDither;
538
caryclark1864bfa2015-07-30 06:41:39 -0700539 typedef GM INHERITED;
540};
fmalita063675b2015-10-12 10:41:48 -0700541DEF_GM( return new RadialGradient4GM(true); )
542DEF_GM( return new RadialGradient4GM(false); )
caryclark1864bfa2015-07-30 06:41:39 -0700543
caryclark159fa572015-07-30 12:35:48 -0700544class LinearGradientGM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700545public:
546 LinearGradientGM(bool dither) : fDither(dither) { }
caryclark159fa572015-07-30 12:35:48 -0700547
548protected:
fmalita063675b2015-10-12 10:41:48 -0700549 SkString onShortName() override {
550 return SkString(fDither ? "linear_gradient" : "linear_gradient_nodither");
551 }
552
caryclark159fa572015-07-30 12:35:48 -0700553 const SkScalar kWidthBump = 30.f;
554 const SkScalar kHeight = 5.f;
555 const SkScalar kMinWidth = 540.f;
556
557 SkISize onISize() override { return SkISize::Make(500, 500); }
558
559 void onOnceBeforeDraw() override {
560 SkPoint pts[2] = { {0, 0}, {0, 0} };
561 const SkColor colors[] = { SK_ColorWHITE, SK_ColorWHITE, 0xFF008200, 0xFF008200,
562 SK_ColorWHITE, SK_ColorWHITE };
563 const SkScalar unitPos[] = { 0, 50, 70, 500, 540 };
564 SkScalar pos[6];
565 pos[5] = 1;
566 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
567 pts[1].fX = 500.f + index * kWidthBump;
568 for (int inner = 0; inner < (int) SK_ARRAY_COUNT(unitPos); ++inner) {
569 pos[inner] = unitPos[inner] / (kMinWidth + index * kWidthBump);
570 }
571 fShader[index].reset(SkGradientShader::CreateLinear(pts, colors, pos,
572 SK_ARRAY_COUNT(gColors), SkShader::kClamp_TileMode));
573 }
574 }
575
576 void onDraw(SkCanvas* canvas) override {
577 SkPaint paint;
578 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700579 paint.setDither(fDither);
caryclark159fa572015-07-30 12:35:48 -0700580 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
581 paint.setShader(fShader[index]);
582 canvas->drawRect(SkRect::MakeLTRB(0, index * kHeight, kMinWidth + index * kWidthBump,
583 (index + 1) * kHeight), paint);
584 }
585 }
586
587private:
fmalita063675b2015-10-12 10:41:48 -0700588 SkAutoTUnref<SkShader> fShader[100];
589 bool fDither;
590
caryclark159fa572015-07-30 12:35:48 -0700591 typedef GM INHERITED;
592};
fmalita063675b2015-10-12 10:41:48 -0700593DEF_GM( return new LinearGradientGM(true); )
594DEF_GM( return new LinearGradientGM(false); )
caryclark159fa572015-07-30 12:35:48 -0700595
fmalita8b78bd62015-11-20 13:58:24 -0800596class LinearGradientTinyGM : public GM {
fmalitabc590c02016-02-22 09:12:33 -0800597public:
598 LinearGradientTinyGM(uint32_t flags, const char* suffix = nullptr)
599 : fName("linear_gradient_tiny")
600 , fFlags(flags) {
601 fName.append(suffix);
602 }
603
fmalita8b78bd62015-11-20 13:58:24 -0800604protected:
605 SkString onShortName() override {
fmalitabc590c02016-02-22 09:12:33 -0800606 return fName;
fmalita8b78bd62015-11-20 13:58:24 -0800607 }
608
609 SkISize onISize() override {
610 return SkISize::Make(600, 500);
611 }
612
613 void onDraw(SkCanvas* canvas) override {
614 const SkScalar kRectSize = 100;
615 const unsigned kStopCount = 3;
616 const SkColor colors[kStopCount] = { SK_ColorGREEN, SK_ColorRED, SK_ColorGREEN };
617 const struct {
618 SkPoint pts[2];
619 SkScalar pos[kStopCount];
620 } configs[] = {
reedde3aac82015-11-22 13:00:04 -0800621 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999f, 1 }},
622 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000001f, 1 }},
623 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999999f, 1 }},
624 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000000001f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800625
reedde3aac82015-11-22 13:00:04 -0800626 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999f, 1 }},
627 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000001f, 1 }},
628 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999999f, 1 }},
629 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000000001f, 1 }},
630
631 { { SkPoint::Make(0, 0), SkPoint::Make(0.00001f, 0) }, { 0, 0.5f, 1 }},
632 { { SkPoint::Make(9.99999f, 0), SkPoint::Make(10, 0) }, { 0, 0.5f, 1 }},
633 { { SkPoint::Make(0, 0), SkPoint::Make(0, 0.00001f) }, { 0, 0.5f, 1 }},
634 { { SkPoint::Make(0, 9.99999f), SkPoint::Make(0, 10) }, { 0, 0.5f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800635 };
636
637 SkPaint paint;
638 for (unsigned i = 0; i < SK_ARRAY_COUNT(configs); ++i) {
639 SkAutoCanvasRestore acr(canvas, true);
640 SkAutoTUnref<SkShader> gradient(
641 SkGradientShader::CreateLinear(configs[i].pts, colors, configs[i].pos, kStopCount,
fmalitabc590c02016-02-22 09:12:33 -0800642 SkShader::kClamp_TileMode, fFlags, nullptr));
fmalita8b78bd62015-11-20 13:58:24 -0800643 canvas->translate(kRectSize * ((i % 4) * 1.5f + 0.25f),
644 kRectSize * ((i / 4) * 1.5f + 0.25f));
645
646 paint.setShader(gradient);
647 canvas->drawRect(SkRect::MakeWH(kRectSize, kRectSize), paint);
648 }
649 }
650
651private:
652 typedef GM INHERITED;
fmalita8b78bd62015-11-20 13:58:24 -0800653
fmalitabc590c02016-02-22 09:12:33 -0800654 SkString fName;
655 uint32_t fFlags;
656};
657DEF_GM( return new LinearGradientTinyGM(0); )
658DEF_GM( return new LinearGradientTinyGM(SkLinearGradient::kForce4fContext_PrivateFlag, "_4f"); )
reed@android.com42309d42009-06-22 02:06:35 +0000659}
reedd4eaa252016-01-22 10:35:26 -0800660
661///////////////////////////////////////////////////////////////////////////////////////////////////
662
663struct GradRun {
664 SkColor fColors[4];
665 SkScalar fPos[4];
666 int fCount;
667};
668
669#define SIZE 121
670
671static SkShader* make_linear(const GradRun& run, SkShader::TileMode mode) {
672 const SkPoint pts[] { { 30, 30 }, { SIZE - 30, SIZE - 30 } };
673 return SkGradientShader::CreateLinear(pts, run.fColors, run.fPos, run.fCount, mode);
674}
675
676static SkShader* make_radial(const GradRun& run, SkShader::TileMode mode) {
677 const SkScalar half = SIZE * 0.5f;
678 return SkGradientShader::CreateRadial({half,half}, half - 10,
679 run.fColors, run.fPos, run.fCount, mode);
680}
681
682static SkShader* make_conical(const GradRun& run, SkShader::TileMode mode) {
683 const SkScalar half = SIZE * 0.5f;
684 const SkPoint center { half, half };
685 return SkGradientShader::CreateTwoPointConical(center, 20, center, half - 10,
686 run.fColors, run.fPos, run.fCount, mode);
687}
688
689static SkShader* make_sweep(const GradRun& run, SkShader::TileMode) {
690 const SkScalar half = SIZE * 0.5f;
691 return SkGradientShader::CreateSweep(half, half, run.fColors, run.fPos, run.fCount);
692}
693
694/*
695 * Exercise duplicate color-stops, at the ends, and in the middle
696 *
697 * At the time of this writing, only Linear correctly deals with duplicates at the ends,
698 * and then only correctly on CPU backend.
699 */
700DEF_SIMPLE_GM(gradients_dup_color_stops, canvas, 704, 564) {
701 const SkColor preColor = 0xFFFF0000; // clamp color before start
702 const SkColor postColor = 0xFF0000FF; // clamp color after end
703 const SkColor color0 = 0xFF000000;
704 const SkColor color1 = 0xFF00FF00;
705 const SkColor badColor = 0xFF3388BB; // should never be seen, fills out fixed-size array
706
707 const GradRun runs[] = {
708 { { color0, color1, badColor, badColor },
709 { 0, 1, -1, -1 },
710 2,
711 },
712 { { preColor, color0, color1, badColor },
713 { 0, 0, 1, -1 },
714 3,
715 },
716 { { color0, color1, postColor, badColor },
717 { 0, 1, 1, -1 },
718 3,
719 },
720 { { preColor, color0, color1, postColor },
721 { 0, 0, 1, 1 },
722 4,
723 },
724 { { color0, color0, color1, color1 },
725 { 0, 0.5f, 0.5f, 1 },
726 4,
727 },
728 };
729 SkShader* (*factories[])(const GradRun&, SkShader::TileMode) {
730 make_linear, make_radial, make_conical, make_sweep
731 };
732
733 const SkRect rect = SkRect::MakeWH(SIZE, SIZE);
734 const SkScalar dx = SIZE + 20;
735 const SkScalar dy = SIZE + 20;
736 const SkShader::TileMode mode = SkShader::kClamp_TileMode;
737
738 SkPaint paint;
739 canvas->translate(10, 10 - dy);
740 for (auto factory : factories) {
741 canvas->translate(0, dy);
742 SkAutoCanvasRestore acr(canvas, true);
743 for (const auto& run : runs) {
744 paint.setShader(factory(run, mode))->unref();
745 canvas->drawRect(rect, paint);
746 canvas->translate(dx, 0);
747 }
748 }
749}
fmalitabc590c02016-02-22 09:12:33 -0800750
751static void draw_many_stops(SkCanvas* canvas, uint32_t flags) {
752 const unsigned kStopCount = 200;
753 const SkPoint pts[] = { {50, 50}, {450, 465}};
754
755 SkColor colors[kStopCount];
756 for (unsigned i = 0; i < kStopCount; i++) {
757 switch (i % 5) {
758 case 0: colors[i] = SK_ColorRED; break;
759 case 1: colors[i] = SK_ColorGREEN; break;
760 case 2: colors[i] = SK_ColorGREEN; break;
761 case 3: colors[i] = SK_ColorBLUE; break;
762 case 4: colors[i] = SK_ColorRED; break;
763 }
764 }
765
766 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateLinear(
767 pts, colors, nullptr, SK_ARRAY_COUNT(colors), SkShader::kClamp_TileMode, flags, nullptr));
768
769 SkPaint p;
770 p.setShader(shader);
771
772 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
773}
774
775DEF_SIMPLE_GM(gradient_many_stops, canvas, 500, 500) {
776 draw_many_stops(canvas, 0);
777}
778
779DEF_SIMPLE_GM(gradient_many_stops_4f, canvas, 500, 500) {
780 draw_many_stops(canvas, SkLinearGradient::kForce4fContext_PrivateFlag);
781}