| Index: gm/nonclosedpaths.cpp
|
| diff --git a/gm/nonclosedpaths.cpp b/gm/nonclosedpaths.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..bc37c54a8360ed542819cab54018dc06075d7d06
|
| --- /dev/null
|
| +++ b/gm/nonclosedpaths.cpp
|
| @@ -0,0 +1,145 @@
|
| +/*
|
| + * Copyright 2013 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 "SkCanvas.h"
|
| +#include "SkPath.h"
|
| +
|
| +namespace skiagm {
|
| +
|
| +// This GM tests a grab-bag of non-closed paths. All these paths look like
|
| +// closed rects, but they don't call path.close(). Depending on the stroke
|
| +// settings these slightly different paths give widely different results.
|
| +class NonClosedPathsGM: public GM {
|
| +public:
|
| + NonClosedPathsGM() {}
|
| +
|
| + enum ClosureType {
|
| + TotallyNonClosed, // The last point doesn't coincide with the first one in the contour.
|
| + // The path looks not closed at all.
|
| +
|
| + FakeCloseCorner, // The last point coincides with the first one at a corner.
|
| + // The path looks closed, but final rendering has 2 ends with cap.
|
| +
|
| + FakeCloseMiddle, // The last point coincides with the first one in the middle of a line.
|
| + // The path looks closed, and the final rendering looks closed too.
|
| +
|
| + kClosureTypeCount
|
| + };
|
| +
|
| +protected:
|
| + virtual SkString onShortName() SK_OVERRIDE {
|
| + return SkString("nonclosedpaths");
|
| + }
|
| +
|
| + // 12 * 18 + 3 cases, every case is 100 * 100 pixels.
|
| + virtual SkISize onISize() SK_OVERRIDE {
|
| + return SkISize::Make(1220, 1920);
|
| + }
|
| +
|
| + // Use rect-like geometry for non-closed path, for right angles make it
|
| + // easier to show the visual difference of lineCap and lineJoin.
|
| + static void MakePath(SkPath* path, ClosureType type) {
|
| + if (FakeCloseMiddle == type) {
|
| + path->moveTo(30, 50);
|
| + path->lineTo(30, 30);
|
| + } else {
|
| + path->moveTo(30, 30);
|
| + }
|
| + path->lineTo(70, 30);
|
| + path->lineTo(70, 70);
|
| + path->lineTo(30, 70);
|
| + path->lineTo(30, 50);
|
| + if (FakeCloseCorner == type) {
|
| + path->lineTo(30, 30);
|
| + }
|
| + }
|
| +
|
| + // Set the location for the current test on the canvas
|
| + static void SetLocation(SkCanvas* canvas, int counter, int lineNum) {
|
| + SkScalar x = SK_Scalar1 * 100 * (counter % lineNum) + 10 + SK_Scalar1 / 4;
|
| + SkScalar y = SK_Scalar1 * 100 * (counter / lineNum) + 10 + 3 * SK_Scalar1 / 4;
|
| + canvas->translate(x, y);
|
| + }
|
| +
|
| + virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
|
| + // Stroke widths are:
|
| + // 0(may use hairline rendering), 10(common case for stroke-style)
|
| + // 40 and 50(>= geometry width/height, make the contour filled in fact)
|
| + static const int kStrokeWidth[] = {0, 10, 40, 50};
|
| + size_t numWidths = SK_ARRAY_COUNT(kStrokeWidth);
|
| +
|
| + static const SkPaint::Style kStyle[] = {
|
| + SkPaint::kStroke_Style, SkPaint::kStrokeAndFill_Style
|
| + };
|
| +
|
| + static const SkPaint::Cap kCap[] = {
|
| + SkPaint::kButt_Cap, SkPaint::kRound_Cap, SkPaint::kSquare_Cap
|
| + };
|
| +
|
| + static const SkPaint::Join kJoin[] = {
|
| + SkPaint::kMiter_Join, SkPaint::kRound_Join, SkPaint::kBevel_Join
|
| + };
|
| +
|
| + static const ClosureType kType[] = {
|
| + TotallyNonClosed, FakeCloseCorner, FakeCloseMiddle
|
| + };
|
| +
|
| + int counter = 0;
|
| + SkPaint paint;
|
| + paint.setAntiAlias(true);
|
| +
|
| + // For stroke style painter and fill-and-stroke style painter
|
| + for (size_t type = 0; type < kClosureTypeCount; ++type) {
|
| + for (size_t style = 0; style < SK_ARRAY_COUNT(kStyle); ++style) {
|
| + for (size_t cap = 0; cap < SK_ARRAY_COUNT(kCap); ++cap) {
|
| + for (size_t join = 0; join < SK_ARRAY_COUNT(kJoin); ++join) {
|
| + for (size_t width = 0; width < numWidths; ++width) {
|
| + canvas->save();
|
| + SetLocation(canvas, counter, SkPaint::kJoinCount * numWidths);
|
| +
|
| + SkPath path;
|
| + MakePath(&path, kType[type]);
|
| +
|
| + paint.setStyle(kStyle[style]);
|
| + paint.setStrokeCap(kCap[cap]);
|
| + paint.setStrokeJoin(kJoin[join]);
|
| + paint.setStrokeWidth(SkIntToScalar(kStrokeWidth[width]));
|
| +
|
| + canvas->drawPath(path, paint);
|
| + canvas->restore();
|
| + ++counter;
|
| + }
|
| + }
|
| + }
|
| + }
|
| + }
|
| +
|
| + // For fill style painter
|
| + paint.setStyle(SkPaint::kFill_Style);
|
| + for (size_t type = 0; type < kClosureTypeCount; ++type) {
|
| + canvas->save();
|
| + SetLocation(canvas, counter, SkPaint::kJoinCount * numWidths);
|
| +
|
| + SkPath path;
|
| + MakePath(&path, kType[type]);
|
| +
|
| + canvas->drawPath(path, paint);
|
| + canvas->restore();
|
| + ++counter;
|
| + }
|
| + }
|
| +
|
| +private:
|
| + typedef GM INHERITED;
|
| +};
|
| +
|
| +//////////////////////////////////////////////////////////////////////////////
|
| +
|
| +DEF_GM(return new NonClosedPathsGM;)
|
| +
|
| +}
|
|
|