1 /* 2 * Copyright 2015 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 GrAAConvexTessellator_DEFINED 9 #define GrAAConvexTessellator_DEFINED 10 11 #include "SkColor.h" 12 #include "SkPoint.h" 13 #include "SkScalar.h" 14 #include "SkTDArray.h" 15 16 class SkCanvas; 17 class SkMatrix; 18 class SkPath; 19 20 //#define GR_AA_CONVEX_TESSELLATOR_VIZ 1 21 22 class GrAAConvexTessellator; 23 24 // The AAConvexTessellator holds the global pool of points and the triangulation 25 // that connects them. It also drives the tessellation process. 26 // The outward facing normals of the original polygon are stored (in 'fNorms') to service 27 // computeDepthFromEdge requests. 28 class GrAAConvexTessellator { 29 public: 30 GrAAConvexTessellator(SkScalar targetDepth = 0.5f) fSide(SkPoint::kOn_Side)31 : fSide(SkPoint::kOn_Side) 32 , fTargetDepth(targetDepth) { 33 } 34 setTargetDepth(SkScalar targetDepth)35 void setTargetDepth(SkScalar targetDepth) { fTargetDepth = targetDepth; } targetDepth()36 SkScalar targetDepth() const { return fTargetDepth; } 37 side()38 SkPoint::Side side() const { return fSide; } 39 40 bool tessellate(const SkMatrix& m, const SkPath& path); 41 42 // The next five should only be called after tessellate to extract the result numPts()43 int numPts() const { return fPts.count(); } numIndices()44 int numIndices() const { return fIndices.count(); } 45 lastPoint()46 const SkPoint& lastPoint() const { return fPts.top(); } point(int index)47 const SkPoint& point(int index) const { return fPts[index]; } index(int index)48 int index(int index) const { return fIndices[index]; } depth(int index)49 SkScalar depth(int index) const {return fDepths[index]; } 50 51 #if GR_AA_CONVEX_TESSELLATOR_VIZ 52 void draw(SkCanvas* canvas) const; 53 #endif 54 55 // The tessellator can be reused for multiple paths by rewinding in between 56 void rewind(); 57 58 private: 59 // CandidateVerts holds the vertices for the next ring while they are 60 // being generated. Its main function is to de-dup the points. 61 class CandidateVerts { 62 public: setReserve(int numPts)63 void setReserve(int numPts) { fPts.setReserve(numPts); } rewind()64 void rewind() { fPts.rewind(); } 65 numPts()66 int numPts() const { return fPts.count(); } 67 lastPoint()68 const SkPoint& lastPoint() const { return fPts.top().fPt; } firstPoint()69 const SkPoint& firstPoint() const { return fPts[0].fPt; } point(int index)70 const SkPoint& point(int index) const { return fPts[index].fPt; } 71 originatingIdx(int index)72 int originatingIdx(int index) const { return fPts[index].fOriginatingIdx; } origEdge(int index)73 int origEdge(int index) const { return fPts[index].fOrigEdgeId; } needsToBeNew(int index)74 bool needsToBeNew(int index) const { return fPts[index].fNeedsToBeNew; } 75 addNewPt(const SkPoint & newPt,int originatingIdx,int origEdge,bool needsToBeNew)76 int addNewPt(const SkPoint& newPt, int originatingIdx, int origEdge, bool needsToBeNew) { 77 struct PointData* pt = fPts.push(); 78 pt->fPt = newPt; 79 pt->fOrigEdgeId = origEdge; 80 pt->fOriginatingIdx = originatingIdx; 81 pt->fNeedsToBeNew = needsToBeNew; 82 return fPts.count() - 1; 83 } 84 fuseWithPrior(int origEdgeId)85 int fuseWithPrior(int origEdgeId) { 86 fPts.top().fOrigEdgeId = origEdgeId; 87 fPts.top().fOriginatingIdx = -1; 88 fPts.top().fNeedsToBeNew = true; 89 return fPts.count() - 1; 90 } 91 fuseWithNext()92 int fuseWithNext() { 93 fPts[0].fOriginatingIdx = -1; 94 fPts[0].fNeedsToBeNew = true; 95 return 0; 96 } 97 fuseWithBoth()98 int fuseWithBoth() { 99 if (fPts.count() > 1) { 100 fPts.pop(); 101 } 102 103 fPts[0].fOriginatingIdx = -1; 104 fPts[0].fNeedsToBeNew = true; 105 return 0; 106 } 107 108 private: 109 struct PointData { 110 SkPoint fPt; 111 int fOriginatingIdx; 112 int fOrigEdgeId; 113 bool fNeedsToBeNew; 114 }; 115 116 SkTDArray<struct PointData> fPts; 117 }; 118 119 // The Ring holds a set of indices into the global pool that together define 120 // a single polygon inset. 121 class Ring { 122 public: setReserve(int numPts)123 void setReserve(int numPts) { fPts.setReserve(numPts); } rewind()124 void rewind() { fPts.rewind(); } 125 numPts()126 int numPts() const { return fPts.count(); } 127 addIdx(int index,int origEdgeId)128 void addIdx(int index, int origEdgeId) { 129 struct PointData* pt = fPts.push(); 130 pt->fIndex = index; 131 pt->fOrigEdgeId = origEdgeId; 132 } 133 134 // init should be called after all the indices have been added (via addIdx) 135 void init(const GrAAConvexTessellator& tess); 136 void init(const SkTDArray<SkVector>& norms, const SkTDArray<SkVector>& bisectors); 137 norm(int index)138 const SkPoint& norm(int index) const { return fPts[index].fNorm; } bisector(int index)139 const SkPoint& bisector(int index) const { return fPts[index].fBisector; } index(int index)140 int index(int index) const { return fPts[index].fIndex; } origEdgeID(int index)141 int origEdgeID(int index) const { return fPts[index].fOrigEdgeId; } 142 143 #if GR_AA_CONVEX_TESSELLATOR_VIZ 144 void draw(SkCanvas* canvas, const GrAAConvexTessellator& tess) const; 145 #endif 146 147 private: 148 void computeNormals(const GrAAConvexTessellator& result); 149 void computeBisectors(); 150 151 SkDEBUGCODE(bool isConvex(const GrAAConvexTessellator& tess) const;) 152 153 struct PointData { 154 SkPoint fNorm; 155 SkPoint fBisector; 156 int fIndex; 157 int fOrigEdgeId; 158 }; 159 160 SkTDArray<PointData> fPts; 161 }; 162 movable(int index)163 bool movable(int index) const { return fMovable[index]; } 164 165 // Movable points are those that can be slid along their bisector. 166 // Basically, a point is immovable if it is part of the original 167 // polygon or it results from the fusing of two bisectors. 168 int addPt(const SkPoint& pt, SkScalar depth, bool movable); 169 void popLastPt(); 170 void popFirstPtShuffle(); 171 172 void updatePt(int index, const SkPoint& pt, SkScalar depth); 173 174 void addTri(int i0, int i1, int i2); 175 reservePts(int count)176 void reservePts(int count) { 177 fPts.setReserve(count); 178 fDepths.setReserve(count); 179 fMovable.setReserve(count); 180 } 181 182 SkScalar computeDepthFromEdge(int edgeIdx, const SkPoint& p) const; 183 184 bool computePtAlongBisector(int startIdx, const SkPoint& bisector, 185 int edgeIdx, SkScalar desiredDepth, 186 SkPoint* result) const; 187 188 void terminate(const Ring& lastRing); 189 190 // return false on failure/degenerate path 191 bool extractFromPath(const SkMatrix& m, const SkPath& path); 192 void computeBisectors(); 193 194 void fanRing(const Ring& ring); 195 void createOuterRing(); 196 197 Ring* getNextRing(Ring* lastRing); 198 199 bool createInsetRing(const Ring& lastRing, Ring* nextRing); 200 201 void validate() const; 202 203 204 #ifdef SK_DEBUG 205 SkScalar computeRealDepth(const SkPoint& p) const; 206 void checkAllDepths() const; 207 #endif 208 209 // fPts, fWeights & fMovable should always have the same # of elements 210 SkTDArray<SkPoint> fPts; 211 SkTDArray<SkScalar> fDepths; 212 // movable points are those that can be slid further along their bisector 213 SkTDArray<bool> fMovable; 214 215 // The outward facing normals for the original polygon 216 SkTDArray<SkVector> fNorms; 217 // The inward facing bisector at each point in the original polygon. Only 218 // needed for exterior ring creation and then handed off to the initial ring. 219 SkTDArray<SkVector> fBisectors; 220 SkPoint::Side fSide; // winding of the original polygon 221 222 // The triangulation of the points 223 SkTDArray<int> fIndices; 224 225 Ring fInitialRing; 226 #if GR_AA_CONVEX_TESSELLATOR_VIZ 227 // When visualizing save all the rings 228 SkTDArray<Ring*> fRings; 229 #else 230 Ring fRings[2]; 231 #endif 232 CandidateVerts fCandidateVerts; 233 234 SkScalar fTargetDepth; 235 236 // If some goes wrong with the inset computation the tessellator will 237 // truncate the creation of the inset polygon. In this case the depth 238 // check will complain. 239 SkDEBUGCODE(bool fShouldCheckDepths;) 240 }; 241 242 243 #endif 244 245