blob: c6d048a9058d9fb0637e1c5eb4fff75c289cd4e6 [file] [log] [blame]
schenney@chromium.org4da06ab2011-12-20 15:14:18 +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 */
Ben Wagner7fde8e12019-05-01 17:28:53 -04007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
9#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040010#include "include/core/SkColor.h"
11#include "include/core/SkFont.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkPaint.h"
Mike Reed92f6eb12020-08-25 11:48:41 -040013#include "include/core/SkPathBuilder.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040014#include "include/core/SkPoint.h"
15#include "include/core/SkRect.h"
16#include "include/core/SkScalar.h"
17#include "include/core/SkSize.h"
18#include "include/core/SkString.h"
19#include "include/core/SkTypeface.h"
20#include "include/core/SkTypes.h"
Kevin Lubick0d4d1142023-02-13 09:13:10 -050021#include "src/base/SkRandom.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "tools/ToolUtils.h"
Kevin Lubicke836c3a2023-10-20 06:55:35 -040023#include "tools/fonts/FontToolUtils.h"
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000024
25namespace skiagm {
26
27class DegenerateSegmentsGM : public GM {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000028 struct PathAndName {
29 SkPath fPath;
30 const char* fName1;
31 const char* fName2;
32 };
33
Leandro Lovisolo24fa2112023-08-15 19:05:17 +000034 SkString getName() const override { return SkString("degeneratesegments"); }
rmistry@google.comd6176b02012-08-23 18:14:13 +000035
Leandro Lovisolo8f023882023-08-15 21:13:52 +000036 SkISize getISize() override { return {896, 930}; }
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000037
Mike Reed92f6eb12020-08-25 11:48:41 -040038 typedef SkPoint (*AddSegmentFunc)(SkPathBuilder&, SkPoint&);
rmistry@google.comd6176b02012-08-23 18:14:13 +000039
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000040 // We need to use explicit commands here, instead of addPath, because we
schenney@chromium.org6630d8d2012-01-04 21:05:51 +000041 // do not want the moveTo that is added at the beginning of a path to
42 // appear in the appended path.
Mike Reed92f6eb12020-08-25 11:48:41 -040043 static SkPoint AddMove(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000044 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
45 path.moveTo(moveToPt);
46 return moveToPt;
47 }
48
Mike Reed92f6eb12020-08-25 11:48:41 -040049 static SkPoint AddMoveClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000050 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
51 path.moveTo(moveToPt);
52 path.close();
53 return moveToPt;
54 }
55
Mike Reed92f6eb12020-08-25 11:48:41 -040056 static SkPoint AddDegenLine(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000057 path.lineTo(startPt);
58 return startPt;
59 }
60
Mike Reed92f6eb12020-08-25 11:48:41 -040061 static SkPoint AddMoveDegenLine(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000062 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
63 path.moveTo(moveToPt);
64 path.lineTo(moveToPt);
65 return moveToPt;
66 }
67
Mike Reed92f6eb12020-08-25 11:48:41 -040068 static SkPoint AddMoveDegenLineClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000069 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
70 path.moveTo(moveToPt);
71 path.lineTo(moveToPt);
72 path.close();
73 return moveToPt;
74 }
75
Mike Reed92f6eb12020-08-25 11:48:41 -040076 static SkPoint AddDegenQuad(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000077 path.quadTo(startPt, startPt);
78 return startPt;
79 }
80
Mike Reed92f6eb12020-08-25 11:48:41 -040081 static SkPoint AddMoveDegenQuad(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000082 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
83 path.moveTo(moveToPt);
84 path.quadTo(moveToPt, moveToPt);
85 return moveToPt;
86 }
87
Mike Reed92f6eb12020-08-25 11:48:41 -040088 static SkPoint AddMoveDegenQuadClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000089 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
90 path.moveTo(moveToPt);
91 path.quadTo(moveToPt, moveToPt);
92 path.close();
93 return moveToPt;
94 }
95
Mike Reed92f6eb12020-08-25 11:48:41 -040096 static SkPoint AddDegenCubic(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +000097 path.cubicTo(startPt, startPt, startPt);
98 return startPt;
99 }
100
Mike Reed92f6eb12020-08-25 11:48:41 -0400101 static SkPoint AddMoveDegenCubic(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000102 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
103 path.moveTo(moveToPt);
104 path.cubicTo(moveToPt, moveToPt, moveToPt);
105 return moveToPt;
106 }
107
Mike Reed92f6eb12020-08-25 11:48:41 -0400108 static SkPoint AddMoveDegenCubicClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000109 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
110 path.moveTo(moveToPt);
111 path.cubicTo(moveToPt, moveToPt, moveToPt);
112 path.close();
113 return moveToPt;
114 }
115
Mike Reed92f6eb12020-08-25 11:48:41 -0400116 static SkPoint AddClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000117 path.close();
118 return startPt;
119 }
120
Mike Reed92f6eb12020-08-25 11:48:41 -0400121 static SkPoint AddLine(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000122 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
123 path.lineTo(endPt);
124 return endPt;
125 }
126
Mike Reed92f6eb12020-08-25 11:48:41 -0400127 static SkPoint AddMoveLine(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000128 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
129 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
130 path.moveTo(moveToPt);
131 path.lineTo(endPt);
132 return endPt;
133 }
134
Mike Reed92f6eb12020-08-25 11:48:41 -0400135 static SkPoint AddMoveLineClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000136 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
137 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
138 path.moveTo(moveToPt);
139 path.lineTo(endPt);
140 path.close();
141 return endPt;
142 }
143
Mike Reed92f6eb12020-08-25 11:48:41 -0400144 static SkPoint AddQuad(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000145 SkPoint midPt = startPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
146 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
147 path.quadTo(midPt, endPt);
148 return endPt;
149 }
150
Mike Reed92f6eb12020-08-25 11:48:41 -0400151 static SkPoint AddMoveQuad(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000152 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
153 SkPoint midPt = moveToPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
154 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
155 path.moveTo(moveToPt);
156 path.quadTo(midPt, endPt);
157 return endPt;
158 }
159
Mike Reed92f6eb12020-08-25 11:48:41 -0400160 static SkPoint AddMoveQuadClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000161 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
162 SkPoint midPt = moveToPt + SkPoint::Make(20*SK_Scalar1, 5*SK_Scalar1);
163 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
164 path.moveTo(moveToPt);
165 path.quadTo(midPt, endPt);
166 path.close();
167 return endPt;
168 }
169
Mike Reed92f6eb12020-08-25 11:48:41 -0400170 static SkPoint AddCubic(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000171 SkPoint t1Pt = startPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
172 SkPoint t2Pt = startPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
173 SkPoint endPt = startPt + SkPoint::Make(40*SK_Scalar1, 0);
174 path.cubicTo(t1Pt, t2Pt, endPt);
175 return endPt;
176 }
177
Mike Reed92f6eb12020-08-25 11:48:41 -0400178 static SkPoint AddMoveCubic(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000179 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
180 SkPoint t1Pt = moveToPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
181 SkPoint t2Pt = moveToPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
182 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
183 path.moveTo(moveToPt);
184 path.cubicTo(t1Pt, t2Pt, endPt);
185 return endPt;
186 }
187
Mike Reed92f6eb12020-08-25 11:48:41 -0400188 static SkPoint AddMoveCubicClose(SkPathBuilder& path, SkPoint& startPt) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000189 SkPoint moveToPt = startPt + SkPoint::Make(0, 10*SK_Scalar1);
190 SkPoint t1Pt = moveToPt + SkPoint::Make(15*SK_Scalar1, 5*SK_Scalar1);
191 SkPoint t2Pt = moveToPt + SkPoint::Make(25*SK_Scalar1, 5*SK_Scalar1);
192 SkPoint endPt = moveToPt + SkPoint::Make(40*SK_Scalar1, 0);
193 path.moveTo(moveToPt);
194 path.cubicTo(t1Pt, t2Pt, endPt);
195 path.close();
196 return endPt;
197 }
198
Mike Reed92f6eb12020-08-25 11:48:41 -0400199 void drawPath(SkPath path, SkCanvas* canvas, SkColor color,
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000200 const SkRect& clip, SkPaint::Cap cap, SkPaint::Join join,
Mike Reed7d34dc72019-11-26 12:17:17 -0500201 SkPaint::Style style, SkPathFillType fill,
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000202 SkScalar strokeWidth) {
203 path.setFillType(fill);
204 SkPaint paint;
205 paint.setStrokeCap(cap);
206 paint.setStrokeWidth(strokeWidth);
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000207 paint.setStrokeJoin(join);
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000208 paint.setColor(color);
209 paint.setStyle(style);
210 canvas->save();
211 canvas->clipRect(clip);
212 canvas->drawPath(path, paint);
213 canvas->restore();
214 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000215
Hal Canaryfa3305a2019-07-18 12:36:54 -0400216 void onDraw(SkCanvas* canvas) override {
217 constexpr AddSegmentFunc gSegmentFunctions[] = {
218 AddMove,
219 AddMoveClose,
220 AddDegenLine,
221 AddMoveDegenLine,
222 AddMoveDegenLineClose,
223 AddDegenQuad,
224 AddMoveDegenQuad,
225 AddMoveDegenQuadClose,
226 AddDegenCubic,
227 AddMoveDegenCubic,
228 AddMoveDegenCubicClose,
229 AddClose,
230 AddLine,
231 AddMoveLine,
232 AddMoveLineClose,
233 AddQuad,
234 AddMoveQuad,
235 AddMoveQuadClose,
236 AddCubic,
237 AddMoveCubic,
238 AddMoveCubicClose
239 };
240 const char* gSegmentNames[] = {
241 "Move",
242 "MoveClose",
243 "DegenLine",
244 "MoveDegenLine",
245 "MoveDegenLineClose",
246 "DegenQuad",
247 "MoveDegenQuad",
248 "MoveDegenQuadClose",
249 "DegenCubic",
250 "MoveDegenCubic",
251 "MoveDegenCubicClose",
252 "Close",
253 "Line",
254 "MoveLine",
255 "MoveLineClose",
256 "Quad",
257 "MoveQuad",
258 "MoveQuadClose",
259 "Cubic",
260 "MoveCubic",
261 "MoveCubicClose"
262 };
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000263
264 struct FillAndName {
Mike Reed7d34dc72019-11-26 12:17:17 -0500265 SkPathFillType fFill;
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000266 const char* fName;
267 };
mtkleindbfd7ab2016-09-01 11:24:54 -0700268 constexpr FillAndName gFills[] = {
Mike Reed7d34dc72019-11-26 12:17:17 -0500269 {SkPathFillType::kWinding, "Winding"},
270 {SkPathFillType::kEvenOdd, "Even / Odd"},
271 {SkPathFillType::kInverseWinding, "Inverse Winding"},
272 {SkPathFillType::kInverseEvenOdd, "Inverse Even / Odd"}
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000273 };
274 struct StyleAndName {
275 SkPaint::Style fStyle;
276 const char* fName;
277 };
mtkleindbfd7ab2016-09-01 11:24:54 -0700278 constexpr StyleAndName gStyles[] = {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000279 {SkPaint::kFill_Style, "Fill"},
280 {SkPaint::kStroke_Style, "Stroke 10"},
281 {SkPaint::kStrokeAndFill_Style, "Stroke 10 And Fill"}
282 };
283 struct CapAndName {
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000284 SkPaint::Cap fCap;
285 SkPaint::Join fJoin;
286 const char* fName;
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000287 };
mtkleindbfd7ab2016-09-01 11:24:54 -0700288 constexpr CapAndName gCaps[] = {
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000289 {SkPaint::kButt_Cap, SkPaint::kBevel_Join, "Butt"},
290 {SkPaint::kRound_Cap, SkPaint::kRound_Join, "Round"},
291 {SkPaint::kSquare_Cap, SkPaint::kBevel_Join, "Square"}
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000292 };
293
294 SkPaint titlePaint;
295 titlePaint.setColor(SK_ColorBLACK);
296 titlePaint.setAntiAlias(true);
Kevin Lubicke836c3a2023-10-20 06:55:35 -0400297 SkFont font(ToolUtils::DefaultPortableTypeface(), 15);
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000298 const char title[] = "Random Paths Drawn Into Rectangle Clips With "
299 "Indicated Style, Fill and Linecaps, "
300 "with Stroke width 6";
Mike Reed1af9b482019-01-07 11:01:57 -0500301 canvas->drawString(title, 20, 20, font, titlePaint);
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000302
scroggof9d61012014-12-15 12:54:51 -0800303 SkRandom rand;
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000304 SkRect rect = SkRect::MakeWH(220*SK_Scalar1, 50*SK_Scalar1);
305 canvas->save();
306 canvas->translate(2*SK_Scalar1, 30 * SK_Scalar1); // The title
307 canvas->save();
Herb Derbyc37b3862022-06-21 09:49:17 -0400308 unsigned numSegments = std::size(gSegmentFunctions);
309 unsigned numCaps = std::size(gCaps);
310 unsigned numStyles = std::size(gStyles);
311 unsigned numFills = std::size(gFills);
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000312 for (size_t row = 0; row < 6; ++row) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000313 if (0 < row) {
314 canvas->translate(0, rect.height() + 100*SK_Scalar1);
315 }
316 canvas->save();
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000317 for (size_t column = 0; column < 4; ++column) {
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000318 if (0 < column) {
319 canvas->translate(rect.width() + 4*SK_Scalar1, 0);
320 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000321
Mike Kleinea3f0142019-03-20 11:12:10 -0500322 SkColor color = ToolUtils::color_to_565(0xff007000);
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000323 StyleAndName style = gStyles[(rand.nextU() >> 16) % numStyles];
324 CapAndName cap = gCaps[(rand.nextU() >> 16) % numCaps];
325 FillAndName fill = gFills[(rand.nextU() >> 16) % numFills];
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000326 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);
Mike Reed92f6eb12020-08-25 11:48:41 -0400332 SkPathBuilder path;
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000333 pt = gSegmentFunctions[s1](path, pt);
334 pt = gSegmentFunctions[s2](path, pt);
335 pt = gSegmentFunctions[s3](path, pt);
336 pt = gSegmentFunctions[s4](path, pt);
337 pt = gSegmentFunctions[s5](path, pt);
338
Mike Reed92f6eb12020-08-25 11:48:41 -0400339 this->drawPath(path.detach(), canvas, color, rect,
schenney@chromium.org45cbfdd2011-12-20 21:48:14 +0000340 cap.fCap, cap.fJoin, style.fStyle,
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000341 fill.fFill, SK_Scalar1*6);
342
343 SkPaint rectPaint;
344 rectPaint.setColor(SK_ColorBLACK);
345 rectPaint.setStyle(SkPaint::kStroke_Style);
346 rectPaint.setStrokeWidth(-1);
347 rectPaint.setAntiAlias(true);
348 canvas->drawRect(rect, rectPaint);
349
350 SkPaint labelPaint;
351 labelPaint.setColor(color);
352 labelPaint.setAntiAlias(true);
Mike Reed1af9b482019-01-07 11:01:57 -0500353 font.setSize(10);
354 canvas->drawString(style.fName, 0, rect.height() + 12, font, labelPaint);
355 canvas->drawString(fill.fName, 0, rect.height() + 24, font, labelPaint);
356 canvas->drawString(cap.fName, 0, rect.height() + 36, font, labelPaint);
357 canvas->drawString(gSegmentNames[s1], 0, rect.height() + 48, font, labelPaint);
358 canvas->drawString(gSegmentNames[s2], 0, rect.height() + 60, font, labelPaint);
359 canvas->drawString(gSegmentNames[s3], 0, rect.height() + 72, font, labelPaint);
360 canvas->drawString(gSegmentNames[s4], 0, rect.height() + 84, font, labelPaint);
361 canvas->drawString(gSegmentNames[s5], 0, rect.height() + 96, font, labelPaint);
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000362 }
363 canvas->restore();
364 }
365 canvas->restore();
366 canvas->restore();
367 }
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000368};
369
370//////////////////////////////////////////////////////////////////////////////
371
Hal Canarye964c182019-01-23 10:22:01 -0500372DEF_GM( return new DegenerateSegmentsGM; )
schenney@chromium.org4da06ab2011-12-20 15:14:18 +0000373
John Stilesa6841be2020-08-06 14:11:56 -0400374} // namespace skiagm