|  | /* | 
|  | * Copyright 2015 Google Inc. | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  |  | 
|  | #include "gm/gm.h" | 
|  | #include "include/core/SkCanvas.h" | 
|  | #include "include/core/SkPaint.h" | 
|  | #include "include/core/SkPathBuilder.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/pathops/SkPathOps.h" | 
|  | #include <tuple> | 
|  |  | 
|  | namespace { | 
|  | struct PathDY { | 
|  | SkPath path; | 
|  | SkScalar dy; | 
|  | }; | 
|  | } | 
|  |  | 
|  | typedef PathDY (*MakePathProc)(); | 
|  |  | 
|  | static PathDY make_triangle() { | 
|  | constexpr int gCoord[] = { | 
|  | 10, 20, 15, 5, 30, 30 | 
|  | }; | 
|  | return { | 
|  | SkPathBuilder().moveTo(SkIntToScalar(gCoord[0]), SkIntToScalar(gCoord[1])) | 
|  | .lineTo(SkIntToScalar(gCoord[2]), SkIntToScalar(gCoord[3])) | 
|  | .lineTo(SkIntToScalar(gCoord[4]), SkIntToScalar(gCoord[5])) | 
|  | .close() | 
|  | .offset(10, 0) | 
|  | .detach(), | 
|  | 30 | 
|  | }; | 
|  | } | 
|  |  | 
|  | static PathDY make_rect() { | 
|  | SkRect r = { SkIntToScalar(10), SkIntToScalar(10), | 
|  | SkIntToScalar(30), SkIntToScalar(30) }; | 
|  | return { | 
|  | SkPath::Rect(r.makeOffset(10, 0)), | 
|  | 30 | 
|  | }; | 
|  | } | 
|  |  | 
|  | static PathDY make_oval() { | 
|  | SkRect r = { SkIntToScalar(10), SkIntToScalar(10), | 
|  | SkIntToScalar(30), SkIntToScalar(30) }; | 
|  | return { | 
|  | SkPath::Oval(r.makeOffset(10, 0)), | 
|  | 30 | 
|  | }; | 
|  | } | 
|  |  | 
|  | static PathDY make_star(int n) { | 
|  | const SkScalar c = SkIntToScalar(45); | 
|  | const SkScalar r = SkIntToScalar(20); | 
|  |  | 
|  | SkScalar rad = -SK_ScalarPI / 2; | 
|  | const SkScalar drad = (n >> 1) * SK_ScalarPI * 2 / n; | 
|  |  | 
|  | SkPathBuilder b; | 
|  | b.moveTo(c, c - r); | 
|  | for (int i = 1; i < n; i++) { | 
|  | rad += drad; | 
|  | b.lineTo(c + SkScalarCos(rad) * r, c + SkScalarSin(rad) * r); | 
|  | } | 
|  | b.close(); | 
|  | return { b.detach(), r * 2 * 6 / 5 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_star_5() { return make_star(5); } | 
|  | static PathDY make_star_13() { return make_star(13); } | 
|  |  | 
|  | static PathDY make_three_line() { | 
|  | static SkScalar xOffset = 34.f; | 
|  | static SkScalar yOffset = 50.f; | 
|  | SkPathBuilder b; | 
|  | b.moveTo(-32.5f + xOffset, 0.0f + yOffset); | 
|  | b.lineTo(32.5f + xOffset, 0.0f + yOffset); | 
|  |  | 
|  | b.moveTo(-32.5f + xOffset, 19 + yOffset); | 
|  | b.lineTo(32.5f + xOffset, 19 + yOffset); | 
|  |  | 
|  | b.moveTo(-32.5f + xOffset, -19 + yOffset); | 
|  | b.lineTo(32.5f + xOffset, -19 + yOffset); | 
|  | b.lineTo(-32.5f + xOffset, -19 + yOffset); | 
|  |  | 
|  | b.close(); | 
|  |  | 
|  | return { b.detach(), 70 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_arrow() { | 
|  | static SkScalar xOffset = 34.f; | 
|  | static SkScalar yOffset = 40.f; | 
|  | SkPathBuilder b; | 
|  | b.moveTo(-26.f + xOffset, 0.0f + yOffset); | 
|  | b.lineTo(26.f + xOffset, 0.0f + yOffset); | 
|  |  | 
|  | b.moveTo(-28.f + xOffset, -2.4748745f + yOffset); | 
|  | b.lineTo(0 + xOffset, 25.525126f + yOffset); | 
|  |  | 
|  | b.moveTo(-28.f + xOffset, 2.4748745f + yOffset); | 
|  | b.lineTo(0 + xOffset, -25.525126f + yOffset); | 
|  | b.lineTo(-28.f + xOffset, 2.4748745f + yOffset); | 
|  |  | 
|  | b.close(); | 
|  |  | 
|  | return { b.detach(), 70 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_curve() { | 
|  | static SkScalar xOffset = -382.f; | 
|  | static SkScalar yOffset = -50.f; | 
|  | SkPathBuilder b; | 
|  | b.moveTo(491 + xOffset, 56 + yOffset); | 
|  | b.conicTo(435.93292f + xOffset, 56.000031f + yOffset, | 
|  | 382.61078f + xOffset, 69.752716f + yOffset, | 
|  | 0.9920463f); | 
|  |  | 
|  | return { b.detach(), 40 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_battery() { | 
|  | static SkScalar xOffset = 5.0f; | 
|  |  | 
|  | SkPathBuilder b; | 
|  | b.moveTo(24.67f + xOffset, 0.33000004f); | 
|  | b.lineTo(8.3299999f + xOffset, 0.33000004f); | 
|  | b.lineTo(8.3299999f + xOffset, 5.3299999f); | 
|  | b.lineTo(0.33000004f + xOffset, 5.3299999f); | 
|  | b.lineTo(0.33000004f + xOffset, 50.669998f); | 
|  | b.lineTo(32.669998f + xOffset, 50.669998f); | 
|  | b.lineTo(32.669998f + xOffset, 5.3299999f); | 
|  | b.lineTo(24.67f + xOffset, 5.3299999f); | 
|  | b.lineTo(24.67f + xOffset, 0.33000004f); | 
|  | b.close(); | 
|  |  | 
|  | b.moveTo(25.727224f + xOffset, 12.886665f); | 
|  | b.lineTo(10.907918f + xOffset, 12.886665f); | 
|  | b.lineTo(7.5166659f + xOffset, 28.683645f); | 
|  | b.lineTo(14.810181f + xOffset, 28.683645f); | 
|  | b.lineTo(7.7024879f + xOffset, 46.135998f); | 
|  | b.lineTo(28.049999f + xOffset, 25.136419f); | 
|  | b.lineTo(16.854223f + xOffset, 25.136419f); | 
|  | b.lineTo(25.727224f + xOffset, 12.886665f); | 
|  | b.close(); | 
|  | return { b.detach(), 50 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_battery2() { | 
|  | static SkScalar xOffset = 225.625f; | 
|  |  | 
|  | SkPathBuilder b; | 
|  | b.moveTo(32.669998f + xOffset, 9.8640003f); | 
|  | b.lineTo(0.33000004f + xOffset, 9.8640003f); | 
|  | b.lineTo(0.33000004f + xOffset, 50.669998f); | 
|  | b.lineTo(32.669998f + xOffset, 50.669998f); | 
|  | b.lineTo(32.669998f + xOffset, 9.8640003f); | 
|  | b.close(); | 
|  |  | 
|  | b.moveTo(10.907918f + xOffset, 12.886665f); | 
|  | b.lineTo(25.727224f + xOffset, 12.886665f); | 
|  | b.lineTo(16.854223f + xOffset, 25.136419f); | 
|  | b.lineTo(28.049999f + xOffset, 25.136419f); | 
|  | b.lineTo(7.7024879f + xOffset, 46.135998f); | 
|  | b.lineTo(14.810181f + xOffset, 28.683645f); | 
|  | b.lineTo(7.5166659f + xOffset, 28.683645f); | 
|  | b.lineTo(10.907918f + xOffset, 12.886665f); | 
|  | b.close(); | 
|  |  | 
|  | return { b.detach(), 60 }; | 
|  | } | 
|  |  | 
|  | static PathDY make_ring() { | 
|  | static SkScalar xOffset = 120; | 
|  | static SkScalar yOffset = -270.f; | 
|  |  | 
|  | SkPathBuilder b; | 
|  | b.setFillType(SkPathFillType::kWinding); | 
|  | b.moveTo(xOffset + 144.859f, yOffset + 285.172f); | 
|  | b.lineTo(xOffset + 144.859f, yOffset + 285.172f); | 
|  | b.lineTo(xOffset + 144.859f, yOffset + 285.172f); | 
|  | b.lineTo(xOffset + 143.132f, yOffset + 284.617f); | 
|  | b.lineTo(xOffset + 144.859f, yOffset + 285.172f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.lineTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.lineTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.lineTo(xOffset + 135.367f, yOffset + 288.571f); | 
|  | b.lineTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.cubicTo(xOffset + 137.07f, yOffset + 287.219f, xOffset + 138.242f, yOffset + 287.086f, | 
|  | xOffset + 139.242f, yOffset + 286.578f); | 
|  | b.cubicTo(xOffset + 140.234f, yOffset + 286.078f, xOffset + 141.031f, yOffset + 285.203f, | 
|  | xOffset + 141.406f, yOffset + 284.055f); | 
|  | b.lineTo(xOffset + 144.859f, yOffset + 285.172f); | 
|  | b.cubicTo(xOffset + 143.492f, yOffset + 289.375f, xOffset + 138.992f, yOffset + 291.656f, | 
|  | xOffset + 134.797f, yOffset + 290.297f); | 
|  | b.lineTo(xOffset + 135.922f, yOffset + 286.844f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 129.68f, yOffset + 280.242f); | 
|  | b.lineTo(xOffset + 129.68f, yOffset + 280.242f); | 
|  | b.lineTo(xOffset + 129.68f, yOffset + 280.242f); | 
|  | b.lineTo(xOffset + 131.407f, yOffset + 280.804f); | 
|  | b.lineTo(xOffset + 129.68f, yOffset + 280.242f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 133.133f, yOffset + 281.367f); | 
|  | b.cubicTo(xOffset + 132.758f, yOffset + 282.508f, xOffset + 132.883f, yOffset + 283.687f, | 
|  | xOffset + 133.391f, yOffset + 284.679f); | 
|  | b.cubicTo(xOffset + 133.907f, yOffset + 285.679f, xOffset + 134.774f, yOffset + 286.468f, | 
|  | xOffset + 135.922f, yOffset + 286.843f); | 
|  | b.lineTo(xOffset + 134.797f, yOffset + 290.296f); | 
|  | b.cubicTo(xOffset + 130.602f, yOffset + 288.929f, xOffset + 128.313f, yOffset + 284.437f, | 
|  | xOffset + 129.68f, yOffset + 280.241f); | 
|  | b.lineTo(xOffset + 133.133f, yOffset + 281.367f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 139.742f, yOffset + 275.117f); | 
|  | b.lineTo(xOffset + 139.742f, yOffset + 275.117f); | 
|  | b.lineTo(xOffset + 139.18f, yOffset + 276.844f); | 
|  | b.lineTo(xOffset + 139.742f, yOffset + 275.117f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 138.609f, yOffset + 278.57f); | 
|  | b.cubicTo(xOffset + 137.461f, yOffset + 278.203f, xOffset + 136.297f, yOffset + 278.328f, | 
|  | xOffset + 135.297f, yOffset + 278.836f); | 
|  | b.cubicTo(xOffset + 134.297f, yOffset + 279.344f, xOffset + 133.508f, yOffset + 280.219f, | 
|  | xOffset + 133.133f, yOffset + 281.367f); | 
|  | b.lineTo(xOffset + 129.68f, yOffset + 280.242f); | 
|  | b.cubicTo(xOffset + 131.047f, yOffset + 276.039f, xOffset + 135.539f, yOffset + 273.758f, | 
|  | xOffset + 139.742f, yOffset + 275.117f); | 
|  | b.lineTo(xOffset + 138.609f, yOffset + 278.57f); | 
|  | b.close(); | 
|  | b.moveTo(xOffset + 141.406f, yOffset + 284.055f); | 
|  | b.cubicTo(xOffset + 141.773f, yOffset + 282.907f, xOffset + 141.648f, yOffset + 281.735f, | 
|  | xOffset + 141.148f, yOffset + 280.735f); | 
|  | b.cubicTo(xOffset + 140.625f, yOffset + 279.735f, xOffset + 139.757f, yOffset + 278.946f, | 
|  | xOffset + 138.609f, yOffset + 278.571f); | 
|  | b.lineTo(xOffset + 139.742f, yOffset + 275.118f); | 
|  | b.cubicTo(xOffset + 143.937f, yOffset + 276.493f, xOffset + 146.219f, yOffset + 280.977f, | 
|  | xOffset + 144.859f, yOffset + 285.173f); | 
|  | b.lineTo(xOffset + 141.406f, yOffset + 284.055f); | 
|  | b.close(); | 
|  |  | 
|  | // uncomment to reveal PathOps bug, see https://bugs.chromium.org/p/skia/issues/detail?id=9732 | 
|  | // (void) Simplify(*path, path); | 
|  |  | 
|  | return { b.detach(), 15 }; | 
|  | } | 
|  |  | 
|  | constexpr MakePathProc gProcs[] = { | 
|  | make_triangle, | 
|  | make_rect, | 
|  | make_oval, | 
|  | make_star_5, | 
|  | make_star_13, | 
|  | make_three_line, | 
|  | make_arrow, | 
|  | make_curve, | 
|  | make_battery, | 
|  | make_battery2, | 
|  | make_ring | 
|  | }; | 
|  |  | 
|  | constexpr SkScalar gWidths[] = { | 
|  | 2.0f, | 
|  | 3.0f, | 
|  | 4.0f, | 
|  | 5.0f, | 
|  | 6.0f, | 
|  | 7.0f, | 
|  | 7.0f, | 
|  | 14.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f | 
|  | }; | 
|  | static_assert(SK_ARRAY_COUNT(gWidths) == SK_ARRAY_COUNT(gProcs)); | 
|  |  | 
|  | constexpr SkScalar gMiters[] = { | 
|  | 2.0f, | 
|  | 3.0f, | 
|  | 3.0f, | 
|  | 3.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | 4.0f, | 
|  | }; | 
|  | static_assert(SK_ARRAY_COUNT(gMiters) == SK_ARRAY_COUNT(gProcs)); | 
|  |  | 
|  | constexpr SkScalar gXTranslate[] = { | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | -220.625f, | 
|  | 0.0f, | 
|  | 0.0f, | 
|  | }; | 
|  | static_assert(SK_ARRAY_COUNT(gXTranslate) == SK_ARRAY_COUNT(gProcs)); | 
|  |  | 
|  | #define N   SK_ARRAY_COUNT(gProcs) | 
|  |  | 
|  | // This GM tests out drawing small paths (i.e., for Ganesh, using the Distance | 
|  | // Field path renderer) which are filled, stroked and filledAndStroked. In | 
|  | // particular this ensures that any cache keys in use include the stroking | 
|  | // parameters. | 
|  | class SmallPathsGM : public skiagm::GM { | 
|  | SkPath  fPath[N]; | 
|  | SkScalar fDY[N]; | 
|  | protected: | 
|  | void onOnceBeforeDraw() override { | 
|  | for (size_t i = 0; i < N; i++) { | 
|  | auto [path, dy] = gProcs[i](); | 
|  | fPath[i] = path; | 
|  | fDY[i]   = dy; | 
|  | } | 
|  | } | 
|  |  | 
|  | SkString onShortName() override { | 
|  | return SkString("smallpaths"); | 
|  | } | 
|  |  | 
|  | SkISize onISize() override { | 
|  | return SkISize::Make(640, 512); | 
|  | } | 
|  |  | 
|  | void onDraw(SkCanvas* canvas) override { | 
|  | SkPaint paint; | 
|  | paint.setAntiAlias(true); | 
|  |  | 
|  | // first column: filled paths | 
|  | canvas->save(); | 
|  | for (size_t i = 0; i < N; i++) { | 
|  | canvas->drawPath(fPath[i], paint); | 
|  | canvas->translate(gXTranslate[i], fDY[i]); | 
|  | } | 
|  | canvas->restore(); | 
|  | canvas->translate(SkIntToScalar(120), SkIntToScalar(0)); | 
|  |  | 
|  | // second column: stroked paths | 
|  | canvas->save(); | 
|  | paint.setStyle(SkPaint::kStroke_Style); | 
|  | paint.setStrokeCap(SkPaint::kButt_Cap); | 
|  | for (size_t i = 0; i < N; i++) { | 
|  | paint.setStrokeWidth(gWidths[i]); | 
|  | paint.setStrokeMiter(gMiters[i]); | 
|  | canvas->drawPath(fPath[i], paint); | 
|  | canvas->translate(gXTranslate[i], fDY[i]); | 
|  | } | 
|  | canvas->restore(); | 
|  | canvas->translate(SkIntToScalar(120), SkIntToScalar(0)); | 
|  |  | 
|  | // third column: stroked paths with different widths | 
|  | canvas->save(); | 
|  | paint.setStyle(SkPaint::kStroke_Style); | 
|  | paint.setStrokeCap(SkPaint::kButt_Cap); | 
|  | for (size_t i = 0; i < N; i++) { | 
|  | paint.setStrokeWidth(gWidths[i] + 2.0f); | 
|  | paint.setStrokeMiter(gMiters[i]); | 
|  | canvas->drawPath(fPath[i], paint); | 
|  | canvas->translate(gXTranslate[i], fDY[i]); | 
|  | } | 
|  | canvas->restore(); | 
|  | canvas->translate(SkIntToScalar(120), SkIntToScalar(0)); | 
|  |  | 
|  | // fourth column: stroked and filled paths | 
|  | paint.setStyle(SkPaint::kStrokeAndFill_Style); | 
|  | paint.setStrokeCap(SkPaint::kButt_Cap); | 
|  | for (size_t i = 0; i < N; i++) { | 
|  | paint.setStrokeWidth(gWidths[i]); | 
|  | paint.setStrokeMiter(gMiters[i]); | 
|  | canvas->drawPath(fPath[i], paint); | 
|  | canvas->translate(gXTranslate[i], fDY[i]); | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | private: | 
|  | using INHERITED = skiagm::GM; | 
|  | }; | 
|  |  | 
|  | DEF_GM(return new SmallPathsGM;) |