• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "SkCanvas.h"
9 #include "SkPaint.h"
10 #include "SkRandom.h"
11 
12 namespace skiagm {
13 
14 class LinePathGM : public GM {
15 public:
LinePathGM()16     LinePathGM() {}
17 
18 protected:
onShortName()19     SkString onShortName() {
20         return SkString("linepath");
21     }
22 
onISize()23     SkISize onISize() { return make_isize(1240, 390); }
24 
drawPath(SkPath & path,SkCanvas * canvas,SkColor color,const SkRect & clip,SkPaint::Cap cap,SkPaint::Join join,SkPaint::Style style,SkPath::FillType fill,SkScalar strokeWidth)25     void drawPath(SkPath& path,SkCanvas* canvas,SkColor color,
26                   const SkRect& clip,SkPaint::Cap cap, SkPaint::Join join,
27                   SkPaint::Style style, SkPath::FillType fill,
28                   SkScalar strokeWidth) {
29         path.setFillType(fill);
30         SkPaint paint;
31         paint.setStrokeCap(cap);
32         paint.setStrokeWidth(strokeWidth);
33         paint.setStrokeJoin(join);
34         paint.setColor(color);
35         paint.setStyle(style);
36         canvas->save();
37         canvas->clipRect(clip);
38         canvas->drawPath(path, paint);
39         canvas->restore();
40     }
41 
onDraw(SkCanvas * canvas)42     virtual void onDraw(SkCanvas* canvas) {
43         struct FillAndName {
44             SkPath::FillType fFill;
45             const char*      fName;
46         };
47         static const FillAndName gFills[] = {
48             {SkPath::kWinding_FillType, "Winding"},
49             {SkPath::kEvenOdd_FillType, "Even / Odd"},
50             {SkPath::kInverseWinding_FillType, "Inverse Winding"},
51             {SkPath::kInverseEvenOdd_FillType, "Inverse Even / Odd"},
52         };
53         struct StyleAndName {
54             SkPaint::Style fStyle;
55             const char*    fName;
56         };
57         static const StyleAndName gStyles[] = {
58             {SkPaint::kFill_Style, "Fill"},
59             {SkPaint::kStroke_Style, "Stroke"},
60             {SkPaint::kStrokeAndFill_Style, "Stroke And Fill"},
61         };
62         struct CapAndName {
63             SkPaint::Cap  fCap;
64             SkPaint::Join fJoin;
65             const char*   fName;
66         };
67         static const CapAndName gCaps[] = {
68             {SkPaint::kButt_Cap, SkPaint::kBevel_Join, "Butt"},
69             {SkPaint::kRound_Cap, SkPaint::kRound_Join, "Round"},
70             {SkPaint::kSquare_Cap, SkPaint::kBevel_Join, "Square"}
71         };
72         struct PathAndName {
73             SkPath      fPath;
74             const char* fName;
75         };
76         PathAndName path;
77         path.fPath.moveTo(25*SK_Scalar1, 15*SK_Scalar1);
78         path.fPath.lineTo(75*SK_Scalar1, 15*SK_Scalar1);
79         path.fName = "moveTo-line";
80 
81         SkPaint titlePaint;
82         titlePaint.setColor(SK_ColorBLACK);
83         titlePaint.setAntiAlias(true);
84         titlePaint.setLCDRenderText(true);
85         titlePaint.setTextSize(15 * SK_Scalar1);
86         const char title[] = "Line Drawn Into Rectangle Clips With "
87                              "Indicated Style, Fill and Linecaps, with stroke width 10";
88         canvas->drawText(title, strlen(title),
89                             20 * SK_Scalar1,
90                             20 * SK_Scalar1,
91                             titlePaint);
92 
93         SkLCGRandom rand;
94         SkRect rect = SkRect::MakeWH(100*SK_Scalar1, 30*SK_Scalar1);
95         canvas->save();
96         canvas->translate(10 * SK_Scalar1, 30 * SK_Scalar1);
97         canvas->save();
98         for (size_t cap = 0; cap < SK_ARRAY_COUNT(gCaps); ++cap) {
99             if (0 < cap) {
100                 canvas->translate((rect.width() + 40 * SK_Scalar1) * SK_ARRAY_COUNT(gStyles), 0);
101             }
102             canvas->save();
103             for (size_t fill = 0; fill < SK_ARRAY_COUNT(gFills); ++fill) {
104                 if (0 < fill) {
105                     canvas->translate(0, rect.height() + 40 * SK_Scalar1);
106                 }
107                 canvas->save();
108                 for (size_t style = 0; style < SK_ARRAY_COUNT(gStyles); ++style) {
109                     if (0 < style) {
110                         canvas->translate(rect.width() + 40 * SK_Scalar1, 0);
111                     }
112 
113                     SkColor color = 0xff007000;
114                     this->drawPath(path.fPath, canvas, color, rect,
115                                     gCaps[cap].fCap, gCaps[cap].fJoin, gStyles[style].fStyle,
116                                     gFills[fill].fFill, SK_Scalar1*10);
117 
118                     SkPaint rectPaint;
119                     rectPaint.setColor(SK_ColorBLACK);
120                     rectPaint.setStyle(SkPaint::kStroke_Style);
121                     rectPaint.setStrokeWidth(-1);
122                     rectPaint.setAntiAlias(true);
123                     canvas->drawRect(rect, rectPaint);
124 
125                     SkPaint labelPaint;
126                     labelPaint.setColor(color);
127                     labelPaint.setAntiAlias(true);
128                     labelPaint.setLCDRenderText(true);
129                     labelPaint.setTextSize(10 * SK_Scalar1);
130                     canvas->drawText(gStyles[style].fName,
131                                         strlen(gStyles[style].fName),
132                                         0, rect.height() + 12 * SK_Scalar1,
133                                         labelPaint);
134                     canvas->drawText(gFills[fill].fName,
135                                         strlen(gFills[fill].fName),
136                                         0, rect.height() + 24 * SK_Scalar1,
137                                         labelPaint);
138                     canvas->drawText(gCaps[cap].fName,
139                                         strlen(gCaps[cap].fName),
140                                         0, rect.height() + 36 * SK_Scalar1,
141                                         labelPaint);
142                 }
143                 canvas->restore();
144             }
145             canvas->restore();
146         }
147         canvas->restore();
148         canvas->restore();
149     }
150 
151 private:
152     typedef GM INHERITED;
153 };
154 
155 class LineClosePathGM : public GM {
156 public:
LineClosePathGM()157     LineClosePathGM() {}
158 
159 protected:
onShortName()160     SkString onShortName() {
161         return SkString("lineclosepath");
162     }
163 
onISize()164     SkISize onISize() { return make_isize(1240, 390); }
165 
drawPath(SkPath & path,SkCanvas * canvas,SkColor color,const SkRect & clip,SkPaint::Cap cap,SkPaint::Join join,SkPaint::Style style,SkPath::FillType fill,SkScalar strokeWidth)166     void drawPath(SkPath& path,SkCanvas* canvas,SkColor color,
167                   const SkRect& clip,SkPaint::Cap cap, SkPaint::Join join,
168                   SkPaint::Style style, SkPath::FillType fill,
169                   SkScalar strokeWidth) {
170         path.setFillType(fill);
171         SkPaint paint;
172         paint.setStrokeCap(cap);
173         paint.setStrokeWidth(strokeWidth);
174         paint.setStrokeJoin(join);
175         paint.setColor(color);
176         paint.setStyle(style);
177         canvas->save();
178         canvas->clipRect(clip);
179         canvas->drawPath(path, paint);
180         canvas->restore();
181     }
182 
onDraw(SkCanvas * canvas)183     virtual void onDraw(SkCanvas* canvas) {
184         struct FillAndName {
185             SkPath::FillType fFill;
186             const char*      fName;
187         };
188         static const FillAndName gFills[] = {
189             {SkPath::kWinding_FillType, "Winding"},
190             {SkPath::kEvenOdd_FillType, "Even / Odd"},
191             {SkPath::kInverseWinding_FillType, "Inverse Winding"},
192             {SkPath::kInverseEvenOdd_FillType, "Inverse Even / Odd"},
193         };
194         struct StyleAndName {
195             SkPaint::Style fStyle;
196             const char*    fName;
197         };
198         static const StyleAndName gStyles[] = {
199             {SkPaint::kFill_Style, "Fill"},
200             {SkPaint::kStroke_Style, "Stroke"},
201             {SkPaint::kStrokeAndFill_Style, "Stroke And Fill"},
202         };
203         struct CapAndName {
204             SkPaint::Cap  fCap;
205             SkPaint::Join fJoin;
206             const char*   fName;
207         };
208         static const CapAndName gCaps[] = {
209             {SkPaint::kButt_Cap, SkPaint::kBevel_Join, "Butt"},
210             {SkPaint::kRound_Cap, SkPaint::kRound_Join, "Round"},
211             {SkPaint::kSquare_Cap, SkPaint::kBevel_Join, "Square"}
212         };
213         struct PathAndName {
214             SkPath      fPath;
215             const char* fName;
216         };
217         PathAndName path;
218         path.fPath.moveTo(25*SK_Scalar1, 15*SK_Scalar1);
219         path.fPath.lineTo(75*SK_Scalar1, 15*SK_Scalar1);
220         path.fPath.close();
221         path.fName = "moveTo-line-close";
222 
223         SkPaint titlePaint;
224         titlePaint.setColor(SK_ColorBLACK);
225         titlePaint.setAntiAlias(true);
226         titlePaint.setLCDRenderText(true);
227         titlePaint.setTextSize(15 * SK_Scalar1);
228         const char title[] = "Line Closed Drawn Into Rectangle Clips With "
229                              "Indicated Style, Fill and Linecaps, with stroke width 10";
230         canvas->drawText(title, strlen(title),
231                             20 * SK_Scalar1,
232                             20 * SK_Scalar1,
233                             titlePaint);
234 
235         SkLCGRandom rand;
236         SkRect rect = SkRect::MakeWH(100*SK_Scalar1, 30*SK_Scalar1);
237         canvas->save();
238         canvas->translate(10 * SK_Scalar1, 30 * SK_Scalar1);
239         canvas->save();
240         for (size_t cap = 0; cap < SK_ARRAY_COUNT(gCaps); ++cap) {
241             if (0 < cap) {
242                 canvas->translate((rect.width() + 40 * SK_Scalar1) * SK_ARRAY_COUNT(gStyles), 0);
243             }
244             canvas->save();
245             for (size_t fill = 0; fill < SK_ARRAY_COUNT(gFills); ++fill) {
246                 if (0 < fill) {
247                     canvas->translate(0, rect.height() + 40 * SK_Scalar1);
248                 }
249                 canvas->save();
250                 for (size_t style = 0; style < SK_ARRAY_COUNT(gStyles); ++style) {
251                     if (0 < style) {
252                         canvas->translate(rect.width() + 40 * SK_Scalar1, 0);
253                     }
254 
255                     SkColor color = 0xff007000;
256                     this->drawPath(path.fPath, canvas, color, rect,
257                                     gCaps[cap].fCap, gCaps[cap].fJoin, gStyles[style].fStyle,
258                                     gFills[fill].fFill, SK_Scalar1*10);
259 
260                     SkPaint rectPaint;
261                     rectPaint.setColor(SK_ColorBLACK);
262                     rectPaint.setStyle(SkPaint::kStroke_Style);
263                     rectPaint.setStrokeWidth(-1);
264                     rectPaint.setAntiAlias(true);
265                     canvas->drawRect(rect, rectPaint);
266 
267                     SkPaint labelPaint;
268                     labelPaint.setColor(color);
269                     labelPaint.setAntiAlias(true);
270                     labelPaint.setLCDRenderText(true);
271                     labelPaint.setTextSize(10 * SK_Scalar1);
272                     canvas->drawText(gStyles[style].fName,
273                                         strlen(gStyles[style].fName),
274                                         0, rect.height() + 12 * SK_Scalar1,
275                                         labelPaint);
276                     canvas->drawText(gFills[fill].fName,
277                                         strlen(gFills[fill].fName),
278                                         0, rect.height() + 24 * SK_Scalar1,
279                                         labelPaint);
280                     canvas->drawText(gCaps[cap].fName,
281                                         strlen(gCaps[cap].fName),
282                                         0, rect.height() + 36 * SK_Scalar1,
283                                         labelPaint);
284                 }
285                 canvas->restore();
286             }
287             canvas->restore();
288         }
289         canvas->restore();
290         canvas->restore();
291     }
292 
293 private:
294     typedef GM INHERITED;
295 };
296 
297 //////////////////////////////////////////////////////////////////////////////
298 
LinePathFactory(void *)299 static GM* LinePathFactory(void*) { return new LinePathGM; }
300 static GMRegistry regLinePath(LinePathFactory);
301 
LineClosePathFactory(void *)302 static GM* LineClosePathFactory(void*) { return new LineClosePathGM; }
303 static GMRegistry regLineClosePath(LineClosePathFactory);
304 
305 }
306