/************************************************************************* ** NumericRanges.hpp ** ** ** ** This file is part of dvisvgm -- a fast DVI to SVG converter ** ** Copyright (C) 2005-2025 Martin Gieseking ** ** ** ** This program is free software; you can redistribute it and/or ** ** modify it under the terms of the GNU General Public License as ** ** published by the Free Software Foundation; either version 3 of ** ** the License, or (at your option) any later version. ** ** ** ** This program 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. ** ** ** ** You should have received a copy of the GNU General Public License ** ** along with this program; if not, see . ** *************************************************************************/ #ifndef NUMERICRANGES_HPP #define NUMERICRANGES_HPP #include #include #include #include "algorithm.hpp" template class NumericRanges { public: using Range = std::pair; using Container = std::list; using ConstIterator = typename Container::const_iterator; class ValueIterator { friend class NumericRanges; public: bool valid () const {return _currentRangeIt != _endRangeIt;} T operator * () const {return _currentValue;} bool operator != (const ValueIterator &it) const { return _currentRangeIt != it._currentRangeIt || (_currentRangeIt != _endRangeIt && _currentValue != it._currentValue); } bool operator == (const ValueIterator &it) const { return _currentRangeIt == it._currentRangeIt && (_currentRangeIt == _endRangeIt || _currentValue == it._currentValue); } void operator ++ () { if (_currentValue < _currentRangeIt->second) ++_currentValue; else if (++_currentRangeIt != _endRangeIt) _currentValue = _currentRangeIt->first; } ValueIterator operator ++ (int) { ValueIterator it = *this; ++(*this); return it; } protected: ValueIterator (const NumericRanges &ranges) : _currentRangeIt(ranges.begin()), _endRangeIt(ranges.end()), _currentValue() { if (_currentRangeIt != ranges.end()) _currentValue = _currentRangeIt->first; } private: ConstIterator _currentRangeIt; const ConstIterator _endRangeIt; T _currentValue; }; public: NumericRanges () = default; NumericRanges (const std::initializer_list &ranges); void addRange (T value) {addRange(value, value);} void addRange (T first, T last); void addRange (const Range &range) {addRange(range.first, range.second);} bool valueExists (T value) const; void clear () {_ranges.clear();} bool empty () const {return _ranges.empty();} size_t size () const {return _ranges.size();} ConstIterator begin () const {return _ranges.begin();} ConstIterator end () const {return _ranges.end();} ValueIterator valueIterator () const {return ValueIterator(*this);} const Container& ranges () const {return _ranges;} private: Container _ranges; }; template NumericRanges::NumericRanges (const std::initializer_list &ranges) { for (const Range &r : ranges) addRange(r); } /** Adds a numeric range to the collection. * @param[in] first left bound of new range * @param[in] last right bound of new range */ template void NumericRanges::addRange (T first, T last) { if (first > last) std::swap(first, last); auto it = _ranges.begin(); while (it != _ranges.end() && first > it->first+1 && first > it->second+1) ++it; if (it == _ranges.end() || last < it->first-1 || first > it->second+1) it = _ranges.insert(it, Range(first, last)); else if ((first < it->first && last >= it->first-1) || (first <= it->second+1 && last > it->second)) { it->first = std::min(it->first, first); it->second = std::max(it->second, last); } // merge adjacent ranges if (it != _ranges.end()) { auto l = it; auto r = it; if (l == _ranges.begin()) l = _ranges.end(); else --l; ++r; bool l_modified = false; bool r_modified = false; if (l != _ranges.end() && l->second >= it->first-1) { l->first = std::min(l->first, it->first); l->second = std::max(l->second, it->second); l_modified = true; } if (r != _ranges.end() && r->first <= it->second+1) { r->first = std::min(r->first, it->first); r->second = std::max(r->second, it->second); r_modified = true; } if (l_modified || r_modified) { _ranges.erase(it); if (l_modified && r_modified && l->second >= r->first-1) { l->first = std::min(l->first, r->first); l->second = std::max(l->second, r->second); _ranges.erase(r); } } } } template bool NumericRanges::valueExists (T value) const { auto it = algo::find_if(_ranges, [&](const Range &r) { return value <= r.second && value >= r.first; }); return it != _ranges.end(); } #endif