Index: gm/shapes.cpp |
diff --git a/gm/shapes.cpp b/gm/shapes.cpp |
new file mode 100644 |
index 0000000000000000000000000000000000000000..02b6a9827be02022a7c612f24d13bff14ab70a41 |
--- /dev/null |
+++ b/gm/shapes.cpp |
@@ -0,0 +1,164 @@ |
+ |
+/* |
+ * Copyright 2016 Google Inc. |
+ * |
+ * Use of this source code is governed by a BSD-style license that can be |
+ * found in the LICENSE file. |
+ */ |
+ |
+#include "gm.h" |
+#include "SkRandom.h" |
+#include "SkRRect.h" |
+ |
+namespace skiagm { |
+ |
+/* |
+ * This is the base class for two GMs that cover various corner cases with primitive Skia shapes |
+ * (zero radius, near-zero radius, inner shape overlap, etc.) It uses an xfermode of darken to help |
+ * double-blended and/or dropped pixels stand out. |
+ */ |
+class ShapesGM : public GM { |
+protected: |
+ ShapesGM(const char* name, bool antialias) : fName(name), fAntialias(antialias) { |
+ fShapes.push_back().setOval(SkRect::MakeXYWH(-5, 25, 200, 100)); |
+ fRotations.push_back(21); |
+ |
+ fShapes.push_back().setRect(SkRect::MakeXYWH(95, 75, 125, 100)); |
+ fRotations.push_back(94); |
+ |
+ fShapes.push_back().setRectXY(SkRect::MakeXYWH(0, 75, 150, 100), 1e-5f, 1e-5f); |
+ fRotations.push_back(132); |
+ |
+ fShapes.push_back().setRectXY(SkRect::MakeXYWH(15, -20, 100, 100), 20, 15); |
+ fRotations.push_back(282); |
+ |
+ fSimpleShapeCount = fShapes.count(); |
+ |
+ fShapes.push_back().setNinePatch(SkRect::MakeXYWH(140, -50, 90, 110), 10, 5, 25, 35); |
+ fRotations.push_back(0); |
+ |
+ fShapes.push_back().setNinePatch(SkRect::MakeXYWH(160, -60, 60, 90), 10, 60, 50, 30); |
+ fRotations.push_back(-35); |
+ |
+ fShapes.push_back().setNinePatch(SkRect::MakeXYWH(220, -120, 60, 90), 1, 89, 59, 1); |
+ fRotations.push_back(65); |
+ |
+ SkVector radii[4] = {{4, 6}, {12, 8}, {24, 16}, {32, 48}}; |
+ fShapes.push_back().setRectRadii(SkRect::MakeXYWH(150, -129, 80, 160), radii); |
+ fRotations.push_back(265); |
+ |
+ SkVector radii2[4] = {{0, 0}, {80, 60}, {0, 0}, {80, 60}}; |
+ fShapes.push_back().setRectRadii(SkRect::MakeXYWH(180, -30, 80, 60), radii2); |
+ fRotations.push_back(295); |
+ |
+ if (!antialias) { |
+ fName.append("_bw"); |
+ } |
+ } |
+ |
+ SkString onShortName() override final { return fName; } |
+ SkISize onISize() override { return SkISize::Make(500, 500); } |
+ |
+ void onOnceBeforeDraw() override { |
+ fPaint.setXfermodeMode(SkXfermode::kDarken_Mode); |
+ fPaint.setAntiAlias(fAntialias); |
+ } |
+ |
+ void onDraw(SkCanvas* canvas) override { |
+ canvas->clear(SK_ColorWHITE); |
+ |
+ canvas->save(); |
+ canvas->translate(canvas->imageInfo().width() / 2.f, canvas->imageInfo().height() / 2.f); |
+ this->drawShapes(canvas); |
+ canvas->restore(); |
+ } |
+ |
+ virtual void drawShapes(SkCanvas* canvas) const = 0; |
+ |
+protected: |
+ SkString fName; |
+ bool fAntialias; |
+ SkPaint fPaint; |
+ SkTArray<SkRRect> fShapes; |
+ SkTArray<SkScalar> fRotations; |
+ int fSimpleShapeCount; |
+ |
+private: |
+ typedef GM INHERITED; |
+}; |
+ |
+class SimpleShapesGM : public ShapesGM { |
+public: |
+ SimpleShapesGM(bool antialias) : INHERITED("simpleshapes", antialias) {} |
+ |
+private: |
+ void drawShapes(SkCanvas* canvas) const override { |
+ SkRandom rand(2); |
+ for (int i = 0; i < fShapes.count(); i++) { |
+ SkPaint paint(fPaint); |
+ paint.setColor(rand.nextU() & ~0x808080); |
+ paint.setAlpha(128); // Use alpha to detect double blends. |
+ const SkRRect& shape = fShapes[i]; |
+ canvas->save(); |
+ canvas->rotate(fRotations[i]); |
+ switch (shape.getType()) { |
+ case SkRRect::kRect_Type: |
+ canvas->drawRect(shape.rect(), paint); |
+ break; |
+ case SkRRect::kOval_Type: |
+ canvas->drawOval(shape.rect(), paint); |
+ break; |
+ default: |
+ canvas->drawRRect(shape, paint); |
+ break; |
+ } |
+ canvas->restore(); |
+ } |
+ } |
+ |
+ typedef ShapesGM INHERITED; |
+}; |
+ |
+class InnerShapesGM : public ShapesGM { |
+public: |
+ InnerShapesGM(bool antialias) : INHERITED("innershapes", antialias) {} |
+ |
+private: |
+ void drawShapes(SkCanvas* canvas) const override { |
+ SkRandom rand; |
+ for (int i = 0; i < fShapes.count(); i++) { |
+ const SkRRect& outer = fShapes[i]; |
+ const SkRRect& inner = fShapes[(i * 7 + 11) % fSimpleShapeCount]; |
+ float s = 0.95f * SkTMin(outer.rect().width() / inner.rect().width(), |
+ outer.rect().height() / inner.rect().height()); |
+ SkMatrix innerXform; |
+ float dx = (rand.nextF() - 0.5f) * (outer.rect().width() - s * inner.rect().width()); |
+ float dy = (rand.nextF() - 0.5f) * (outer.rect().height() - s * inner.rect().height()); |
+ innerXform.setTranslate(outer.rect().centerX() + dx, outer.rect().centerY() + dy); |
+ if (s < 1) { |
+ innerXform.preScale(s, s); |
+ } |
+ innerXform.preTranslate(-inner.rect().centerX(), -inner.rect().centerY()); |
+ SkRRect xformedInner; |
+ inner.transform(innerXform, &xformedInner); |
+ SkPaint paint(fPaint); |
+ paint.setColor(rand.nextU() & ~0x808080); |
+ paint.setAlpha(128); // Use alpha to detect double blends. |
+ canvas->save(); |
+ canvas->rotate(fRotations[i]); |
+ canvas->drawDRRect(outer, xformedInner, paint); |
+ canvas->restore(); |
+ } |
+ } |
+ |
+ typedef ShapesGM INHERITED; |
+}; |
+ |
+////////////////////////////////////////////////////////////////////////////// |
+ |
+DEF_GM( return new SimpleShapesGM(true); ) |
+DEF_GM( return new SimpleShapesGM(false); ) |
+DEF_GM( return new InnerShapesGM(true); ) |
+DEF_GM( return new InnerShapesGM(false); ) |
+ |
+} |