/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2013 - Raw Material Software Ltd. Permission is granted to use this software under the terms of either: a) the GPL v2 (or any later version) b) the Affero GPL v3 Details of these licenses can be found at: www.gnu.org/licenses JUCE is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. ------------------------------------------------------------------------------ To release a closed-source product which uses JUCE, commercial licenses are available: visit www.juce.com for more information. ============================================================================== */ StretchableObjectResizer::StretchableObjectResizer() {} StretchableObjectResizer::~StretchableObjectResizer() {} void StretchableObjectResizer::addItem (const double size, const double minSize, const double maxSize, const int order) { // the order must be >= 0 but less than the maximum integer value. jassert (order >= 0 && order < std::numeric_limits::max()); jassert (maxSize >= minSize); Item item; item.size = size; item.minSize = minSize; item.maxSize = maxSize; item.order = order; items.add (item); } double StretchableObjectResizer::getItemSize (const int index) const noexcept { return isPositiveAndBelow (index, items.size()) ? items.getReference (index).size : 0.0; } void StretchableObjectResizer::resizeToFit (const double targetSize) { int order = 0; for (;;) { double currentSize = 0; double minSize = 0; double maxSize = 0; int nextHighestOrder = std::numeric_limits::max(); for (int i = 0; i < items.size(); ++i) { const Item& it = items.getReference(i); currentSize += it.size; if (it.order <= order) { minSize += it.minSize; maxSize += it.maxSize; } else { minSize += it.size; maxSize += it.size; nextHighestOrder = jmin (nextHighestOrder, it.order); } } const double thisIterationTarget = jlimit (minSize, maxSize, targetSize); if (thisIterationTarget >= currentSize) { const double availableExtraSpace = maxSize - currentSize; const double targetAmountOfExtraSpace = thisIterationTarget - currentSize; const double scale = availableExtraSpace > 0 ? targetAmountOfExtraSpace / availableExtraSpace : 1.0; for (int i = 0; i < items.size(); ++i) { Item& it = items.getReference(i); if (it.order <= order) it.size = jlimit (it.minSize, it.maxSize, it.size + (it.maxSize - it.size) * scale); } } else { const double amountOfSlack = currentSize - minSize; const double targetAmountOfSlack = thisIterationTarget - minSize; const double scale = targetAmountOfSlack / amountOfSlack; for (int i = 0; i < items.size(); ++i) { Item& it = items.getReference(i); if (it.order <= order) it.size = jmax (it.minSize, it.minSize + (it.size - it.minSize) * scale); } } if (nextHighestOrder < std::numeric_limits::max()) order = nextHighestOrder; else break; } }