blob: bd393829e0a332866deb36f67022eb65b1aaa486 [file] [log] [blame]
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * 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.
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +00006 */
7
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +00008#include "gm.h"
bungemand3ebb482015-08-05 13:57:49 -07009#include "SkPath.h"
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000010#include "SkRandom.h"
caryclark5cb00a92015-08-26 09:04:55 -070011#include "SkDashPathEffect.h"
12#include "SkParsePath.h"
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000013
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000014#define W 400
15#define H 400
16#define N 50
17
18static const SkScalar SW = SkIntToScalar(W);
19static const SkScalar SH = SkIntToScalar(H);
20
scroggof9d61012014-12-15 12:54:51 -080021static void rnd_rect(SkRect* r, SkPaint* paint, SkRandom& rand) {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000022 SkScalar x = rand.nextUScalar1() * W;
23 SkScalar y = rand.nextUScalar1() * H;
24 SkScalar w = rand.nextUScalar1() * (W >> 2);
25 SkScalar h = rand.nextUScalar1() * (H >> 2);
epoger@google.com17b78942011-08-26 14:40:38 +000026 SkScalar hoffset = rand.nextSScalar1();
27 SkScalar woffset = rand.nextSScalar1();
rmistry@google.comae933ce2012-08-23 18:19:56 +000028
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000029 r->set(x, y, x + w, y + h);
epoger@google.com17b78942011-08-26 14:40:38 +000030 r->offset(-w/2 + woffset, -h/2 + hoffset);
rmistry@google.comae933ce2012-08-23 18:19:56 +000031
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000032 paint->setColor(rand.nextU());
33 paint->setAlpha(0xFF);
34}
35
rmistry@google.comae933ce2012-08-23 18:19:56 +000036
reed@google.com4384fab2012-06-05 16:14:23 +000037class StrokesGM : public skiagm::GM {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000038public:
39 StrokesGM() {}
rmistry@google.comae933ce2012-08-23 18:19:56 +000040
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000041protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +000042
mtklein36352bf2015-03-25 18:17:31 -070043 SkString onShortName() override {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000044 return SkString("strokes_round");
45 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000046
mtklein36352bf2015-03-25 18:17:31 -070047 SkISize onISize() override {
reed@google.com4384fab2012-06-05 16:14:23 +000048 return SkISize::Make(W, H*2);
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000049 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000050
mtklein36352bf2015-03-25 18:17:31 -070051 void onDraw(SkCanvas* canvas) override {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000052 SkPaint paint;
53 paint.setStyle(SkPaint::kStroke_Style);
54 paint.setStrokeWidth(SkIntToScalar(9)/2);
rmistry@google.comae933ce2012-08-23 18:19:56 +000055
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000056 for (int y = 0; y < 2; y++) {
57 paint.setAntiAlias(!!y);
58 SkAutoCanvasRestore acr(canvas, true);
59 canvas->translate(0, SH * y);
60 canvas->clipRect(SkRect::MakeLTRB(
61 SkIntToScalar(2), SkIntToScalar(2)
62 , SW - SkIntToScalar(2), SH - SkIntToScalar(2)
63 ));
rmistry@google.comae933ce2012-08-23 18:19:56 +000064
scroggof9d61012014-12-15 12:54:51 -080065 SkRandom rand;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000066 for (int i = 0; i < N; i++) {
67 SkRect r;
68 rnd_rect(&r, &paint, rand);
69 canvas->drawOval(r, paint);
70 rnd_rect(&r, &paint, rand);
71 canvas->drawRoundRect(r, r.width()/4, r.height()/4, paint);
72 rnd_rect(&r, &paint, rand);
73 }
74 }
75 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000076
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000077private:
reed@google.com4384fab2012-06-05 16:14:23 +000078 typedef skiagm::GM INHERITED;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +000079};
80
caryclark5cb00a92015-08-26 09:04:55 -070081/* See
82 https://code.google.com/p/chromium/issues/detail?id=422974 and
83 http://jsfiddle.net/1xnku3sg/2/
84 */
85class ZeroLenStrokesGM : public skiagm::GM {
86 SkPath fMoveHfPath, fMoveZfPath, fDashedfPath, fRefPath[4];
caryclark6651a322015-09-09 13:20:49 -070087 SkPath fCubicPath, fQuadPath, fLinePath;
caryclark5cb00a92015-08-26 09:04:55 -070088protected:
89 void onOnceBeforeDraw() override {
90
91 SkAssertResult(SkParsePath::FromSVGString("M0,0h0M10,0h0M20,0h0", &fMoveHfPath));
92 SkAssertResult(SkParsePath::FromSVGString("M0,0zM10,0zM20,0z", &fMoveZfPath));
93 SkAssertResult(SkParsePath::FromSVGString("M0,0h25", &fDashedfPath));
caryclark6651a322015-09-09 13:20:49 -070094 SkAssertResult(SkParsePath::FromSVGString("M 0 0 C 0 0 0 0 0 0", &fCubicPath));
95 SkAssertResult(SkParsePath::FromSVGString("M 0 0 Q 0 0 0 0", &fQuadPath));
96 SkAssertResult(SkParsePath::FromSVGString("M 0 0 L 0 0", &fLinePath));
caryclark5cb00a92015-08-26 09:04:55 -070097
98 for (int i = 0; i < 3; ++i) {
99 fRefPath[0].addCircle(i * 10.f, 0, 5);
100 fRefPath[1].addCircle(i * 10.f, 0, 10);
101 fRefPath[2].addRect(i * 10.f - 4, -2, i * 10.f + 4, 6);
102 fRefPath[3].addRect(i * 10.f - 10, -10, i * 10.f + 10, 10);
103 }
104 }
105
106 SkString onShortName() override {
107 return SkString("zeroPath");
108 }
109
110 SkISize onISize() override {
111 return SkISize::Make(W, H*2);
112 }
113
114 void onDraw(SkCanvas* canvas) override {
115 SkPaint fillPaint, strokePaint, dashPaint;
116 fillPaint.setAntiAlias(true);
117 strokePaint = fillPaint;
118 strokePaint.setStyle(SkPaint::kStroke_Style);
119 for (int i = 0; i < 2; ++i) {
120 fillPaint.setAlpha(255);
121 strokePaint.setAlpha(255);
122 strokePaint.setStrokeWidth(i ? 8.f : 10.f);
123 strokePaint.setStrokeCap(i ? SkPaint::kSquare_Cap : SkPaint::kRound_Cap);
124 canvas->save();
125 canvas->translate(10 + i * 100.f, 10);
126 canvas->drawPath(fMoveHfPath, strokePaint);
127 canvas->translate(0, 20);
128 canvas->drawPath(fMoveZfPath, strokePaint);
129 dashPaint = strokePaint;
130 const SkScalar intervals[] = { 0, 10 };
131 dashPaint.setPathEffect(SkDashPathEffect::Create(intervals, 2, 0))->unref();
132 SkPath fillPath;
133 dashPaint.getFillPath(fDashedfPath, &fillPath);
134 canvas->translate(0, 20);
135 canvas->drawPath(fDashedfPath, dashPaint);
136 canvas->translate(0, 20);
137 canvas->drawPath(fRefPath[i * 2], fillPaint);
138 strokePaint.setStrokeWidth(20);
139 strokePaint.setAlpha(127);
140 canvas->translate(0, 50);
141 canvas->drawPath(fMoveHfPath, strokePaint);
142 canvas->translate(0, 30);
143 canvas->drawPath(fMoveZfPath, strokePaint);
144 canvas->translate(0, 30);
145 fillPaint.setAlpha(127);
146 canvas->drawPath(fRefPath[1 + i * 2], fillPaint);
caryclark6651a322015-09-09 13:20:49 -0700147 canvas->translate(0, 30);
148 canvas->drawPath(fCubicPath, strokePaint);
149 canvas->translate(0, 30);
150 canvas->drawPath(fQuadPath, strokePaint);
151 canvas->translate(0, 30);
152 canvas->drawPath(fLinePath, strokePaint);
caryclark5cb00a92015-08-26 09:04:55 -0700153 canvas->restore();
154 }
155 }
156
157private:
158 typedef skiagm::GM INHERITED;
159};
160
caryclark950305e2015-10-26 08:17:04 -0700161class TeenyStrokesGM : public skiagm::GM {
162
163 SkString onShortName() override {
164 return SkString("teenyStrokes");
165 }
166
167 SkISize onISize() override {
168 return SkISize::Make(W, H*2);
169 }
170
171 static void line(SkScalar scale, SkCanvas* canvas, SkColor color) {
172 SkPaint p;
173 p.setAntiAlias(true);
174 p.setStyle(SkPaint::kStroke_Style);
175 p.setColor(color);
176 canvas->translate(50, 0);
177 canvas->save();
178 p.setStrokeWidth(scale * 5);
179 canvas->scale(1 / scale, 1 / scale);
180 canvas->drawLine(20 * scale, 20 * scale, 20 * scale, 100 * scale, p);
181 canvas->drawLine(20 * scale, 20 * scale, 100 * scale, 100 * scale, p);
182 canvas->restore();
183 }
184
185 void onDraw(SkCanvas* canvas) override {
186 line(0.00005f, canvas, SK_ColorBLACK);
187 line(0.000045f, canvas, SK_ColorRED);
188 line(0.0000035f, canvas, SK_ColorGREEN);
189 line(0.000003f, canvas, SK_ColorBLUE);
190 line(0.000002f, canvas, SK_ColorBLACK);
191 }
192private:
193 typedef skiagm::GM INHERITED;
194};
195
196
reed@google.com4384fab2012-06-05 16:14:23 +0000197class Strokes2GM : public skiagm::GM {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000198 SkPath fPath;
caryclark63c684a2015-02-25 09:04:04 -0800199protected:
mtklein36352bf2015-03-25 18:17:31 -0700200 void onOnceBeforeDraw() override {
scroggof9d61012014-12-15 12:54:51 -0800201 SkRandom rand;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000202 fPath.moveTo(0, 0);
203 for (int i = 0; i < 13; i++) {
204 SkScalar x = rand.nextUScalar1() * (W >> 1);
205 SkScalar y = rand.nextUScalar1() * (H >> 1);
206 fPath.lineTo(x, y);
207 }
208 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000209
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000210
mtklein36352bf2015-03-25 18:17:31 -0700211 SkString onShortName() override {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000212 return SkString("strokes_poly");
213 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000214
mtklein36352bf2015-03-25 18:17:31 -0700215 SkISize onISize() override {
reed@google.com4384fab2012-06-05 16:14:23 +0000216 return SkISize::Make(W, H*2);
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000217 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000218
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000219 static void rotate(SkScalar angle, SkScalar px, SkScalar py, SkCanvas* canvas) {
220 SkMatrix matrix;
221 matrix.setRotate(angle, px, py);
222 canvas->concat(matrix);
223 }
224
mtklein36352bf2015-03-25 18:17:31 -0700225 void onDraw(SkCanvas* canvas) override {
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000226 canvas->drawColor(SK_ColorWHITE);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000227
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000228 SkPaint paint;
229 paint.setStyle(SkPaint::kStroke_Style);
230 paint.setStrokeWidth(SkIntToScalar(9)/2);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000231
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000232 for (int y = 0; y < 2; y++) {
233 paint.setAntiAlias(!!y);
234 SkAutoCanvasRestore acr(canvas, true);
235 canvas->translate(0, SH * y);
236 canvas->clipRect(SkRect::MakeLTRB(SkIntToScalar(2),
237 SkIntToScalar(2),
238 SW - SkIntToScalar(2),
239 SH - SkIntToScalar(2)));
rmistry@google.comae933ce2012-08-23 18:19:56 +0000240
scroggof9d61012014-12-15 12:54:51 -0800241 SkRandom rand;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000242 for (int i = 0; i < N/2; i++) {
243 SkRect r;
244 rnd_rect(&r, &paint, rand);
245 rotate(SkIntToScalar(15), SW/2, SH/2, canvas);
246 canvas->drawPath(fPath, paint);
247 }
248 }
249 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000250
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000251private:
reed@google.com4384fab2012-06-05 16:14:23 +0000252 typedef skiagm::GM INHERITED;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000253};
254
255//////////////////////////////////////////////////////////////////////////////
256
reed@google.com4384fab2012-06-05 16:14:23 +0000257static SkRect inset(const SkRect& r) {
258 SkRect rr(r);
259 rr.inset(r.width()/10, r.height()/10);
260 return rr;
mike@reedtribe.orgf2c21cd2011-06-18 00:15:04 +0000261}
262
reed@google.com4384fab2012-06-05 16:14:23 +0000263class Strokes3GM : public skiagm::GM {
264 static void make0(SkPath* path, const SkRect& bounds, SkString* title) {
265 path->addRect(bounds, SkPath::kCW_Direction);
266 path->addRect(inset(bounds), SkPath::kCW_Direction);
267 title->set("CW CW");
268 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000269
reed@google.com4384fab2012-06-05 16:14:23 +0000270 static void make1(SkPath* path, const SkRect& bounds, SkString* title) {
271 path->addRect(bounds, SkPath::kCW_Direction);
272 path->addRect(inset(bounds), SkPath::kCCW_Direction);
273 title->set("CW CCW");
274 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000275
reed@google.com4384fab2012-06-05 16:14:23 +0000276 static void make2(SkPath* path, const SkRect& bounds, SkString* title) {
277 path->addOval(bounds, SkPath::kCW_Direction);
278 path->addOval(inset(bounds), SkPath::kCW_Direction);
279 title->set("CW CW");
280 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000281
reed@google.com4384fab2012-06-05 16:14:23 +0000282 static void make3(SkPath* path, const SkRect& bounds, SkString* title) {
283 path->addOval(bounds, SkPath::kCW_Direction);
284 path->addOval(inset(bounds), SkPath::kCCW_Direction);
285 title->set("CW CCW");
286 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000287
reed@google.com4384fab2012-06-05 16:14:23 +0000288 static void make4(SkPath* path, const SkRect& bounds, SkString* title) {
289 path->addRect(bounds, SkPath::kCW_Direction);
290 SkRect r = bounds;
291 r.inset(bounds.width() / 10, -bounds.height() / 10);
292 path->addOval(r, SkPath::kCW_Direction);
293 title->set("CW CW");
294 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000295
reed@google.com4384fab2012-06-05 16:14:23 +0000296 static void make5(SkPath* path, const SkRect& bounds, SkString* title) {
297 path->addRect(bounds, SkPath::kCW_Direction);
298 SkRect r = bounds;
299 r.inset(bounds.width() / 10, -bounds.height() / 10);
300 path->addOval(r, SkPath::kCCW_Direction);
301 title->set("CW CCW");
302 }
303
304public:
305 Strokes3GM() {}
rmistry@google.comae933ce2012-08-23 18:19:56 +0000306
reed@google.com4384fab2012-06-05 16:14:23 +0000307protected:
commit-bot@chromium.orga90c6802014-04-30 13:20:45 +0000308
mtklein36352bf2015-03-25 18:17:31 -0700309 SkString onShortName() override {
reed@google.com4384fab2012-06-05 16:14:23 +0000310 return SkString("strokes3");
311 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000312
mtklein36352bf2015-03-25 18:17:31 -0700313 SkISize onISize() override {
caryclark37604572015-02-23 06:51:04 -0800314 return SkISize::Make(1500, 1500);
reed@google.com4384fab2012-06-05 16:14:23 +0000315 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000316
mtklein36352bf2015-03-25 18:17:31 -0700317 void onDraw(SkCanvas* canvas) override {
reed@google.com4384fab2012-06-05 16:14:23 +0000318 SkPaint origPaint;
319 origPaint.setAntiAlias(true);
320 origPaint.setStyle(SkPaint::kStroke_Style);
321 SkPaint fillPaint(origPaint);
322 fillPaint.setColor(SK_ColorRED);
323 SkPaint strokePaint(origPaint);
caryclark12596012015-07-29 05:27:47 -0700324 strokePaint.setColor(sk_tool_utils::color_to_565(0xFF4444FF));
reed@google.com4384fab2012-06-05 16:14:23 +0000325
326 void (*procs[])(SkPath*, const SkRect&, SkString*) = {
327 make0, make1, make2, make3, make4, make5
328 };
329
caryclark37604572015-02-23 06:51:04 -0800330 canvas->translate(SkIntToScalar(20), SkIntToScalar(80));
reed@google.com4384fab2012-06-05 16:14:23 +0000331
332 SkRect bounds = SkRect::MakeWH(SkIntToScalar(50), SkIntToScalar(50));
333 SkScalar dx = bounds.width() * 4/3;
334 SkScalar dy = bounds.height() * 5;
335
336 for (size_t i = 0; i < SK_ARRAY_COUNT(procs); ++i) {
337 SkPath orig;
338 SkString str;
339 procs[i](&orig, bounds, &str);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000340
reed@google.com4384fab2012-06-05 16:14:23 +0000341 canvas->save();
342 for (int j = 0; j < 13; ++j) {
343 strokePaint.setStrokeWidth(SK_Scalar1 * j * j);
344 canvas->drawPath(orig, strokePaint);
345 canvas->drawPath(orig, origPaint);
346 SkPath fill;
347 strokePaint.getFillPath(orig, &fill);
348 canvas->drawPath(fill, fillPaint);
349 canvas->translate(dx + strokePaint.getStrokeWidth(), 0);
350 }
351 canvas->restore();
352 canvas->translate(0, dy);
353 }
354 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000355
reed@google.com4384fab2012-06-05 16:14:23 +0000356private:
357 typedef skiagm::GM INHERITED;
358};
359
caryclark612f70d2015-05-19 11:05:37 -0700360class Strokes4GM : public skiagm::GM {
361public:
362 Strokes4GM() {}
363
364protected:
365
366 SkString onShortName() override {
367 return SkString("strokes_zoomed");
368 }
369
370 SkISize onISize() override {
371 return SkISize::Make(W, H*2);
372 }
373
374 void onDraw(SkCanvas* canvas) override {
375 SkPaint paint;
376 paint.setStyle(SkPaint::kStroke_Style);
377 paint.setStrokeWidth(0.055f);
378
379 canvas->scale(1000, 1000);
380 canvas->drawCircle(0, 2, 1.97f, paint);
381 }
382
383private:
384 typedef skiagm::GM INHERITED;
385};
386
caryclark45398df2015-08-25 13:19:06 -0700387// Test stroking for curves that produce degenerate tangents when t is 0 or 1 (see bug 4191)
388class Strokes5GM : public skiagm::GM {
389public:
390 Strokes5GM() {}
391
392protected:
393
394 SkString onShortName() override {
395 return SkString("zero_control_stroke");
396 }
397
398 SkISize onISize() override {
399 return SkISize::Make(W, H*2);
400 }
401
402 void onDraw(SkCanvas* canvas) override {
403 SkPaint p;
404 p.setColor(SK_ColorRED);
405 p.setAntiAlias(true);
406 p.setStyle(SkPaint::kStroke_Style);
407 p.setStrokeWidth(40);
408 p.setStrokeCap(SkPaint::kButt_Cap);
409
410 SkPath path;
411 path.moveTo(157.474f,111.753f);
412 path.cubicTo(128.5f,111.5f,35.5f,29.5f,35.5f,29.5f);
413 canvas->drawPath(path, p);
414 path.reset();
415 path.moveTo(250, 50);
416 path.quadTo(280, 80, 280, 80);
417 canvas->drawPath(path, p);
418 path.reset();
419 path.moveTo(150, 50);
420 path.conicTo(180, 80, 180, 80, 0.707f);
421 canvas->drawPath(path, p);
422
423 path.reset();
424 path.moveTo(157.474f,311.753f);
425 path.cubicTo(157.474f,311.753f,85.5f,229.5f,35.5f,229.5f);
426 canvas->drawPath(path, p);
427 path.reset();
428 path.moveTo(280, 250);
429 path.quadTo(280, 250, 310, 280);
430 canvas->drawPath(path, p);
431 path.reset();
432 path.moveTo(180, 250);
433 path.conicTo(180, 250, 210, 280, 0.707f);
434 canvas->drawPath(path, p);
435 }
436
437private:
438 typedef skiagm::GM INHERITED;
439};
440
caryclark612f70d2015-05-19 11:05:37 -0700441
reed@google.com4384fab2012-06-05 16:14:23 +0000442//////////////////////////////////////////////////////////////////////////////
443
444static skiagm::GM* F0(void*) { return new StrokesGM; }
445static skiagm::GM* F1(void*) { return new Strokes2GM; }
446static skiagm::GM* F2(void*) { return new Strokes3GM; }
caryclark612f70d2015-05-19 11:05:37 -0700447static skiagm::GM* F3(void*) { return new Strokes4GM; }
caryclark45398df2015-08-25 13:19:06 -0700448static skiagm::GM* F4(void*) { return new Strokes5GM; }
reed@google.com4384fab2012-06-05 16:14:23 +0000449
450static skiagm::GMRegistry R0(F0);
451static skiagm::GMRegistry R1(F1);
452static skiagm::GMRegistry R2(F2);
caryclark612f70d2015-05-19 11:05:37 -0700453static skiagm::GMRegistry R3(F3);
caryclark45398df2015-08-25 13:19:06 -0700454static skiagm::GMRegistry R4(F4);
caryclark5cb00a92015-08-26 09:04:55 -0700455
halcanary385fe4d2015-08-26 13:07:48 -0700456DEF_GM( return new ZeroLenStrokesGM; )
caryclark950305e2015-10-26 08:17:04 -0700457DEF_GM( return new TeenyStrokesGM; )