1 /*
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 */
7 #include "gm.h"
8 #include "sk_tool_utils.h"
9 #include "SkCanvas.h"
10 #include "SkPaint.h"
11 #include "SkPath.h"
12 #include "SkRandom.h"
13
14 namespace skiagm {
15
16 class DegenerateSegmentsGM : public GM {
17 public:
DegenerateSegmentsGM()18 DegenerateSegmentsGM() {}
19
20 protected:
21 struct PathAndName {
22 SkPath fPath;
23 const char* fName1;
24 const char* fName2;
25 };
26
onShortName()27 SkString onShortName() {
28 return SkString("degeneratesegments");
29 }
30
onISize()31 SkISize onISize() { return SkISize::Make(896, 930); }
32
33 typedef SkPoint (*AddSegmentFunc)(SkPath&, SkPoint&);
34
35 // We need to use explicit commands here, instead of addPath, because we
36 // do not want the moveTo that is added at the beginning of a path to
37 // appear in the appended path.
AddMove(SkPath & path,SkPoint & startPt)38 static SkPoint AddMove(SkPath& path, SkPoint& startPt) {
39 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
40 path.moveTo(moveToPt);
41 return moveToPt;
42 }
43
AddMoveClose(SkPath & path,SkPoint & startPt)44 static SkPoint AddMoveClose(SkPath& path, SkPoint& startPt) {
45 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
46 path.moveTo(moveToPt);
47 path.close();
48 return moveToPt;
49 }
50
AddDegenLine(SkPath & path,SkPoint & startPt)51 static SkPoint AddDegenLine(SkPath& path, SkPoint& startPt) {
52 path.lineTo(startPt);
53 return startPt;
54 }
55
AddMoveDegenLine(SkPath & path,SkPoint & startPt)56 static SkPoint AddMoveDegenLine(SkPath& path, SkPoint& startPt) {
57 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
58 path.moveTo(moveToPt);
59 path.lineTo(moveToPt);
60 return moveToPt;
61 }
62
AddMoveDegenLineClose(SkPath & path,SkPoint & startPt)63 static SkPoint AddMoveDegenLineClose(SkPath& path, SkPoint& startPt) {
64 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
65 path.moveTo(moveToPt);
66 path.lineTo(moveToPt);
67 path.close();
68 return moveToPt;
69 }
70
AddDegenQuad(SkPath & path,SkPoint & startPt)71 static SkPoint AddDegenQuad(SkPath& path, SkPoint& startPt) {
72 path.quadTo(startPt, startPt);
73 return startPt;
74 }
75
AddMoveDegenQuad(SkPath & path,SkPoint & startPt)76 static SkPoint AddMoveDegenQuad(SkPath& path, SkPoint& startPt) {
77 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
78 path.moveTo(moveToPt);
79 path.quadTo(moveToPt, moveToPt);
80 return moveToPt;
81 }
82
AddMoveDegenQuadClose(SkPath & path,SkPoint & startPt)83 static SkPoint AddMoveDegenQuadClose(SkPath& path, SkPoint& startPt) {
84 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
85 path.moveTo(moveToPt);
86 path.quadTo(moveToPt, moveToPt);
87 path.close();
88 return moveToPt;
89 }
90
AddDegenCubic(SkPath & path,SkPoint & startPt)91 static SkPoint AddDegenCubic(SkPath& path, SkPoint& startPt) {
92 path.cubicTo(startPt, startPt, startPt);
93 return startPt;
94 }
95
AddMoveDegenCubic(SkPath & path,SkPoint & startPt)96 static SkPoint AddMoveDegenCubic(SkPath& path, SkPoint& startPt) {
97 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
98 path.moveTo(moveToPt);
99 path.cubicTo(moveToPt, moveToPt, moveToPt);
100 return moveToPt;
101 }
102
AddMoveDegenCubicClose(SkPath & path,SkPoint & startPt)103 static SkPoint AddMoveDegenCubicClose(SkPath& path, SkPoint& startPt) {
104 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
105 path.moveTo(moveToPt);
106 path.cubicTo(moveToPt, moveToPt, moveToPt);
107 path.close();
108 return moveToPt;
109 }
110
AddClose(SkPath & path,SkPoint & startPt)111 static SkPoint AddClose(SkPath& path, SkPoint& startPt) {
112 path.close();
113 return startPt;
114 }
115
AddLine(SkPath & path,SkPoint & startPt)116 static SkPoint AddLine(SkPath& path, SkPoint& startPt) {
117 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
118 path.lineTo(endPt);
119 return endPt;
120 }
121
AddMoveLine(SkPath & path,SkPoint & startPt)122 static SkPoint AddMoveLine(SkPath& path, SkPoint& startPt) {
123 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
124 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
125 path.moveTo(moveToPt);
126 path.lineTo(endPt);
127 return endPt;
128 }
129
AddMoveLineClose(SkPath & path,SkPoint & startPt)130 static SkPoint AddMoveLineClose(SkPath& path, SkPoint& startPt) {
131 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
132 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
133 path.moveTo(moveToPt);
134 path.lineTo(endPt);
135 path.close();
136 return endPt;
137 }
138
AddQuad(SkPath & path,SkPoint & startPt)139 static SkPoint AddQuad(SkPath& path, SkPoint& startPt) {
140 SkPoint midPt = startPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
141 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
142 path.quadTo(midPt, endPt);
143 return endPt;
144 }
145
AddMoveQuad(SkPath & path,SkPoint & startPt)146 static SkPoint AddMoveQuad(SkPath& path, SkPoint& startPt) {
147 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
148 SkPoint midPt = moveToPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
149 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
150 path.moveTo(moveToPt);
151 path.quadTo(midPt, endPt);
152 return endPt;
153 }
154
AddMoveQuadClose(SkPath & path,SkPoint & startPt)155 static SkPoint AddMoveQuadClose(SkPath& path, SkPoint& startPt) {
156 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
157 SkPoint midPt = moveToPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
158 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
159 path.moveTo(moveToPt);
160 path.quadTo(midPt, endPt);
161 path.close();
162 return endPt;
163 }
164
AddCubic(SkPath & path,SkPoint & startPt)165 static SkPoint AddCubic(SkPath& path, SkPoint& startPt) {
166 SkPoint t1Pt = startPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
167 SkPoint t2Pt = startPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
168 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
169 path.cubicTo(t1Pt, t2Pt, endPt);
170 return endPt;
171 }
172
AddMoveCubic(SkPath & path,SkPoint & startPt)173 static SkPoint AddMoveCubic(SkPath& path, SkPoint& startPt) {
174 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
175 SkPoint t1Pt = moveToPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
176 SkPoint t2Pt = moveToPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
177 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
178 path.moveTo(moveToPt);
179 path.cubicTo(t1Pt, t2Pt, endPt);
180 return endPt;
181 }
182
AddMoveCubicClose(SkPath & path,SkPoint & startPt)183 static SkPoint AddMoveCubicClose(SkPath& path, SkPoint& startPt) {
184 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
185 SkPoint t1Pt = moveToPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
186 SkPoint t2Pt = moveToPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
187 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
188 path.moveTo(moveToPt);
189 path.cubicTo(t1Pt, t2Pt, endPt);
190 path.close();
191 return endPt;
192 }
193
drawPath(SkPath & path,SkCanvas * canvas,SkColor color,const SkRect & clip,SkPaint::Cap cap,SkPaint::Join join,SkPaint::Style style,SkPath::FillType fill,SkScalar strokeWidth)194 void drawPath(SkPath& path, SkCanvas* canvas, SkColor color,
195 const SkRect& clip, SkPaint::Cap cap, SkPaint::Join join,
196 SkPaint::Style style, SkPath::FillType fill,
197 SkScalar strokeWidth) {
198 path.setFillType(fill);
199 SkPaint paint;
200 paint.setStrokeCap(cap);
201 paint.setStrokeWidth(strokeWidth);
202 paint.setStrokeJoin(join);
203 paint.setColor(color);
204 paint.setStyle(style);
205 canvas->save();
206 canvas->clipRect(clip);
207 canvas->drawPath(path, paint);
208 canvas->restore();
209 }
210
onDraw(SkCanvas * canvas)211 virtual void onDraw(SkCanvas* canvas) {
212 constexpr AddSegmentFunc gSegmentFunctions[] = {
213 AddMove,
214 AddMoveClose,
215 AddDegenLine,
216 AddMoveDegenLine,
217 AddMoveDegenLineClose,
218 AddDegenQuad,
219 AddMoveDegenQuad,
220 AddMoveDegenQuadClose,
221 AddDegenCubic,
222 AddMoveDegenCubic,
223 AddMoveDegenCubicClose,
224 AddClose,
225 AddLine,
226 AddMoveLine,
227 AddMoveLineClose,
228 AddQuad,
229 AddMoveQuad,
230 AddMoveQuadClose,
231 AddCubic,
232 AddMoveCubic,
233 AddMoveCubicClose
234 };
235 const char* gSegmentNames[] = {
236 "Move",
237 "MoveClose",
238 "DegenLine",
239 "MoveDegenLine",
240 "MoveDegenLineClose",
241 "DegenQuad",
242 "MoveDegenQuad",
243 "MoveDegenQuadClose",
244 "DegenCubic",
245 "MoveDegenCubic",
246 "MoveDegenCubicClose",
247 "Close",
248 "Line",
249 "MoveLine",
250 "MoveLineClose",
251 "Quad",
252 "MoveQuad",
253 "MoveQuadClose",
254 "Cubic",
255 "MoveCubic",
256 "MoveCubicClose"
257 };
258
259 struct FillAndName {
260 SkPath::FillType fFill;
261 const char* fName;
262 };
263 constexpr FillAndName gFills[] = {
264 {SkPath::kWinding_FillType, "Winding"},
265 {SkPath::kEvenOdd_FillType, "Even / Odd"},
266 {SkPath::kInverseWinding_FillType, "Inverse Winding"},
267 {SkPath::kInverseEvenOdd_FillType, "Inverse Even / Odd"}
268 };
269 struct StyleAndName {
270 SkPaint::Style fStyle;
271 const char* fName;
272 };
273 constexpr StyleAndName gStyles[] = {
274 {SkPaint::kFill_Style, "Fill"},
275 {SkPaint::kStroke_Style, "Stroke 10"},
276 {SkPaint::kStrokeAndFill_Style, "Stroke 10 And Fill"}
277 };
278 struct CapAndName {
279 SkPaint::Cap fCap;
280 SkPaint::Join fJoin;
281 const char* fName;
282 };
283 constexpr CapAndName gCaps[] = {
284 {SkPaint::kButt_Cap, SkPaint::kBevel_Join, "Butt"},
285 {SkPaint::kRound_Cap, SkPaint::kRound_Join, "Round"},
286 {SkPaint::kSquare_Cap, SkPaint::kBevel_Join, "Square"}
287 };
288
289 SkPaint titlePaint;
290 titlePaint.setColor(SK_ColorBLACK);
291 titlePaint.setAntiAlias(true);
292 sk_tool_utils::set_portable_typeface(&titlePaint);
293 titlePaint.setTextSize(15 * SK_Scalar1);
294 const char title[] = "Random Paths Drawn Into Rectangle Clips With "
295 "Indicated Style, Fill and Linecaps, "
296 "with Stroke width 6";
297 canvas->drawText(title, strlen(title),
298 20 * SK_Scalar1,
299 20 * SK_Scalar1,
300 titlePaint);
301
302 SkRandom rand;
303 SkRect rect = SkRect::MakeWH(220*SK_Scalar1, 50*SK_Scalar1);
304 canvas->save();
305 canvas->translate(2*SK_Scalar1, 30 * SK_Scalar1); // The title
306 canvas->save();
307 unsigned numSegments = SK_ARRAY_COUNT(gSegmentFunctions);
308 unsigned numCaps = SK_ARRAY_COUNT(gCaps);
309 unsigned numStyles = SK_ARRAY_COUNT(gStyles);
310 unsigned numFills = SK_ARRAY_COUNT(gFills);
311 for (size_t row = 0; row < 6; ++row) {
312 if (0 < row) {
313 canvas->translate(0, rect.height() + 100*SK_Scalar1);
314 }
315 canvas->save();
316 for (size_t column = 0; column < 4; ++column) {
317 if (0 < column) {
318 canvas->translate(rect.width() + 4*SK_Scalar1, 0);
319 }
320
321 SkColor color = sk_tool_utils::color_to_565(0xff007000);
322 StyleAndName style = gStyles[(rand.nextU() >> 16) % numStyles];
323 CapAndName cap = gCaps[(rand.nextU() >> 16) % numCaps];
324 FillAndName fill = gFills[(rand.nextU() >> 16) % numFills];
325 SkPath path;
326 unsigned s1 = (rand.nextU() >> 16) % numSegments;
327 unsigned s2 = (rand.nextU() >> 16) % numSegments;
328 unsigned s3 = (rand.nextU() >> 16) % numSegments;
329 unsigned s4 = (rand.nextU() >> 16) % numSegments;
330 unsigned s5 = (rand.nextU() >> 16) % numSegments;
331 SkPoint pt = SkPoint::Make(10*SK_Scalar1, 0);
332 pt = gSegmentFunctions[s1](path, pt);
333 pt = gSegmentFunctions[s2](path, pt);
334 pt = gSegmentFunctions[s3](path, pt);
335 pt = gSegmentFunctions[s4](path, pt);
336 pt = gSegmentFunctions[s5](path, pt);
337
338 this->drawPath(path, canvas, color, rect,
339 cap.fCap, cap.fJoin, style.fStyle,
340 fill.fFill, SK_Scalar1*6);
341
342 SkPaint rectPaint;
343 rectPaint.setColor(SK_ColorBLACK);
344 rectPaint.setStyle(SkPaint::kStroke_Style);
345 rectPaint.setStrokeWidth(-1);
346 rectPaint.setAntiAlias(true);
347 canvas->drawRect(rect, rectPaint);
348
349 SkPaint labelPaint;
350 labelPaint.setColor(color);
351 labelPaint.setAntiAlias(true);
352 sk_tool_utils::set_portable_typeface(&labelPaint);
353 labelPaint.setTextSize(10 * SK_Scalar1);
354 canvas->drawText(style.fName,
355 strlen(style.fName),
356 0, rect.height() + 12 * SK_Scalar1,
357 labelPaint);
358 canvas->drawText(fill.fName,
359 strlen(fill.fName),
360 0, rect.height() + 24 * SK_Scalar1,
361 labelPaint);
362 canvas->drawText(cap.fName,
363 strlen(cap.fName),
364 0, rect.height() + 36 * SK_Scalar1,
365 labelPaint);
366 canvas->drawText(gSegmentNames[s1],
367 strlen(gSegmentNames[s1]),
368 0, rect.height() + 48 * SK_Scalar1,
369 labelPaint);
370 canvas->drawText(gSegmentNames[s2],
371 strlen(gSegmentNames[s2]),
372 0, rect.height() + 60 * SK_Scalar1,
373 labelPaint);
374 canvas->drawText(gSegmentNames[s3],
375 strlen(gSegmentNames[s3]),
376 0, rect.height() + 72 * SK_Scalar1,
377 labelPaint);
378 canvas->drawText(gSegmentNames[s4],
379 strlen(gSegmentNames[s4]),
380 0, rect.height() + 84 * SK_Scalar1,
381 labelPaint);
382 canvas->drawText(gSegmentNames[s5],
383 strlen(gSegmentNames[s5]),
384 0, rect.height() + 96 * SK_Scalar1,
385 labelPaint);
386 }
387 canvas->restore();
388 }
389 canvas->restore();
390 canvas->restore();
391 }
392
393 private:
394 typedef GM INHERITED;
395 };
396
397 //////////////////////////////////////////////////////////////////////////////
398
MyFactory(void *)399 static GM* MyFactory(void*) { return new DegenerateSegmentsGM; }
400 static GMRegistry reg(MyFactory);
401
402 }
403