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 8 #include "gm.h" 9 #include "SkCanvas.h" 10 #include "SkPath.h" 11 #include "SkTArray.h" 12 13 class ConicPathsGM : public skiagm::GM { 14 protected: 15 onShortName()16 SkString onShortName() override { 17 return SkString("conicpaths"); 18 } 19 onISize()20 SkISize onISize() override { 21 return SkISize::Make(920, 960); 22 } 23 onOnceBeforeDraw()24 void onOnceBeforeDraw() override { 25 { 26 const SkScalar w = SkScalarSqrt(2)/2; 27 SkPath* conicCirlce = &fPaths.push_back(); 28 conicCirlce->moveTo(0, 0); 29 conicCirlce->conicTo(0, 50, 50, 50, w); 30 conicCirlce->rConicTo(50, 0, 50, -50, w); 31 conicCirlce->rConicTo(0, -50, -50, -50, w); 32 conicCirlce->rConicTo(-50, 0, -50, 50, w); 33 34 } 35 { 36 SkPath* hyperbola = &fPaths.push_back(); 37 hyperbola->moveTo(0, 0); 38 hyperbola->conicTo(0, 100, 100, 100, 2); 39 } 40 { 41 SkPath* thinHyperbola = &fPaths.push_back(); 42 thinHyperbola->moveTo(0, 0); 43 thinHyperbola->conicTo(100, 100, 5, 0, 2); 44 } 45 { 46 SkPath* veryThinHyperbola = &fPaths.push_back(); 47 veryThinHyperbola->moveTo(0, 0); 48 veryThinHyperbola->conicTo(100, 100, 1, 0, 2); 49 } 50 { 51 SkPath* closedHyperbola = &fPaths.push_back(); 52 closedHyperbola->moveTo(0, 0); 53 closedHyperbola->conicTo(100, 100, 0, 0, 2); 54 } 55 { 56 // using 1 as weight defaults to using quadTo 57 SkPath* nearParabola = &fPaths.push_back(); 58 nearParabola->moveTo(0, 0); 59 nearParabola->conicTo(0, 100, 100, 100, 0.999f); 60 } 61 { 62 SkPath* thinEllipse = &fPaths.push_back(); 63 thinEllipse->moveTo(0, 0); 64 thinEllipse->conicTo(100, 100, 5, 0, SK_ScalarHalf); 65 } 66 { 67 SkPath* veryThinEllipse = &fPaths.push_back(); 68 veryThinEllipse->moveTo(0, 0); 69 veryThinEllipse->conicTo(100, 100, 1, 0, SK_ScalarHalf); 70 } 71 { 72 SkPath* closedEllipse = &fPaths.push_back(); 73 closedEllipse->moveTo(0, 0); 74 closedEllipse->conicTo(100, 100, 0, 0, SK_ScalarHalf); 75 } 76 { 77 const SkScalar w = SkScalarSqrt(2)/2; 78 fGiantCircle.moveTo(2.1e+11f, -1.05e+11f); 79 fGiantCircle.conicTo(2.1e+11f, 0, 1.05e+11f, 0, w); 80 fGiantCircle.conicTo(0, 0, 0, -1.05e+11f, w); 81 fGiantCircle.conicTo(0, -2.1e+11f, 1.05e+11f, -2.1e+11f, w); 82 fGiantCircle.conicTo(2.1e+11f, -2.1e+11f, 2.1e+11f, -1.05e+11f, w); 83 84 } 85 } 86 drawGiantCircle(SkCanvas * canvas)87 void drawGiantCircle(SkCanvas* canvas) { 88 SkPaint paint; 89 canvas->drawPath(fGiantCircle, paint); 90 } 91 onDraw(SkCanvas * canvas)92 void onDraw(SkCanvas* canvas) override { 93 const SkAlpha kAlphaValue[] = { 0xFF, 0x40 }; 94 95 const SkScalar margin = 15; 96 canvas->translate(margin, margin); 97 98 SkPaint paint; 99 for (int p = 0; p < fPaths.count(); ++p) { 100 canvas->save(); 101 for (size_t a = 0; a < SK_ARRAY_COUNT(kAlphaValue); ++a) { 102 paint.setARGB(kAlphaValue[a], 0, 0, 0); 103 for (int aa = 0; aa < 2; ++aa) { 104 paint.setAntiAlias(SkToBool(aa)); 105 for (int fh = 0; fh < 2; ++fh) { 106 paint.setStyle(fh ? SkPaint::kStroke_Style : SkPaint::kFill_Style); 107 108 const SkRect& bounds = fPaths[p].getBounds(); 109 canvas->save(); 110 canvas->translate(-bounds.fLeft, -bounds.fTop); 111 canvas->drawPath(fPaths[p], paint); 112 canvas->restore(); 113 114 canvas->translate(110, 0); 115 } 116 } 117 } 118 canvas->restore(); 119 canvas->translate(0, 110); 120 } 121 canvas->restore(); 122 123 this->drawGiantCircle(canvas); 124 } 125 126 private: 127 SkTArray<SkPath> fPaths; 128 SkPath fGiantCircle; 129 typedef skiagm::GM INHERITED; 130 }; 131 DEF_GM(return new ConicPathsGM;) 132 133 ////////////////////////////////////////////////////////////////////////////// 134 135 /* arc should be on top of circle */ 136 DEF_SIMPLE_GM(arccirclegap, canvas, 250, 250) { 137 canvas->translate(50, 100); 138 SkPoint c = { 1052.5390625f, 506.8760978034711f }; 139 SkScalar radius = 1096.702150363923f; 140 SkPaint paint; 141 paint.setAntiAlias(true); 142 paint.setStyle(SkPaint::kStroke_Style); 143 canvas->drawCircle(c.fX, c.fY, radius, paint); 144 SkPath path; 145 path.moveTo(288.88884710654133f, -280.26680862609f); 146 path.arcTo(0, 0, -39.00216443306411f, 400.6058925796476f, radius); 147 paint.setColor(0xff007f00); 148 canvas->drawPath(path, paint); 149 } 150 151 /* circle should be antialiased */ 152 DEF_SIMPLE_GM(largecircle, canvas, 250, 250) { 153 canvas->translate(50, 100); 154 SkPoint c = { 1052.5390625f, 506.8760978034711f }; 155 SkScalar radius = 1096.702150363923f; 156 SkPaint paint; 157 paint.setAntiAlias(true); 158 paint.setStyle(SkPaint::kStroke_Style); 159 canvas->drawCircle(c.fX, c.fY, radius, paint); 160 } 161 162 DEF_SIMPLE_GM(crbug_640176, canvas, 250, 250) { 163 SkPath path; 164 path.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0 165 path.lineTo(SkBits2Float(0x42cfd89a), SkBits2Float(0xc2700000)); // 103.923f, -60 166 path.lineTo(SkBits2Float(0x42cfd899), SkBits2Float(0xc2700006)); // 103.923f, -60 167 path.conicTo(SkBits2Float(0x42f00000), SkBits2Float(0xc2009d9c), 168 SkBits2Float(0x42f00001), SkBits2Float(0x00000000), 169 SkBits2Float(0x3f7746ea)); // 120, -32.1539f, 120, 0, 0.965926f 170 171 SkPaint paint; 172 paint.setAntiAlias(true); 173 canvas->translate(125, 125); 174 canvas->drawPath(path, paint); 175 } 176