xref: /aosp_15_r20/external/skia/src/pathops/SkPathWriter.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2012 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathWriter.h"
8*c8dee2aaSAndroid Build Coastguard Worker 
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkMath.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkTSort.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkOpSegment.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkOpSpan.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsDebug.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsTypes.h"
16*c8dee2aaSAndroid Build Coastguard Worker 
17*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
18*c8dee2aaSAndroid Build Coastguard Worker 
19*c8dee2aaSAndroid Build Coastguard Worker // wrap path to keep track of whether the contour is initialized and non-empty
SkPathWriter(SkPath & path)20*c8dee2aaSAndroid Build Coastguard Worker SkPathWriter::SkPathWriter(SkPath& path)
21*c8dee2aaSAndroid Build Coastguard Worker     : fPathPtr(&path)
22*c8dee2aaSAndroid Build Coastguard Worker {
23*c8dee2aaSAndroid Build Coastguard Worker     init();
24*c8dee2aaSAndroid Build Coastguard Worker }
25*c8dee2aaSAndroid Build Coastguard Worker 
close()26*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::close() {
27*c8dee2aaSAndroid Build Coastguard Worker     if (fCurrent.isEmpty()) {
28*c8dee2aaSAndroid Build Coastguard Worker         return;
29*c8dee2aaSAndroid Build Coastguard Worker     }
30*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(this->isClosed());
31*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
32*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.close();\n");
33*c8dee2aaSAndroid Build Coastguard Worker #endif
34*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.close();
35*c8dee2aaSAndroid Build Coastguard Worker     fPathPtr->addPath(fCurrent);
36*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.reset();
37*c8dee2aaSAndroid Build Coastguard Worker     init();
38*c8dee2aaSAndroid Build Coastguard Worker }
39*c8dee2aaSAndroid Build Coastguard Worker 
conicTo(const SkPoint & pt1,const SkOpPtT * pt2,SkScalar weight)40*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::conicTo(const SkPoint& pt1, const SkOpPtT* pt2, SkScalar weight) {
41*c8dee2aaSAndroid Build Coastguard Worker     SkPoint pt2pt = this->update(pt2);
42*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
43*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.conicTo(%1.9g,%1.9g, %1.9g,%1.9g, %1.9g);\n",
44*c8dee2aaSAndroid Build Coastguard Worker             pt1.fX, pt1.fY, pt2pt.fX, pt2pt.fY, weight);
45*c8dee2aaSAndroid Build Coastguard Worker #endif
46*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.conicTo(pt1, pt2pt, weight);
47*c8dee2aaSAndroid Build Coastguard Worker }
48*c8dee2aaSAndroid Build Coastguard Worker 
cubicTo(const SkPoint & pt1,const SkPoint & pt2,const SkOpPtT * pt3)49*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::cubicTo(const SkPoint& pt1, const SkPoint& pt2, const SkOpPtT* pt3) {
50*c8dee2aaSAndroid Build Coastguard Worker     SkPoint pt3pt = this->update(pt3);
51*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
52*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.cubicTo(%1.9g,%1.9g, %1.9g,%1.9g, %1.9g,%1.9g);\n",
53*c8dee2aaSAndroid Build Coastguard Worker             pt1.fX, pt1.fY, pt2.fX, pt2.fY, pt3pt.fX, pt3pt.fY);
54*c8dee2aaSAndroid Build Coastguard Worker #endif
55*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.cubicTo(pt1, pt2, pt3pt);
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker 
deferredLine(const SkOpPtT * pt)58*c8dee2aaSAndroid Build Coastguard Worker bool SkPathWriter::deferredLine(const SkOpPtT* pt) {
59*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fFirstPtT);
60*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fDefer[0]);
61*c8dee2aaSAndroid Build Coastguard Worker     if (fDefer[0] == pt) {
62*c8dee2aaSAndroid Build Coastguard Worker         // FIXME: why we're adding a degenerate line? Caller should have preflighted this.
63*c8dee2aaSAndroid Build Coastguard Worker         return true;
64*c8dee2aaSAndroid Build Coastguard Worker     }
65*c8dee2aaSAndroid Build Coastguard Worker     if (pt->contains(fDefer[0])) {
66*c8dee2aaSAndroid Build Coastguard Worker         // FIXME: why we're adding a degenerate line?
67*c8dee2aaSAndroid Build Coastguard Worker         return true;
68*c8dee2aaSAndroid Build Coastguard Worker     }
69*c8dee2aaSAndroid Build Coastguard Worker     if (this->matchedLast(pt)) {
70*c8dee2aaSAndroid Build Coastguard Worker         return false;
71*c8dee2aaSAndroid Build Coastguard Worker     }
72*c8dee2aaSAndroid Build Coastguard Worker     if (fDefer[1] && this->changedSlopes(pt)) {
73*c8dee2aaSAndroid Build Coastguard Worker         this->lineTo();
74*c8dee2aaSAndroid Build Coastguard Worker         fDefer[0] = fDefer[1];
75*c8dee2aaSAndroid Build Coastguard Worker     }
76*c8dee2aaSAndroid Build Coastguard Worker     fDefer[1] = pt;
77*c8dee2aaSAndroid Build Coastguard Worker     return true;
78*c8dee2aaSAndroid Build Coastguard Worker }
79*c8dee2aaSAndroid Build Coastguard Worker 
deferredMove(const SkOpPtT * pt)80*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::deferredMove(const SkOpPtT* pt) {
81*c8dee2aaSAndroid Build Coastguard Worker     if (!fDefer[1]) {
82*c8dee2aaSAndroid Build Coastguard Worker         fFirstPtT = fDefer[0] = pt;
83*c8dee2aaSAndroid Build Coastguard Worker         return;
84*c8dee2aaSAndroid Build Coastguard Worker     }
85*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fDefer[0]);
86*c8dee2aaSAndroid Build Coastguard Worker     if (!this->matchedLast(pt)) {
87*c8dee2aaSAndroid Build Coastguard Worker         this->finishContour();
88*c8dee2aaSAndroid Build Coastguard Worker         fFirstPtT = fDefer[0] = pt;
89*c8dee2aaSAndroid Build Coastguard Worker     }
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker 
finishContour()92*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::finishContour() {
93*c8dee2aaSAndroid Build Coastguard Worker     if (!this->matchedLast(fDefer[0])) {
94*c8dee2aaSAndroid Build Coastguard Worker         if (!fDefer[1]) {
95*c8dee2aaSAndroid Build Coastguard Worker           return;
96*c8dee2aaSAndroid Build Coastguard Worker         }
97*c8dee2aaSAndroid Build Coastguard Worker         this->lineTo();
98*c8dee2aaSAndroid Build Coastguard Worker     }
99*c8dee2aaSAndroid Build Coastguard Worker     if (fCurrent.isEmpty()) {
100*c8dee2aaSAndroid Build Coastguard Worker         return;
101*c8dee2aaSAndroid Build Coastguard Worker     }
102*c8dee2aaSAndroid Build Coastguard Worker     if (this->isClosed()) {
103*c8dee2aaSAndroid Build Coastguard Worker         this->close();
104*c8dee2aaSAndroid Build Coastguard Worker     } else {
105*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(fDefer[1]);
106*c8dee2aaSAndroid Build Coastguard Worker         fEndPtTs.push_back(fFirstPtT);
107*c8dee2aaSAndroid Build Coastguard Worker         fEndPtTs.push_back(fDefer[1]);
108*c8dee2aaSAndroid Build Coastguard Worker         fPartials.push_back(fCurrent);
109*c8dee2aaSAndroid Build Coastguard Worker         this->init();
110*c8dee2aaSAndroid Build Coastguard Worker     }
111*c8dee2aaSAndroid Build Coastguard Worker }
112*c8dee2aaSAndroid Build Coastguard Worker 
init()113*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::init() {
114*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.reset();
115*c8dee2aaSAndroid Build Coastguard Worker     fFirstPtT = fDefer[0] = fDefer[1] = nullptr;
116*c8dee2aaSAndroid Build Coastguard Worker }
117*c8dee2aaSAndroid Build Coastguard Worker 
isClosed() const118*c8dee2aaSAndroid Build Coastguard Worker bool SkPathWriter::isClosed() const {
119*c8dee2aaSAndroid Build Coastguard Worker     return this->matchedLast(fFirstPtT);
120*c8dee2aaSAndroid Build Coastguard Worker }
121*c8dee2aaSAndroid Build Coastguard Worker 
lineTo()122*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::lineTo() {
123*c8dee2aaSAndroid Build Coastguard Worker     if (fCurrent.isEmpty()) {
124*c8dee2aaSAndroid Build Coastguard Worker         this->moveTo();
125*c8dee2aaSAndroid Build Coastguard Worker     }
126*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
127*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.lineTo(%1.9g,%1.9g);\n", fDefer[1]->fPt.fX, fDefer[1]->fPt.fY);
128*c8dee2aaSAndroid Build Coastguard Worker #endif
129*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.lineTo(fDefer[1]->fPt);
130*c8dee2aaSAndroid Build Coastguard Worker }
131*c8dee2aaSAndroid Build Coastguard Worker 
matchedLast(const SkOpPtT * test) const132*c8dee2aaSAndroid Build Coastguard Worker bool SkPathWriter::matchedLast(const SkOpPtT* test) const {
133*c8dee2aaSAndroid Build Coastguard Worker     if (test == fDefer[1]) {
134*c8dee2aaSAndroid Build Coastguard Worker         return true;
135*c8dee2aaSAndroid Build Coastguard Worker     }
136*c8dee2aaSAndroid Build Coastguard Worker     if (!test) {
137*c8dee2aaSAndroid Build Coastguard Worker         return false;
138*c8dee2aaSAndroid Build Coastguard Worker     }
139*c8dee2aaSAndroid Build Coastguard Worker     if (!fDefer[1]) {
140*c8dee2aaSAndroid Build Coastguard Worker         return false;
141*c8dee2aaSAndroid Build Coastguard Worker     }
142*c8dee2aaSAndroid Build Coastguard Worker     return test->contains(fDefer[1]);
143*c8dee2aaSAndroid Build Coastguard Worker }
144*c8dee2aaSAndroid Build Coastguard Worker 
moveTo()145*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::moveTo() {
146*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
147*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.moveTo(%1.9g,%1.9g);\n", fFirstPtT->fPt.fX, fFirstPtT->fPt.fY);
148*c8dee2aaSAndroid Build Coastguard Worker #endif
149*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.moveTo(fFirstPtT->fPt);
150*c8dee2aaSAndroid Build Coastguard Worker }
151*c8dee2aaSAndroid Build Coastguard Worker 
quadTo(const SkPoint & pt1,const SkOpPtT * pt2)152*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::quadTo(const SkPoint& pt1, const SkOpPtT* pt2) {
153*c8dee2aaSAndroid Build Coastguard Worker     SkPoint pt2pt = this->update(pt2);
154*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
155*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("path.quadTo(%1.9g,%1.9g, %1.9g,%1.9g);\n",
156*c8dee2aaSAndroid Build Coastguard Worker             pt1.fX, pt1.fY, pt2pt.fX, pt2pt.fY);
157*c8dee2aaSAndroid Build Coastguard Worker #endif
158*c8dee2aaSAndroid Build Coastguard Worker     fCurrent.quadTo(pt1, pt2pt);
159*c8dee2aaSAndroid Build Coastguard Worker }
160*c8dee2aaSAndroid Build Coastguard Worker 
161*c8dee2aaSAndroid Build Coastguard Worker // if last point to be written matches the current path's first point, alter the
162*c8dee2aaSAndroid Build Coastguard Worker // last to avoid writing a degenerate lineTo when the path is closed
update(const SkOpPtT * pt)163*c8dee2aaSAndroid Build Coastguard Worker SkPoint SkPathWriter::update(const SkOpPtT* pt) {
164*c8dee2aaSAndroid Build Coastguard Worker     if (!fDefer[1]) {
165*c8dee2aaSAndroid Build Coastguard Worker         this->moveTo();
166*c8dee2aaSAndroid Build Coastguard Worker     } else if (!this->matchedLast(fDefer[0])) {
167*c8dee2aaSAndroid Build Coastguard Worker         this->lineTo();
168*c8dee2aaSAndroid Build Coastguard Worker     }
169*c8dee2aaSAndroid Build Coastguard Worker     SkPoint result = pt->fPt;
170*c8dee2aaSAndroid Build Coastguard Worker     if (fFirstPtT && result != fFirstPtT->fPt && fFirstPtT->contains(pt)) {
171*c8dee2aaSAndroid Build Coastguard Worker         result = fFirstPtT->fPt;
172*c8dee2aaSAndroid Build Coastguard Worker     }
173*c8dee2aaSAndroid Build Coastguard Worker     fDefer[0] = fDefer[1] = pt;  // set both to know that there is not a pending deferred line
174*c8dee2aaSAndroid Build Coastguard Worker     return result;
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker 
someAssemblyRequired()177*c8dee2aaSAndroid Build Coastguard Worker bool SkPathWriter::someAssemblyRequired() {
178*c8dee2aaSAndroid Build Coastguard Worker     this->finishContour();
179*c8dee2aaSAndroid Build Coastguard Worker     return !fEndPtTs.empty();
180*c8dee2aaSAndroid Build Coastguard Worker }
181*c8dee2aaSAndroid Build Coastguard Worker 
changedSlopes(const SkOpPtT * ptT) const182*c8dee2aaSAndroid Build Coastguard Worker bool SkPathWriter::changedSlopes(const SkOpPtT* ptT) const {
183*c8dee2aaSAndroid Build Coastguard Worker     if (matchedLast(fDefer[0])) {
184*c8dee2aaSAndroid Build Coastguard Worker         return false;
185*c8dee2aaSAndroid Build Coastguard Worker     }
186*c8dee2aaSAndroid Build Coastguard Worker     SkVector deferDxdy = fDefer[1]->fPt - fDefer[0]->fPt;
187*c8dee2aaSAndroid Build Coastguard Worker     SkVector lineDxdy = ptT->fPt - fDefer[1]->fPt;
188*c8dee2aaSAndroid Build Coastguard Worker     return deferDxdy.fX * lineDxdy.fY != deferDxdy.fY * lineDxdy.fX;
189*c8dee2aaSAndroid Build Coastguard Worker }
190*c8dee2aaSAndroid Build Coastguard Worker 
191*c8dee2aaSAndroid Build Coastguard Worker class DistanceLessThan {
192*c8dee2aaSAndroid Build Coastguard Worker public:
DistanceLessThan(double * distances)193*c8dee2aaSAndroid Build Coastguard Worker     DistanceLessThan(double* distances) : fDistances(distances) { }
194*c8dee2aaSAndroid Build Coastguard Worker     double* fDistances;
operator ()(const int one,const int two) const195*c8dee2aaSAndroid Build Coastguard Worker     bool operator()(const int one, const int two) const {
196*c8dee2aaSAndroid Build Coastguard Worker         return fDistances[one] < fDistances[two];
197*c8dee2aaSAndroid Build Coastguard Worker     }
198*c8dee2aaSAndroid Build Coastguard Worker };
199*c8dee2aaSAndroid Build Coastguard Worker 
200*c8dee2aaSAndroid Build Coastguard Worker     /*
201*c8dee2aaSAndroid Build Coastguard Worker         check start and end of each contour
202*c8dee2aaSAndroid Build Coastguard Worker         if not the same, record them
203*c8dee2aaSAndroid Build Coastguard Worker         match them up
204*c8dee2aaSAndroid Build Coastguard Worker         connect closest
205*c8dee2aaSAndroid Build Coastguard Worker         reassemble contour pieces into new path
206*c8dee2aaSAndroid Build Coastguard Worker     */
assemble()207*c8dee2aaSAndroid Build Coastguard Worker void SkPathWriter::assemble() {
208*c8dee2aaSAndroid Build Coastguard Worker     if (!this->someAssemblyRequired()) {
209*c8dee2aaSAndroid Build Coastguard Worker         return;
210*c8dee2aaSAndroid Build Coastguard Worker     }
211*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_PATH_CONSTRUCTION
212*c8dee2aaSAndroid Build Coastguard Worker     SkDebugf("%s\n", __FUNCTION__);
213*c8dee2aaSAndroid Build Coastguard Worker #endif
214*c8dee2aaSAndroid Build Coastguard Worker     SkOpPtT const* const* runs = fEndPtTs.begin();  // starts, ends of partial contours
215*c8dee2aaSAndroid Build Coastguard Worker     int endCount = fEndPtTs.size(); // all starts and ends
216*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(endCount > 0);
217*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(endCount == fPartials.size() * 2);
218*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_ASSEMBLE
219*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < endCount; index += 2) {
220*c8dee2aaSAndroid Build Coastguard Worker         const SkOpPtT* eStart = runs[index];
221*c8dee2aaSAndroid Build Coastguard Worker         const SkOpPtT* eEnd = runs[index + 1];
222*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(eStart != eEnd);
223*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!eStart->contains(eEnd));
224*c8dee2aaSAndroid Build Coastguard Worker         SkDebugf("%s contour start=(%1.9g,%1.9g) end=(%1.9g,%1.9g)\n", __FUNCTION__,
225*c8dee2aaSAndroid Build Coastguard Worker                 eStart->fPt.fX, eStart->fPt.fY, eEnd->fPt.fX, eEnd->fPt.fY);
226*c8dee2aaSAndroid Build Coastguard Worker     }
227*c8dee2aaSAndroid Build Coastguard Worker #endif
228*c8dee2aaSAndroid Build Coastguard Worker     // lengthen any partial contour adjacent to a simple segment
229*c8dee2aaSAndroid Build Coastguard Worker     for (int pIndex = 0; pIndex < endCount; pIndex++) {
230*c8dee2aaSAndroid Build Coastguard Worker         SkOpPtT* opPtT = const_cast<SkOpPtT*>(runs[pIndex]);
231*c8dee2aaSAndroid Build Coastguard Worker         SkPath p;
232*c8dee2aaSAndroid Build Coastguard Worker         SkPathWriter partWriter(p);
233*c8dee2aaSAndroid Build Coastguard Worker         do {
234*c8dee2aaSAndroid Build Coastguard Worker             if (!zero_or_one(opPtT->fT)) {
235*c8dee2aaSAndroid Build Coastguard Worker                 break;
236*c8dee2aaSAndroid Build Coastguard Worker             }
237*c8dee2aaSAndroid Build Coastguard Worker             SkOpSpanBase* opSpanBase = opPtT->span();
238*c8dee2aaSAndroid Build Coastguard Worker             SkOpSpanBase* start = opPtT->fT ? opSpanBase->prev() : opSpanBase->upCast()->next();
239*c8dee2aaSAndroid Build Coastguard Worker             int step = opPtT->fT ? 1 : -1;
240*c8dee2aaSAndroid Build Coastguard Worker             const SkOpSegment* opSegment = opSpanBase->segment();
241*c8dee2aaSAndroid Build Coastguard Worker             const SkOpSegment* nextSegment = opSegment->isSimple(&start, &step);
242*c8dee2aaSAndroid Build Coastguard Worker             if (!nextSegment) {
243*c8dee2aaSAndroid Build Coastguard Worker                 break;
244*c8dee2aaSAndroid Build Coastguard Worker             }
245*c8dee2aaSAndroid Build Coastguard Worker             SkOpSpanBase* opSpanEnd = start->t() ? start->prev() : start->upCast()->next();
246*c8dee2aaSAndroid Build Coastguard Worker             if (start->starter(opSpanEnd)->alreadyAdded()) {
247*c8dee2aaSAndroid Build Coastguard Worker                 break;
248*c8dee2aaSAndroid Build Coastguard Worker             }
249*c8dee2aaSAndroid Build Coastguard Worker             nextSegment->addCurveTo(start, opSpanEnd, &partWriter);
250*c8dee2aaSAndroid Build Coastguard Worker             opPtT = opSpanEnd->ptT();
251*c8dee2aaSAndroid Build Coastguard Worker             SkOpPtT** runsPtr = const_cast<SkOpPtT**>(&runs[pIndex]);
252*c8dee2aaSAndroid Build Coastguard Worker             *runsPtr = opPtT;
253*c8dee2aaSAndroid Build Coastguard Worker         } while (true);
254*c8dee2aaSAndroid Build Coastguard Worker         partWriter.finishContour();
255*c8dee2aaSAndroid Build Coastguard Worker         const TArray<SkPath>& partPartials = partWriter.partials();
256*c8dee2aaSAndroid Build Coastguard Worker         if (partPartials.empty()) {
257*c8dee2aaSAndroid Build Coastguard Worker             continue;
258*c8dee2aaSAndroid Build Coastguard Worker         }
259*c8dee2aaSAndroid Build Coastguard Worker         // if pIndex is even, reverse and prepend to fPartials; otherwise, append
260*c8dee2aaSAndroid Build Coastguard Worker         SkPath& partial = const_cast<SkPath&>(fPartials[pIndex >> 1]);
261*c8dee2aaSAndroid Build Coastguard Worker         const SkPath& part = partPartials[0];
262*c8dee2aaSAndroid Build Coastguard Worker         if (pIndex & 1) {
263*c8dee2aaSAndroid Build Coastguard Worker             partial.addPath(part, SkPath::kExtend_AddPathMode);
264*c8dee2aaSAndroid Build Coastguard Worker         } else {
265*c8dee2aaSAndroid Build Coastguard Worker             SkPath reverse;
266*c8dee2aaSAndroid Build Coastguard Worker             reverse.reverseAddPath(part);
267*c8dee2aaSAndroid Build Coastguard Worker             reverse.addPath(partial, SkPath::kExtend_AddPathMode);
268*c8dee2aaSAndroid Build Coastguard Worker             partial = reverse;
269*c8dee2aaSAndroid Build Coastguard Worker         }
270*c8dee2aaSAndroid Build Coastguard Worker     }
271*c8dee2aaSAndroid Build Coastguard Worker     SkTDArray<int> sLink, eLink;
272*c8dee2aaSAndroid Build Coastguard Worker     int linkCount = endCount / 2; // number of partial contours
273*c8dee2aaSAndroid Build Coastguard Worker     sLink.append(linkCount);
274*c8dee2aaSAndroid Build Coastguard Worker     eLink.append(linkCount);
275*c8dee2aaSAndroid Build Coastguard Worker     int rIndex, iIndex;
276*c8dee2aaSAndroid Build Coastguard Worker     for (rIndex = 0; rIndex < linkCount; ++rIndex) {
277*c8dee2aaSAndroid Build Coastguard Worker         sLink[rIndex] = eLink[rIndex] = SK_MaxS32;
278*c8dee2aaSAndroid Build Coastguard Worker     }
279*c8dee2aaSAndroid Build Coastguard Worker     const int entries = endCount * (endCount - 1) / 2;  // folded triangle
280*c8dee2aaSAndroid Build Coastguard Worker     STArray<8, double, true> distances(entries);
281*c8dee2aaSAndroid Build Coastguard Worker     STArray<8, int, true> sortedDist(entries);
282*c8dee2aaSAndroid Build Coastguard Worker     STArray<8, int, true> distLookup(entries);
283*c8dee2aaSAndroid Build Coastguard Worker     int rRow = 0;
284*c8dee2aaSAndroid Build Coastguard Worker     int dIndex = 0;
285*c8dee2aaSAndroid Build Coastguard Worker     for (rIndex = 0; rIndex < endCount - 1; ++rIndex) {
286*c8dee2aaSAndroid Build Coastguard Worker         const SkOpPtT* oPtT = runs[rIndex];
287*c8dee2aaSAndroid Build Coastguard Worker         for (iIndex = rIndex + 1; iIndex < endCount; ++iIndex) {
288*c8dee2aaSAndroid Build Coastguard Worker             const SkOpPtT* iPtT = runs[iIndex];
289*c8dee2aaSAndroid Build Coastguard Worker             double dx = iPtT->fPt.fX - oPtT->fPt.fX;
290*c8dee2aaSAndroid Build Coastguard Worker             double dy = iPtT->fPt.fY - oPtT->fPt.fY;
291*c8dee2aaSAndroid Build Coastguard Worker             double dist = dx * dx + dy * dy;
292*c8dee2aaSAndroid Build Coastguard Worker             distLookup.push_back(rRow + iIndex);
293*c8dee2aaSAndroid Build Coastguard Worker             distances.push_back(dist);  // oStart distance from iStart
294*c8dee2aaSAndroid Build Coastguard Worker             sortedDist.push_back(dIndex++);
295*c8dee2aaSAndroid Build Coastguard Worker         }
296*c8dee2aaSAndroid Build Coastguard Worker         rRow += endCount;
297*c8dee2aaSAndroid Build Coastguard Worker     }
298*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dIndex == entries);
299*c8dee2aaSAndroid Build Coastguard Worker     SkTQSort<int>(sortedDist.begin(), sortedDist.end(), DistanceLessThan(distances.begin()));
300*c8dee2aaSAndroid Build Coastguard Worker     int remaining = linkCount;  // number of start/end pairs
301*c8dee2aaSAndroid Build Coastguard Worker     for (rIndex = 0; rIndex < entries; ++rIndex) {
302*c8dee2aaSAndroid Build Coastguard Worker         int pair = sortedDist[rIndex];
303*c8dee2aaSAndroid Build Coastguard Worker         pair = distLookup[pair];
304*c8dee2aaSAndroid Build Coastguard Worker         int row = pair / endCount;
305*c8dee2aaSAndroid Build Coastguard Worker         int col = pair - row * endCount;
306*c8dee2aaSAndroid Build Coastguard Worker         int ndxOne = row >> 1;
307*c8dee2aaSAndroid Build Coastguard Worker         bool endOne = row & 1;
308*c8dee2aaSAndroid Build Coastguard Worker         int* linkOne = endOne ? eLink.begin() : sLink.begin();
309*c8dee2aaSAndroid Build Coastguard Worker         if (linkOne[ndxOne] != SK_MaxS32) {
310*c8dee2aaSAndroid Build Coastguard Worker             continue;
311*c8dee2aaSAndroid Build Coastguard Worker         }
312*c8dee2aaSAndroid Build Coastguard Worker         int ndxTwo = col >> 1;
313*c8dee2aaSAndroid Build Coastguard Worker         bool endTwo = col & 1;
314*c8dee2aaSAndroid Build Coastguard Worker         int* linkTwo = endTwo ? eLink.begin() : sLink.begin();
315*c8dee2aaSAndroid Build Coastguard Worker         if (linkTwo[ndxTwo] != SK_MaxS32) {
316*c8dee2aaSAndroid Build Coastguard Worker             continue;
317*c8dee2aaSAndroid Build Coastguard Worker         }
318*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(&linkOne[ndxOne] != &linkTwo[ndxTwo]);
319*c8dee2aaSAndroid Build Coastguard Worker         bool flip = endOne == endTwo;
320*c8dee2aaSAndroid Build Coastguard Worker         linkOne[ndxOne] = flip ? ~ndxTwo : ndxTwo;
321*c8dee2aaSAndroid Build Coastguard Worker         linkTwo[ndxTwo] = flip ? ~ndxOne : ndxOne;
322*c8dee2aaSAndroid Build Coastguard Worker         if (!--remaining) {
323*c8dee2aaSAndroid Build Coastguard Worker             break;
324*c8dee2aaSAndroid Build Coastguard Worker         }
325*c8dee2aaSAndroid Build Coastguard Worker     }
326*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(!remaining);
327*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_ASSEMBLE
328*c8dee2aaSAndroid Build Coastguard Worker     for (rIndex = 0; rIndex < linkCount; ++rIndex) {
329*c8dee2aaSAndroid Build Coastguard Worker         int s = sLink[rIndex];
330*c8dee2aaSAndroid Build Coastguard Worker         int e = eLink[rIndex];
331*c8dee2aaSAndroid Build Coastguard Worker         SkDebugf("%s %c%d <- s%d - e%d -> %c%d\n", __FUNCTION__, s < 0 ? 's' : 'e',
332*c8dee2aaSAndroid Build Coastguard Worker                 s < 0 ? ~s : s, rIndex, rIndex, e < 0 ? 'e' : 's', e < 0 ? ~e : e);
333*c8dee2aaSAndroid Build Coastguard Worker     }
334*c8dee2aaSAndroid Build Coastguard Worker #endif
335*c8dee2aaSAndroid Build Coastguard Worker     rIndex = 0;
336*c8dee2aaSAndroid Build Coastguard Worker     do {
337*c8dee2aaSAndroid Build Coastguard Worker         bool forward = true;
338*c8dee2aaSAndroid Build Coastguard Worker         bool first = true;
339*c8dee2aaSAndroid Build Coastguard Worker         int sIndex = sLink[rIndex];
340*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(sIndex != SK_MaxS32);
341*c8dee2aaSAndroid Build Coastguard Worker         sLink[rIndex] = SK_MaxS32;
342*c8dee2aaSAndroid Build Coastguard Worker         int eIndex;
343*c8dee2aaSAndroid Build Coastguard Worker         if (sIndex < 0) {
344*c8dee2aaSAndroid Build Coastguard Worker             eIndex = sLink[~sIndex];
345*c8dee2aaSAndroid Build Coastguard Worker             sLink[~sIndex] = SK_MaxS32;
346*c8dee2aaSAndroid Build Coastguard Worker         } else {
347*c8dee2aaSAndroid Build Coastguard Worker             eIndex = eLink[sIndex];
348*c8dee2aaSAndroid Build Coastguard Worker             eLink[sIndex] = SK_MaxS32;
349*c8dee2aaSAndroid Build Coastguard Worker         }
350*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(eIndex != SK_MaxS32);
351*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_ASSEMBLE
352*c8dee2aaSAndroid Build Coastguard Worker         SkDebugf("%s sIndex=%c%d eIndex=%c%d\n", __FUNCTION__, sIndex < 0 ? 's' : 'e',
353*c8dee2aaSAndroid Build Coastguard Worker                     sIndex < 0 ? ~sIndex : sIndex, eIndex < 0 ? 's' : 'e',
354*c8dee2aaSAndroid Build Coastguard Worker                     eIndex < 0 ? ~eIndex : eIndex);
355*c8dee2aaSAndroid Build Coastguard Worker #endif
356*c8dee2aaSAndroid Build Coastguard Worker         do {
357*c8dee2aaSAndroid Build Coastguard Worker             const SkPath& contour = fPartials[rIndex];
358*c8dee2aaSAndroid Build Coastguard Worker             if (!first) {
359*c8dee2aaSAndroid Build Coastguard Worker                 SkPoint prior, next;
360*c8dee2aaSAndroid Build Coastguard Worker                 if (!fPathPtr->getLastPt(&prior)) {
361*c8dee2aaSAndroid Build Coastguard Worker                     return;
362*c8dee2aaSAndroid Build Coastguard Worker                 }
363*c8dee2aaSAndroid Build Coastguard Worker                 if (forward) {
364*c8dee2aaSAndroid Build Coastguard Worker                     next = contour.getPoint(0);
365*c8dee2aaSAndroid Build Coastguard Worker                 } else {
366*c8dee2aaSAndroid Build Coastguard Worker                     SkAssertResult(contour.getLastPt(&next));
367*c8dee2aaSAndroid Build Coastguard Worker                 }
368*c8dee2aaSAndroid Build Coastguard Worker                 if (prior != next) {
369*c8dee2aaSAndroid Build Coastguard Worker                     /* TODO: if there is a gap between open path written so far and path to come,
370*c8dee2aaSAndroid Build Coastguard Worker                        connect by following segments from one to the other, rather than introducing
371*c8dee2aaSAndroid Build Coastguard Worker                        a diagonal to connect the two.
372*c8dee2aaSAndroid Build Coastguard Worker                      */
373*c8dee2aaSAndroid Build Coastguard Worker                 }
374*c8dee2aaSAndroid Build Coastguard Worker             }
375*c8dee2aaSAndroid Build Coastguard Worker             if (forward) {
376*c8dee2aaSAndroid Build Coastguard Worker                 fPathPtr->addPath(contour,
377*c8dee2aaSAndroid Build Coastguard Worker                         first ? SkPath::kAppend_AddPathMode : SkPath::kExtend_AddPathMode);
378*c8dee2aaSAndroid Build Coastguard Worker             } else {
379*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(!first);
380*c8dee2aaSAndroid Build Coastguard Worker                 fPathPtr->reversePathTo(contour);
381*c8dee2aaSAndroid Build Coastguard Worker             }
382*c8dee2aaSAndroid Build Coastguard Worker             if (first) {
383*c8dee2aaSAndroid Build Coastguard Worker                 first = false;
384*c8dee2aaSAndroid Build Coastguard Worker             }
385*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_ASSEMBLE
386*c8dee2aaSAndroid Build Coastguard Worker             SkDebugf("%s rIndex=%d eIndex=%s%d close=%d\n", __FUNCTION__, rIndex,
387*c8dee2aaSAndroid Build Coastguard Worker                 eIndex < 0 ? "~" : "", eIndex < 0 ? ~eIndex : eIndex,
388*c8dee2aaSAndroid Build Coastguard Worker                 sIndex == ((rIndex != eIndex) ^ forward ? eIndex : ~eIndex));
389*c8dee2aaSAndroid Build Coastguard Worker #endif
390*c8dee2aaSAndroid Build Coastguard Worker             if (sIndex == ((rIndex != eIndex) ^ forward ? eIndex : ~eIndex)) {
391*c8dee2aaSAndroid Build Coastguard Worker                 fPathPtr->close();
392*c8dee2aaSAndroid Build Coastguard Worker                 break;
393*c8dee2aaSAndroid Build Coastguard Worker             }
394*c8dee2aaSAndroid Build Coastguard Worker             if (forward) {
395*c8dee2aaSAndroid Build Coastguard Worker                 eIndex = eLink[rIndex];
396*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(eIndex != SK_MaxS32);
397*c8dee2aaSAndroid Build Coastguard Worker                 eLink[rIndex] = SK_MaxS32;
398*c8dee2aaSAndroid Build Coastguard Worker                 if (eIndex >= 0) {
399*c8dee2aaSAndroid Build Coastguard Worker                     SkASSERT(sLink[eIndex] == rIndex);
400*c8dee2aaSAndroid Build Coastguard Worker                     sLink[eIndex] = SK_MaxS32;
401*c8dee2aaSAndroid Build Coastguard Worker                 } else {
402*c8dee2aaSAndroid Build Coastguard Worker                     SkASSERT(eLink[~eIndex] == ~rIndex);
403*c8dee2aaSAndroid Build Coastguard Worker                     eLink[~eIndex] = SK_MaxS32;
404*c8dee2aaSAndroid Build Coastguard Worker                 }
405*c8dee2aaSAndroid Build Coastguard Worker             } else {
406*c8dee2aaSAndroid Build Coastguard Worker                 eIndex = sLink[rIndex];
407*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(eIndex != SK_MaxS32);
408*c8dee2aaSAndroid Build Coastguard Worker                 sLink[rIndex] = SK_MaxS32;
409*c8dee2aaSAndroid Build Coastguard Worker                 if (eIndex >= 0) {
410*c8dee2aaSAndroid Build Coastguard Worker                     SkASSERT(eLink[eIndex] == rIndex);
411*c8dee2aaSAndroid Build Coastguard Worker                     eLink[eIndex] = SK_MaxS32;
412*c8dee2aaSAndroid Build Coastguard Worker                 } else {
413*c8dee2aaSAndroid Build Coastguard Worker                     SkASSERT(sLink[~eIndex] == ~rIndex);
414*c8dee2aaSAndroid Build Coastguard Worker                     sLink[~eIndex] = SK_MaxS32;
415*c8dee2aaSAndroid Build Coastguard Worker                 }
416*c8dee2aaSAndroid Build Coastguard Worker             }
417*c8dee2aaSAndroid Build Coastguard Worker             rIndex = eIndex;
418*c8dee2aaSAndroid Build Coastguard Worker             if (rIndex < 0) {
419*c8dee2aaSAndroid Build Coastguard Worker                 forward ^= 1;
420*c8dee2aaSAndroid Build Coastguard Worker                 rIndex = ~rIndex;
421*c8dee2aaSAndroid Build Coastguard Worker             }
422*c8dee2aaSAndroid Build Coastguard Worker         } while (true);
423*c8dee2aaSAndroid Build Coastguard Worker         for (rIndex = 0; rIndex < linkCount; ++rIndex) {
424*c8dee2aaSAndroid Build Coastguard Worker             if (sLink[rIndex] != SK_MaxS32) {
425*c8dee2aaSAndroid Build Coastguard Worker                 break;
426*c8dee2aaSAndroid Build Coastguard Worker             }
427*c8dee2aaSAndroid Build Coastguard Worker         }
428*c8dee2aaSAndroid Build Coastguard Worker     } while (rIndex < linkCount);
429*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_ASSEMBLE
430*c8dee2aaSAndroid Build Coastguard Worker     for (rIndex = 0; rIndex < linkCount; ++rIndex) {
431*c8dee2aaSAndroid Build Coastguard Worker        SkASSERT(sLink[rIndex] == SK_MaxS32);
432*c8dee2aaSAndroid Build Coastguard Worker        SkASSERT(eLink[rIndex] == SK_MaxS32);
433*c8dee2aaSAndroid Build Coastguard Worker     }
434*c8dee2aaSAndroid Build Coastguard Worker #endif
435*c8dee2aaSAndroid Build Coastguard Worker }
436