1 /* 2 * Copyright 2016 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #ifndef GrStyle_DEFINED 9 #define GrStyle_DEFINED 10 11 #include "GrTypes.h" 12 #include "SkPathEffect.h" 13 #include "SkStrokeRec.h" 14 #include "SkTemplates.h" 15 16 /** 17 * Represents the various ways that a GrShape can be styled. It has fill/stroking information 18 * as well as an optional path effect. If the path effect represents dashing, the dashing 19 * information is extracted from the path effect and stored explicitly. 20 * 21 * This will replace GrStrokeInfo as GrShape is deployed. 22 */ 23 class GrStyle { 24 public: 25 /** 26 * A style object that represents a fill with no path effect. 27 * TODO: constexpr with C++14 28 */ 29 static const GrStyle& SimpleFill() { 30 static const GrStyle kFill(SkStrokeRec::kFill_InitStyle); 31 return kFill; 32 } 33 34 /** 35 * A style object that represents a hairline stroke with no path effect. 36 * TODO: constexpr with C++14 37 */ 38 static const GrStyle& SimpleHairline() { 39 static const GrStyle kHairline(SkStrokeRec::kHairline_InitStyle); 40 return kHairline; 41 } 42 43 enum class Apply { 44 kPathEffectOnly, 45 kPathEffectAndStrokeRec 46 }; 47 48 /** 49 * Optional flags for computing keys that may remove unnecessary variation in the key due to 50 * style settings that don't affect particular classes of geometry. 51 */ 52 enum KeyFlags { 53 // The shape being styled has no open contours. 54 kClosed_KeyFlag = 0x1, 55 // The shape being styled doesn't have any joins and so isn't affected by join type. 56 kNoJoins_KeyFlag = 0x2 57 }; 58 59 /** 60 * Computes the key length for a GrStyle. The return will be negative if it cannot be turned 61 * into a key. This occurs when there is a path effect that is not a dash. The key can 62 * either reflect just the path effect (if one) or the path effect and the strokerec. Note 63 * that a simple fill has a zero sized key. 64 */ 65 static int KeySize(const GrStyle&, Apply, uint32_t flags = 0); 66 67 /** 68 * Writes a unique key for the style into the provided buffer. This function assumes the buffer 69 * has room for at least KeySize() values. It assumes that KeySize() returns a non-negative 70 * value for the combination of GrStyle, Apply and flags params. This is written so that the key 71 * for just dash application followed by the key for the remaining SkStrokeRec is the same as 72 * the key for applying dashing and SkStrokeRec all at once. 73 */ 74 static void WriteKey(uint32_t*, const GrStyle&, Apply, SkScalar scale, uint32_t flags = 0); 75 76 GrStyle() : GrStyle(SkStrokeRec::kFill_InitStyle) {} 77 78 explicit GrStyle(SkStrokeRec::InitStyle initStyle) : fStrokeRec(initStyle) {} 79 80 GrStyle(const SkStrokeRec& strokeRec, sk_sp<SkPathEffect> pe) : fStrokeRec(strokeRec) { 81 this->initPathEffect(std::move(pe)); 82 } 83 84 GrStyle(const GrStyle& that) = default; 85 86 explicit GrStyle(const SkPaint& paint) : fStrokeRec(paint) { 87 this->initPathEffect(paint.refPathEffect()); 88 } 89 90 explicit GrStyle(const SkPaint& paint, SkPaint::Style overrideStyle) 91 : fStrokeRec(paint, overrideStyle) { 92 this->initPathEffect(paint.refPathEffect()); 93 } 94 95 GrStyle& operator=(const GrStyle& that) { 96 fPathEffect = that.fPathEffect; 97 fDashInfo = that.fDashInfo; 98 fStrokeRec = that.fStrokeRec; 99 return *this; 100 } 101 102 void resetToInitStyle(SkStrokeRec::InitStyle fillOrHairline) { 103 fDashInfo.reset(); 104 fPathEffect.reset(nullptr); 105 if (SkStrokeRec::kFill_InitStyle == fillOrHairline) { 106 fStrokeRec.setFillStyle(); 107 } else { 108 fStrokeRec.setHairlineStyle(); 109 } 110 } 111 112 /** Is this style a fill with no path effect? */ 113 bool isSimpleFill() const { return fStrokeRec.isFillStyle() && !fPathEffect; } 114 115 /** Is this style a hairline with no path effect? */ 116 bool isSimpleHairline() const { return fStrokeRec.isHairlineStyle() && !fPathEffect; } 117 118 SkPathEffect* pathEffect() const { return fPathEffect.get(); } 119 sk_sp<SkPathEffect> refPathEffect() const { return fPathEffect; } 120 121 bool hasPathEffect() const { return SkToBool(fPathEffect.get()); } 122 123 bool hasNonDashPathEffect() const { return fPathEffect.get() && !this->isDashed(); } 124 125 bool isDashed() const { return SkPathEffect::kDash_DashType == fDashInfo.fType; } 126 SkScalar dashPhase() const { 127 SkASSERT(this->isDashed()); 128 return fDashInfo.fPhase; 129 } 130 int dashIntervalCnt() const { 131 SkASSERT(this->isDashed()); 132 return fDashInfo.fIntervals.count(); 133 } 134 const SkScalar* dashIntervals() const { 135 SkASSERT(this->isDashed()); 136 return fDashInfo.fIntervals.get(); 137 } 138 139 const SkStrokeRec& strokeRec() const { return fStrokeRec; } 140 141 /** Hairline or fill styles without path effects make no alterations to a geometry. */ 142 bool applies() const { 143 return this->pathEffect() || (!fStrokeRec.isFillStyle() && !fStrokeRec.isHairlineStyle()); 144 } 145 146 static SkScalar MatrixToScaleFactor(const SkMatrix& matrix) { 147 // getMaxScale will return -1 if the matrix has perspective. In that case we can use a scale 148 // factor of 1. This isn't necessarily a good choice and in the future we might consider 149 // taking a bounds here for the perspective case. 150 return SkScalarAbs(matrix.getMaxScale()); 151 } 152 /** 153 * Applies just the path effect and returns remaining stroke information. This will fail if 154 * there is no path effect. dst may or may not have been overwritten on failure. Scale controls 155 * geometric approximations made by the path effect. It is typically computed from the view 156 * matrix. 157 */ 158 bool SK_WARN_UNUSED_RESULT applyPathEffectToPath(SkPath* dst, SkStrokeRec* remainingStoke, 159 const SkPath& src, SkScalar scale) const; 160 161 /** 162 * If this succeeds then the result path should be filled or hairlined as indicated by the 163 * returned SkStrokeRec::InitStyle value. Will fail if there is no path effect and the 164 * strokerec doesn't change the geometry. When this fails the outputs may or may not have 165 * been overwritten. Scale controls geometric approximations made by the path effect and 166 * stroker. It is typically computed from the view matrix. 167 */ 168 bool SK_WARN_UNUSED_RESULT applyToPath(SkPath* dst, SkStrokeRec::InitStyle* fillOrHairline, 169 const SkPath& src, SkScalar scale) const; 170 171 /** Given bounds of a path compute the bounds of path with the style applied. */ 172 void adjustBounds(SkRect* dst, const SkRect& src) const { 173 if (this->pathEffect()) { 174 this->pathEffect()->computeFastBounds(dst, src); 175 // This may not be the correct SkStrokeRec to use. skbug.com/5299 176 // It happens to work for dashing. 177 SkScalar radius = fStrokeRec.getInflationRadius(); 178 dst->outset(radius, radius); 179 } else { 180 SkScalar radius = fStrokeRec.getInflationRadius(); 181 *dst = src.makeOutset(radius, radius); 182 } 183 } 184 185 private: 186 void initPathEffect(sk_sp<SkPathEffect> pe); 187 188 struct DashInfo { 189 DashInfo() : fType(SkPathEffect::kNone_DashType) {} 190 DashInfo(const DashInfo& that) { *this = that; } 191 DashInfo& operator=(const DashInfo& that) { 192 fType = that.fType; 193 fPhase = that.fPhase; 194 fIntervals.reset(that.fIntervals.count()); 195 sk_careful_memcpy(fIntervals.get(), that.fIntervals.get(), 196 sizeof(SkScalar) * that.fIntervals.count()); 197 return *this; 198 } 199 void reset() { 200 fType = SkPathEffect::kNone_DashType; 201 fIntervals.reset(0); 202 } 203 SkPathEffect::DashType fType; 204 SkScalar fPhase; 205 SkAutoSTArray<4, SkScalar> fIntervals; 206 }; 207 208 bool applyPathEffect(SkPath* dst, SkStrokeRec* strokeRec, const SkPath& src) const; 209 210 SkStrokeRec fStrokeRec; 211 sk_sp<SkPathEffect> fPathEffect; 212 DashInfo fDashInfo; 213 }; 214 215 #endif 216