Index: src/pathops/SkPathWriter.cpp |
diff --git a/src/pathops/SkPathWriter.cpp b/src/pathops/SkPathWriter.cpp |
index bd8c72134c1cef41e492927a37b9d4550d49afd4..2c3391bb44c521a615f36fd42de766be335bc28f 100644 |
--- a/src/pathops/SkPathWriter.cpp |
+++ b/src/pathops/SkPathWriter.cpp |
@@ -4,181 +4,356 @@ |
* Use of this source code is governed by a BSD-style license that can be |
* found in the LICENSE file. |
*/ |
+#include "SkOpSpan.h" |
#include "SkPathOpsPoint.h" |
#include "SkPathWriter.h" |
+#include "SkTSort.h" |
// wrap path to keep track of whether the contour is initialized and non-empty |
SkPathWriter::SkPathWriter(SkPath& path) |
: fPathPtr(&path) |
- , fCloses(0) |
- , fMoves(0) |
{ |
init(); |
} |
void SkPathWriter::close() { |
- if (!fHasMove) { |
+ if (fCurrent.isEmpty()) { |
return; |
} |
- bool callClose = isClosed(); |
- lineTo(); |
- if (fEmpty) { |
- return; |
- } |
- if (callClose) { |
+ SkASSERT(this->isClosed()); |
#if DEBUG_PATH_CONSTRUCTION |
- SkDebugf("path.close();\n"); |
+ SkDebugf("path.close();\n"); |
#endif |
- fPathPtr->close(); |
- fCloses++; |
- } |
+ fCurrent.close(); |
+ fPathPtr->addPath(fCurrent); |
+ fCurrent.reset(); |
init(); |
} |
-void SkPathWriter::conicTo(const SkPoint& pt1, const SkPoint& pt2, SkScalar weight) { |
- lineTo(); |
- if (fEmpty && AlmostEqualUlps(fDefer[0], pt1) && AlmostEqualUlps(pt1, pt2)) { |
- deferredLine(pt2); |
- return; |
- } |
- moveTo(); |
- fDefer[1] = pt2; |
- nudge(); |
- fDefer[0] = fDefer[1]; |
+void SkPathWriter::conicTo(const SkPoint& pt1, const SkOpPtT* pt2, SkScalar weight) { |
+ this->update(pt2); |
#if DEBUG_PATH_CONSTRUCTION |
SkDebugf("path.conicTo(%1.9g,%1.9g, %1.9g,%1.9g, %1.9g);\n", |
- pt1.fX, pt1.fY, fDefer[1].fX, fDefer[1].fY, weight); |
+ pt1.fX, pt1.fY, pt2->fPt.fX, pt2->fPt.fY, weight); |
#endif |
- fPathPtr->conicTo(pt1.fX, pt1.fY, fDefer[1].fX, fDefer[1].fY, weight); |
- fEmpty = false; |
+ fCurrent.conicTo(pt1, pt2->fPt, weight); |
} |
-void SkPathWriter::cubicTo(const SkPoint& pt1, const SkPoint& pt2, const SkPoint& pt3) { |
- lineTo(); |
- if (fEmpty && AlmostEqualUlps(fDefer[0], pt1) && AlmostEqualUlps(pt1, pt2) |
- && AlmostEqualUlps(pt2, pt3)) { |
- deferredLine(pt3); |
- return; |
- } |
- moveTo(); |
- fDefer[1] = pt3; |
- nudge(); |
- fDefer[0] = fDefer[1]; |
+void SkPathWriter::cubicTo(const SkPoint& pt1, const SkPoint& pt2, const SkOpPtT* pt3) { |
+ this->update(pt3); |
#if DEBUG_PATH_CONSTRUCTION |
SkDebugf("path.cubicTo(%1.9g,%1.9g, %1.9g,%1.9g, %1.9g,%1.9g);\n", |
- pt1.fX, pt1.fY, pt2.fX, pt2.fY, fDefer[1].fX, fDefer[1].fY); |
+ pt1.fX, pt1.fY, pt2.fX, pt2.fY, pt3->fPt.fX, pt3->fPt.fY); |
#endif |
- fPathPtr->cubicTo(pt1.fX, pt1.fY, pt2.fX, pt2.fY, fDefer[1].fX, fDefer[1].fY); |
- fEmpty = false; |
+ fCurrent.cubicTo(pt1, pt2, pt3->fPt); |
} |
-void SkPathWriter::deferredLine(const SkPoint& pt) { |
- if (pt == fDefer[1]) { |
+void SkPathWriter::deferredLine(const SkOpPtT* pt) { |
+ SkASSERT(fFirstPtT); |
+ SkASSERT(fDefer[0]); |
+ if (fDefer[0] == pt) { |
+ // FIXME: why we're adding a degenerate line? Caller should have preflighted this. |
+ return; |
+ } |
+ if (pt->contains(fDefer[0])) { |
+ // FIXME: why we're adding a degenerate line? |
return; |
} |
- if (changedSlopes(pt)) { |
- lineTo(); |
+ SkASSERT(!this->matchedLast(pt)); |
+ if (fDefer[1] && this->changedSlopes(pt)) { |
+ this->lineTo(); |
fDefer[0] = fDefer[1]; |
} |
fDefer[1] = pt; |
} |
-void SkPathWriter::deferredMove(const SkPoint& pt) { |
- fMoved = true; |
- fHasMove = true; |
- fEmpty = true; |
- fDefer[0] = fDefer[1] = pt; |
-} |
- |
-void SkPathWriter::deferredMoveLine(const SkPoint& pt) { |
- if (!fHasMove) { |
- deferredMove(pt); |
+void SkPathWriter::deferredMove(const SkOpPtT* pt) { |
+ if (!fDefer[1]) { |
+ fFirstPtT = fDefer[0] = pt; |
+ return; |
+ } |
+ SkASSERT(fDefer[0]); |
+ if (!this->matchedLast(pt)) { |
+ this->finishContour(); |
+ fFirstPtT = fDefer[0] = pt; |
} |
- deferredLine(pt); |
} |
-bool SkPathWriter::hasMove() const { |
- return fHasMove; |
+void SkPathWriter::finishContour() { |
+ if (!this->matchedLast(fDefer[0])) { |
+ this->lineTo(); |
+ } |
+ if (fCurrent.isEmpty()) { |
+ return; |
+ } |
+ if (this->isClosed()) { |
+ this->close(); |
+ } else { |
+ SkASSERT(fDefer[1]); |
+ fEndPtTs.push(fFirstPtT); |
+ fEndPtTs.push(fDefer[1]); |
+ fPartials.push_back(fCurrent); |
+ this->init(); |
+ } |
} |
void SkPathWriter::init() { |
- fEmpty = true; |
- fHasMove = false; |
- fMoved = false; |
+ fCurrent.reset(); |
+ fFirstPtT = fDefer[0] = fDefer[1] = nullptr; |
} |
bool SkPathWriter::isClosed() const { |
- return !fEmpty && SkDPoint::ApproximatelyEqual(fFirstPt, fDefer[1]); |
+ return this->matchedLast(fFirstPtT); |
} |
void SkPathWriter::lineTo() { |
- if (fDefer[0] == fDefer[1]) { |
- return; |
+ if (fCurrent.isEmpty()) { |
+ this->moveTo(); |
} |
- moveTo(); |
- nudge(); |
- fEmpty = false; |
#if DEBUG_PATH_CONSTRUCTION |
- SkDebugf("path.lineTo(%1.9g,%1.9g);\n", fDefer[1].fX, fDefer[1].fY); |
+ SkDebugf("path.lineTo(%1.9g,%1.9g);\n", fDefer[1]->fPt.fX, fDefer[1]->fPt.fY); |
#endif |
- fPathPtr->lineTo(fDefer[1].fX, fDefer[1].fY); |
- fDefer[0] = fDefer[1]; |
+ fCurrent.lineTo(fDefer[1]->fPt); |
} |
-const SkPath* SkPathWriter::nativePath() const { |
- return fPathPtr; |
+bool SkPathWriter::matchedLast(const SkOpPtT* test) const { |
+ if (test == fDefer[1]) { |
+ return true; |
+ } |
+ if (!test) { |
+ return false; |
+ } |
+ if (!fDefer[1]) { |
+ return false; |
+ } |
+ return test->contains(fDefer[1]); |
} |
-void SkPathWriter::nudge() { |
- if (fEmpty || !AlmostEqualUlps(fDefer[1].fX, fFirstPt.fX) |
- || !AlmostEqualUlps(fDefer[1].fY, fFirstPt.fY)) { |
- return; |
- } |
- fDefer[1] = fFirstPt; |
+void SkPathWriter::moveTo() { |
+#if DEBUG_PATH_CONSTRUCTION |
+ SkDebugf("path.moveTo(%1.9g,%1.9g);\n", fFirstPtT->fPt.fX, fFirstPtT->fPt.fY); |
+#endif |
+ fCurrent.moveTo(fFirstPtT->fPt); |
} |
-void SkPathWriter::quadTo(const SkPoint& pt1, const SkPoint& pt2) { |
- lineTo(); |
- if (fEmpty && AlmostEqualUlps(fDefer[0], pt1) && AlmostEqualUlps(pt1, pt2)) { |
- deferredLine(pt2); |
- return; |
- } |
- moveTo(); |
- fDefer[1] = pt2; |
- nudge(); |
- fDefer[0] = fDefer[1]; |
+void SkPathWriter::quadTo(const SkPoint& pt1, const SkOpPtT* pt2) { |
+ this->update(pt2); |
#if DEBUG_PATH_CONSTRUCTION |
SkDebugf("path.quadTo(%1.9g,%1.9g, %1.9g,%1.9g);\n", |
- pt1.fX, pt1.fY, fDefer[1].fX, fDefer[1].fY); |
+ pt1.fX, pt1.fY, pt2->fPt.fX, pt2->fPt.fY); |
#endif |
- fPathPtr->quadTo(pt1.fX, pt1.fY, fDefer[1].fX, fDefer[1].fY); |
- fEmpty = false; |
+ fCurrent.quadTo(pt1, pt2->fPt); |
} |
-bool SkPathWriter::someAssemblyRequired() const { |
- return fCloses < fMoves; |
+void SkPathWriter::update(const SkOpPtT* pt) { |
+ if (!fDefer[1]) { |
+ this->moveTo(); |
+ } else if (!this->matchedLast(fDefer[0])) { |
+ this->lineTo(); |
+ } |
+ fDefer[0] = fDefer[1] = pt; // set both to know that there is not a pending deferred line |
+} |
+ |
+bool SkPathWriter::someAssemblyRequired() { |
+ this->finishContour(); |
+ return fEndPtTs.count() > 0; |
} |
-bool SkPathWriter::changedSlopes(const SkPoint& pt) const { |
- if (fDefer[0] == fDefer[1]) { |
+bool SkPathWriter::changedSlopes(const SkOpPtT* ptT) const { |
+ if (matchedLast(fDefer[0])) { |
return false; |
} |
- SkScalar deferDx = fDefer[1].fX - fDefer[0].fX; |
- SkScalar deferDy = fDefer[1].fY - fDefer[0].fY; |
- SkScalar lineDx = pt.fX - fDefer[1].fX; |
- SkScalar lineDy = pt.fY - fDefer[1].fY; |
- return deferDx * lineDy != deferDy * lineDx; |
+ SkVector deferDxdy = fDefer[1]->fPt - fDefer[0]->fPt; |
+ SkVector lineDxdy = ptT->fPt - fDefer[1]->fPt; |
+ return deferDxdy.fX * lineDxdy.fY != deferDxdy.fY * lineDxdy.fX; |
} |
-void SkPathWriter::moveTo() { |
- if (!fMoved) { |
+class DistanceLessThan { |
+public: |
+ DistanceLessThan(double* distances) : fDistances(distances) { } |
+ double* fDistances; |
+ bool operator()(const int one, const int two) { |
+ return fDistances[one] < fDistances[two]; |
+ } |
+}; |
+ |
+ /* |
+ check start and end of each contour |
+ if not the same, record them |
+ match them up |
+ connect closest |
+ reassemble contour pieces into new path |
+ */ |
+void SkPathWriter::assemble() { |
+#if DEBUG_SHOW_TEST_NAME |
+ SkDebugf("</div>\n"); |
+#endif |
+ if (!this->someAssemblyRequired()) { |
return; |
} |
- fFirstPt = fDefer[0]; |
#if DEBUG_PATH_CONSTRUCTION |
- SkDebugf("path.moveTo(%1.9g,%1.9g);\n", fDefer[0].fX, fDefer[0].fY); |
+ SkDebugf("%s\n", __FUNCTION__); |
+#endif |
+ SkOpPtT const* const* runs = fEndPtTs.begin(); // starts, ends of partial contours |
+ int endCount = fEndPtTs.count(); // all starts and ends |
+ SkASSERT(endCount > 0); |
+ SkASSERT(endCount == fPartials.count() * 2); |
+#if DEBUG_ASSEMBLE |
+ for (int index = 0; index < endCount; index += 2) { |
+ const SkOpPtT* eStart = runs[index]; |
+ const SkOpPtT* eEnd = runs[index + 1]; |
+ SkASSERT(eStart != eEnd); |
+ SkASSERT(!eStart->contains(eEnd)); |
+ SkDebugf("%s contour start=(%1.9g,%1.9g) end=(%1.9g,%1.9g)\n", __FUNCTION__, |
+ eStart->fPt.fX, eStart->fPt.fY, eEnd->fPt.fX, eEnd->fPt.fY); |
+ } |
+#endif |
+ SkTDArray<int> sLink, eLink; |
+ int linkCount = endCount / 2; // number of partial contours |
+ sLink.append(linkCount); |
+ eLink.append(linkCount); |
+ int rIndex, iIndex; |
+ for (rIndex = 0; rIndex < linkCount; ++rIndex) { |
+ sLink[rIndex] = eLink[rIndex] = SK_MaxS32; |
+ } |
+ const int entries = endCount * (endCount - 1) / 2; // folded triangle |
+ SkSTArray<8, double, true> distances(entries); |
+ SkSTArray<8, int, true> sortedDist(entries); |
+ SkSTArray<8, int, true> distLookup(entries); |
+ int rRow = 0; |
+ int dIndex = 0; |
+ for (rIndex = 0; rIndex < endCount - 1; ++rIndex) { |
+ const SkOpPtT* oPtT = runs[rIndex]; |
+ for (iIndex = rIndex + 1; iIndex < endCount; ++iIndex) { |
+ const SkOpPtT* iPtT = runs[iIndex]; |
+ double dx = iPtT->fPt.fX - oPtT->fPt.fX; |
+ double dy = iPtT->fPt.fY - oPtT->fPt.fY; |
+ double dist = dx * dx + dy * dy; |
+ distLookup.push_back(rRow + iIndex); |
+ distances.push_back(dist); // oStart distance from iStart |
+ sortedDist.push_back(dIndex++); |
+ } |
+ rRow += endCount; |
+ } |
+ SkASSERT(dIndex == entries); |
+ SkTQSort<int>(sortedDist.begin(), sortedDist.end() - 1, DistanceLessThan(distances.begin())); |
+ int remaining = linkCount; // number of start/end pairs |
+ for (rIndex = 0; rIndex < entries; ++rIndex) { |
+ int pair = sortedDist[rIndex]; |
+ pair = distLookup[pair]; |
+ int row = pair / endCount; |
+ int col = pair - row * endCount; |
+ int ndxOne = row >> 1; |
+ bool endOne = row & 1; |
+ int* linkOne = endOne ? eLink.begin() : sLink.begin(); |
+ if (linkOne[ndxOne] != SK_MaxS32) { |
+ continue; |
+ } |
+ int ndxTwo = col >> 1; |
+ bool endTwo = col & 1; |
+ int* linkTwo = endTwo ? eLink.begin() : sLink.begin(); |
+ if (linkTwo[ndxTwo] != SK_MaxS32) { |
+ continue; |
+ } |
+ SkASSERT(&linkOne[ndxOne] != &linkTwo[ndxTwo]); |
+ bool flip = endOne == endTwo; |
+ linkOne[ndxOne] = flip ? ~ndxTwo : ndxTwo; |
+ linkTwo[ndxTwo] = flip ? ~ndxOne : ndxOne; |
+ if (!--remaining) { |
+ break; |
+ } |
+ } |
+ SkASSERT(!remaining); |
+#if DEBUG_ASSEMBLE |
+ for (rIndex = 0; rIndex < linkCount; ++rIndex) { |
+ int s = sLink[rIndex]; |
+ int e = eLink[rIndex]; |
+ SkDebugf("%s %c%d <- s%d - e%d -> %c%d\n", __FUNCTION__, s < 0 ? 's' : 'e', |
+ s < 0 ? ~s : s, rIndex, rIndex, e < 0 ? 'e' : 's', e < 0 ? ~e : e); |
+ } |
+#endif |
+ rIndex = 0; |
+ do { |
+ bool forward = true; |
+ bool first = true; |
+ int sIndex = sLink[rIndex]; |
+ SkASSERT(sIndex != SK_MaxS32); |
+ sLink[rIndex] = SK_MaxS32; |
+ int eIndex; |
+ if (sIndex < 0) { |
+ eIndex = sLink[~sIndex]; |
+ sLink[~sIndex] = SK_MaxS32; |
+ } else { |
+ eIndex = eLink[sIndex]; |
+ eLink[sIndex] = SK_MaxS32; |
+ } |
+ SkASSERT(eIndex != SK_MaxS32); |
+#if DEBUG_ASSEMBLE |
+ SkDebugf("%s sIndex=%c%d eIndex=%c%d\n", __FUNCTION__, sIndex < 0 ? 's' : 'e', |
+ sIndex < 0 ? ~sIndex : sIndex, eIndex < 0 ? 's' : 'e', |
+ eIndex < 0 ? ~eIndex : eIndex); |
+#endif |
+ do { |
+ const SkPath& contour = fPartials[rIndex]; |
+ if (forward) { |
+ fPathPtr->addPath(contour, |
+ first ? SkPath::kAppend_AddPathMode : SkPath::kExtend_AddPathMode); |
+ } else { |
+ SkASSERT(!first); |
+ fPathPtr->reverseAddPath(contour); |
+ } |
+ if (first) { |
+ first = false; |
+ } |
+#if DEBUG_ASSEMBLE |
+ SkDebugf("%s rIndex=%d eIndex=%s%d close=%d\n", __FUNCTION__, rIndex, |
+ eIndex < 0 ? "~" : "", eIndex < 0 ? ~eIndex : eIndex, |
+ sIndex == ((rIndex != eIndex) ^ forward ? eIndex : ~eIndex)); |
+#endif |
+ if (sIndex == ((rIndex != eIndex) ^ forward ? eIndex : ~eIndex)) { |
+ fPathPtr->close(); |
+ break; |
+ } |
+ if (forward) { |
+ eIndex = eLink[rIndex]; |
+ SkASSERT(eIndex != SK_MaxS32); |
+ eLink[rIndex] = SK_MaxS32; |
+ if (eIndex >= 0) { |
+ SkASSERT(sLink[eIndex] == rIndex); |
+ sLink[eIndex] = SK_MaxS32; |
+ } else { |
+ SkASSERT(eLink[~eIndex] == ~rIndex); |
+ eLink[~eIndex] = SK_MaxS32; |
+ } |
+ } else { |
+ eIndex = sLink[rIndex]; |
+ SkASSERT(eIndex != SK_MaxS32); |
+ sLink[rIndex] = SK_MaxS32; |
+ if (eIndex >= 0) { |
+ SkASSERT(eLink[eIndex] == rIndex); |
+ eLink[eIndex] = SK_MaxS32; |
+ } else { |
+ SkASSERT(sLink[~eIndex] == ~rIndex); |
+ sLink[~eIndex] = SK_MaxS32; |
+ } |
+ } |
+ rIndex = eIndex; |
+ if (rIndex < 0) { |
+ forward ^= 1; |
+ rIndex = ~rIndex; |
+ } |
+ } while (true); |
+ for (rIndex = 0; rIndex < linkCount; ++rIndex) { |
+ if (sLink[rIndex] != SK_MaxS32) { |
+ break; |
+ } |
+ } |
+ } while (rIndex < linkCount); |
+#if DEBUG_ASSEMBLE |
+ for (rIndex = 0; rIndex < linkCount; ++rIndex) { |
+ SkASSERT(sLink[rIndex] == SK_MaxS32); |
+ SkASSERT(eLink[rIndex] == SK_MaxS32); |
+ } |
#endif |
- fPathPtr->moveTo(fDefer[0].fX, fDefer[0].fY); |
- fMoved = false; |
- fMoves++; |
+ return; |
} |