2015-01-29 18:48:16 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2015 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "gm/gm.h"
|
|
|
|
#include "include/core/SkCanvas.h"
|
2019-04-30 17:44:26 +00:00
|
|
|
#include "include/core/SkColor.h"
|
|
|
|
#include "include/core/SkPaint.h"
|
2020-08-03 15:02:20 +00:00
|
|
|
#include "include/core/SkPathBuilder.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkPathMeasure.h"
|
2019-04-30 17:44:26 +00:00
|
|
|
#include "include/core/SkPoint.h"
|
|
|
|
#include "include/core/SkRect.h"
|
|
|
|
#include "include/core/SkScalar.h"
|
|
|
|
#include "include/core/SkSize.h"
|
|
|
|
#include "include/core/SkString.h"
|
|
|
|
#include "include/core/SkTypes.h"
|
|
|
|
#include "include/private/SkFloatingPoint.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/utils/SkRandom.h"
|
|
|
|
#include "tools/ToolUtils.h"
|
2019-07-11 20:32:53 +00:00
|
|
|
#include "tools/timer/TimeUtils.h"
|
2015-01-29 18:48:16 +00:00
|
|
|
|
|
|
|
class AddArcGM : public skiagm::GM {
|
2015-02-02 03:01:04 +00:00
|
|
|
public:
|
|
|
|
AddArcGM() : fRotate(0) {}
|
|
|
|
|
2015-01-29 18:48:16 +00:00
|
|
|
protected:
|
2015-03-26 01:17:31 +00:00
|
|
|
SkString onShortName() override { return SkString("addarc"); }
|
2015-01-29 18:48:16 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkISize onISize() override { return SkISize::Make(1040, 1040); }
|
2015-01-29 18:48:16 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onDraw(SkCanvas* canvas) override {
|
2015-01-29 18:48:16 +00:00
|
|
|
canvas->translate(20, 20);
|
|
|
|
|
|
|
|
SkRect r = SkRect::MakeWH(1000, 1000);
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2015-01-29 18:48:16 +00:00
|
|
|
paint.setStrokeWidth(15);
|
|
|
|
|
|
|
|
const SkScalar inset = paint.getStrokeWidth() + 4;
|
|
|
|
const SkScalar sweepAngle = 345;
|
|
|
|
SkRandom rand;
|
|
|
|
|
2015-02-02 03:01:04 +00:00
|
|
|
SkScalar sign = 1;
|
2015-01-29 18:48:16 +00:00
|
|
|
while (r.width() > paint.getStrokeWidth() * 3) {
|
2019-03-20 16:12:10 +00:00
|
|
|
paint.setColor(ToolUtils::color_to_565(rand.nextU() | (0xFF << 24)));
|
2015-01-29 18:48:16 +00:00
|
|
|
SkScalar startAngle = rand.nextUScalar1() * 360;
|
|
|
|
|
2015-02-02 03:01:04 +00:00
|
|
|
SkScalar speed = SkScalarSqrt(16 / r.width()) * 0.5f;
|
|
|
|
startAngle += fRotate * 360 * speed * sign;
|
|
|
|
|
2020-08-25 15:48:41 +00:00
|
|
|
SkPathBuilder path;
|
2015-01-29 18:48:16 +00:00
|
|
|
path.addArc(r, startAngle, sweepAngle);
|
2020-12-29 17:18:21 +00:00
|
|
|
canvas->drawPath(path.detach().setIsVolatile(true), paint);
|
2015-01-29 18:48:16 +00:00
|
|
|
|
|
|
|
r.inset(inset, inset);
|
2015-02-02 03:01:04 +00:00
|
|
|
sign = -sign;
|
2015-01-29 18:48:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 20:32:53 +00:00
|
|
|
bool onAnimate(double nanos) override {
|
|
|
|
fRotate = TimeUtils::Scaled(1e-9 * nanos, 1, 360);
|
2015-02-02 03:01:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-01-29 18:48:16 +00:00
|
|
|
private:
|
2015-02-02 03:01:04 +00:00
|
|
|
SkScalar fRotate;
|
2020-09-03 02:42:33 +00:00
|
|
|
using INHERITED = skiagm::GM;
|
2015-01-29 18:48:16 +00:00
|
|
|
};
|
|
|
|
DEF_GM( return new AddArcGM; )
|
2015-02-09 21:01:05 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#define R 400
|
|
|
|
|
2019-01-23 15:40:02 +00:00
|
|
|
DEF_SIMPLE_GM(addarc_meas, canvas, 2*R + 40, 2*R + 40) {
|
2015-02-09 21:01:05 +00:00
|
|
|
canvas->translate(R + 20, R + 20);
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2015-02-09 21:01:05 +00:00
|
|
|
|
|
|
|
SkPaint measPaint;
|
|
|
|
measPaint.setAntiAlias(true);
|
|
|
|
measPaint.setColor(SK_ColorRED);
|
|
|
|
|
|
|
|
const SkRect oval = SkRect::MakeLTRB(-R, -R, R, R);
|
|
|
|
canvas->drawOval(oval, paint);
|
|
|
|
|
|
|
|
for (SkScalar deg = 0; deg < 360; deg += 10) {
|
|
|
|
const SkScalar rad = SkDegreesToRadians(deg);
|
|
|
|
SkScalar rx = SkScalarCos(rad) * R;
|
|
|
|
SkScalar ry = SkScalarSin(rad) * R;
|
|
|
|
|
|
|
|
canvas->drawLine(0, 0, rx, ry, paint);
|
|
|
|
|
2020-08-25 15:48:41 +00:00
|
|
|
SkPathMeasure meas(SkPathBuilder().addArc(oval, 0, deg).detach(), false);
|
2015-02-09 21:01:05 +00:00
|
|
|
SkScalar arcLen = rad * R;
|
|
|
|
SkPoint pos;
|
2015-08-27 14:41:13 +00:00
|
|
|
if (meas.getPosTan(arcLen, &pos, nullptr)) {
|
2017-05-15 17:35:35 +00:00
|
|
|
canvas->drawLine({0, 0}, pos, measPaint);
|
2015-02-09 21:01:05 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-23 15:40:02 +00:00
|
|
|
}
|
2015-02-11 01:44:26 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Emphasize drawing a stroked oval (containing conics) and then scaling the results up,
|
|
|
|
// to ensure that we compute the stroke taking the CTM into account
|
|
|
|
//
|
|
|
|
class StrokeCircleGM : public skiagm::GM {
|
|
|
|
public:
|
|
|
|
StrokeCircleGM() : fRotate(0) {}
|
2016-03-29 16:03:52 +00:00
|
|
|
|
2015-02-11 01:44:26 +00:00
|
|
|
protected:
|
2015-03-26 01:17:31 +00:00
|
|
|
SkString onShortName() override { return SkString("strokecircle"); }
|
2016-03-29 16:03:52 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkISize onISize() override { return SkISize::Make(520, 520); }
|
2016-03-29 16:03:52 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onDraw(SkCanvas* canvas) override {
|
2015-02-11 01:44:26 +00:00
|
|
|
canvas->scale(20, 20);
|
|
|
|
canvas->translate(13, 13);
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2015-02-11 01:44:26 +00:00
|
|
|
paint.setStrokeWidth(SK_Scalar1 / 2);
|
|
|
|
|
|
|
|
const SkScalar delta = paint.getStrokeWidth() * 3 / 2;
|
|
|
|
SkRect r = SkRect::MakeXYWH(-12, -12, 24, 24);
|
|
|
|
SkRandom rand;
|
|
|
|
|
|
|
|
SkScalar sign = 1;
|
|
|
|
while (r.width() > paint.getStrokeWidth() * 2) {
|
|
|
|
SkAutoCanvasRestore acr(canvas, true);
|
|
|
|
canvas->rotate(fRotate * sign);
|
|
|
|
|
2019-03-20 16:12:10 +00:00
|
|
|
paint.setColor(ToolUtils::color_to_565(rand.nextU() | (0xFF << 24)));
|
2015-02-11 01:44:26 +00:00
|
|
|
canvas->drawOval(r, paint);
|
|
|
|
r.inset(delta, delta);
|
|
|
|
sign = -sign;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 20:32:53 +00:00
|
|
|
bool onAnimate(double nanos) override {
|
|
|
|
fRotate = TimeUtils::Scaled(1e-9 * nanos, 60, 360);
|
2015-02-11 01:44:26 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SkScalar fRotate;
|
|
|
|
|
2020-09-03 02:42:33 +00:00
|
|
|
using INHERITED = skiagm::GM;
|
2015-02-11 01:44:26 +00:00
|
|
|
};
|
|
|
|
DEF_GM( return new StrokeCircleGM; )
|
2015-02-17 19:43:14 +00:00
|
|
|
|
|
|
|
//////////////////////
|
|
|
|
|
2016-10-03 20:49:37 +00:00
|
|
|
// Fill circles and rotate them to test our Analytic Anti-Aliasing.
|
|
|
|
// This test is based on StrokeCircleGM.
|
|
|
|
class FillCircleGM : public skiagm::GM {
|
|
|
|
public:
|
|
|
|
FillCircleGM() : fRotate(0) {}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
SkString onShortName() override { return SkString("fillcircle"); }
|
|
|
|
|
|
|
|
SkISize onISize() override { return SkISize::Make(520, 520); }
|
|
|
|
|
|
|
|
void onDraw(SkCanvas* canvas) override {
|
|
|
|
canvas->scale(20, 20);
|
|
|
|
canvas->translate(13, 13);
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2016-10-03 20:49:37 +00:00
|
|
|
paint.setStrokeWidth(SK_Scalar1 / 2);
|
|
|
|
|
|
|
|
const SkScalar strokeWidth = paint.getStrokeWidth();
|
|
|
|
const SkScalar delta = strokeWidth * 3 / 2;
|
|
|
|
SkRect r = SkRect::MakeXYWH(-12, -12, 24, 24);
|
|
|
|
SkRandom rand;
|
|
|
|
|
|
|
|
// Reset style to fill. We only need stroke stype for producing delta and strokeWidth
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(false);
|
2016-10-03 20:49:37 +00:00
|
|
|
|
|
|
|
SkScalar sign = 1;
|
|
|
|
while (r.width() > strokeWidth * 2) {
|
|
|
|
SkAutoCanvasRestore acr(canvas, true);
|
|
|
|
canvas->rotate(fRotate * sign);
|
2019-03-20 16:12:10 +00:00
|
|
|
paint.setColor(ToolUtils::color_to_565(rand.nextU() | (0xFF << 24)));
|
2016-10-03 20:49:37 +00:00
|
|
|
canvas->drawOval(r, paint);
|
|
|
|
r.inset(delta, delta);
|
|
|
|
sign = -sign;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 20:32:53 +00:00
|
|
|
bool onAnimate(double nanos) override {
|
|
|
|
fRotate = TimeUtils::Scaled(1e-9 * nanos, 60, 360);
|
2016-10-03 20:49:37 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SkScalar fRotate;
|
|
|
|
|
2020-09-03 02:42:33 +00:00
|
|
|
using INHERITED = skiagm::GM;
|
2016-10-03 20:49:37 +00:00
|
|
|
};
|
|
|
|
DEF_GM( return new FillCircleGM; )
|
|
|
|
|
|
|
|
//////////////////////
|
|
|
|
|
2020-08-03 15:02:20 +00:00
|
|
|
static void html_canvas_arc(SkPathBuilder* path, SkScalar x, SkScalar y, SkScalar r, SkScalar start,
|
2016-10-06 12:42:23 +00:00
|
|
|
SkScalar end, bool ccw, bool callArcTo) {
|
2015-02-17 19:43:14 +00:00
|
|
|
SkRect bounds = { x - r, y - r, x + r, y + r };
|
|
|
|
SkScalar sweep = ccw ? end - start : start - end;
|
2016-10-06 12:42:23 +00:00
|
|
|
if (callArcTo)
|
|
|
|
path->arcTo(bounds, start, sweep, false);
|
|
|
|
else
|
|
|
|
path->addArc(bounds, start, sweep);
|
2015-02-17 19:43:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Lifted from canvas-arc-circumference-fill-diffs.html
|
2019-01-23 15:40:02 +00:00
|
|
|
DEF_SIMPLE_GM(manyarcs, canvas, 620, 330) {
|
2015-02-17 19:43:14 +00:00
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2015-02-17 19:43:14 +00:00
|
|
|
|
|
|
|
canvas->translate(10, 10);
|
|
|
|
|
|
|
|
// 20 angles.
|
|
|
|
SkScalar sweepAngles[] = {
|
|
|
|
-123.7f, -2.3f, -2, -1, -0.3f, -0.000001f, 0, 0.000001f, 0.3f, 0.7f,
|
|
|
|
1, 1.3f, 1.5f, 1.7f, 1.99999f, 2, 2.00001f, 2.3f, 4.3f, 3934723942837.3f
|
|
|
|
};
|
2022-06-21 13:49:17 +00:00
|
|
|
for (size_t i = 0; i < std::size(sweepAngles); ++i) {
|
2015-02-17 19:43:14 +00:00
|
|
|
sweepAngles[i] *= 180;
|
|
|
|
}
|
2016-03-29 16:03:52 +00:00
|
|
|
|
2015-02-17 19:43:14 +00:00
|
|
|
SkScalar startAngles[] = { -1, -0.5f, 0, 0.5f };
|
2022-06-21 13:49:17 +00:00
|
|
|
for (size_t i = 0; i < std::size(startAngles); ++i) {
|
2015-02-17 19:43:14 +00:00
|
|
|
startAngles[i] *= 180;
|
|
|
|
}
|
2016-03-29 16:03:52 +00:00
|
|
|
|
2015-02-17 19:43:14 +00:00
|
|
|
bool anticlockwise = false;
|
|
|
|
SkScalar sign = 1;
|
2022-06-21 13:49:17 +00:00
|
|
|
for (size_t i = 0; i < std::size(startAngles) * 2; ++i) {
|
|
|
|
if (i == std::size(startAngles)) {
|
2015-02-17 19:43:14 +00:00
|
|
|
anticlockwise = true;
|
|
|
|
sign = -1;
|
|
|
|
}
|
2022-06-21 13:49:17 +00:00
|
|
|
SkScalar startAngle = startAngles[i % std::size(startAngles)] * sign;
|
2015-02-17 19:43:14 +00:00
|
|
|
canvas->save();
|
2022-06-21 13:49:17 +00:00
|
|
|
for (size_t j = 0; j < std::size(sweepAngles); ++j) {
|
2020-08-03 15:02:20 +00:00
|
|
|
SkPathBuilder path;
|
2015-02-17 19:43:14 +00:00
|
|
|
path.moveTo(0, 2);
|
|
|
|
html_canvas_arc(&path, 18, 15, 10, startAngle, startAngle + (sweepAngles[j] * sign),
|
2016-10-06 12:42:23 +00:00
|
|
|
anticlockwise, true);
|
2015-02-17 19:43:14 +00:00
|
|
|
path.lineTo(0, 28);
|
2020-12-29 17:18:21 +00:00
|
|
|
canvas->drawPath(path.detach().setIsVolatile(true), paint);
|
2015-02-17 19:43:14 +00:00
|
|
|
canvas->translate(30, 0);
|
|
|
|
}
|
|
|
|
canvas->restore();
|
|
|
|
canvas->translate(0, 40);
|
|
|
|
}
|
2019-01-23 15:40:02 +00:00
|
|
|
}
|
2016-10-06 12:42:23 +00:00
|
|
|
|
|
|
|
// Lifted from https://bugs.chromium.org/p/chromium/issues/detail?id=640031
|
2019-01-23 15:40:02 +00:00
|
|
|
DEF_SIMPLE_GM(tinyanglearcs, canvas, 620, 330) {
|
2016-10-06 12:42:23 +00:00
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
2020-05-19 01:25:44 +00:00
|
|
|
paint.setStroke(true);
|
2016-10-06 12:42:23 +00:00
|
|
|
|
|
|
|
canvas->translate(50, 50);
|
|
|
|
|
2016-10-19 17:24:28 +00:00
|
|
|
SkScalar outerRadius = 100000.0f;
|
|
|
|
SkScalar innerRadius = outerRadius - 20.0f;
|
|
|
|
SkScalar centerX = 50;
|
|
|
|
SkScalar centerY = outerRadius;
|
|
|
|
SkScalar startAngles[] = { 1.5f * SK_ScalarPI , 1.501f * SK_ScalarPI };
|
|
|
|
SkScalar sweepAngle = 10.0f / outerRadius;
|
|
|
|
|
2022-06-21 13:49:17 +00:00
|
|
|
for (size_t i = 0; i < std::size(startAngles); ++i) {
|
2020-08-03 15:02:20 +00:00
|
|
|
SkPathBuilder path;
|
2016-10-19 17:24:28 +00:00
|
|
|
SkScalar endAngle = startAngles[i] + sweepAngle;
|
|
|
|
path.moveTo(centerX + innerRadius * sk_float_cos(startAngles[i]),
|
|
|
|
centerY + innerRadius * sk_float_sin(startAngles[i]));
|
|
|
|
path.lineTo(centerX + outerRadius * sk_float_cos(startAngles[i]),
|
|
|
|
centerY + outerRadius * sk_float_sin(startAngles[i]));
|
|
|
|
// A combination of tiny sweepAngle + large radius, we should draw a line.
|
|
|
|
html_canvas_arc(&path, centerX, outerRadius, outerRadius,
|
|
|
|
startAngles[i] * 180 / SK_ScalarPI, endAngle * 180 / SK_ScalarPI,
|
|
|
|
true, true);
|
|
|
|
path.lineTo(centerX + innerRadius * sk_float_cos(endAngle),
|
|
|
|
centerY + innerRadius * sk_float_sin(endAngle));
|
|
|
|
html_canvas_arc(&path, centerX, outerRadius, innerRadius,
|
|
|
|
endAngle * 180 / SK_ScalarPI, startAngles[i] * 180 / SK_ScalarPI,
|
|
|
|
true, false);
|
2020-08-03 15:02:20 +00:00
|
|
|
canvas->drawPath(path.detach(), paint);
|
2016-10-19 17:24:28 +00:00
|
|
|
canvas->translate(20, 0);
|
|
|
|
}
|
2019-01-23 15:40:02 +00:00
|
|
|
}
|