2012-02-13 15:18:34 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
2019-05-01 21:28:53 +00:00
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "gm/gm.h"
|
|
|
|
#include "include/core/SkCanvas.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/core/SkMatrix.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkPaint.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/core/SkPath.h"
|
|
|
|
#include "include/core/SkPathEffect.h"
|
|
|
|
#include "include/core/SkRect.h"
|
|
|
|
#include "include/core/SkRefCnt.h"
|
|
|
|
#include "include/core/SkScalar.h"
|
|
|
|
#include "include/core/SkSize.h"
|
|
|
|
#include "include/core/SkString.h"
|
|
|
|
#include "include/core/SkTypes.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/effects/Sk1DPathEffect.h"
|
|
|
|
#include "include/effects/Sk2DPathEffect.h"
|
|
|
|
#include "include/effects/SkCornerPathEffect.h"
|
|
|
|
#include "include/effects/SkDashPathEffect.h"
|
|
|
|
#include "include/effects/SkDiscretePathEffect.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/effects/SkOpPathEffect.h"
|
|
|
|
#include "include/pathops/SkPathOps.h"
|
|
|
|
|
|
|
|
#include <initializer_list>
|
2012-02-13 15:18:34 +00:00
|
|
|
|
|
|
|
namespace skiagm {
|
|
|
|
|
|
|
|
static void compose_pe(SkPaint* paint) {
|
|
|
|
SkPathEffect* pe = paint->getPathEffect();
|
2016-03-18 18:22:57 +00:00
|
|
|
sk_sp<SkPathEffect> corner = SkCornerPathEffect::Make(25);
|
|
|
|
sk_sp<SkPathEffect> compose;
|
2012-02-13 15:18:34 +00:00
|
|
|
if (pe) {
|
2017-02-26 03:34:32 +00:00
|
|
|
compose = SkPathEffect::MakeCompose(sk_ref_sp(pe), corner);
|
2012-02-13 15:18:34 +00:00
|
|
|
} else {
|
|
|
|
compose = corner;
|
|
|
|
}
|
2016-03-18 18:22:57 +00:00
|
|
|
paint->setPathEffect(compose);
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void hair_pe(SkPaint* paint) {
|
|
|
|
paint->setStrokeWidth(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hair2_pe(SkPaint* paint) {
|
|
|
|
paint->setStrokeWidth(0);
|
|
|
|
compose_pe(paint);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stroke_pe(SkPaint* paint) {
|
|
|
|
paint->setStrokeWidth(12);
|
|
|
|
compose_pe(paint);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dash_pe(SkPaint* paint) {
|
|
|
|
SkScalar inter[] = { 20, 10, 10, 10 };
|
|
|
|
paint->setStrokeWidth(12);
|
2016-03-18 18:22:57 +00:00
|
|
|
paint->setPathEffect(SkDashPathEffect::Make(inter, SK_ARRAY_COUNT(inter), 0));
|
2012-02-13 15:18:34 +00:00
|
|
|
compose_pe(paint);
|
|
|
|
}
|
|
|
|
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr int gXY[] = {
|
2012-02-13 15:18:34 +00:00
|
|
|
4, 0, 0, -4, 8, -4, 12, 0, 8, 4, 0, 4
|
|
|
|
};
|
|
|
|
|
|
|
|
static void scale(SkPath* path, SkScalar scale) {
|
|
|
|
SkMatrix m;
|
|
|
|
m.setScale(scale, scale);
|
|
|
|
path->transform(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void one_d_pe(SkPaint* paint) {
|
|
|
|
SkPath path;
|
|
|
|
path.moveTo(SkIntToScalar(gXY[0]), SkIntToScalar(gXY[1]));
|
|
|
|
for (unsigned i = 2; i < SK_ARRAY_COUNT(gXY); i += 2)
|
|
|
|
path.lineTo(SkIntToScalar(gXY[i]), SkIntToScalar(gXY[i+1]));
|
|
|
|
path.close();
|
|
|
|
path.offset(SkIntToScalar(-6), 0);
|
2012-06-06 12:10:45 +00:00
|
|
|
scale(&path, 1.5f);
|
2012-08-23 18:19:56 +00:00
|
|
|
|
2016-03-18 18:22:57 +00:00
|
|
|
paint->setPathEffect(SkPath1DPathEffect::Make(path, SkIntToScalar(21), 0,
|
|
|
|
SkPath1DPathEffect::kRotate_Style));
|
2012-02-13 15:18:34 +00:00
|
|
|
compose_pe(paint);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef void (*PE_Proc)(SkPaint*);
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr PE_Proc gPE[] = { hair_pe, hair2_pe, stroke_pe, dash_pe, one_d_pe };
|
2012-02-13 15:18:34 +00:00
|
|
|
|
|
|
|
static void fill_pe(SkPaint* paint) {
|
|
|
|
paint->setStyle(SkPaint::kFill_Style);
|
2015-08-27 14:41:13 +00:00
|
|
|
paint->setPathEffect(nullptr);
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void discrete_pe(SkPaint* paint) {
|
2016-03-18 18:22:57 +00:00
|
|
|
paint->setPathEffect(SkDiscretePathEffect::Make(10, 4));
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
2016-03-18 18:22:57 +00:00
|
|
|
static sk_sp<SkPathEffect> MakeTileEffect() {
|
2012-02-13 15:18:34 +00:00
|
|
|
SkMatrix m;
|
|
|
|
m.setScale(SkIntToScalar(12), SkIntToScalar(12));
|
|
|
|
|
|
|
|
SkPath path;
|
|
|
|
path.addCircle(0, 0, SkIntToScalar(5));
|
2012-08-23 18:19:56 +00:00
|
|
|
|
2016-03-18 18:22:57 +00:00
|
|
|
return SkPath2DPathEffect::Make(m, path);
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void tile_pe(SkPaint* paint) {
|
2016-03-18 18:22:57 +00:00
|
|
|
paint->setPathEffect(MakeTileEffect());
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr PE_Proc gPE2[] = { fill_pe, discrete_pe, tile_pe };
|
2012-02-13 15:18:34 +00:00
|
|
|
|
|
|
|
class PathEffectGM : public GM {
|
|
|
|
public:
|
|
|
|
PathEffectGM() {}
|
|
|
|
|
|
|
|
protected:
|
2014-04-30 13:20:45 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkString onShortName() override {
|
2012-02-13 15:18:34 +00:00
|
|
|
return SkString("patheffect");
|
|
|
|
}
|
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
SkISize onISize() override { return SkISize::Make(800, 600); }
|
2012-02-13 15:18:34 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onDraw(SkCanvas* canvas) override {
|
2012-02-13 15:18:34 +00:00
|
|
|
SkPaint paint;
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
|
|
|
|
|
|
SkPath path;
|
|
|
|
path.moveTo(20, 20);
|
|
|
|
path.lineTo(70, 120);
|
|
|
|
path.lineTo(120, 30);
|
|
|
|
path.lineTo(170, 80);
|
|
|
|
path.lineTo(240, 50);
|
|
|
|
|
|
|
|
canvas->save();
|
2018-08-17 15:01:03 +00:00
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPE); i++) {
|
2012-02-13 15:18:34 +00:00
|
|
|
gPE[i](&paint);
|
|
|
|
canvas->drawPath(path, paint);
|
|
|
|
canvas->translate(0, 75);
|
|
|
|
}
|
|
|
|
canvas->restore();
|
|
|
|
|
|
|
|
path.reset();
|
|
|
|
SkRect r = { 0, 0, 250, 120 };
|
2019-11-22 18:34:02 +00:00
|
|
|
path.addOval(r, SkPathDirection::kCW);
|
2012-02-13 15:18:34 +00:00
|
|
|
r.inset(50, 50);
|
2019-11-22 18:34:02 +00:00
|
|
|
path.addRect(r, SkPathDirection::kCCW);
|
2012-02-13 15:18:34 +00:00
|
|
|
|
|
|
|
canvas->translate(320, 20);
|
2018-08-17 15:01:03 +00:00
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPE2); i++) {
|
2012-02-13 15:18:34 +00:00
|
|
|
gPE2[i](&paint);
|
|
|
|
canvas->drawPath(path, paint);
|
|
|
|
canvas->translate(0, 160);
|
|
|
|
}
|
2012-03-26 14:36:55 +00:00
|
|
|
|
2018-08-17 15:01:03 +00:00
|
|
|
const SkIRect rect = SkIRect::MakeXYWH(20, 20, 60, 60);
|
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPE); i++) {
|
2012-03-26 14:36:55 +00:00
|
|
|
SkPaint p;
|
|
|
|
p.setAntiAlias(true);
|
|
|
|
p.setStyle(SkPaint::kFill_Style);
|
|
|
|
gPE[i](&p);
|
|
|
|
canvas->drawIRect(rect, p);
|
|
|
|
canvas->translate(75, 0);
|
|
|
|
}
|
2012-02-13 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
|
2019-01-23 15:22:01 +00:00
|
|
|
DEF_GM( return new PathEffectGM; )
|
2012-02-13 15:18:34 +00:00
|
|
|
|
2020-08-06 18:11:56 +00:00
|
|
|
} // namespace skiagm
|
2018-07-18 17:28:42 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
class ComboPathEfectsGM : public skiagm::GM {
|
|
|
|
public:
|
|
|
|
ComboPathEfectsGM() {}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
SkString onShortName() override {
|
|
|
|
return SkString("combo-patheffects");
|
|
|
|
}
|
|
|
|
|
|
|
|
SkISize onISize() override { return SkISize::Make(360, 630); }
|
|
|
|
|
|
|
|
void onDraw(SkCanvas* canvas) override {
|
|
|
|
SkPath path0, path1, path2;
|
|
|
|
path0.addCircle(100, 100, 60);
|
|
|
|
path1.moveTo(20, 20); path1.cubicTo(20, 180, 140, 0, 140, 140);
|
|
|
|
|
|
|
|
sk_sp<SkPathEffect> effects[] = {
|
|
|
|
nullptr,
|
|
|
|
SkStrokePathEffect::Make(20, SkPaint::kRound_Join, SkPaint::kRound_Cap, 0),
|
2018-07-19 17:47:44 +00:00
|
|
|
SkMergePathEffect::Make(nullptr,
|
|
|
|
SkStrokePathEffect::Make(20, SkPaint::kRound_Join,
|
|
|
|
SkPaint::kRound_Cap, 0),
|
|
|
|
kDifference_SkPathOp),
|
|
|
|
SkMergePathEffect::Make(SkMatrixPathEffect::MakeTranslate(50, 30),
|
|
|
|
SkStrokePathEffect::Make(20, SkPaint::kRound_Join,
|
|
|
|
SkPaint::kRound_Cap, 0),
|
|
|
|
kReverseDifference_SkPathOp),
|
2018-07-18 17:28:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
SkPaint wireframe;
|
|
|
|
wireframe.setStyle(SkPaint::kStroke_Style);
|
|
|
|
wireframe.setAntiAlias(true);
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setColor(0xFF8888FF);
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
|
2020-07-31 00:24:57 +00:00
|
|
|
for (const SkPath& path : { path0, path1 }) {
|
2018-07-18 17:28:42 +00:00
|
|
|
canvas->save();
|
2020-07-31 00:24:57 +00:00
|
|
|
for (const sk_sp<SkPathEffect>& pe : effects) {
|
2018-07-18 17:28:42 +00:00
|
|
|
paint.setPathEffect(pe);
|
|
|
|
canvas->drawPath(path, paint);
|
|
|
|
canvas->drawPath(path, wireframe);
|
|
|
|
|
|
|
|
canvas->translate(0, 150);
|
|
|
|
}
|
|
|
|
canvas->restore();
|
|
|
|
canvas->translate(180, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
DEF_GM(return new ComboPathEfectsGM;)
|
|
|
|
|
2020-05-20 13:55:58 +00:00
|
|
|
#include "include/effects/SkStrokeAndFillPathEffect.h"
|
|
|
|
|
|
|
|
// Test that we can replicate SkPaint::kStrokeAndFill_Style
|
|
|
|
// with a patheffect. We expect the 2nd and 3rd columns to draw the same.
|
|
|
|
DEF_SIMPLE_GM(stroke_and_fill_patheffect, canvas, 900, 450) {
|
|
|
|
const float kStrokeWidth = 20;
|
|
|
|
|
2020-08-03 02:14:43 +00:00
|
|
|
typedef SkPath (*Maker)();
|
2020-05-20 13:55:58 +00:00
|
|
|
const Maker makers[] = {
|
2020-08-03 02:14:43 +00:00
|
|
|
[]() { return SkPath::Oval({0, 0, 100, 100}, SkPathDirection::kCW); },
|
|
|
|
[]() { return SkPath::Oval({0, 0, 100, 100}, SkPathDirection::kCCW); },
|
|
|
|
[]() {
|
|
|
|
const SkPoint pts[] = {
|
|
|
|
{0, 0}, {100, 100}, {0, 100}, {100, 0},
|
|
|
|
};
|
|
|
|
return SkPath::Polygon(pts, SK_ARRAY_COUNT(pts), true);
|
2020-05-20 13:55:58 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const struct {
|
|
|
|
SkPaint::Style fStyle;
|
|
|
|
float fWidth;
|
|
|
|
bool fUsePE;
|
|
|
|
bool fExpectStrokeAndFill;
|
|
|
|
} rec[] = {
|
|
|
|
{ SkPaint::kStroke_Style, 0, false, false },
|
|
|
|
{ SkPaint::kFill_Style, 0, true, false },
|
|
|
|
{ SkPaint::kStroke_Style, 0, true, false },
|
|
|
|
{ SkPaint::kStrokeAndFill_Style, kStrokeWidth, false, true },
|
|
|
|
{ SkPaint::kStroke_Style, kStrokeWidth, true, true },
|
|
|
|
{ SkPaint::kStrokeAndFill_Style, kStrokeWidth, true, true },
|
|
|
|
};
|
|
|
|
|
|
|
|
SkPaint paint;
|
|
|
|
canvas->translate(20, 20);
|
|
|
|
for (auto maker : makers) {
|
2020-08-03 02:14:43 +00:00
|
|
|
const SkPath path = maker();
|
2020-05-20 13:55:58 +00:00
|
|
|
canvas->save();
|
|
|
|
for (const auto& r : rec) {
|
|
|
|
paint.setStyle(r.fStyle);
|
|
|
|
paint.setStrokeWidth(r.fWidth);
|
|
|
|
paint.setPathEffect(r.fUsePE ? SkStrokeAndFillPathEffect::Make() : nullptr);
|
|
|
|
paint.setColor(r.fExpectStrokeAndFill ? SK_ColorGRAY : SK_ColorBLACK);
|
|
|
|
|
|
|
|
canvas->drawPath(path, paint);
|
|
|
|
canvas->translate(150, 0);
|
|
|
|
}
|
|
|
|
canvas->restore();
|
|
|
|
|
|
|
|
canvas->translate(0, 150);
|
|
|
|
}
|
|
|
|
}
|