OLD | NEW |
---|---|
(Empty) | |
1 | |
2 /* | |
3 * Copyright 2014 Google Inc. | |
4 * | |
5 * Use of this source code is governed by a BSD-style license that can be | |
6 * found in the LICENSE file. | |
7 */ | |
8 #include "gm.h" | |
9 #include "SkGradientShader.h" | |
10 | |
11 namespace skiagm { | |
12 | |
13 struct GradData { | |
14 int fCount; | |
15 const SkColor* fColors; | |
16 const SkScalar* fPos; | |
17 }; | |
18 | |
19 static const SkColor gColors[] = { | |
20 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK | |
21 }; | |
22 static const SkScalar gPos0[] = { 0, SK_Scalar1 }; | |
23 static const SkScalar gPos1[] = { SK_Scalar1/4, SK_Scalar1*3/4 }; | |
24 static const SkScalar gPos2[] = { | |
25 0, SK_Scalar1/8, SK_Scalar1/2, SK_Scalar1*7/8, SK_Scalar1 | |
26 }; | |
27 | |
28 static const SkScalar gPosClamp[] = {0.0f, 0.0f, 1.0f, 1.0f}; | |
29 static const SkColor gColorClamp[] = { | |
30 SK_ColorRED, SK_ColorGREEN, SK_ColorGREEN, SK_ColorBLUE | |
31 }; | |
32 | |
33 static const GradData gGradData[] = { | |
34 { 2, gColors, gPos0 }, | |
35 { 2, gColors, gPos1 }, | |
36 { 5, gColors, gPos2 }, | |
37 { 4, gColorClamp, gPosClamp } | |
38 }; | |
39 | |
40 static SkShader* Make2ConicalOutside(const SkPoint pts[2], const GradData& data, | |
41 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
42 SkPoint center0, center1; | |
43 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 10); | |
44 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
45 center0.set(pts[0].fX + radius0, pts[0].fY + radius0); | |
46 center1.set(pts[1].fX - radius1, pts[1].fY - radius1); | |
47 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
48 center1, radius1, | |
49 data.fColors, data.fPos, | |
50 data.fCount, tm, mapper); | |
51 } | |
52 | |
53 static SkShader* Make2ConicalOutsideFlip(const SkPoint pts[2], const GradData& d ata, | |
54 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
55 SkPoint center0, center1; | |
56 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 10); | |
57 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
58 center0.set(pts[0].fX + radius0, pts[0].fY + radius0); | |
59 center1.set(pts[1].fX - radius1, pts[1].fY - radius1); | |
60 return SkGradientShader::CreateTwoPointConical(center1, radius1, | |
61 center0, radius0, | |
62 data.fColors, data.fPos, | |
63 data.fCount, tm, mapper); | |
64 } | |
65 | |
66 static SkShader* Make2ConicalInside(const SkPoint pts[2], const GradData& data, | |
67 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
68 SkPoint center0, center1; | |
69 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
70 SkScalarAve(pts[0].fY, pts[1].fY)); | |
71 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
72 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
73 return SkGradientShader::CreateTwoPointConical( | |
74 center1, (pts[1].fX - pts[0].f X) / 7, | |
75 center0, (pts[1].fX - pts[0].f X) / 2, | |
76 data.fColors, data.fPos, data. fCount, tm, mapper); | |
77 } | |
78 | |
79 static SkShader* Make2ConicalInsideFlip(const SkPoint pts[2], const GradData& da ta, | |
80 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
81 SkPoint center0, center1; | |
82 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
83 SkScalarAve(pts[0].fY, pts[1].fY)); | |
84 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
85 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
86 return SkGradientShader::CreateTwoPointConical( | |
87 center0, (pts[1].fX - pts[0].f X) / 2, | |
88 center1, (pts[1].fX - pts[0].f X) / 7, | |
89 data.fColors, data.fPos, data. fCount, tm, mapper); | |
90 } | |
91 | |
92 static SkShader* Make2ConicalInsideCenter(const SkPoint pts[2], const GradData& data, | |
93 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
94 SkPoint center0, center1; | |
95 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
96 SkScalarAve(pts[0].fY, pts[1].fY)); | |
97 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
98 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
99 return SkGradientShader::CreateTwoPointConical( | |
100 center0, (pts[1].fX - pts[0].f X) / 7, | |
101 center0, (pts[1].fX - pts[0].f X) / 2, | |
102 data.fColors, data.fPos, data. fCount, tm, mapper); | |
103 } | |
104 | |
105 static SkShader* Make2ConicalZeroRad(const SkPoint pts[2], const GradData& data, | |
106 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
107 SkPoint center0, center1; | |
108 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
109 SkScalarAve(pts[0].fY, pts[1].fY)); | |
110 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
111 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
112 return SkGradientShader::CreateTwoPointConical( | |
113 center1, 0.0, | |
114 center0, (pts[1].fX - pts[0].f X) / 2, | |
115 data.fColors, data.fPos, data. fCount, tm, mapper); | |
116 } | |
117 | |
118 static SkShader* Make2ConicalZeroRadFlip(const SkPoint pts[2], const GradData& d ata, | |
119 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
120 SkPoint center0, center1; | |
121 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
122 SkScalarAve(pts[0].fY, pts[1].fY)); | |
123 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
124 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
125 return SkGradientShader::CreateTwoPointConical( | |
126 center1, (pts[1].fX - pts[0].f X) / 2, | |
127 center0, 0.0, | |
128 data.fColors, data.fPos, data. fCount, tm, mapper); | |
129 } | |
130 | |
131 static SkShader* Make2ConicalZeroRadCenter(const SkPoint pts[2], const GradData& data, | |
132 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
133 SkPoint center0, center1; | |
134 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
135 SkScalarAve(pts[0].fY, pts[1].fY)); | |
136 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
137 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
138 return SkGradientShader::CreateTwoPointConical( | |
139 center0, 0.0, | |
140 center0, (pts[1].fX - pts[0].f X) / 2, | |
141 data.fColors, data.fPos, data. fCount, tm, mapper); | |
142 } | |
143 | |
144 static SkShader* Make2ConicalZeroRadOutside(const SkPoint pts[2], const GradData & data, | |
145 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
146 SkPoint center0, center1; | |
147 SkScalar radius0 = 0.0; | |
148 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
149 center0.set(pts[0].fX + radius0, pts[0].fY + radius0); | |
150 center1.set(pts[1].fX - radius1, pts[1].fY - radius1); | |
151 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
152 center1, radius1, | |
153 data.fColors, data.fPos, | |
154 data.fCount, tm, mapper); | |
155 } | |
156 | |
157 static SkShader* Make2ConicalZeroRadFlipOutside(const SkPoint pts[2], const Grad Data& data, | |
158 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
159 SkPoint center0, center1; | |
160 SkScalar radius0 = 0.0; | |
161 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
162 center0.set(pts[0].fX + radius0, pts[0].fY + radius0); | |
163 center1.set(pts[1].fX - radius1, pts[1].fY - radius1); | |
164 return SkGradientShader::CreateTwoPointConical(center1, radius1, | |
165 center0, radius0, | |
166 data.fColors, data.fPos, | |
167 data.fCount, tm, mapper); | |
168 } | |
169 | |
170 static SkShader* Make2ConicalEdgeX(const SkPoint pts[2], const GradData& data, | |
171 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
172 SkPoint center0, center1; | |
173 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 7); | |
174 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
175 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
176 SkScalarAve(pts[0].fY, pts[1].fY)); | |
177 center0.set(center1.fX + radius1, center1.fY); | |
178 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
179 center1, radius1, | |
180 data.fColors, data.fPos, | |
181 data.fCount, tm, mapper); | |
182 } | |
183 | |
184 static SkShader* Make2ConicalEdgeY(const SkPoint pts[2], const GradData& data, | |
185 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
186 SkPoint center0, center1; | |
187 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 7); | |
188 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
189 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
190 SkScalarAve(pts[0].fY, pts[1].fY)); | |
191 center0.set(center1.fX, center1.fY + radius1); | |
192 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
193 center1, radius1, | |
194 data.fColors, data.fPos, | |
195 data.fCount, tm, mapper); | |
196 } | |
197 | |
198 static SkShader* Make2ConicalZeroRadEdgeX(const SkPoint pts[2], const GradData& data, | |
199 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
200 SkPoint center0, center1; | |
201 SkScalar radius0 = 0.0; | |
202 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
203 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
204 SkScalarAve(pts[0].fY, pts[1].fY)); | |
205 center0.set(center1.fX + radius1, center1.fY); | |
206 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
207 center1, radius1, | |
208 data.fColors, data.fPos, | |
209 data.fCount, tm, mapper); | |
210 } | |
211 | |
212 static SkShader* Make2ConicalZeroRadEdgeY(const SkPoint pts[2], const GradData& data, | |
213 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
214 SkPoint center0, center1; | |
215 SkScalar radius0 = 0.0; | |
216 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
217 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
218 SkScalarAve(pts[0].fY, pts[1].fY)); | |
219 center0.set(center1.fX, center1.fY + radius1); | |
220 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
221 center1, radius1, | |
222 data.fColors, data.fPos, | |
223 data.fCount, tm, mapper); | |
224 } | |
225 | |
226 static SkShader* Make2ConicalTouchX(const SkPoint pts[2], const GradData& data, | |
227 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
228 SkPoint center0, center1; | |
229 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 7); | |
230 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
231 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
232 SkScalarAve(pts[0].fY, pts[1].fY)); | |
233 center0.set(center1.fX - radius1 + radius0, center1.fY); | |
234 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
235 center1, radius1, | |
236 data.fColors, data.fPos, | |
237 data.fCount, tm, mapper); | |
238 } | |
239 | |
240 static SkShader* Make2ConicalTouchY(const SkPoint pts[2], const GradData& data, | |
241 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
242 SkPoint center0, center1; | |
243 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 7); | |
244 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3); | |
245 center1.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
246 SkScalarAve(pts[0].fY, pts[1].fY)); | |
247 center0.set(center1.fX, center1.fY + radius1 - radius0); | |
248 return SkGradientShader::CreateTwoPointConical(center0, radius0, | |
249 center1, radius1, | |
250 data.fColors, data.fPos, | |
251 data.fCount, tm, mapper); | |
252 } | |
253 | |
254 static SkShader* Make2ConicalInsideSmallRad(const SkPoint pts[2], const GradData & data, | |
255 SkShader::TileMode tm, SkUnitMapper* mapper) { | |
256 SkPoint center0, center1; | |
257 center0.set(SkScalarAve(pts[0].fX, pts[1].fX), | |
258 SkScalarAve(pts[0].fY, pts[1].fY)); | |
259 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), | |
260 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); | |
261 return SkGradientShader::CreateTwoPointConical( | |
262 center0, 0.0000000000000000001 , | |
263 center0, (pts[1].fX - pts[0].f X) / 2, | |
264 data.fColors, data.fPos, data. fCount, tm, mapper); | |
265 } | |
266 | |
267 typedef SkShader* (*GradMaker)(const SkPoint pts[2], const GradData& data, | |
268 SkShader::TileMode tm, SkUnitMapper* mapper); | |
269 | |
270 static const GradMaker gGradMakersOutside[] = { | |
271 Make2ConicalOutside, Make2ConicalOutsideFlip, | |
272 Make2ConicalZeroRadOutside, Make2ConicalZeroRadFlipOutside | |
273 }; | |
274 | |
275 static const GradMaker gGradMakersInside[] = { | |
276 Make2ConicalInside, Make2ConicalInsideFlip, Make2ConicalInsideCenter, | |
277 Make2ConicalZeroRad, Make2ConicalZeroRadFlip, Make2ConicalZeroRadCenter, | |
278 }; | |
279 | |
280 static const GradMaker gGradMakersEdgeCases[] = { | |
281 Make2ConicalEdgeX, Make2ConicalEdgeY, | |
282 Make2ConicalZeroRadEdgeX, Make2ConicalZeroRadEdgeY, | |
283 Make2ConicalTouchX, Make2ConicalTouchY, | |
284 Make2ConicalInsideSmallRad | |
285 }; | |
286 | |
287 | |
288 static const struct { | |
289 const GradMaker* fMaker; | |
290 const int fCount; | |
291 const char* fName; | |
292 } gGradCases[] = { | |
293 { gGradMakersOutside, 4, "outside" }, | |
bsalomon
2014/04/22 15:19:59
Can 4, 6, and 7 be replaced by SK_ARRAY_COUNT?
| |
294 { gGradMakersInside, 6, "inside" }, | |
295 { gGradMakersEdgeCases, 7, "edge" }, | |
296 }; | |
297 | |
298 enum GradCaseType { // these must match the order in gGradCases | |
299 kOutside_GradCaseType, | |
300 kInside_GradCaseType, | |
301 kEdge_GradCaseType, | |
302 }; | |
303 | |
304 /////////////////////////////////////////////////////////////////////////////// | |
305 | |
306 class ConicalGradientsGM : public GM { | |
307 public: | |
308 ConicalGradientsGM(GradCaseType gradCaseType) { | |
309 this->setBGColor(0xFFDDDDDD); | |
310 fName.printf("gradients_2pt_conical_%s", gGradCases[gradCaseType].fName) ; | |
311 fGradCaseType = gradCaseType; | |
312 } | |
313 | |
314 protected: | |
315 SkString onShortName() { | |
316 return fName; | |
317 } | |
318 | |
319 virtual SkISize onISize() { return SkISize::Make(840, 815); } | |
320 | |
321 virtual void onDraw(SkCanvas* canvas) { | |
322 | |
323 SkPoint pts[2] = { | |
324 { 0, 0 }, | |
325 { SkIntToScalar(100), SkIntToScalar(100) } | |
326 }; | |
327 SkShader::TileMode tm = SkShader::kClamp_TileMode; | |
328 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) }; | |
329 SkPaint paint; | |
330 paint.setAntiAlias(true); | |
331 | |
332 canvas->translate(SkIntToScalar(20), SkIntToScalar(20)); | |
333 | |
334 const GradMaker* gradMaker = gGradCases[fGradCaseType].fMaker; | |
335 const int count = gGradCases[fGradCaseType].fCount; | |
336 | |
337 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) { | |
338 canvas->save(); | |
339 for (int j = 0; j < count; j++) { | |
340 SkShader* shader = gradMaker[j](pts, gGradData[i], tm, NULL); | |
341 | |
342 if (i == 3) { // if the clamp case | |
343 SkMatrix scale; | |
344 scale.setScale(0.5f, 0.5f); | |
345 scale.postTranslate(25.f, 25.f); | |
346 shader->setLocalMatrix(scale); | |
347 } | |
348 | |
349 paint.setShader(shader); | |
350 canvas->drawRect(r, paint); | |
351 shader->unref(); | |
352 canvas->translate(0, SkIntToScalar(120)); | |
353 } | |
354 canvas->restore(); | |
355 canvas->translate(SkIntToScalar(120), 0); | |
356 } | |
357 } | |
358 | |
359 private: | |
360 typedef GM INHERITED; | |
361 | |
362 GradCaseType fGradCaseType; | |
363 SkString fName; | |
364 }; | |
365 /////////////////////////////////////////////////////////////////////////////// | |
366 | |
367 static GM* MyFactory1(void*) { return new ConicalGradientsGM(kInside_GradCaseTyp e); } | |
368 static GMRegistry reg1(MyFactory1); | |
369 | |
370 static GM* MyFactory2(void*) { return new ConicalGradientsGM(kOutside_GradCaseTy pe); } | |
371 static GMRegistry reg2(MyFactory2); | |
372 | |
373 static GM* MyFactory3(void*) { return new ConicalGradientsGM(kEdge_GradCaseType) ; } | |
374 static GMRegistry reg3(MyFactory3); | |
375 } | |
376 | |
OLD | NEW |