/* ============================================================================== This file is part of the JUCE framework. Copyright (c) Raw Material Software Limited JUCE is an open source framework subject to commercial or open source licensing. By downloading, installing, or using the JUCE framework, or combining the JUCE framework with any other source code, object code, content or any other copyrightable work, you agree to the terms of the JUCE End User Licence Agreement, and all incorporated terms including the JUCE Privacy Policy and the JUCE Website Terms of Service, as applicable, which will bind you. If you do not agree to the terms of these agreements, we will not license the JUCE framework to you, and you must discontinue the installation or download process and cease use of the JUCE framework. JUCE End User Licence Agreement: https://juce.com/legal/juce-8-licence/ JUCE Privacy Policy: https://juce.com/juce-privacy-policy JUCE Website Terms of Service: https://juce.com/juce-website-terms-of-service/ Or: You may also use this code under the terms of the AGPLv3: https://www.gnu.org/licenses/agpl-3.0.en.html THE JUCE FRAMEWORK IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, ARE DISCLAIMED. ============================================================================== */ namespace juce { #if JUCE_MSVC && JUCE_DEBUG #pragma optimize ("t", on) #endif //============================================================================== PathFlatteningIterator::PathFlatteningIterator (const Path& pathToUse, const AffineTransform& t, float tolerance) : x2 (0), y2 (0), closesSubPath (false), subPathIndex (-1), path (pathToUse), transform (t), source (path.data.begin()), toleranceSquared (tolerance * tolerance), isIdentityTransform (t.isIdentity()) { stackPos = stackBase; } PathFlatteningIterator::~PathFlatteningIterator() { } bool PathFlatteningIterator::isLastInSubpath() const noexcept { return stackPos == stackBase.get() && (source == path.data.end() || isMarker (*source, Path::moveMarker)); } bool PathFlatteningIterator::next() { x1 = x2; y1 = y2; float x3 = 0; float y3 = 0; float x4 = 0; float y4 = 0; for (;;) { float type; if (stackPos == stackBase.get()) { if (source == path.data.end()) return false; type = *source++; if (! isMarker (type, Path::closeSubPathMarker)) { x2 = *source++; y2 = *source++; if (isMarker (type, Path::quadMarker)) { x3 = *source++; y3 = *source++; if (! isIdentityTransform) transform.transformPoints (x2, y2, x3, y3); } else if (isMarker (type, Path::cubicMarker)) { x3 = *source++; y3 = *source++; x4 = *source++; y4 = *source++; if (! isIdentityTransform) transform.transformPoints (x2, y2, x3, y3, x4, y4); } else { if (! isIdentityTransform) transform.transformPoint (x2, y2); } } } else { type = *--stackPos; if (! isMarker (type, Path::closeSubPathMarker)) { x2 = *--stackPos; y2 = *--stackPos; if (isMarker (type, Path::quadMarker)) { x3 = *--stackPos; y3 = *--stackPos; } else if (isMarker (type, Path::cubicMarker)) { x3 = *--stackPos; y3 = *--stackPos; x4 = *--stackPos; y4 = *--stackPos; } } } if (isMarker (type, Path::lineMarker)) { ++subPathIndex; closesSubPath = stackPos == stackBase.get() && source != path.data.end() && isMarker (*source, Path::closeSubPathMarker) && approximatelyEqual (x2, subPathCloseX) && approximatelyEqual (y2, subPathCloseY); return true; } if (isMarker (type, Path::quadMarker)) { const size_t offset = (size_t) (stackPos - stackBase); if (offset >= stackSize - 10) { stackSize <<= 1; stackBase.realloc (stackSize); stackPos = stackBase + offset; } auto m1x = (x1 + x2) * 0.5f; auto m1y = (y1 + y2) * 0.5f; auto m2x = (x2 + x3) * 0.5f; auto m2y = (y2 + y3) * 0.5f; auto m3x = (m1x + m2x) * 0.5f; auto m3y = (m1y + m2y) * 0.5f; auto errorX = m3x - x2; auto errorY = m3y - y2; auto outsideTolerance = errorX * errorX + errorY * errorY > toleranceSquared; auto canBeSubdivided = (! approximatelyEqual (m3x, m1x) && ! approximatelyEqual (m3x, m2x)) || (! approximatelyEqual (m3y, m1y) && ! approximatelyEqual (m3y, m2y)); if (outsideTolerance && canBeSubdivided) { *stackPos++ = y3; *stackPos++ = x3; *stackPos++ = m2y; *stackPos++ = m2x; *stackPos++ = Path::quadMarker; *stackPos++ = m3y; *stackPos++ = m3x; *stackPos++ = m1y; *stackPos++ = m1x; *stackPos++ = Path::quadMarker; } else { *stackPos++ = y3; *stackPos++ = x3; *stackPos++ = Path::lineMarker; *stackPos++ = m3y; *stackPos++ = m3x; *stackPos++ = Path::lineMarker; } jassert (stackPos < stackBase + stackSize); } else if (isMarker (type, Path::cubicMarker)) { const size_t offset = (size_t) (stackPos - stackBase); if (offset >= stackSize - 16) { stackSize <<= 1; stackBase.realloc (stackSize); stackPos = stackBase + offset; } auto m1x = (x1 + x2) * 0.5f; auto m1y = (y1 + y2) * 0.5f; auto m2x = (x3 + x2) * 0.5f; auto m2y = (y3 + y2) * 0.5f; auto m3x = (x3 + x4) * 0.5f; auto m3y = (y3 + y4) * 0.5f; auto m4x = (m1x + m2x) * 0.5f; auto m4y = (m1y + m2y) * 0.5f; auto m5x = (m3x + m2x) * 0.5f; auto m5y = (m3y + m2y) * 0.5f; auto error1X = m4x - x2; auto error1Y = m4y - y2; auto error2X = m5x - x3; auto error2Y = m5y - y3; auto outsideTolerance = error1X * error1X + error1Y * error1Y > toleranceSquared || error2X * error2X + error2Y * error2Y > toleranceSquared; auto canBeSubdivided = (! approximatelyEqual (m4x, m1x) && ! approximatelyEqual (m4x, m2x)) || (! approximatelyEqual (m4y, m1y) && ! approximatelyEqual (m4y, m2y)) || (! approximatelyEqual (m5x, m3x) && ! approximatelyEqual (m5x, m2x)) || (! approximatelyEqual (m5y, m3y) && ! approximatelyEqual (m5y, m2y)); if (outsideTolerance && canBeSubdivided) { *stackPos++ = y4; *stackPos++ = x4; *stackPos++ = m3y; *stackPos++ = m3x; *stackPos++ = m5y; *stackPos++ = m5x; *stackPos++ = Path::cubicMarker; *stackPos++ = (m4y + m5y) * 0.5f; *stackPos++ = (m4x + m5x) * 0.5f; *stackPos++ = m4y; *stackPos++ = m4x; *stackPos++ = m1y; *stackPos++ = m1x; *stackPos++ = Path::cubicMarker; } else { *stackPos++ = y4; *stackPos++ = x4; *stackPos++ = Path::lineMarker; *stackPos++ = m5y; *stackPos++ = m5x; *stackPos++ = Path::lineMarker; *stackPos++ = m4y; *stackPos++ = m4x; *stackPos++ = Path::lineMarker; } } else if (isMarker (type, Path::closeSubPathMarker)) { if (! approximatelyEqual (x2, subPathCloseX) || ! approximatelyEqual (y2, subPathCloseY)) { x1 = x2; y1 = y2; x2 = subPathCloseX; y2 = subPathCloseY; closesSubPath = true; return true; } } else { jassert (isMarker (type, Path::moveMarker)); subPathIndex = -1; subPathCloseX = x1 = x2; subPathCloseY = y1 = y2; } } } #if JUCE_MSVC && JUCE_DEBUG #pragma optimize ("", on) // resets optimisations to the project defaults #endif } // namespace juce