/* * Copyright 2012 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 "sk_tool_utils.h" #include "SkCanvas.h" #include "SkPath.h" #define STROKE_WIDTH SkIntToScalar(20) static void draw_path(SkCanvas* canvas, const SkPath& path, const SkRect& rect, SkPaint::Join join, int doFill) { SkPaint paint; paint.setAntiAlias(true); paint.setStyle(doFill ? SkPaint::kStrokeAndFill_Style : SkPaint::kStroke_Style); paint.setColor(sk_tool_utils::color_to_565(SK_ColorGRAY)); paint.setStrokeWidth(STROKE_WIDTH); paint.setStrokeJoin(join); canvas->drawRect(rect, paint); paint.setStyle(SkPaint::kStroke_Style); paint.setStrokeWidth(0); paint.setColor(SK_ColorRED); canvas->drawPath(path, paint); paint.setStrokeWidth(3); paint.setStrokeJoin(SkPaint::kMiter_Join); int n = path.countPoints(); SkAutoTArray points(n); path.getPoints(points.get(), n); canvas->drawPoints(SkCanvas::kPoints_PointMode, n, points.get(), paint); } /* * Test calling SkStroker for rectangles. Cases to cover: * * geometry: normal, small (smaller than stroke-width), empty, inverted * joint-type for the corners */ class StrokeRectGM : public skiagm::GM { public: StrokeRectGM() {} protected: SkString onShortName() override { return SkString("strokerect"); } SkISize onISize() override { return SkISize::Make(1400, 740); } void onDraw(SkCanvas* canvas) override { canvas->drawColor(SK_ColorWHITE); canvas->translate(STROKE_WIDTH*3/2, STROKE_WIDTH*3/2); SkPaint paint; paint.setStyle(SkPaint::kStroke_Style); paint.setStrokeWidth(STROKE_WIDTH); constexpr SkPaint::Join gJoins[] = { SkPaint::kMiter_Join, SkPaint::kRound_Join, SkPaint::kBevel_Join }; constexpr SkScalar W = 80; constexpr SkScalar H = 80; constexpr SkRect gRects[] = { { 0, 0, W, H }, { W, 0, 0, H }, { 0, H, W, 0 }, { 0, 0, STROKE_WIDTH, H }, { 0, 0, W, STROKE_WIDTH }, { 0, 0, STROKE_WIDTH/2, STROKE_WIDTH/2 }, { 0, 0, W, 0 }, { 0, 0, 0, H }, { 0, 0, 0, 0 }, { 0, 0, W, FLT_EPSILON }, { 0, 0, FLT_EPSILON, H }, { 0, 0, FLT_EPSILON, FLT_EPSILON }, }; for (int doFill = 0; doFill <= 1; ++doFill) { for (size_t i = 0; i < SK_ARRAY_COUNT(gJoins); ++i) { SkPaint::Join join = gJoins[i]; paint.setStrokeJoin(join); SkAutoCanvasRestore acr(canvas, true); for (size_t j = 0; j < SK_ARRAY_COUNT(gRects); ++j) { const SkRect& r = gRects[j]; SkPath path, fillPath; path.addRect(r); paint.getFillPath(path, &fillPath); draw_path(canvas, fillPath, r, join, doFill); canvas->translate(W + 2 * STROKE_WIDTH, 0); } acr.restore(); canvas->translate(0, H + 2 * STROKE_WIDTH); } paint.setStyle(SkPaint::kStrokeAndFill_Style); } } private: typedef GM INHERITED; }; DEF_GM(return new StrokeRectGM;) /////////////////////////////////////////////////////////////////////////////////////////////////// /* * Exercise rect-stroking (which is specialized from paths) when the resulting stroke-width is * non-square. See https://bugs.chromium.org/p/skia/issues/detail?id=5408 */ DEF_SIMPLE_GM(strokerect_anisotropic_5408, canvas, 200, 50) { SkPaint p; p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(6); canvas->scale(10, 1); SkRect r = SkRect::MakeXYWH(5, 20, 10, 10); canvas->drawRect(r, p); }