egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2013 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 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "gm/gm.h" |
| 9 | #include "include/core/SkCanvas.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 10 | #include "include/core/SkColor.h" |
| 11 | #include "include/core/SkPaint.h" |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 12 | #include "include/core/SkPathBuilder.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 13 | #include "include/core/SkPoint.h" |
| 14 | #include "include/core/SkRect.h" |
| 15 | #include "include/core/SkScalar.h" |
| 16 | #include "include/core/SkSize.h" |
| 17 | #include "include/core/SkString.h" |
| 18 | #include "include/core/SkTypes.h" |
Kevin Lubick | dc6cc02 | 2023-01-13 11:24:27 -0500 | [diff] [blame] | 19 | #include "include/private/base/SkFloatBits.h" |
| 20 | #include "include/private/base/SkTArray.h" |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 21 | |
Herb Derby | ec96c21 | 2023-03-06 10:31:22 -0500 | [diff] [blame] | 22 | using namespace skia_private; |
| 23 | |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 24 | class ConicPathsGM : public skiagm::GM { |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 25 | protected: |
Leandro Lovisolo | 24fa211 | 2023-08-15 19:05:17 +0000 | [diff] [blame] | 26 | SkString getName() const override { return SkString("conicpaths"); } |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 27 | |
Leandro Lovisolo | 8f02388 | 2023-08-15 21:13:52 +0000 | [diff] [blame] | 28 | SkISize getISize() override { return SkISize::Make(920, 960); } |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 29 | |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 30 | template <typename Proc> void append_path(Proc proc) { |
| 31 | SkPathBuilder b; |
| 32 | proc(&b); |
| 33 | fPaths.push_back(b.detach()); |
| 34 | } |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 35 | |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 36 | void onOnceBeforeDraw() override { |
| 37 | this->append_path([](SkPathBuilder* conicCircle) { |
| 38 | const SkScalar w = SkScalarSqrt(2)/2; |
| 39 | conicCircle->moveTo(0, 0); |
| 40 | conicCircle->conicTo(0, 50, 50, 50, w); |
| 41 | conicCircle->rConicTo(50, 0, 50, -50, w); |
| 42 | conicCircle->rConicTo(0, -50, -50, -50, w); |
| 43 | conicCircle->rConicTo(-50, 0, -50, 50, w); |
| 44 | }); |
| 45 | |
| 46 | this->append_path([](SkPathBuilder* hyperbola) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 47 | hyperbola->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 48 | hyperbola->conicTo(0, 100, 100, 100, 2); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 49 | }); |
| 50 | |
| 51 | this->append_path([](SkPathBuilder* thinHyperbola) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 52 | thinHyperbola->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 53 | thinHyperbola->conicTo(100, 100, 5, 0, 2); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 54 | }); |
| 55 | |
| 56 | this->append_path([](SkPathBuilder* veryThinHyperbola) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 57 | veryThinHyperbola->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 58 | veryThinHyperbola->conicTo(100, 100, 1, 0, 2); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 59 | }); |
| 60 | |
| 61 | this->append_path([](SkPathBuilder* closedHyperbola) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 62 | closedHyperbola->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 63 | closedHyperbola->conicTo(100, 100, 0, 0, 2); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 64 | }); |
| 65 | |
| 66 | this->append_path([](SkPathBuilder* nearParabola) { |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 67 | // using 1 as weight defaults to using quadTo |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 68 | nearParabola->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 69 | nearParabola->conicTo(0, 100, 100, 100, 0.999f); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 70 | }); |
| 71 | |
| 72 | this->append_path([](SkPathBuilder* thinEllipse) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 73 | thinEllipse->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 74 | thinEllipse->conicTo(100, 100, 5, 0, SK_ScalarHalf); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 75 | }); |
| 76 | |
| 77 | this->append_path([](SkPathBuilder* veryThinEllipse) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 78 | veryThinEllipse->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 79 | veryThinEllipse->conicTo(100, 100, 1, 0, SK_ScalarHalf); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 80 | }); |
| 81 | |
| 82 | this->append_path([](SkPathBuilder* closedEllipse) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 83 | closedEllipse->moveTo(0, 0); |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 84 | closedEllipse->conicTo(100, 100, 0, 0, SK_ScalarHalf); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 85 | }); |
egdaniel | 5a23a14 | 2015-02-25 06:41:47 -0800 | [diff] [blame] | 86 | |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 87 | { |
| 88 | SkPathBuilder b; |
| 89 | const SkScalar w = SkScalarSqrt(2)/2; |
| 90 | b.moveTo(2.1e+11f, -1.05e+11f); |
| 91 | b.conicTo(2.1e+11f, 0, 1.05e+11f, 0, w); |
| 92 | b.conicTo(0, 0, 0, -1.05e+11f, w); |
| 93 | b.conicTo(0, -2.1e+11f, 1.05e+11f, -2.1e+11f, w); |
| 94 | b.conicTo(2.1e+11f, -2.1e+11f, 2.1e+11f, -1.05e+11f, w); |
| 95 | fGiantCircle = b.detach(); |
egdaniel | 5a23a14 | 2015-02-25 06:41:47 -0800 | [diff] [blame] | 96 | } |
| 97 | } |
| 98 | |
| 99 | void drawGiantCircle(SkCanvas* canvas) { |
| 100 | SkPaint paint; |
| 101 | canvas->drawPath(fGiantCircle, paint); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 102 | } |
| 103 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 104 | void onDraw(SkCanvas* canvas) override { |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 105 | const SkAlpha kAlphaValue[] = { 0xFF, 0x40 }; |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 106 | |
reed | d1bd1d7 | 2014-12-31 20:07:01 -0800 | [diff] [blame] | 107 | const SkScalar margin = 15; |
| 108 | canvas->translate(margin, margin); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 109 | |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 110 | SkPaint paint; |
Herb Derby | ffacce5 | 2022-11-09 10:51:34 -0500 | [diff] [blame] | 111 | for (int p = 0; p < fPaths.size(); ++p) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 112 | canvas->save(); |
Herb Derby | c37b386 | 2022-06-21 09:49:17 -0400 | [diff] [blame] | 113 | for (size_t a = 0; a < std::size(kAlphaValue); ++a) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 114 | paint.setARGB(kAlphaValue[a], 0, 0, 0); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 115 | for (int aa = 0; aa < 2; ++aa) { |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 116 | paint.setAntiAlias(SkToBool(aa)); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 117 | for (int fh = 0; fh < 2; ++fh) { |
Mike Reed | 1963009 | 2020-05-18 21:25:44 -0400 | [diff] [blame] | 118 | paint.setStroke(fh != 0); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 119 | |
| 120 | const SkRect& bounds = fPaths[p].getBounds(); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 121 | canvas->save(); |
| 122 | canvas->translate(-bounds.fLeft, -bounds.fTop); |
| 123 | canvas->drawPath(fPaths[p], paint); |
| 124 | canvas->restore(); |
| 125 | |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 126 | canvas->translate(110, 0); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 127 | } |
| 128 | } |
| 129 | } |
reed | 40c85e4 | 2015-01-05 10:01:25 -0800 | [diff] [blame] | 130 | canvas->restore(); |
| 131 | canvas->translate(0, 110); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 132 | } |
| 133 | canvas->restore(); |
egdaniel | 5a23a14 | 2015-02-25 06:41:47 -0800 | [diff] [blame] | 134 | |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 135 | this->drawGiantCircle(canvas); |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 136 | } |
| 137 | |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 138 | private: |
Herb Derby | ec96c21 | 2023-03-06 10:31:22 -0500 | [diff] [blame] | 139 | TArray<SkPath> fPaths; |
egdaniel | 5a23a14 | 2015-02-25 06:41:47 -0800 | [diff] [blame] | 140 | SkPath fGiantCircle; |
John Stiles | 7571f9e | 2020-09-02 22:42:33 -0400 | [diff] [blame] | 141 | using INHERITED = skiagm::GM; |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 142 | }; |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 143 | DEF_GM(return new ConicPathsGM;) |
egdaniel@google.com | def9f6e | 2013-06-20 16:54:31 +0000 | [diff] [blame] | 144 | |
| 145 | ////////////////////////////////////////////////////////////////////////////// |
| 146 | |
caryclark | 2b39ffc | 2016-01-20 07:46:05 -0800 | [diff] [blame] | 147 | /* arc should be on top of circle */ |
| 148 | DEF_SIMPLE_GM(arccirclegap, canvas, 250, 250) { |
| 149 | canvas->translate(50, 100); |
| 150 | SkPoint c = { 1052.5390625f, 506.8760978034711f }; |
| 151 | SkScalar radius = 1096.702150363923f; |
| 152 | SkPaint paint; |
| 153 | paint.setAntiAlias(true); |
Mike Reed | 1963009 | 2020-05-18 21:25:44 -0400 | [diff] [blame] | 154 | paint.setStroke(true); |
Hal Canary | 23e474c | 2017-05-15 13:35:35 -0400 | [diff] [blame] | 155 | canvas->drawCircle(c, radius, paint); |
Mike Reed | cfb130c | 2020-08-03 11:02:20 -0400 | [diff] [blame] | 156 | SkPath path = SkPathBuilder().moveTo(288.88884710654133f, -280.26680862609f) |
| 157 | .arcTo({0, 0}, {-39.00216443306411f, 400.6058925796476f}, radius) |
| 158 | .detach(); |
caryclark | 2b39ffc | 2016-01-20 07:46:05 -0800 | [diff] [blame] | 159 | paint.setColor(0xff007f00); |
| 160 | canvas->drawPath(path, paint); |
| 161 | } |
caryclark | 531191f | 2016-08-24 11:59:30 -0700 | [diff] [blame] | 162 | |
Jim Van Verth | 6750e91 | 2016-12-19 14:45:19 -0500 | [diff] [blame] | 163 | /* circle should be antialiased */ |
| 164 | DEF_SIMPLE_GM(largecircle, canvas, 250, 250) { |
| 165 | canvas->translate(50, 100); |
| 166 | SkPoint c = { 1052.5390625f, 506.8760978034711f }; |
| 167 | SkScalar radius = 1096.702150363923f; |
| 168 | SkPaint paint; |
| 169 | paint.setAntiAlias(true); |
Mike Reed | 1963009 | 2020-05-18 21:25:44 -0400 | [diff] [blame] | 170 | paint.setStroke(true); |
Hal Canary | 23e474c | 2017-05-15 13:35:35 -0400 | [diff] [blame] | 171 | canvas->drawCircle(c, radius, paint); |
Jim Van Verth | 6750e91 | 2016-12-19 14:45:19 -0500 | [diff] [blame] | 172 | } |
| 173 | |
Jim Van Verth | 20ae25c | 2019-03-29 08:50:41 -0400 | [diff] [blame] | 174 | /* ovals should not be blurry */ |
| 175 | DEF_SIMPLE_GM(largeovals, canvas, 250, 250) { |
| 176 | // Test EllipseOp |
| 177 | SkRect r = SkRect::MakeXYWH(-520, -520, 5000, 4000); |
| 178 | SkPaint paint; |
| 179 | paint.setAntiAlias(true); |
Mike Reed | 1963009 | 2020-05-18 21:25:44 -0400 | [diff] [blame] | 180 | paint.setStroke(true); |
Jim Van Verth | 20ae25c | 2019-03-29 08:50:41 -0400 | [diff] [blame] | 181 | paint.setStrokeWidth(100); |
| 182 | canvas->drawOval(r, paint); |
| 183 | r.offset(-15, -15); |
| 184 | paint.setColor(SK_ColorDKGRAY); |
| 185 | // we use stroke and fill to avoid falling into the SimpleFill path |
| 186 | paint.setStyle(SkPaint::kStrokeAndFill_Style); |
| 187 | paint.setStrokeWidth(1); |
| 188 | canvas->drawOval(r, paint); |
| 189 | |
| 190 | // Test DIEllipseOp |
| 191 | canvas->rotate(1.0f); |
| 192 | r.offset(55, 55); |
| 193 | paint.setColor(SK_ColorGRAY); |
Mike Reed | 1963009 | 2020-05-18 21:25:44 -0400 | [diff] [blame] | 194 | paint.setStroke(true); |
Jim Van Verth | 20ae25c | 2019-03-29 08:50:41 -0400 | [diff] [blame] | 195 | paint.setStrokeWidth(100); |
| 196 | canvas->drawOval(r, paint); |
| 197 | r.offset(-15, -15); |
| 198 | paint.setColor(SK_ColorLTGRAY); |
| 199 | paint.setStyle(SkPaint::kStrokeAndFill_Style); |
| 200 | paint.setStrokeWidth(1); |
| 201 | canvas->drawOval(r, paint); |
| 202 | } |
| 203 | |
caryclark | 531191f | 2016-08-24 11:59:30 -0700 | [diff] [blame] | 204 | DEF_SIMPLE_GM(crbug_640176, canvas, 250, 250) { |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 205 | SkPathBuilder path; |
caryclark | 531191f | 2016-08-24 11:59:30 -0700 | [diff] [blame] | 206 | path.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0 |
| 207 | path.lineTo(SkBits2Float(0x42cfd89a), SkBits2Float(0xc2700000)); // 103.923f, -60 |
| 208 | path.lineTo(SkBits2Float(0x42cfd899), SkBits2Float(0xc2700006)); // 103.923f, -60 |
| 209 | path.conicTo(SkBits2Float(0x42f00000), SkBits2Float(0xc2009d9c), |
| 210 | SkBits2Float(0x42f00001), SkBits2Float(0x00000000), |
| 211 | SkBits2Float(0x3f7746ea)); // 120, -32.1539f, 120, 0, 0.965926f |
| 212 | |
| 213 | SkPaint paint; |
| 214 | paint.setAntiAlias(true); |
| 215 | canvas->translate(125, 125); |
Mike Reed | 2243d77 | 2020-07-31 21:57:51 -0400 | [diff] [blame] | 216 | canvas->drawPath(path.detach(), paint); |
caryclark | 531191f | 2016-08-24 11:59:30 -0700 | [diff] [blame] | 217 | } |