2 * Copyright 2015 Google Inc.
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
9 #include "SkAnimTimer.h"
11 #include "SkPathMeasure.h"
14 class AddArcGM : public skiagm::GM {
16 AddArcGM() : fRotate(0) {}
19 SkString onShortName() override { return SkString("addarc"); }
21 SkISize onISize() override { return SkISize::Make(1040, 1040); }
23 void onDraw(SkCanvas* canvas) override {
24 canvas->translate(20, 20);
26 SkRect r = SkRect::MakeWH(1000, 1000);
29 paint.setAntiAlias(true);
30 paint.setStyle(SkPaint::kStroke_Style);
31 paint.setStrokeWidth(15);
33 const SkScalar inset = paint.getStrokeWidth() + 4;
34 const SkScalar sweepAngle = 345;
38 while (r.width() > paint.getStrokeWidth() * 3) {
39 paint.setColor(rand.nextU() | (0xFF << 24));
40 SkScalar startAngle = rand.nextUScalar1() * 360;
42 SkScalar speed = SkScalarSqrt(16 / r.width()) * 0.5f;
43 startAngle += fRotate * 360 * speed * sign;
46 path.addArc(r, startAngle, sweepAngle);
47 canvas->drawPath(path, paint);
49 r.inset(inset, inset);
54 bool onAnimate(const SkAnimTimer& timer) override {
55 fRotate = timer.scaled(1, 360);
61 typedef skiagm::GM INHERITED;
63 DEF_GM( return new AddArcGM; )
65 ///////////////////////////////////////////////////
69 class AddArcMeasGM : public skiagm::GM {
74 SkString onShortName() override { return SkString("addarc_meas"); }
76 SkISize onISize() override { return SkISize::Make(2*R + 40, 2*R + 40); }
78 void onDraw(SkCanvas* canvas) override {
79 canvas->translate(R + 20, R + 20);
82 paint.setAntiAlias(true);
83 paint.setStyle(SkPaint::kStroke_Style);
86 measPaint.setAntiAlias(true);
87 measPaint.setColor(SK_ColorRED);
89 const SkRect oval = SkRect::MakeLTRB(-R, -R, R, R);
90 canvas->drawOval(oval, paint);
92 for (SkScalar deg = 0; deg < 360; deg += 10) {
93 const SkScalar rad = SkDegreesToRadians(deg);
94 SkScalar rx = SkScalarCos(rad) * R;
95 SkScalar ry = SkScalarSin(rad) * R;
97 canvas->drawLine(0, 0, rx, ry, paint);
100 path.addArc(oval, 0, deg);
101 SkPathMeasure meas(path, false);
102 SkScalar arcLen = rad * R;
104 if (meas.getPosTan(arcLen, &pos, NULL)) {
105 canvas->drawLine(0, 0, pos.x(), pos.y(), measPaint);
111 typedef skiagm::GM INHERITED;
113 DEF_GM( return new AddArcMeasGM; )
115 ///////////////////////////////////////////////////
117 // Emphasize drawing a stroked oval (containing conics) and then scaling the results up,
118 // to ensure that we compute the stroke taking the CTM into account
120 class StrokeCircleGM : public skiagm::GM {
122 StrokeCircleGM() : fRotate(0) {}
125 SkString onShortName() override { return SkString("strokecircle"); }
127 SkISize onISize() override { return SkISize::Make(520, 520); }
129 void onDraw(SkCanvas* canvas) override {
130 canvas->scale(20, 20);
131 canvas->translate(13, 13);
134 paint.setAntiAlias(true);
135 paint.setStyle(SkPaint::kStroke_Style);
136 paint.setStrokeWidth(SK_Scalar1 / 2);
138 const SkScalar delta = paint.getStrokeWidth() * 3 / 2;
139 SkRect r = SkRect::MakeXYWH(-12, -12, 24, 24);
143 while (r.width() > paint.getStrokeWidth() * 2) {
144 SkAutoCanvasRestore acr(canvas, true);
145 canvas->rotate(fRotate * sign);
147 paint.setColor(rand.nextU() | (0xFF << 24));
148 canvas->drawOval(r, paint);
149 r.inset(delta, delta);
154 bool onAnimate(const SkAnimTimer& timer) override {
155 fRotate = timer.scaled(60, 360);
162 typedef skiagm::GM INHERITED;
164 DEF_GM( return new StrokeCircleGM; )
166 //////////////////////
168 static void html_canvas_arc(SkPath* path, SkScalar x, SkScalar y, SkScalar r, SkScalar start,
169 SkScalar end, bool ccw) {
170 SkRect bounds = { x - r, y - r, x + r, y + r };
171 SkScalar sweep = ccw ? end - start : start - end;
172 path->arcTo(bounds, start, sweep, false);
175 // Lifted from canvas-arc-circumference-fill-diffs.html
176 class ManyArcsGM : public skiagm::GM {
181 SkString onShortName() override { return SkString("manyarcs"); }
183 SkISize onISize() override { return SkISize::Make(620, 330); }
185 void onDraw(SkCanvas* canvas) override {
187 paint.setAntiAlias(true);
188 paint.setStyle(SkPaint::kStroke_Style);
190 canvas->translate(10, 10);
193 SkScalar sweepAngles[] = {
194 -123.7f, -2.3f, -2, -1, -0.3f, -0.000001f, 0, 0.000001f, 0.3f, 0.7f,
195 1, 1.3f, 1.5f, 1.7f, 1.99999f, 2, 2.00001f, 2.3f, 4.3f, 3934723942837.3f
197 for (size_t i = 0; i < SK_ARRAY_COUNT(sweepAngles); ++i) {
198 sweepAngles[i] *= 180;
201 SkScalar startAngles[] = { -1, -0.5f, 0, 0.5f };
202 for (size_t i = 0; i < SK_ARRAY_COUNT(startAngles); ++i) {
203 startAngles[i] *= 180;
206 bool anticlockwise = false;
208 for (size_t i = 0; i < SK_ARRAY_COUNT(startAngles) * 2; ++i) {
209 if (i == SK_ARRAY_COUNT(startAngles)) {
210 anticlockwise = true;
213 SkScalar startAngle = startAngles[i % SK_ARRAY_COUNT(startAngles)] * sign;
215 for (size_t j = 0; j < SK_ARRAY_COUNT(sweepAngles); ++j) {
218 html_canvas_arc(&path, 18, 15, 10, startAngle, startAngle + (sweepAngles[j] * sign),
221 canvas->drawPath(path, paint);
222 canvas->translate(30, 0);
225 canvas->translate(0, 40);
230 typedef skiagm::GM INHERITED;
232 DEF_GM( return new ManyArcsGM; )