blob: eeb5bcb9ff86ccba7e8116b6e501667b12bff702 [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
caryclarkcb100712016-02-26 05:59:40 -0800307/* bug.skia.org/517
308<canvas id="canvas"></canvas>
309<script>
310var c = document.getElementById("canvas");
311var ctx = c.getContext("2d");
312ctx.fillStyle = '#ff0';
313ctx.fillRect(0, 0, 100, 50);
314
315var g = ctx.createRadialGradient(200, 25, 20, 200, 25, 10);
316g.addColorStop(0, '#0f0');
317g.addColorStop(0.003, '#f00'); // 0.004 makes this work
318g.addColorStop(1, '#ff0');
319ctx.fillStyle = g;
320ctx.fillRect(0, 0, 100, 50);
321</script>
322*/
323
324// should draw only green
325DEF_SIMPLE_GM(small_color_stop, canvas, 100, 150) {
326 SkColor colors[] = { SK_ColorGREEN, SK_ColorRED, SK_ColorYELLOW };
327 SkScalar pos[] = { 0, 0.003f, SK_Scalar1 }; // 0.004f makes this work
328 SkPoint c0 = { 200, 25 };
329 SkScalar r0 = 20;
330 SkPoint c1 = { 200, 25 };
331 SkScalar r1 = 10;
332 SkShader* s = SkGradientShader::CreateTwoPointConical(c0, r0, c1, r1, colors,
333 pos, SK_ARRAY_COUNT(pos),
334 SkShader::kClamp_TileMode);
335 SkPaint paint;
336 paint.setColor(SK_ColorYELLOW);
337 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
338 paint.setShader(s)->unref();
339 canvas->drawRect(SkRect::MakeWH(100, 150), paint);
340}
341
342
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000343/// Tests correctness of *optimized* codepaths in gradients.
344
345class ClampedGradientsGM : public GM {
346public:
fmalita063675b2015-10-12 10:41:48 -0700347 ClampedGradientsGM(bool dither) : fDither(dither) {}
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000348
349protected:
fmalita063675b2015-10-12 10:41:48 -0700350 SkString onShortName() {
351 return SkString(fDither ? "clamped_gradients" : "clamped_gradients_nodither");
352 }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000353
edisonn@google.com1da3a802013-09-19 17:55:49 +0000354 virtual SkISize onISize() { return SkISize::Make(640, 510); }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000355
356 void drawBG(SkCanvas* canvas) {
caryclark12596012015-07-29 05:27:47 -0700357 canvas->drawColor(sk_tool_utils::color_to_565(0xFFDDDDDD));
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000358 }
359
360 virtual void onDraw(SkCanvas* canvas) {
361 this->drawBG(canvas);
362
363 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(300) };
364 SkPaint paint;
fmalita063675b2015-10-12 10:41:48 -0700365 paint.setDither(fDither);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000366 paint.setAntiAlias(true);
367
368 SkPoint center;
369 center.iset(0, 300);
370 canvas->translate(SkIntToScalar(20), SkIntToScalar(20));
371 SkShader* shader = SkGradientShader::CreateRadial(
372 SkPoint(center),
halcanary96fcdcc2015-08-27 07:41:13 -0700373 SkIntToScalar(200), gColors, nullptr, 5,
commit-bot@chromium.org83f23d82014-05-22 12:27:41 +0000374 SkShader::kClamp_TileMode);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000375 paint.setShader(shader);
376 canvas->drawRect(r, paint);
377 shader->unref();
378 }
379
380private:
fmalita063675b2015-10-12 10:41:48 -0700381 bool fDither;
382
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000383 typedef GM INHERITED;
384};
fmalita063675b2015-10-12 10:41:48 -0700385DEF_GM( return new ClampedGradientsGM(true); )
386DEF_GM( return new ClampedGradientsGM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000387
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000388/// Checks quality of large radial gradients, which may display
389/// some banding.
390
391class RadialGradientGM : public GM {
392public:
393 RadialGradientGM() {}
394
395protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000396
mtklein36352bf2015-03-25 18:17:31 -0700397 SkString onShortName() override { return SkString("radial_gradient"); }
398 SkISize onISize() override { return SkISize::Make(1280, 1280); }
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000399 void drawBG(SkCanvas* canvas) {
400 canvas->drawColor(0xFF000000);
401 }
mtklein36352bf2015-03-25 18:17:31 -0700402 void onDraw(SkCanvas* canvas) override {
reed@google.combb0948f2012-01-31 14:44:13 +0000403 const SkISize dim = this->getISize();
404
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000405 this->drawBG(canvas);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000406
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000407 SkPaint paint;
408 paint.setDither(true);
409 SkPoint center;
reed@google.combb0948f2012-01-31 14:44:13 +0000410 center.set(SkIntToScalar(dim.width())/2, SkIntToScalar(dim.height())/2);
411 SkScalar radius = SkIntToScalar(dim.width())/2;
412 const SkColor colors[] = { 0x7f7f7f7f, 0x7f7f7f7f, 0xb2000000 };
commit-bot@chromium.org4b413c82013-11-25 19:44:07 +0000413 const SkScalar pos[] = { 0.0f,
414 0.35f,
415 1.0f };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000416 SkShader* shader =
417 SkGradientShader::CreateRadial(center, radius, colors,
reed@google.combb0948f2012-01-31 14:44:13 +0000418 pos, SK_ARRAY_COUNT(pos),
419 SkShader::kClamp_TileMode);
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000420 paint.setShader(shader)->unref();
reed@google.combb0948f2012-01-31 14:44:13 +0000421 SkRect r = {
422 0, 0, SkIntToScalar(dim.width()), SkIntToScalar(dim.height())
423 };
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000424 canvas->drawRect(r, paint);
425 }
426private:
427 typedef GM INHERITED;
428};
reed3d9005c2015-04-23 10:30:27 -0700429DEF_GM( return new RadialGradientGM; )
tomhudson@google.comb18e58c2012-01-30 20:00:13 +0000430
mtklein@google.com361a72f2013-08-19 18:43:34 +0000431class RadialGradient2GM : public GM {
432public:
fmalita063675b2015-10-12 10:41:48 -0700433 RadialGradient2GM(bool dither) : fDither(dither) {}
mtklein@google.com361a72f2013-08-19 18:43:34 +0000434
435protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000436
fmalita063675b2015-10-12 10:41:48 -0700437 SkString onShortName() override {
438 return SkString(fDither ? "radial_gradient2" : "radial_gradient2_nodither");
439 }
440
mtklein36352bf2015-03-25 18:17:31 -0700441 SkISize onISize() override { return SkISize::Make(800, 400); }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000442 void drawBG(SkCanvas* canvas) {
443 canvas->drawColor(0xFF000000);
444 }
445
446 // Reproduces the example given in bug 7671058.
mtklein36352bf2015-03-25 18:17:31 -0700447 void onDraw(SkCanvas* canvas) override {
mtklein@google.com361a72f2013-08-19 18:43:34 +0000448 SkPaint paint1, paint2, paint3;
449 paint1.setStyle(SkPaint::kFill_Style);
450 paint2.setStyle(SkPaint::kFill_Style);
451 paint3.setStyle(SkPaint::kFill_Style);
452
453 const SkColor sweep_colors[] =
454 { 0xFFFF0000, 0xFFFFFF00, 0xFF00FF00, 0xFF00FFFF, 0xFF0000FF, 0xFFFF00FF, 0xFFFF0000 };
455 const SkColor colors1[] = { 0xFFFFFFFF, 0x00000000 };
456 const SkColor colors2[] = { 0xFF000000, 0x00000000 };
457
458 const SkScalar cx = 200, cy = 200, radius = 150;
459 SkPoint center;
460 center.set(cx, cy);
461
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000462 // We can either interpolate endpoints and premultiply each point (default, more precision),
463 // or premultiply the endpoints first, avoiding the need to premultiply each point (cheap).
464 const uint32_t flags[] = { 0, SkGradientShader::kInterpolateColorsInPremul_Flag };
mtklein@google.com361a72f2013-08-19 18:43:34 +0000465
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000466 for (size_t i = 0; i < SK_ARRAY_COUNT(flags); i++) {
467 SkAutoTUnref<SkShader> sweep(
468 SkGradientShader::CreateSweep(cx, cy, sweep_colors,
halcanary96fcdcc2015-08-27 07:41:13 -0700469 nullptr, SK_ARRAY_COUNT(sweep_colors),
470 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000471 SkAutoTUnref<SkShader> radial1(
472 SkGradientShader::CreateRadial(center, radius, colors1,
halcanary96fcdcc2015-08-27 07:41:13 -0700473 nullptr, SK_ARRAY_COUNT(colors1),
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000474 SkShader::kClamp_TileMode,
halcanary96fcdcc2015-08-27 07:41:13 -0700475 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000476 SkAutoTUnref<SkShader> radial2(
477 SkGradientShader::CreateRadial(center, radius, colors2,
halcanary96fcdcc2015-08-27 07:41:13 -0700478 nullptr, SK_ARRAY_COUNT(colors2),
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000479 SkShader::kClamp_TileMode,
halcanary96fcdcc2015-08-27 07:41:13 -0700480 flags[i], nullptr));
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000481 paint1.setShader(sweep);
fmalita063675b2015-10-12 10:41:48 -0700482 paint1.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000483 paint2.setShader(radial1);
fmalita063675b2015-10-12 10:41:48 -0700484 paint2.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000485 paint3.setShader(radial2);
fmalita063675b2015-10-12 10:41:48 -0700486 paint3.setDither(fDither);
mtklein@google.com3ef7eea2013-09-16 13:02:52 +0000487
488 canvas->drawCircle(cx, cy, radius, paint1);
489 canvas->drawCircle(cx, cy, radius, paint3);
490 canvas->drawCircle(cx, cy, radius, paint2);
491
492 canvas->translate(400, 0);
493 }
mtklein@google.com361a72f2013-08-19 18:43:34 +0000494 }
495
496private:
fmalita063675b2015-10-12 10:41:48 -0700497 bool fDither;
498
mtklein@google.com361a72f2013-08-19 18:43:34 +0000499 typedef GM INHERITED;
500};
fmalita063675b2015-10-12 10:41:48 -0700501DEF_GM( return new RadialGradient2GM(true); )
502DEF_GM( return new RadialGradient2GM(false); )
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000503
reed3d9005c2015-04-23 10:30:27 -0700504// Shallow radial (shows banding on raster)
505class RadialGradient3GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700506public:
507 RadialGradient3GM(bool dither) : fDither(dither) { }
reed@android.com42309d42009-06-22 02:06:35 +0000508
reed3d9005c2015-04-23 10:30:27 -0700509protected:
fmalita063675b2015-10-12 10:41:48 -0700510 SkString onShortName() override {
511 return SkString(fDither ? "radial_gradient3" : "radial_gradient3_nodither");
512 }
reed@android.com42309d42009-06-22 02:06:35 +0000513
reed3d9005c2015-04-23 10:30:27 -0700514 SkISize onISize() override { return SkISize::Make(500, 500); }
reed@google.comac864a92011-06-27 18:11:17 +0000515
reed3d9005c2015-04-23 10:30:27 -0700516 bool runAsBench() const override { return true; }
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000517
reed3d9005c2015-04-23 10:30:27 -0700518 void onOnceBeforeDraw() override {
519 const SkPoint center = { 0, 0 };
520 const SkScalar kRadius = 3000;
521 const SkColor gColors[] = { 0xFFFFFFFF, 0xFF000000 };
halcanary96fcdcc2015-08-27 07:41:13 -0700522 fShader.reset(SkGradientShader::CreateRadial(center, kRadius, gColors, nullptr, 2,
mtklein1113da72015-04-27 12:08:01 -0700523 SkShader::kClamp_TileMode));
reed3d9005c2015-04-23 10:30:27 -0700524 }
rileya@google.com5cf2c282012-07-09 14:42:16 +0000525
reed3d9005c2015-04-23 10:30:27 -0700526 void onDraw(SkCanvas* canvas) override {
527 SkPaint paint;
528 paint.setShader(fShader);
fmalita063675b2015-10-12 10:41:48 -0700529 paint.setDither(fDither);
reed3d9005c2015-04-23 10:30:27 -0700530 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
531 }
532
533private:
fmalita063675b2015-10-12 10:41:48 -0700534 SkAutoTUnref<SkShader> fShader;
535 bool fDither;
536
reed3d9005c2015-04-23 10:30:27 -0700537 typedef GM INHERITED;
538};
fmalita063675b2015-10-12 10:41:48 -0700539DEF_GM( return new RadialGradient3GM(true); )
540DEF_GM( return new RadialGradient3GM(false); )
rileya@google.com5cf2c282012-07-09 14:42:16 +0000541
caryclark1864bfa2015-07-30 06:41:39 -0700542class RadialGradient4GM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700543public:
544 RadialGradient4GM(bool dither) : fDither(dither) { }
caryclark1864bfa2015-07-30 06:41:39 -0700545
546protected:
fmalita063675b2015-10-12 10:41:48 -0700547 SkString onShortName() override {
548 return SkString(fDither ? "radial_gradient4" : "radial_gradient4_nodither");
549 }
caryclark1864bfa2015-07-30 06:41:39 -0700550
551 SkISize onISize() override { return SkISize::Make(500, 500); }
552
553 void onOnceBeforeDraw() override {
554 const SkPoint center = { 250, 250 };
555 const SkScalar kRadius = 250;
556 const SkColor colors[] = { SK_ColorRED, SK_ColorRED, SK_ColorWHITE, SK_ColorWHITE,
557 SK_ColorRED };
558 const SkScalar pos[] = { 0, .4f, .4f, .8f, .8f, 1 };
559 fShader.reset(SkGradientShader::CreateRadial(center, kRadius, colors, pos,
560 SK_ARRAY_COUNT(gColors), SkShader::kClamp_TileMode));
561 }
562
563 void onDraw(SkCanvas* canvas) override {
564 SkPaint paint;
565 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700566 paint.setDither(fDither);
caryclark1864bfa2015-07-30 06:41:39 -0700567 paint.setShader(fShader);
568 canvas->drawRect(SkRect::MakeWH(500, 500), paint);
569 }
570
571private:
fmalita063675b2015-10-12 10:41:48 -0700572 SkAutoTUnref<SkShader> fShader;
573 bool fDither;
574
caryclark1864bfa2015-07-30 06:41:39 -0700575 typedef GM INHERITED;
576};
fmalita063675b2015-10-12 10:41:48 -0700577DEF_GM( return new RadialGradient4GM(true); )
578DEF_GM( return new RadialGradient4GM(false); )
caryclark1864bfa2015-07-30 06:41:39 -0700579
caryclark159fa572015-07-30 12:35:48 -0700580class LinearGradientGM : public GM {
fmalita063675b2015-10-12 10:41:48 -0700581public:
582 LinearGradientGM(bool dither) : fDither(dither) { }
caryclark159fa572015-07-30 12:35:48 -0700583
584protected:
fmalita063675b2015-10-12 10:41:48 -0700585 SkString onShortName() override {
586 return SkString(fDither ? "linear_gradient" : "linear_gradient_nodither");
587 }
588
caryclark159fa572015-07-30 12:35:48 -0700589 const SkScalar kWidthBump = 30.f;
590 const SkScalar kHeight = 5.f;
591 const SkScalar kMinWidth = 540.f;
592
593 SkISize onISize() override { return SkISize::Make(500, 500); }
594
595 void onOnceBeforeDraw() override {
596 SkPoint pts[2] = { {0, 0}, {0, 0} };
597 const SkColor colors[] = { SK_ColorWHITE, SK_ColorWHITE, 0xFF008200, 0xFF008200,
598 SK_ColorWHITE, SK_ColorWHITE };
599 const SkScalar unitPos[] = { 0, 50, 70, 500, 540 };
600 SkScalar pos[6];
601 pos[5] = 1;
602 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
603 pts[1].fX = 500.f + index * kWidthBump;
604 for (int inner = 0; inner < (int) SK_ARRAY_COUNT(unitPos); ++inner) {
605 pos[inner] = unitPos[inner] / (kMinWidth + index * kWidthBump);
606 }
607 fShader[index].reset(SkGradientShader::CreateLinear(pts, colors, pos,
608 SK_ARRAY_COUNT(gColors), SkShader::kClamp_TileMode));
609 }
610 }
611
612 void onDraw(SkCanvas* canvas) override {
613 SkPaint paint;
614 paint.setAntiAlias(true);
fmalita063675b2015-10-12 10:41:48 -0700615 paint.setDither(fDither);
caryclark159fa572015-07-30 12:35:48 -0700616 for (int index = 0; index < (int) SK_ARRAY_COUNT(fShader); ++index) {
617 paint.setShader(fShader[index]);
618 canvas->drawRect(SkRect::MakeLTRB(0, index * kHeight, kMinWidth + index * kWidthBump,
619 (index + 1) * kHeight), paint);
620 }
621 }
622
623private:
fmalita063675b2015-10-12 10:41:48 -0700624 SkAutoTUnref<SkShader> fShader[100];
625 bool fDither;
626
caryclark159fa572015-07-30 12:35:48 -0700627 typedef GM INHERITED;
628};
fmalita063675b2015-10-12 10:41:48 -0700629DEF_GM( return new LinearGradientGM(true); )
630DEF_GM( return new LinearGradientGM(false); )
caryclark159fa572015-07-30 12:35:48 -0700631
fmalita8b78bd62015-11-20 13:58:24 -0800632class LinearGradientTinyGM : public GM {
fmalitabc590c02016-02-22 09:12:33 -0800633public:
634 LinearGradientTinyGM(uint32_t flags, const char* suffix = nullptr)
635 : fName("linear_gradient_tiny")
636 , fFlags(flags) {
637 fName.append(suffix);
638 }
639
fmalita8b78bd62015-11-20 13:58:24 -0800640protected:
641 SkString onShortName() override {
fmalitabc590c02016-02-22 09:12:33 -0800642 return fName;
fmalita8b78bd62015-11-20 13:58:24 -0800643 }
644
645 SkISize onISize() override {
646 return SkISize::Make(600, 500);
647 }
648
649 void onDraw(SkCanvas* canvas) override {
650 const SkScalar kRectSize = 100;
651 const unsigned kStopCount = 3;
652 const SkColor colors[kStopCount] = { SK_ColorGREEN, SK_ColorRED, SK_ColorGREEN };
653 const struct {
654 SkPoint pts[2];
655 SkScalar pos[kStopCount];
656 } configs[] = {
reedde3aac82015-11-22 13:00:04 -0800657 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999f, 1 }},
658 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000001f, 1 }},
659 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.999999999f, 1 }},
660 { { SkPoint::Make(0, 0), SkPoint::Make(10, 0) }, { 0, 0.000000001f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800661
reedde3aac82015-11-22 13:00:04 -0800662 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999f, 1 }},
663 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000001f, 1 }},
664 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.999999999f, 1 }},
665 { { SkPoint::Make(0, 0), SkPoint::Make(0, 10) }, { 0, 0.000000001f, 1 }},
666
667 { { SkPoint::Make(0, 0), SkPoint::Make(0.00001f, 0) }, { 0, 0.5f, 1 }},
668 { { SkPoint::Make(9.99999f, 0), SkPoint::Make(10, 0) }, { 0, 0.5f, 1 }},
669 { { SkPoint::Make(0, 0), SkPoint::Make(0, 0.00001f) }, { 0, 0.5f, 1 }},
670 { { SkPoint::Make(0, 9.99999f), SkPoint::Make(0, 10) }, { 0, 0.5f, 1 }},
fmalita8b78bd62015-11-20 13:58:24 -0800671 };
672
673 SkPaint paint;
674 for (unsigned i = 0; i < SK_ARRAY_COUNT(configs); ++i) {
675 SkAutoCanvasRestore acr(canvas, true);
676 SkAutoTUnref<SkShader> gradient(
677 SkGradientShader::CreateLinear(configs[i].pts, colors, configs[i].pos, kStopCount,
fmalitabc590c02016-02-22 09:12:33 -0800678 SkShader::kClamp_TileMode, fFlags, nullptr));
fmalita8b78bd62015-11-20 13:58:24 -0800679 canvas->translate(kRectSize * ((i % 4) * 1.5f + 0.25f),
680 kRectSize * ((i / 4) * 1.5f + 0.25f));
681
682 paint.setShader(gradient);
683 canvas->drawRect(SkRect::MakeWH(kRectSize, kRectSize), paint);
684 }
685 }
686
687private:
688 typedef GM INHERITED;
fmalita8b78bd62015-11-20 13:58:24 -0800689
fmalitabc590c02016-02-22 09:12:33 -0800690 SkString fName;
691 uint32_t fFlags;
692};
693DEF_GM( return new LinearGradientTinyGM(0); )
694DEF_GM( return new LinearGradientTinyGM(SkLinearGradient::kForce4fContext_PrivateFlag, "_4f"); )
reed@android.com42309d42009-06-22 02:06:35 +0000695}
reedd4eaa252016-01-22 10:35:26 -0800696
697///////////////////////////////////////////////////////////////////////////////////////////////////
698
699struct GradRun {
700 SkColor fColors[4];
701 SkScalar fPos[4];
702 int fCount;
703};
704
705#define SIZE 121
706
707static SkShader* make_linear(const GradRun& run, SkShader::TileMode mode) {
708 const SkPoint pts[] { { 30, 30 }, { SIZE - 30, SIZE - 30 } };
709 return SkGradientShader::CreateLinear(pts, run.fColors, run.fPos, run.fCount, mode);
710}
711
712static SkShader* make_radial(const GradRun& run, SkShader::TileMode mode) {
713 const SkScalar half = SIZE * 0.5f;
714 return SkGradientShader::CreateRadial({half,half}, half - 10,
715 run.fColors, run.fPos, run.fCount, mode);
716}
717
718static SkShader* make_conical(const GradRun& run, SkShader::TileMode mode) {
719 const SkScalar half = SIZE * 0.5f;
720 const SkPoint center { half, half };
721 return SkGradientShader::CreateTwoPointConical(center, 20, center, half - 10,
722 run.fColors, run.fPos, run.fCount, mode);
723}
724
725static SkShader* make_sweep(const GradRun& run, SkShader::TileMode) {
726 const SkScalar half = SIZE * 0.5f;
727 return SkGradientShader::CreateSweep(half, half, run.fColors, run.fPos, run.fCount);
728}
729
730/*
731 * Exercise duplicate color-stops, at the ends, and in the middle
732 *
733 * At the time of this writing, only Linear correctly deals with duplicates at the ends,
734 * and then only correctly on CPU backend.
735 */
736DEF_SIMPLE_GM(gradients_dup_color_stops, canvas, 704, 564) {
737 const SkColor preColor = 0xFFFF0000; // clamp color before start
738 const SkColor postColor = 0xFF0000FF; // clamp color after end
739 const SkColor color0 = 0xFF000000;
740 const SkColor color1 = 0xFF00FF00;
741 const SkColor badColor = 0xFF3388BB; // should never be seen, fills out fixed-size array
742
743 const GradRun runs[] = {
744 { { color0, color1, badColor, badColor },
745 { 0, 1, -1, -1 },
746 2,
747 },
748 { { preColor, color0, color1, badColor },
749 { 0, 0, 1, -1 },
750 3,
751 },
752 { { color0, color1, postColor, badColor },
753 { 0, 1, 1, -1 },
754 3,
755 },
756 { { preColor, color0, color1, postColor },
757 { 0, 0, 1, 1 },
758 4,
759 },
760 { { color0, color0, color1, color1 },
761 { 0, 0.5f, 0.5f, 1 },
762 4,
763 },
764 };
765 SkShader* (*factories[])(const GradRun&, SkShader::TileMode) {
766 make_linear, make_radial, make_conical, make_sweep
767 };
768
769 const SkRect rect = SkRect::MakeWH(SIZE, SIZE);
770 const SkScalar dx = SIZE + 20;
771 const SkScalar dy = SIZE + 20;
772 const SkShader::TileMode mode = SkShader::kClamp_TileMode;
773
774 SkPaint paint;
775 canvas->translate(10, 10 - dy);
776 for (auto factory : factories) {
777 canvas->translate(0, dy);
778 SkAutoCanvasRestore acr(canvas, true);
779 for (const auto& run : runs) {
780 paint.setShader(factory(run, mode))->unref();
781 canvas->drawRect(rect, paint);
782 canvas->translate(dx, 0);
783 }
784 }
785}
fmalitabc590c02016-02-22 09:12:33 -0800786
787static void draw_many_stops(SkCanvas* canvas, uint32_t flags) {
788 const unsigned kStopCount = 200;
789 const SkPoint pts[] = { {50, 50}, {450, 465}};
790
791 SkColor colors[kStopCount];
792 for (unsigned i = 0; i < kStopCount; i++) {
793 switch (i % 5) {
794 case 0: colors[i] = SK_ColorRED; break;
795 case 1: colors[i] = SK_ColorGREEN; break;
796 case 2: colors[i] = SK_ColorGREEN; break;
797 case 3: colors[i] = SK_ColorBLUE; break;
798 case 4: colors[i] = SK_ColorRED; break;
799 }
800 }
801
802 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateLinear(
803 pts, colors, nullptr, SK_ARRAY_COUNT(colors), SkShader::kClamp_TileMode, flags, nullptr));
804
805 SkPaint p;
806 p.setShader(shader);
807
808 canvas->drawRect(SkRect::MakeXYWH(0, 0, 500, 500), p);
809}
810
811DEF_SIMPLE_GM(gradient_many_stops, canvas, 500, 500) {
812 draw_many_stops(canvas, 0);
813}
814
815DEF_SIMPLE_GM(gradient_many_stops_4f, canvas, 500, 500) {
816 draw_many_stops(canvas, SkLinearGradient::kForce4fContext_PrivateFlag);
817}