diff options
author | Florin Malita <fmalita@chromium.org> | 2018-03-07 16:37:38 -0500 |
---|---|---|
committer | Skia Commit-Bot <skia-commit-bot@chromium.org> | 2018-03-07 22:01:20 +0000 |
commit | 19f292247ef1a16bfff49d14f44c97fb030c40cc (patch) | |
tree | 2ffb4a3542b874d35f71d465ca45ca13b0b77ca2 /experimental | |
parent | 02bd2950e04952398930ed179bb72e08ad5ed3d3 (diff) |
[sksg] Cache transformed TrimEffect geometry
Apply the effect at revalidation time, and cache the result.
TBR=
Change-Id: I166fc0e4e2869bea51e5e45e5a2a50df2f034691
Reviewed-on: https://skia-review.googlesource.com/112801
Reviewed-by: Florin Malita <fmalita@chromium.org>
Commit-Queue: Florin Malita <fmalita@chromium.org>
Diffstat (limited to 'experimental')
-rw-r--r-- | experimental/sksg/geometry/SkSGTrimEffect.cpp | 52 | ||||
-rw-r--r-- | experimental/sksg/geometry/SkSGTrimEffect.h | 1 |
2 files changed, 31 insertions, 22 deletions
diff --git a/experimental/sksg/geometry/SkSGTrimEffect.cpp b/experimental/sksg/geometry/SkSGTrimEffect.cpp index 0b664fd436..195b09ece0 100644 --- a/experimental/sksg/geometry/SkSGTrimEffect.cpp +++ b/experimental/sksg/geometry/SkSGTrimEffect.cpp @@ -10,6 +10,7 @@ #include "SkCanvas.h" #include "SkDashPathEffect.h" #include "SkPathMeasure.h" +#include "SkStrokeRec.h" namespace sksg { @@ -23,15 +24,24 @@ TrimEffect::~TrimEffect() { } void TrimEffect::onClip(SkCanvas* canvas, bool antiAlias) const { - canvas->clipPath(fChild->asPath(), SkClipOp::kIntersect, antiAlias); + canvas->clipPath(fTrimmedPath, SkClipOp::kIntersect, antiAlias); } -// TODO -// This is a quick hack to get something on the screen. What we really want here is to apply -// the geometry transformation and cache the result on revalidation. Or an SkTrimPathEffect. void TrimEffect::onDraw(SkCanvas* canvas, const SkPaint& paint) const { SkASSERT(!paint.getPathEffect()); + canvas->drawPath(fTrimmedPath, paint); +} + +SkPath TrimEffect::onAsPath() const { + return fTrimmedPath; +} + +SkRect TrimEffect::onRevalidate(InvalidationController* ic, const SkMatrix& ctm) { + SkASSERT(this->hasInval()); + + const auto childbounds = fChild->revalidate(ic, ctm); + const auto path = fChild->asPath(); SkScalar pathLen = 0; SkPathMeasure measure(path, false); @@ -44,26 +54,24 @@ void TrimEffect::onDraw(SkCanvas* canvas, const SkPaint& paint) const { offset = pathLen * fOffset, len = end - start; - if (len <= 0) { - return; - } - - const SkScalar dashes[] = { len, pathLen - len }; - SkPaint dashedPaint(paint); - dashedPaint.setPathEffect(SkDashPathEffect::Make(dashes, - SK_ARRAY_COUNT(dashes), - -start - offset)); + fTrimmedPath.reset(); - canvas->drawPath(path, dashedPaint); -} - -SkPath TrimEffect::onAsPath() const { - return fChild->asPath(); -} + if (len > 0) { + // If the trim is positioned exactly at the beginning of the path, we don't expect any + // visible wrap-around. But due to limited precision / accumulated error, the dash effect + // may sometimes start one extra dash at the very end (a flickering dot during animation). + // To avoid this, we bump the dash len by |epsilon|. + const SkScalar dashes[] = { len, pathLen + SK_ScalarNearlyZero - len }; + auto dashEffect = SkDashPathEffect::Make(dashes, + SK_ARRAY_COUNT(dashes), + -start - offset); + if (dashEffect) { + SkStrokeRec rec(SkStrokeRec::kHairline_InitStyle); + SkAssertResult(dashEffect->filterPath(&fTrimmedPath, path, &rec, &childbounds)); + } + } -SkRect TrimEffect::onRevalidate(InvalidationController* ic, const SkMatrix& ctm) { - SkASSERT(this->hasInval()); - return fChild->revalidate(ic, ctm); + return fTrimmedPath.computeTightBounds(); } } // namespace sksg diff --git a/experimental/sksg/geometry/SkSGTrimEffect.h b/experimental/sksg/geometry/SkSGTrimEffect.h index e86ede83e6..4e950a7ade 100644 --- a/experimental/sksg/geometry/SkSGTrimEffect.h +++ b/experimental/sksg/geometry/SkSGTrimEffect.h @@ -44,6 +44,7 @@ private: const sk_sp<GeometryNode> fChild; + SkPath fTrimmedPath; SkScalar fStart = 0, // starting t fEnd = 1, // ending t fOffset = 0; // t offset |