/* * CDE - Common Desktop Environment * * Copyright (c) 1993-2012, The Open Group. All rights reserved. * * These libraries and programs are free software; you can * redistribute them and/or modify them under the terms of the GNU * Lesser General Public License as published by the Free Software * Foundation; either version 2 of the License, or (at your option) * any later version. * * These libraries and programs are distributed in the hope that * they will be useful, but WITHOUT ANY WARRANTY; without even the * implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU Lesser General Public License for more * details. * * You should have received a copy of the GNU Lesser General Public * License along with these librararies and programs; if not, write * to the Free Software Foundation, Inc., 51 Franklin Street, Fifth * Floor, Boston, MA 02110-1301 USA */ /* $XConsortium: ISet.C /main/1 1996/07/29 16:54:10 cde-hp $ */ // Copyright (c) 1994 James Clark // See the file COPYING for copying permission. #ifndef ISet_DEF_INCLUDED #define ISet_DEF_INCLUDED 1 #include #ifdef SP_NAMESPACE namespace SP_NAMESPACE { #endif template ISet::ISet() { } template ISet::~ISet() { } template ISet::ISet(const T *v, size_t n) { for (size_t i = 0; i < n; i++) add(v[i]); } template Boolean ISet::contains(T x) const { for (size_t i = 0; i < r_.size(); i++) if (r_[i].max >= x) return r_[i].min <= x ? 1 : 0; return 0; } template void ISet::addRange(T min, T max) { size_t i; if (min == 0) i = 0; else { for (i = r_.size(); i > 0 && min - 1 <= r_[i - 1].max; i--) ; } // r_[i - 1].max < min - 1 <= r_[i].max if (i < r_.size() && (r_[i].min == 0 || max >= r_[i].min - 1)) { // we can coelesce if (min < r_[i].min) r_[i].min = min; if (max > r_[i].max) { r_[i].max = max; size_t j; for (j = i + 1; j < r_.size() && r_[i].max >= r_[j].min - 1; j++) r_[i].max = r_[j].max; // get rid of i + 1 ... j - 1 if (j > i + 1) { for (size_t k = j; k < r_.size(); k++) r_[k - (j - i - 1)] = r_[k]; r_.resize(r_.size() - (j - i - 1)); } } } else { // r_[i - 1].max < min - 1 // max + 1 < r_[i].min r_.resize(r_.size() + 1); for (size_t j = r_.size() - 1; j > i; j--) r_[j] = r_[j - 1]; r_[i].max = max; r_[i].min = min; } } template void ISet::remove(T c) { for (size_t i = 0; i < r_.size(); i++) if (r_[i].max >= c) { if (r_[i].min <= c) { if (r_[i].min == r_[i].max) { while (++i < r_.size()) r_[i - 1] = r_[i]; r_.resize(r_.size() - 1); } else if (c == r_[i].min) r_[i].min += 1; else if (c == r_[i].max) r_[i].max -= 1; else { r_.resize(r_.size() + 1); // split the range // subtracting 2 is safe since we know that the length is >= 2 for (size_t j = r_.size() - 2; j > i; j--) r_[j + 1] = r_[j]; r_[i + 1].max = r_[i].max; r_[i + 1].min = c + 1; r_[i].max = c - 1; } } break; } } template void ISet::check() { for (size_t i = 0; i < r_.size(); i++) { if (r_[i].min > r_[i].max) abort(); // adjacent ranges must be coalesced if (i > 0 && r_[i].min - 1 <= r_[i - 1].max) abort(); } } template void ISet::clear() { r_.resize(0); } #ifdef SP_NAMESPACE } #endif #endif /* not ISet_DEF_INCLUDED */