blob: 9aba9c653027b1754e770a322655db9dba11f7ca [file] [log] [blame]
reed19d8f9f2015-01-29 10:48:16 -08001/*
2 * Copyright 2015 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.h"
reed76113a92015-02-02 12:55:02 -08009#include "SkAnimTimer.h"
reed19d8f9f2015-01-29 10:48:16 -080010#include "SkCanvas.h"
reed61adb1b2015-02-09 13:01:05 -080011#include "SkPathMeasure.h"
reed19d8f9f2015-01-29 10:48:16 -080012#include "SkRandom.h"
13
14class AddArcGM : public skiagm::GM {
reedd9adfe62015-02-01 19:01:04 -080015public:
16 AddArcGM() : fRotate(0) {}
17
reed19d8f9f2015-01-29 10:48:16 -080018protected:
mtklein36352bf2015-03-25 18:17:31 -070019 SkString onShortName() override { return SkString("addarc"); }
reed19d8f9f2015-01-29 10:48:16 -080020
mtklein36352bf2015-03-25 18:17:31 -070021 SkISize onISize() override { return SkISize::Make(1040, 1040); }
reed19d8f9f2015-01-29 10:48:16 -080022
mtklein36352bf2015-03-25 18:17:31 -070023 void onDraw(SkCanvas* canvas) override {
reed19d8f9f2015-01-29 10:48:16 -080024 canvas->translate(20, 20);
25
26 SkRect r = SkRect::MakeWH(1000, 1000);
27
28 SkPaint paint;
29 paint.setAntiAlias(true);
30 paint.setStyle(SkPaint::kStroke_Style);
31 paint.setStrokeWidth(15);
32
33 const SkScalar inset = paint.getStrokeWidth() + 4;
34 const SkScalar sweepAngle = 345;
35 SkRandom rand;
36
reedd9adfe62015-02-01 19:01:04 -080037 SkScalar sign = 1;
reed19d8f9f2015-01-29 10:48:16 -080038 while (r.width() > paint.getStrokeWidth() * 3) {
caryclark12596012015-07-29 05:27:47 -070039 paint.setColor(sk_tool_utils::color_to_565(rand.nextU() | (0xFF << 24)));
reed19d8f9f2015-01-29 10:48:16 -080040 SkScalar startAngle = rand.nextUScalar1() * 360;
41
reedd9adfe62015-02-01 19:01:04 -080042 SkScalar speed = SkScalarSqrt(16 / r.width()) * 0.5f;
43 startAngle += fRotate * 360 * speed * sign;
44
reed19d8f9f2015-01-29 10:48:16 -080045 SkPath path;
46 path.addArc(r, startAngle, sweepAngle);
47 canvas->drawPath(path, paint);
48
49 r.inset(inset, inset);
reedd9adfe62015-02-01 19:01:04 -080050 sign = -sign;
reed19d8f9f2015-01-29 10:48:16 -080051 }
52 }
53
mtklein36352bf2015-03-25 18:17:31 -070054 bool onAnimate(const SkAnimTimer& timer) override {
reed76113a92015-02-02 12:55:02 -080055 fRotate = timer.scaled(1, 360);
reedd9adfe62015-02-01 19:01:04 -080056 return true;
57 }
58
reed19d8f9f2015-01-29 10:48:16 -080059private:
reedd9adfe62015-02-01 19:01:04 -080060 SkScalar fRotate;
reed19d8f9f2015-01-29 10:48:16 -080061 typedef skiagm::GM INHERITED;
62};
63DEF_GM( return new AddArcGM; )
reed61adb1b2015-02-09 13:01:05 -080064
65///////////////////////////////////////////////////
66
67#define R 400
68
69class AddArcMeasGM : public skiagm::GM {
70public:
71 AddArcMeasGM() {}
72
73protected:
mtklein36352bf2015-03-25 18:17:31 -070074 SkString onShortName() override { return SkString("addarc_meas"); }
reed61adb1b2015-02-09 13:01:05 -080075
mtklein36352bf2015-03-25 18:17:31 -070076 SkISize onISize() override { return SkISize::Make(2*R + 40, 2*R + 40); }
reed61adb1b2015-02-09 13:01:05 -080077
mtklein36352bf2015-03-25 18:17:31 -070078 void onDraw(SkCanvas* canvas) override {
reed61adb1b2015-02-09 13:01:05 -080079 canvas->translate(R + 20, R + 20);
80
81 SkPaint paint;
82 paint.setAntiAlias(true);
83 paint.setStyle(SkPaint::kStroke_Style);
84
85 SkPaint measPaint;
86 measPaint.setAntiAlias(true);
87 measPaint.setColor(SK_ColorRED);
88
89 const SkRect oval = SkRect::MakeLTRB(-R, -R, R, R);
90 canvas->drawOval(oval, paint);
91
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;
96
97 canvas->drawLine(0, 0, rx, ry, paint);
98
99 SkPath path;
100 path.addArc(oval, 0, deg);
101 SkPathMeasure meas(path, false);
102 SkScalar arcLen = rad * R;
103 SkPoint pos;
halcanary96fcdcc2015-08-27 07:41:13 -0700104 if (meas.getPosTan(arcLen, &pos, nullptr)) {
reed61adb1b2015-02-09 13:01:05 -0800105 canvas->drawLine(0, 0, pos.x(), pos.y(), measPaint);
106 }
107 }
108 }
109
110private:
111 typedef skiagm::GM INHERITED;
112};
113DEF_GM( return new AddArcMeasGM; )
reed8ed666d2015-02-10 17:44:26 -0800114
115///////////////////////////////////////////////////
116
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
119//
120class StrokeCircleGM : public skiagm::GM {
121public:
122 StrokeCircleGM() : fRotate(0) {}
halcanary9d524f22016-03-29 09:03:52 -0700123
reed8ed666d2015-02-10 17:44:26 -0800124protected:
mtklein36352bf2015-03-25 18:17:31 -0700125 SkString onShortName() override { return SkString("strokecircle"); }
halcanary9d524f22016-03-29 09:03:52 -0700126
mtklein36352bf2015-03-25 18:17:31 -0700127 SkISize onISize() override { return SkISize::Make(520, 520); }
halcanary9d524f22016-03-29 09:03:52 -0700128
mtklein36352bf2015-03-25 18:17:31 -0700129 void onDraw(SkCanvas* canvas) override {
reed8ed666d2015-02-10 17:44:26 -0800130 canvas->scale(20, 20);
131 canvas->translate(13, 13);
132
133 SkPaint paint;
134 paint.setAntiAlias(true);
135 paint.setStyle(SkPaint::kStroke_Style);
136 paint.setStrokeWidth(SK_Scalar1 / 2);
137
138 const SkScalar delta = paint.getStrokeWidth() * 3 / 2;
139 SkRect r = SkRect::MakeXYWH(-12, -12, 24, 24);
140 SkRandom rand;
141
142 SkScalar sign = 1;
143 while (r.width() > paint.getStrokeWidth() * 2) {
144 SkAutoCanvasRestore acr(canvas, true);
145 canvas->rotate(fRotate * sign);
146
caryclark12596012015-07-29 05:27:47 -0700147 paint.setColor(sk_tool_utils::color_to_565(rand.nextU() | (0xFF << 24)));
reed8ed666d2015-02-10 17:44:26 -0800148 canvas->drawOval(r, paint);
149 r.inset(delta, delta);
150 sign = -sign;
151 }
152 }
153
mtklein36352bf2015-03-25 18:17:31 -0700154 bool onAnimate(const SkAnimTimer& timer) override {
reed8ed666d2015-02-10 17:44:26 -0800155 fRotate = timer.scaled(60, 360);
156 return true;
157 }
158
159private:
160 SkScalar fRotate;
161
162 typedef skiagm::GM INHERITED;
163};
164DEF_GM( return new StrokeCircleGM; )
reed9e779d42015-02-17 11:43:14 -0800165
166//////////////////////
167
liyuqiane60d8552016-10-03 13:49:37 -0700168// Fill circles and rotate them to test our Analytic Anti-Aliasing.
169// This test is based on StrokeCircleGM.
170class FillCircleGM : public skiagm::GM {
171public:
172 FillCircleGM() : fRotate(0) {}
173
174protected:
175 SkString onShortName() override { return SkString("fillcircle"); }
176
177 SkISize onISize() override { return SkISize::Make(520, 520); }
178
179 void onDraw(SkCanvas* canvas) override {
180 canvas->scale(20, 20);
181 canvas->translate(13, 13);
182
183 SkPaint paint;
184 paint.setAntiAlias(true);
185 paint.setStyle(SkPaint::kStroke_Style);
186 paint.setStrokeWidth(SK_Scalar1 / 2);
187
188 const SkScalar strokeWidth = paint.getStrokeWidth();
189 const SkScalar delta = strokeWidth * 3 / 2;
190 SkRect r = SkRect::MakeXYWH(-12, -12, 24, 24);
191 SkRandom rand;
192
193 // Reset style to fill. We only need stroke stype for producing delta and strokeWidth
194 paint.setStyle(SkPaint::kFill_Style);
195
196 SkScalar sign = 1;
197 while (r.width() > strokeWidth * 2) {
198 SkAutoCanvasRestore acr(canvas, true);
199 canvas->rotate(fRotate * sign);
200 paint.setColor(sk_tool_utils::color_to_565(rand.nextU() | (0xFF << 24)));
201 canvas->drawOval(r, paint);
202 r.inset(delta, delta);
203 sign = -sign;
204 }
205 }
206
207 bool onAnimate(const SkAnimTimer& timer) override {
208 fRotate = timer.scaled(60, 360);
209 return true;
210 }
211
212private:
213 SkScalar fRotate;
214
215 typedef skiagm::GM INHERITED;
216};
217DEF_GM( return new FillCircleGM; )
218
219//////////////////////
220
reed9e779d42015-02-17 11:43:14 -0800221static void html_canvas_arc(SkPath* path, SkScalar x, SkScalar y, SkScalar r, SkScalar start,
xidachen6069dda2016-10-06 05:42:23 -0700222 SkScalar end, bool ccw, bool callArcTo) {
reed9e779d42015-02-17 11:43:14 -0800223 SkRect bounds = { x - r, y - r, x + r, y + r };
224 SkScalar sweep = ccw ? end - start : start - end;
xidachen6069dda2016-10-06 05:42:23 -0700225 if (callArcTo)
226 path->arcTo(bounds, start, sweep, false);
227 else
228 path->addArc(bounds, start, sweep);
reed9e779d42015-02-17 11:43:14 -0800229}
230
231// Lifted from canvas-arc-circumference-fill-diffs.html
232class ManyArcsGM : public skiagm::GM {
233public:
234 ManyArcsGM() {}
halcanary9d524f22016-03-29 09:03:52 -0700235
reed9e779d42015-02-17 11:43:14 -0800236protected:
mtklein36352bf2015-03-25 18:17:31 -0700237 SkString onShortName() override { return SkString("manyarcs"); }
halcanary9d524f22016-03-29 09:03:52 -0700238
mtklein36352bf2015-03-25 18:17:31 -0700239 SkISize onISize() override { return SkISize::Make(620, 330); }
halcanary9d524f22016-03-29 09:03:52 -0700240
mtklein36352bf2015-03-25 18:17:31 -0700241 void onDraw(SkCanvas* canvas) override {
reed9e779d42015-02-17 11:43:14 -0800242 SkPaint paint;
243 paint.setAntiAlias(true);
244 paint.setStyle(SkPaint::kStroke_Style);
245
246 canvas->translate(10, 10);
247
248 // 20 angles.
249 SkScalar sweepAngles[] = {
250 -123.7f, -2.3f, -2, -1, -0.3f, -0.000001f, 0, 0.000001f, 0.3f, 0.7f,
251 1, 1.3f, 1.5f, 1.7f, 1.99999f, 2, 2.00001f, 2.3f, 4.3f, 3934723942837.3f
252 };
253 for (size_t i = 0; i < SK_ARRAY_COUNT(sweepAngles); ++i) {
254 sweepAngles[i] *= 180;
255 }
halcanary9d524f22016-03-29 09:03:52 -0700256
reed9e779d42015-02-17 11:43:14 -0800257 SkScalar startAngles[] = { -1, -0.5f, 0, 0.5f };
258 for (size_t i = 0; i < SK_ARRAY_COUNT(startAngles); ++i) {
259 startAngles[i] *= 180;
260 }
halcanary9d524f22016-03-29 09:03:52 -0700261
reed9e779d42015-02-17 11:43:14 -0800262 bool anticlockwise = false;
263 SkScalar sign = 1;
264 for (size_t i = 0; i < SK_ARRAY_COUNT(startAngles) * 2; ++i) {
265 if (i == SK_ARRAY_COUNT(startAngles)) {
266 anticlockwise = true;
267 sign = -1;
268 }
269 SkScalar startAngle = startAngles[i % SK_ARRAY_COUNT(startAngles)] * sign;
270 canvas->save();
271 for (size_t j = 0; j < SK_ARRAY_COUNT(sweepAngles); ++j) {
272 SkPath path;
273 path.moveTo(0, 2);
274 html_canvas_arc(&path, 18, 15, 10, startAngle, startAngle + (sweepAngles[j] * sign),
xidachen6069dda2016-10-06 05:42:23 -0700275 anticlockwise, true);
reed9e779d42015-02-17 11:43:14 -0800276 path.lineTo(0, 28);
277 canvas->drawPath(path, paint);
278 canvas->translate(30, 0);
279 }
280 canvas->restore();
281 canvas->translate(0, 40);
282 }
283 }
halcanary9d524f22016-03-29 09:03:52 -0700284
reed9e779d42015-02-17 11:43:14 -0800285private:
286 typedef skiagm::GM INHERITED;
287};
288DEF_GM( return new ManyArcsGM; )
xidachen6069dda2016-10-06 05:42:23 -0700289
290// Lifted from https://bugs.chromium.org/p/chromium/issues/detail?id=640031
291class TinyAngleBigRadiusArcsGM : public skiagm::GM {
292public:
293 TinyAngleBigRadiusArcsGM() {}
294
295protected:
296 SkString onShortName() override { return SkString("tinyanglearcs"); }
297
298 SkISize onISize() override { return SkISize::Make(620, 330); }
299
300 void onDraw(SkCanvas* canvas) override {
301 SkPaint paint;
302 paint.setAntiAlias(true);
303 paint.setStyle(SkPaint::kStroke_Style);
304
305 canvas->translate(50, 50);
306
xidachen95e34a32016-10-19 10:24:28 -0700307 SkScalar outerRadius = 100000.0f;
308 SkScalar innerRadius = outerRadius - 20.0f;
309 SkScalar centerX = 50;
310 SkScalar centerY = outerRadius;
311 SkScalar startAngles[] = { 1.5f * SK_ScalarPI , 1.501f * SK_ScalarPI };
312 SkScalar sweepAngle = 10.0f / outerRadius;
313
314 for (size_t i = 0; i < SK_ARRAY_COUNT(startAngles); ++i) {
315 SkPath path;
316 SkScalar endAngle = startAngles[i] + sweepAngle;
317 path.moveTo(centerX + innerRadius * sk_float_cos(startAngles[i]),
318 centerY + innerRadius * sk_float_sin(startAngles[i]));
319 path.lineTo(centerX + outerRadius * sk_float_cos(startAngles[i]),
320 centerY + outerRadius * sk_float_sin(startAngles[i]));
321 // A combination of tiny sweepAngle + large radius, we should draw a line.
322 html_canvas_arc(&path, centerX, outerRadius, outerRadius,
323 startAngles[i] * 180 / SK_ScalarPI, endAngle * 180 / SK_ScalarPI,
324 true, true);
325 path.lineTo(centerX + innerRadius * sk_float_cos(endAngle),
326 centerY + innerRadius * sk_float_sin(endAngle));
327 html_canvas_arc(&path, centerX, outerRadius, innerRadius,
328 endAngle * 180 / SK_ScalarPI, startAngles[i] * 180 / SK_ScalarPI,
329 true, false);
330 canvas->drawPath(path, paint);
331 canvas->translate(20, 0);
332 }
xidachen6069dda2016-10-06 05:42:23 -0700333 }
334
335private:
336 typedef skiagm::GM INHERITED;
337};
338DEF_GM( return new TinyAngleBigRadiusArcsGM; )