| Index: src/gpu/GrStyle.cpp
|
| diff --git a/src/gpu/GrStyle.cpp b/src/gpu/GrStyle.cpp
|
| index 40a148bb4a9c6e88063d064aa3232ef6fda50202..7d70baa1bcda4fc7e7395e8d1481849236f28d88 100644
|
| --- a/src/gpu/GrStyle.cpp
|
| +++ b/src/gpu/GrStyle.cpp
|
| @@ -7,27 +7,142 @@
|
|
|
| #include "GrStyle.h"
|
|
|
| +int GrStyle::KeySize(const GrStyle &style, Apply apply) {
|
| + GR_STATIC_ASSERT(sizeof(uint32_t) == sizeof(SkScalar));
|
| + int size = 0;
|
| + if (style.isDashed()) {
|
| + // One scalar for dash phase and one for each dash value.
|
| + size += 1 + style.dashIntervalCnt();
|
| + } else if (style.pathEffect()) {
|
| + // No key for a generic path effect.
|
| + return -1;
|
| + }
|
| +
|
| + if (Apply::kPathEffectOnly == apply) {
|
| + return size;
|
| + }
|
| +
|
| + if (style.strokeRec().needToApply()) {
|
| + // One for style/cap/join, 2 for miter and width.
|
| + size += 3;
|
| + }
|
| + return size;
|
| +}
|
| +
|
| +void GrStyle::WriteKey(uint32_t *key, const GrStyle &style, Apply apply) {
|
| + SkASSERT(key);
|
| + SkASSERT(KeySize(style, apply) >= 0);
|
| + GR_STATIC_ASSERT(sizeof(uint32_t) == sizeof(SkScalar));
|
| +
|
| + int i = 0;
|
| + if (style.isDashed()) {
|
| + GR_STATIC_ASSERT(sizeof(style.dashPhase()) == sizeof(uint32_t));
|
| + SkScalar phase = style.dashPhase();
|
| + memcpy(&key[i++], &phase, sizeof(SkScalar));
|
| +
|
| + int32_t count = style.dashIntervalCnt();
|
| + // Dash count should always be even.
|
| + SkASSERT(0 == (count & 0x1));
|
| + const SkScalar *intervals = style.dashIntervals();
|
| + int intervalByteCnt = count * sizeof(SkScalar);
|
| + memcpy(&key[i], intervals, intervalByteCnt);
|
| + i += count;
|
| + } else {
|
| + SkASSERT(!style.pathEffect());
|
| + }
|
| +
|
| + if (Apply::kPathEffectAndStrokeRec == apply && style.strokeRec().needToApply()) {
|
| + enum {
|
| + kStyleBits = 2,
|
| + kJoinBits = 2,
|
| + kCapBits = 32 - kStyleBits - kJoinBits,
|
| +
|
| + kJoinShift = kStyleBits,
|
| + kCapShift = kJoinShift + kJoinBits,
|
| + };
|
| + GR_STATIC_ASSERT(SkStrokeRec::kStyleCount <= (1 << kStyleBits));
|
| + GR_STATIC_ASSERT(SkPaint::kJoinCount <= (1 << kJoinBits));
|
| + GR_STATIC_ASSERT(SkPaint::kCapCount <= (1 << kCapBits));
|
| + key[i++] = style.strokeRec().getStyle() |
|
| + style.strokeRec().getJoin() << kJoinShift |
|
| + style.strokeRec().getCap() << kCapShift;
|
| +
|
| + SkScalar scalar;
|
| + // Miter limit only affects miter joins
|
| + scalar = SkPaint::kMiter_Join == style.strokeRec().getJoin()
|
| + ? style.strokeRec().getMiter()
|
| + : -1.f;
|
| + memcpy(&key[i++], &scalar, sizeof(scalar));
|
| +
|
| + scalar = style.strokeRec().getWidth();
|
| + memcpy(&key[i++], &scalar, sizeof(scalar));
|
| + }
|
| + SkASSERT(KeySize(style, apply) == i);
|
| +}
|
| +
|
| void GrStyle::initPathEffect(SkPathEffect* pe) {
|
| + SkASSERT(!fPathEffect)
|
| + SkASSERT(SkPathEffect::kNone_DashType == fDashInfo.fType);
|
| + SkASSERT(0 == fDashInfo.fIntervals.count());
|
| if (!pe) {
|
| - fDashInfo.fType = SkPathEffect::kNone_DashType;
|
| return;
|
| }
|
| SkPathEffect::DashInfo info;
|
| if (SkPathEffect::kDash_DashType == pe->asADash(&info)) {
|
| - if (fStrokeRec.getStyle() == SkStrokeRec::kFill_Style) {
|
| - fPathEffect.reset(nullptr);
|
| - } else {
|
| - fPathEffect.reset(SkSafeRef(pe));
|
| + if (fStrokeRec.getStyle() != SkStrokeRec::kFill_Style) {
|
| fDashInfo.fType = SkPathEffect::kDash_DashType;
|
| fDashInfo.fIntervals.reset(info.fCount);
|
| fDashInfo.fPhase = info.fPhase;
|
| info.fIntervals = fDashInfo.fIntervals.get();
|
| pe->asADash(&info);
|
| - return;
|
| + fPathEffect.reset(SkSafeRef(pe));
|
| }
|
| } else {
|
| fPathEffect.reset(SkSafeRef(pe));
|
| }
|
| - fDashInfo.fType = SkPathEffect::kNone_DashType;
|
| - fDashInfo.fIntervals.reset(0);
|
| +}
|
| +
|
| +static inline bool apply_path_effect(SkPath* dst, SkStrokeRec* strokeRec,
|
| + const sk_sp<SkPathEffect>& pe, const SkPath& src) {
|
| + if (!pe) {
|
| + return false;
|
| + }
|
| + if (!pe->filterPath(dst, src, strokeRec, nullptr)) {
|
| + return false;
|
| + }
|
| + dst->setIsVolatile(true);
|
| + return true;
|
| +}
|
| +
|
| +bool GrStyle::applyPathEffectToPath(SkPath *dst, SkStrokeRec *remainingStroke,
|
| + const SkPath &src) const {
|
| + SkASSERT(dst);
|
| + SkStrokeRec strokeRec = fStrokeRec;
|
| + if (!apply_path_effect(dst, &strokeRec, fPathEffect, src)) {
|
| + return false;
|
| + }
|
| + *remainingStroke = strokeRec;
|
| + return true;
|
| +}
|
| +
|
| +bool GrStyle::applyToPath(SkPath* dst, SkStrokeRec::InitStyle* style, const SkPath& src) const {
|
| + SkASSERT(style);
|
| + SkASSERT(dst);
|
| + SkStrokeRec strokeRec = fStrokeRec;
|
| + if (!apply_path_effect(dst, &strokeRec, fPathEffect, src)) {
|
| + return false;
|
| + }
|
| + if (strokeRec.needToApply()) {
|
| + if (!strokeRec.applyToPath(dst, *dst)) {
|
| + return false;
|
| + }
|
| + *style = SkStrokeRec::kFill_InitStyle;
|
| + } else {
|
| + SkASSERT(SkStrokeRec::kFill_Style == strokeRec.getStyle() ||
|
| + SkStrokeRec::kHairline_Style == strokeRec.getStyle());
|
| + *style = strokeRec.getStyle() == SkStrokeRec::kFill_Style
|
| + ? SkStrokeRec::kFill_InitStyle
|
| + : SkStrokeRec::kHairline_InitStyle;
|
| + }
|
| + return true;
|
| }
|
|
|