• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2016 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 
8 #include "gm/gm.h"
9 #include "include/core/SkCanvas.h"
10 #include "include/core/SkColor.h"
11 #include "include/core/SkPaint.h"
12 #include "include/core/SkPath.h"
13 #include "include/core/SkPathEffect.h"
14 #include "include/core/SkRect.h"
15 #include "include/core/SkScalar.h"
16 #include "include/core/SkTypes.h"
17 #include "include/effects/SkDashPathEffect.h"
18 #include "include/private/SkFloatBits.h"
19 #include "include/private/SkTArray.h"
20 
21 #include <functional>
22 
23 constexpr SkScalar kStarts[] = {0.f, 10.f, 30.f, 45.f, 90.f, 165.f, 180.f, 270.f};
24 constexpr SkScalar kSweeps[] = {1.f, 45.f, 90.f, 130.f, 180.f, 184.f, 300.f, 355.f};
25 constexpr SkScalar kDiameter = 40.f;
26 constexpr SkRect kRect = {0.f, 0.f, kDiameter, kDiameter};
27 constexpr int kW = 1000;
28 constexpr int kH = 1000;
29 constexpr SkScalar kPad = 20.f;
30 
draw_arcs(SkCanvas * canvas,std::function<void (SkPaint *)> configureStyle)31 void draw_arcs(SkCanvas* canvas, std::function<void(SkPaint*)> configureStyle) {
32     // Draws grid of arcs with different start/sweep angles in red and their complement arcs in
33     // blue.
34     auto drawGrid = [canvas, &configureStyle] (SkScalar x, SkScalar y, bool useCenter, bool aa) {
35         SkPaint p0;
36         p0.setColor(SK_ColorRED);
37         p0.setAntiAlias(aa);
38         // Set a reasonable stroke width that configureStyle can override.
39         p0.setStrokeWidth(15.f);
40         SkPaint p1 = p0;
41         p1.setColor(SK_ColorBLUE);
42         // Use alpha so we see magenta on overlap between arc and its complement.
43         p0.setAlpha(100);
44         p1.setAlpha(100);
45         configureStyle(&p0);
46         configureStyle(&p1);
47 
48         canvas->save();
49         canvas->translate(kPad + x, kPad + y);
50         for (auto start : kStarts) {
51             canvas->save();
52             for (auto sweep : kSweeps) {
53                 canvas->drawArc(kRect, start, sweep, useCenter, p0);
54                 canvas->drawArc(kRect, start, -(360.f - sweep), useCenter, p1);
55                 canvas->translate(kRect.width() + kPad, 0.f);
56             }
57             canvas->restore();
58             canvas->translate(0, kRect.height() + kPad);
59         }
60         canvas->restore();
61     };
62     // Draw a grids for combo of enabling/disabling aa and using center.
63     constexpr SkScalar kGridW = kW / 2.f;
64     constexpr SkScalar kGridH = kH / 2.f;
65     drawGrid(0.f   , 0.f   , false, false);
66     drawGrid(kGridW, 0.f   , true , false);
67     drawGrid(0.f   , kGridH, false, true );
68     drawGrid(kGridW, kGridH, true , true );
69     // Draw separators between the grids.
70     SkPaint linePaint;
71     linePaint.setAntiAlias(true);
72     linePaint.setColor(SK_ColorBLACK);
73     canvas->drawLine(kGridW, 0.f   , kGridW,            SkIntToScalar(kH), linePaint);
74     canvas->drawLine(0.f   , kGridH, SkIntToScalar(kW), kGridH,            linePaint);
75 }
76 
77 #define DEF_ARC_GM(name) DEF_SIMPLE_GM(circular_arcs_##name, canvas, kW, kH)
78 
DEF_ARC_GM(fill)79 DEF_ARC_GM(fill) {
80     auto setFill = [] (SkPaint*p) { p->setStyle(SkPaint::kFill_Style); };
81     draw_arcs(canvas, setFill);
82 }
83 
DEF_ARC_GM(hairline)84 DEF_ARC_GM(hairline) {
85     auto setHairline = [] (SkPaint* p) {
86         p->setStyle(SkPaint::kStroke_Style);
87         p->setStrokeWidth(0.f);
88     };
89     draw_arcs(canvas, setHairline);
90 }
91 
DEF_ARC_GM(stroke_butt)92 DEF_ARC_GM(stroke_butt) {
93     auto setStroke = [](SkPaint* p) {
94         p->setStyle(SkPaint::kStroke_Style);
95         p->setStrokeCap(SkPaint::kButt_Cap);
96     };
97     draw_arcs(canvas, setStroke);
98 }
99 
DEF_ARC_GM(stroke_square)100 DEF_ARC_GM(stroke_square) {
101     auto setStroke = [] (SkPaint* p) {
102         p->setStyle(SkPaint::kStroke_Style);
103         p->setStrokeCap(SkPaint::kSquare_Cap);
104     };
105     draw_arcs(canvas, setStroke);
106 }
107 
DEF_ARC_GM(stroke_round)108 DEF_ARC_GM(stroke_round) {
109     auto setStroke = [] (SkPaint* p) {
110         p->setStyle(SkPaint::kStroke_Style);
111         p->setStrokeCap(SkPaint::kRound_Cap);
112     };
113     draw_arcs(canvas, setStroke);
114 }
115 
DEF_ARC_GM(stroke_and_fill_butt)116 DEF_ARC_GM(stroke_and_fill_butt) {
117     auto setStroke = [] (SkPaint* p) {
118         p->setStyle(SkPaint::kStrokeAndFill_Style);
119         p->setStrokeCap(SkPaint::kButt_Cap);
120     };
121     draw_arcs(canvas, setStroke);
122 }
123 
DEF_ARC_GM(stroke_and_fill_square)124 DEF_ARC_GM(stroke_and_fill_square) {
125     auto setStroke = [] (SkPaint* p) {
126         p->setStyle(SkPaint::kStrokeAndFill_Style);
127         p->setStrokeCap(SkPaint::kSquare_Cap);
128     };
129     draw_arcs(canvas, setStroke);
130 }
131 
DEF_ARC_GM(stroke_and_fill_round)132 DEF_ARC_GM(stroke_and_fill_round) {
133     auto setStroke = [] (SkPaint* p) {
134         p->setStyle(SkPaint::kStrokeAndFill_Style);
135         p->setStrokeCap(SkPaint::kRound_Cap);
136     };
137     draw_arcs(canvas, setStroke);
138 }
139 
140 DEF_SIMPLE_GM(circular_arcs_weird, canvas, 1000, 400) {
141     constexpr SkScalar kS = 50;
142     struct Arc {
143         SkRect   fOval;
144         SkScalar fStart;
145         SkScalar fSweep;
146     };
147     const Arc noDrawArcs[] = {
148         // no sweep
149         {SkRect::MakeWH(kS, kS),  0,  0},
150         // empty rect in x
151         {SkRect::MakeWH(-kS, kS), 0, 90},
152         // empty rect in y
153         {SkRect::MakeWH(kS, -kS), 0, 90},
154         // empty rect in x and y
155         {SkRect::MakeWH( 0,   0), 0, 90},
156     };
157     const Arc arcs[] = {
158         // large start
159         {SkRect::MakeWH(kS, kS),   810.f,   90.f},
160         // large negative start
161         {SkRect::MakeWH(kS, kS),  -810.f,   90.f},
162         // exactly 360 sweep
163         {SkRect::MakeWH(kS, kS),     0.f,  360.f},
164         // exactly -360 sweep
165         {SkRect::MakeWH(kS, kS),     0.f, -360.f},
166         // exactly 540 sweep
167         {SkRect::MakeWH(kS, kS),     0.f,  540.f},
168         // exactly -540 sweep
169         {SkRect::MakeWH(kS, kS),     0.f, -540.f},
170         // generic large sweep and large start
171         {SkRect::MakeWH(kS, kS),  1125.f,  990.f},
172     };
173     SkTArray<SkPaint> paints;
174     // fill
175     paints.push_back();
176     // stroke
177     paints.push_back().setStyle(SkPaint::kStroke_Style);
178     paints.back().setStrokeWidth(kS / 6.f);
179     // hairline
180     paints.push_back().setStyle(SkPaint::kStroke_Style);
181     paints.back().setStrokeWidth(0.f);
182     // stroke and fill
183     paints.push_back().setStyle(SkPaint::kStrokeAndFill_Style);
184     paints.back().setStrokeWidth(kS / 6.f);
185     // dash effect
186     paints.push_back().setStyle(SkPaint::kStroke_Style);
187     paints.back().setStrokeWidth(kS / 6.f);
188     constexpr SkScalar kDashIntervals[] = {kS / 15, 2 * kS / 15};
189     paints.back().setPathEffect(SkDashPathEffect::Make(kDashIntervals, 2, 0.f));
190 
191     canvas->translate(kPad, kPad);
192     // This loop should draw nothing.
193     for (auto arc : noDrawArcs) {
194         for (auto paint : paints) {
195             paint.setAntiAlias(true);
196             canvas->drawArc(arc.fOval, arc.fStart, arc.fSweep, false, paint);
197             canvas->drawArc(arc.fOval, arc.fStart, arc.fSweep, true, paint);
198         }
199     }
200 
201     SkPaint linePaint;
202     linePaint.setAntiAlias(true);
203     linePaint.setColor(SK_ColorRED);
204     SkScalar midX   = SK_ARRAY_COUNT(arcs) * (kS + kPad) - kPad/2.f;
205     SkScalar height = paints.count() * (kS + kPad);
206     canvas->drawLine(midX, -kPad, midX, height, linePaint);
207 
208     for (auto paint : paints) {
209         paint.setAntiAlias(true);
210         canvas->save();
211         for (auto arc : arcs) {
212             canvas->drawArc(arc.fOval, arc.fStart, arc.fSweep, false, paint);
213             canvas->translate(kS + kPad, 0.f);
214         }
215         for (auto arc : arcs) {
216             canvas->drawArc(arc.fOval, arc.fStart, arc.fSweep, true, paint);
217             canvas->translate(kS + kPad, 0.f);
218         }
219         canvas->restore();
220         canvas->translate(0, kS + kPad);
221     }
222 }
223 
224 DEF_SIMPLE_GM(onebadarc, canvas, 100, 100) {
225     SkPath path;
226     path.moveTo(SkBits2Float(0x41a00000), SkBits2Float(0x41a00000));  // 20, 20
227     path.lineTo(SkBits2Float(0x4208918c), SkBits2Float(0x4208918c));  // 34.1421f, 34.1421f
228     path.conicTo(SkBits2Float(0x41a00000), SkBits2Float(0x42412318),  // 20, 48.2843f
229             SkBits2Float(0x40bb73a0), SkBits2Float(0x4208918c),       // 5.85786f, 34.1421f
230             SkBits2Float(0x3f3504f3));                                // 0.707107f
231     path.quadTo(SkBits2Float(0x40bb73a0), SkBits2Float(0x4208918c),   // 5.85786f, 34.1421f
232             SkBits2Float(0x40bb73a2), SkBits2Float(0x4208918c));      // 5.85787f, 34.1421f
233     path.lineTo(SkBits2Float(0x41a00000), SkBits2Float(0x41a00000));  // 20, 20
234     path.close();
235     SkPaint p0;
236     p0.setColor(SK_ColorRED);
237     p0.setStrokeWidth(15.f);
238     p0.setStyle(SkPaint::kStroke_Style);
239     p0.setAlpha(100);
240     canvas->translate(20, 0);
241     canvas->drawPath(path, p0);
242 
243     SkRect kRect = { 60, 0, 100, 40};
244     canvas->drawArc(kRect, 45, 90, true, p0);
245 }
246 
247 DEF_SIMPLE_GM(crbug_888453, canvas, 480, 150) {
248     // Two GPU path renderers were using a too-large tolerance when chopping connics to quads.
249     // This manifested as not-very-round circular arcs at certain radii. All the arcs being drawn
250     // here should look like circles.
251     SkPaint fill;
252     fill.setAntiAlias(true);
253     SkPaint hairline = fill;
254     hairline.setStyle(SkPaint::kStroke_Style);
255     SkPaint stroke = hairline;
256     stroke.setStrokeWidth(2.0f);
257     int x = 4;
258     int y0 = 25, y1 = 75, y2 = 125;
259     for (int r = 2; r <= 20; ++r) {
260         canvas->drawArc(SkRect::MakeXYWH(x - r, y0 - r, 2 * r, 2 * r), 0, 360, false, fill);
261         canvas->drawArc(SkRect::MakeXYWH(x - r, y1 - r, 2 * r, 2 * r), 0, 360, false, hairline);
262         canvas->drawArc(SkRect::MakeXYWH(x - r, y2 - r, 2 * r, 2 * r), 0, 360, false, stroke);
263         x += 2 * r + 4;
264     }
265 }
266 
267 DEF_SIMPLE_GM(circular_arc_stroke_matrix, canvas, 820, 1090) {
268     static constexpr SkScalar kRadius = 40.f;
269     static constexpr SkScalar kStrokeWidth = 5.f;
270     static constexpr SkScalar kStart = 89.f;
271     static constexpr SkScalar kSweep = 180.f/SK_ScalarPI; // one radian
272 
273     SkTArray<SkMatrix> matrices;
274     matrices.push_back().setRotate(kRadius, kRadius, 45.f);
275     matrices.push_back(SkMatrix::I());
276     matrices.push_back().setAll(-1,  0,  2*kRadius,
277                                  0,  1,  0,
278                                  0,  0,  1);
279     matrices.push_back().setAll( 1,  0,  0,
280                                  0, -1,  2*kRadius,
281                                  0,  0,  1);
282     matrices.push_back().setAll( 1,  0,  0,
283                                  0, -1,  2*kRadius,
284                                  0,  0,  1);
285     matrices.push_back().setAll( 0, -1,  2*kRadius,
286                                 -1,  0,  2*kRadius,
287                                  0,  0,  1);
288     matrices.push_back().setAll( 0, -1,  2*kRadius,
289                                  1,  0,  0,
290                                  0,  0,  1);
291     matrices.push_back().setAll( 0,  1,  0,
292                                  1,  0,  0,
293                                  0,  0,  1);
294     matrices.push_back().setAll( 0,  1,  0,
295                                 -1,  0,  2*kRadius,
296                                  0,  0,  1);
297     int baseMatrixCnt = matrices.count();
298 
299 
300     SkMatrix tinyCW;
301     tinyCW.setRotate(0.001f, kRadius, kRadius);
302     for (int i = 0; i < baseMatrixCnt; ++i) {
303         matrices.push_back().setConcat(matrices[i], tinyCW);
304     }
305     SkMatrix tinyCCW;
306     tinyCCW.setRotate(-0.001f, kRadius, kRadius);
307     for (int i = 0; i < baseMatrixCnt; ++i) {
308         matrices.push_back().setConcat(matrices[i], tinyCCW);
309     }
310     SkMatrix cw45;
311     cw45.setRotate(45.f, kRadius, kRadius);
312     for (int i = 0; i < baseMatrixCnt; ++i) {
313         matrices.push_back().setConcat(matrices[i], cw45);
314     }
315 
316     int x = 0;
317     int y = 0;
318     static constexpr SkScalar kPad = 2*kStrokeWidth;
319     canvas->translate(kPad, kPad);
320     auto bounds = SkRect::MakeWH(2*kRadius, 2*kRadius);
321     for (auto cap : {SkPaint::kRound_Cap, SkPaint::kButt_Cap, SkPaint::kSquare_Cap}) {
322         for (const auto& m : matrices) {
323             SkPaint paint;
324             paint.setStrokeCap(cap);
325             paint.setAntiAlias(true);
326             paint.setStyle(SkPaint::kStroke_Style);
327             paint.setStrokeWidth(kStrokeWidth);
328             canvas->save();
329                 canvas->translate(x * (2*kRadius + kPad), y * (2*kRadius + kPad));
330                 canvas->concat(m);
331                 paint.setColor(SK_ColorRED);
332                 paint.setAlpha(0x80);
333                 canvas->drawArc(bounds, kStart, kSweep, false, paint);
334                 paint.setColor(SK_ColorBLUE);
335                 paint.setAlpha(0x80);
336                 canvas->drawArc(bounds, kStart, kSweep - 360.f, false, paint);
337             canvas->restore();
338             ++x;
339             if (x == baseMatrixCnt) {
340                 x = 0;
341                 ++y;
342             }
343         }
344     }
345 }
346