|  | /* | 
|  | * Copyright 2006 The Android Open Source Project | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  |  | 
|  | #include "SkPathEffect.h" | 
|  | #include "SkPath.h" | 
|  | #include "SkReadBuffer.h" | 
|  | #include "SkWriteBuffer.h" | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | void SkPathEffect::computeFastBounds(SkRect* dst, const SkRect& src) const { | 
|  | *dst = src; | 
|  | } | 
|  |  | 
|  | bool SkPathEffect::asPoints(PointData* results, const SkPath& src, | 
|  | const SkStrokeRec&, const SkMatrix&, const SkRect*) const { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | SkPathEffect::DashType SkPathEffect::asADash(DashInfo* info) const { | 
|  | return kNone_DashType; | 
|  | } | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | /** \class SkPairPathEffect | 
|  |  | 
|  | Common baseclass for Compose and Sum. This subclass manages two pathEffects, | 
|  | including flattening them. It does nothing in filterPath, and is only useful | 
|  | for managing the lifetimes of its two arguments. | 
|  | */ | 
|  | class SkPairPathEffect : public SkPathEffect { | 
|  | protected: | 
|  | SkPairPathEffect(sk_sp<SkPathEffect> pe0, sk_sp<SkPathEffect> pe1) | 
|  | : fPE0(std::move(pe0)), fPE1(std::move(pe1)) | 
|  | { | 
|  | SkASSERT(fPE0.get()); | 
|  | SkASSERT(fPE1.get()); | 
|  | } | 
|  |  | 
|  | void flatten(SkWriteBuffer& buffer) const override { | 
|  | buffer.writeFlattenable(fPE0.get()); | 
|  | buffer.writeFlattenable(fPE1.get()); | 
|  | } | 
|  |  | 
|  | // these are visible to our subclasses | 
|  | sk_sp<SkPathEffect> fPE0; | 
|  | sk_sp<SkPathEffect> fPE1; | 
|  |  | 
|  | void toString(SkString* str) const override; | 
|  |  | 
|  | private: | 
|  | typedef SkPathEffect INHERITED; | 
|  | }; | 
|  |  | 
|  | void SkPairPathEffect::toString(SkString* str) const { | 
|  | str->appendf("first: "); | 
|  | if (fPE0) { | 
|  | fPE0->toString(str); | 
|  | } | 
|  | str->appendf(" second: "); | 
|  | if (fPE1) { | 
|  | fPE1->toString(str); | 
|  | } | 
|  | } | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | /** \class SkComposePathEffect | 
|  |  | 
|  | This subclass of SkPathEffect composes its two arguments, to create | 
|  | a compound pathEffect. | 
|  | */ | 
|  | class SkComposePathEffect : public SkPairPathEffect { | 
|  | public: | 
|  | /** Construct a pathEffect whose effect is to apply first the inner pathEffect | 
|  | and the the outer pathEffect (e.g. outer(inner(path))) | 
|  | The reference counts for outer and inner are both incremented in the constructor, | 
|  | and decremented in the destructor. | 
|  | */ | 
|  | static sk_sp<SkPathEffect> Make(sk_sp<SkPathEffect> outer, sk_sp<SkPathEffect> inner) { | 
|  | if (!outer) { | 
|  | return inner; | 
|  | } | 
|  | if (!inner) { | 
|  | return outer; | 
|  | } | 
|  | return sk_sp<SkPathEffect>(new SkComposePathEffect(outer, inner)); | 
|  | } | 
|  |  | 
|  | bool filterPath(SkPath* dst, const SkPath& src, SkStrokeRec* rec, | 
|  | const SkRect* cullRect) const override { | 
|  | SkPath          tmp; | 
|  | const SkPath*   ptr = &src; | 
|  |  | 
|  | if (fPE1->filterPath(&tmp, src, rec, cullRect)) { | 
|  | ptr = &tmp; | 
|  | } | 
|  | return fPE0->filterPath(dst, *ptr, rec, cullRect); | 
|  | } | 
|  |  | 
|  |  | 
|  | void toString(SkString* str) const override; | 
|  | SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkComposePathEffect) | 
|  |  | 
|  | #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK | 
|  | bool exposedInAndroidJavaAPI() const override { return true; } | 
|  | #endif | 
|  |  | 
|  | protected: | 
|  | SkComposePathEffect(sk_sp<SkPathEffect> outer, sk_sp<SkPathEffect> inner) | 
|  | : INHERITED(outer, inner) {} | 
|  |  | 
|  | private: | 
|  | // illegal | 
|  | SkComposePathEffect(const SkComposePathEffect&); | 
|  | SkComposePathEffect& operator=(const SkComposePathEffect&); | 
|  | friend class SkPathEffect; | 
|  |  | 
|  | typedef SkPairPathEffect INHERITED; | 
|  | }; | 
|  |  | 
|  | sk_sp<SkFlattenable> SkComposePathEffect::CreateProc(SkReadBuffer& buffer) { | 
|  | sk_sp<SkPathEffect> pe0(buffer.readPathEffect()); | 
|  | sk_sp<SkPathEffect> pe1(buffer.readPathEffect()); | 
|  | return SkComposePathEffect::Make(std::move(pe0), std::move(pe1)); | 
|  | } | 
|  |  | 
|  | void SkComposePathEffect::toString(SkString* str) const { | 
|  | str->appendf("SkComposePathEffect: ("); | 
|  | this->INHERITED::toString(str); | 
|  | str->appendf(")"); | 
|  | } | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | /** \class SkSumPathEffect | 
|  |  | 
|  | This subclass of SkPathEffect applies two pathEffects, one after the other. | 
|  | Its filterPath() returns true if either of the effects succeeded. | 
|  | */ | 
|  | class SkSumPathEffect : public SkPairPathEffect { | 
|  | public: | 
|  | /** Construct a pathEffect whose effect is to apply two effects, in sequence. | 
|  | (e.g. first(path) + second(path)) | 
|  | The reference counts for first and second are both incremented in the constructor, | 
|  | and decremented in the destructor. | 
|  | */ | 
|  | static sk_sp<SkPathEffect> Make(sk_sp<SkPathEffect> first, sk_sp<SkPathEffect> second) { | 
|  | if (!first) { | 
|  | return second; | 
|  | } | 
|  | if (!second) { | 
|  | return first; | 
|  | } | 
|  | return sk_sp<SkPathEffect>(new SkSumPathEffect(first, second)); | 
|  | } | 
|  |  | 
|  | bool filterPath(SkPath* dst, const SkPath& src, SkStrokeRec* rec, | 
|  | const SkRect* cullRect) const override { | 
|  | // use bit-or so that we always call both, even if the first one succeeds | 
|  | return fPE0->filterPath(dst, src, rec, cullRect) | | 
|  | fPE1->filterPath(dst, src, rec, cullRect); | 
|  | } | 
|  |  | 
|  |  | 
|  | void toString(SkString* str) const override; | 
|  | SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkSumPathEffect) | 
|  |  | 
|  | #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK | 
|  | bool exposedInAndroidJavaAPI() const override { return true; } | 
|  | #endif | 
|  |  | 
|  | protected: | 
|  | SkSumPathEffect(sk_sp<SkPathEffect> first, sk_sp<SkPathEffect> second) | 
|  | : INHERITED(first, second) {} | 
|  |  | 
|  | private: | 
|  | // illegal | 
|  | SkSumPathEffect(const SkSumPathEffect&); | 
|  | SkSumPathEffect& operator=(const SkSumPathEffect&); | 
|  | friend class SkPathEffect; | 
|  |  | 
|  | typedef SkPairPathEffect INHERITED; | 
|  | }; | 
|  |  | 
|  | sk_sp<SkFlattenable> SkSumPathEffect::CreateProc(SkReadBuffer& buffer) { | 
|  | sk_sp<SkPathEffect> pe0(buffer.readPathEffect()); | 
|  | sk_sp<SkPathEffect> pe1(buffer.readPathEffect()); | 
|  | return SkSumPathEffect::Make(pe0, pe1); | 
|  | } | 
|  |  | 
|  | void SkSumPathEffect::toString(SkString* str) const { | 
|  | str->appendf("SkSumPathEffect: ("); | 
|  | this->INHERITED::toString(str); | 
|  | str->appendf(")"); | 
|  | } | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | sk_sp<SkPathEffect> SkPathEffect::MakeSum(sk_sp<SkPathEffect> first, sk_sp<SkPathEffect> second) { | 
|  | return SkSumPathEffect::Make(std::move(first), std::move(second)); | 
|  | } | 
|  |  | 
|  | sk_sp<SkPathEffect> SkPathEffect::MakeCompose(sk_sp<SkPathEffect> outer, | 
|  | sk_sp<SkPathEffect> inner) { | 
|  | return SkComposePathEffect::Make(std::move(outer), std::move(inner)); | 
|  | } | 
|  |  | 
|  | SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_START(SkPathEffect) | 
|  | SK_DEFINE_FLATTENABLE_REGISTRAR_ENTRY(SkComposePathEffect) | 
|  | SK_DEFINE_FLATTENABLE_REGISTRAR_ENTRY(SkSumPathEffect) | 
|  | SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_END |