| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677 | /*-----------------------------------------------------------------------------+    Copyright (c) 2010-2010: Joachim Faulhaber+------------------------------------------------------------------------------+   Distributed under the Boost Software License, Version 1.0.      (See accompanying file LICENCE.txt or copy at           http://www.boost.org/LICENSE_1_0.txt)+-----------------------------------------------------------------------------*/#ifndef BOOST_ICL_CONCEPT_INTERVAL_MAP_HPP_JOFA_100920#define BOOST_ICL_CONCEPT_INTERVAL_MAP_HPP_JOFA_100920#include <boost/icl/type_traits/element_type_of.hpp>#include <boost/icl/type_traits/segment_type_of.hpp>#include <boost/icl/type_traits/absorbs_identities.hpp>#include <boost/icl/type_traits/is_combinable.hpp>#include <boost/icl/type_traits/is_interval_splitter.hpp>#include <boost/icl/detail/set_algo.hpp>#include <boost/icl/detail/interval_map_algo.hpp>#include <boost/icl/concept/interval.hpp>#include <boost/icl/concept/joinable.hpp>namespace boost{ namespace icl{template<class Type>inline typename enable_if<is_interval_map<Type>, typename Type::segment_type>::typemake_segment(const typename Type::element_type& element){    typedef typename Type::interval_type interval_type;    typedef typename Type::segment_type  segment_type;    return segment_type(icl::singleton<interval_type>(element.key), element.data);}//==============================================================================//= Containedness<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- bool contains(c T&, c P&) T:{M} P:{b p M} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, bool>::typecontains(const Type& super, const typename Type::element_type& key_value_pair){    typedef typename Type::const_iterator const_iterator;    const_iterator it_ = icl::find(super, key_value_pair.key);    return it_ != super.end() && (*it_).second == key_value_pair.data;}template<class Type>typename enable_if<is_interval_map<Type>, bool>::typecontains(const Type& super, const typename Type::segment_type& sub_segment){    typedef typename Type::interval_type  interval_type;    typedef typename Type::const_iterator const_iterator;    interval_type sub_interval = sub_segment.first;    if(icl::is_empty(sub_interval))         return true;    std::pair<const_iterator, const_iterator> exterior = super.equal_range(sub_interval);    if(exterior.first == exterior.second)        return false;    const_iterator last_overlap = prior(exterior.second);    if(!(sub_segment.second == exterior.first->second) )        return false;    return          icl::contains(hull(exterior.first->first, last_overlap->first), sub_interval)      &&  Interval_Map::is_joinable(super, exterior.first, last_overlap);}template<class Type, class CoType>typename enable_if<is_concept_compatible<is_interval_map, Type, CoType>, bool>::typecontains(const Type& super, const CoType& sub){    return Interval_Set::within(sub, super);}//------------------------------------------------------------------------------//- bool contains(c T&, c P&) T:{M} P:{e i S} key_types : total//------------------------------------------------------------------------------template<class Type, class CoType>typename enable_if< mpl::and_< is_interval_map<Type>                             , is_total<Type>                              , is_cross_derivative<Type, CoType> >            , bool>::typecontains(const Type&, const CoType&){    return true;}//------------------------------------------------------------------------------//- bool contains(c T&, c P&) T:{M} P:{e i S} key_types : partial//------------------------------------------------------------------------------template<class Type>typename enable_if< mpl::and_< is_interval_map<Type>                             , mpl::not_<is_total<Type> > >            , bool>::typecontains(const Type& super, const typename Type::domain_type& key)    {    return icl::find(super, key) != super.end();}template<class Type>typename enable_if< mpl::and_< is_interval_map<Type>                             , mpl::not_<is_total<Type> > >            , bool>::typecontains(const Type& super, const typename Type::interval_type& sub_interval){    typedef typename Type::const_iterator const_iterator;    if(icl::is_empty(sub_interval))         return true;    std::pair<const_iterator, const_iterator> exterior = super.equal_range(sub_interval);    if(exterior.first == exterior.second)        return false;    const_iterator last_overlap = prior(exterior.second);    return          icl::contains(hull(exterior.first->first, last_overlap->first), sub_interval)      &&  Interval_Set::is_joinable(super, exterior.first, last_overlap);}template<class Type, class KeyT>typename enable_if< mpl::and_< is_concept_combinable<is_interval_map, is_interval_set, Type, KeyT>                             , mpl::not_<is_total<Type> > >            , bool>::typecontains(const Type& super, const KeyT& sub){    return Interval_Set::within(sub, super);}//==============================================================================//= Addition<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& add(T&, c P&) T:{M} P:{b p} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&add(Type& object, const typename Type::segment_type& operand){    return object.add(operand);}template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&add(Type& object, const typename Type::element_type& operand){    return icl::add(object, make_segment<Type>(operand));}//------------------------------------------------------------------------------//- T& add(T&, J, c P&) T:{M} P:{p} segment_type//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, typename Type::iterator >::typeadd(Type& object, typename Type::iterator      prior_,            const typename Type::segment_type& operand){    return object.add(prior_, operand);}//==============================================================================//= Insertion<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& insert(T&, c P&) T:{M} P:{b p} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&insert(Type& object, const typename Type::segment_type& operand){    return object.insert(operand);}template<class Type>inline typename enable_if<is_interval_map<Type>, Type>::type&insert(Type& object, const typename Type::element_type& operand){    return icl::insert(object, make_segment<Type>(operand));}//------------------------------------------------------------------------------//- T& insert(T&, J, c P&) T:{M} P:{p} with hint//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, typename Type::iterator>::typeinsert(Type& object, typename Type::iterator      prior,               const typename Type::segment_type& operand){    return object.insert(prior, operand);}//==============================================================================//= Erasure<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& erase(T&, c P&) T:{M} P:{e i} key_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&erase(Type& object, const typename Type::interval_type& operand){    return object.erase(operand);}template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&erase(Type& object, const typename Type::domain_type& operand){    typedef typename Type::interval_type interval_type;    return icl::erase(object, icl::singleton<interval_type>(operand));}//------------------------------------------------------------------------------//- T& erase(T&, c P&) T:{M} P:{b p} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&erase(Type& object, const typename Type::segment_type& operand){    return object.erase(operand);}template<class Type>inline typename enable_if<is_interval_map<Type>, Type>::type&erase(Type& object, const typename Type::element_type& operand){    return icl::erase(object, make_segment<Type>(operand));}//==============================================================================//= Subtraction<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{b p} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type& subtract(Type& object, const typename Type::segment_type& operand){    return object.subtract(operand);}template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&subtract(Type& object, const typename Type::element_type& operand){    return icl::subtract(object, make_segment<Type>(operand));}//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{e i} key_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&subtract(Type& object, const typename Type::domain_type& operand){    return object.erase(operand);}template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&subtract(Type& object, const typename Type::interval_type& operand){    return object.erase(operand);}//==============================================================================//= Selective Update<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& set_at(T&, c P&) T:{M} P:{e i}//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&set_at(Type& object, const typename Type::segment_type& operand){    icl::erase(object, operand.first);     return icl::insert(object, operand); }template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&set_at(Type& object, const typename Type::element_type& operand){    return icl::set_at(object, make_segment<Type>(operand));}//==============================================================================//= Intersection<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{b p} fragment_type//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, void>::typeadd_intersection(Type& section, const Type& object,                  const typename Type::element_type& operand){    //CL typedef typename Type::segment_type segment_type;    object.add_intersection(section, make_segment<Type>(operand));}template<class Type>typename enable_if<is_interval_map<Type>, void>::typeadd_intersection(Type& section, const Type& object,                  const typename Type::segment_type& operand){    object.add_intersection(section, operand);}//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{M'} map fragment_type total//------------------------------------------------------------------------------template<class Type, class MapT>typename enable_if<    mpl::and_< is_total<Type>             , is_concept_compatible<is_interval_map, Type, MapT> >  , void>::typeadd_intersection(Type& section, const Type& object, const MapT& operand){    section += object;    section += operand;}//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{M'} map fragment_type partial//------------------------------------------------------------------------------template<class Type, class MapT>typename enable_if<    mpl::and_< mpl::not_<is_total<Type> >             , is_concept_compatible<is_interval_map, Type, MapT> >  , void>::typeadd_intersection(Type& section, const Type& object, const MapT& operand){    //CL typedef typename Type::segment_type   segment_type;    //CL typedef typename Type::interval_type  interval_type;    typedef typename MapT::const_iterator const_iterator;    if(operand.empty())         return;    const_iterator common_lwb, common_upb;    if(!Set::common_range(common_lwb, common_upb, operand, object))        return;    const_iterator it_ = common_lwb;    while(it_ != common_upb)        add_intersection(section, object, *it_++);}//------------------------------------------------------------------------------//- T& subtract(T&, c P&) T:{M} P:{e i S} key_type//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, void>::typeadd_intersection(Type& section, const Type& object,                  const typename Type::domain_type& key_value){    typedef typename Type::interval_type  interval_type;    typedef typename Type::segment_type   segment_type;    typedef typename Type::const_iterator const_iterator;    const_iterator it_ = icl::find(object, key_value);    if(it_ != object.end())        add(section, segment_type(interval_type(key_value),(*it_).second));}template<class Type>typename enable_if<is_interval_map<Type>, void>::typeadd_intersection(Type& section, const Type& object,                  const typename Type::interval_type& inter_val){    typedef typename Type::interval_type  interval_type;    typedef typename Type::value_type     value_type;    typedef typename Type::const_iterator const_iterator;    typedef typename Type::iterator       iterator;    if(icl::is_empty(inter_val))         return;    std::pair<const_iterator, const_iterator> exterior         = object.equal_range(inter_val);    if(exterior.first == exterior.second)        return;    iterator prior_ = section.end();    for(const_iterator it_=exterior.first; it_ != exterior.second; it_++)     {        interval_type common_interval = (*it_).first & inter_val;         if(!icl::is_empty(common_interval))            prior_ = add(section, prior_,                          value_type(common_interval, (*it_).second) );    }}template<class Type, class KeySetT>typename enable_if<is_concept_combinable<is_interval_map, is_interval_set, Type, KeySetT>, void>::typeadd_intersection(Type& section, const Type& object, const KeySetT& key_set){    typedef typename KeySetT::const_iterator const_iterator;    if(icl::is_empty(key_set))         return;    const_iterator common_lwb, common_upb;    if(!Set::common_range(common_lwb, common_upb, key_set, object))        return;    const_iterator it_ = common_lwb;    while(it_ != common_upb)        add_intersection(section, object, *it_++);}//------------------------------------------------------------------------------//- intersects<IntervalMaps> fragment_types//------------------------------------------------------------------------------template<class Type, class OperandT>typename enable_if<mpl::and_< is_interval_map<Type>                            , is_total<Type>                            , boost::is_same< OperandT                                            , typename segment_type_of<Type>::type> >,                    bool>::typeintersects(const Type&, const OperandT&){    return true;}template<class Type, class OperandT>typename enable_if<mpl::and_< is_interval_map<Type>                            , mpl::not_<is_total<Type> >                            , boost::is_same<OperandT, typename segment_type_of<Type>::type> >,                    bool>::typeintersects(const Type& object, const OperandT& operand){    Type intersection;    icl::add_intersection(intersection, object, operand);    return !icl::is_empty(intersection); }template<class Type, class OperandT>typename enable_if<mpl::and_< is_interval_map<Type>                            , boost::is_same<OperandT, typename element_type_of<Type>::type> >,                    bool>::typeintersects(const Type& object, const OperandT& operand){    return icl::intersects(object, make_segment<Type>(operand)); }//==============================================================================//= Symmetric difference<IntervalMap>//==============================================================================//------------------------------------------------------------------------------//- T& flip(T&, c P&) T:{M} P:{b p} fragment_types//------------------------------------------------------------------------------template<class Type>typename enable_if<is_interval_map<Type>, Type>::type&flip(Type& object, const typename Type::segment_type& operand){    return object.flip(operand);}template<class Type>inline typename enable_if<is_interval_map<Type>, Type>::type&flip(Type& object, const typename Type::element_type& operand){    return icl::flip(object, make_segment<Type>(operand));}//------------------------------------------------------------------------------//- T& flip(T&, c P&) T:{M} P:{M'} total absorber //------------------------------------------------------------------------------template<class Type, class OperandT>typename enable_if< mpl::and_< is_total<Type>                             , absorbs_identities<Type>                             , is_concept_compatible<is_interval_map,                                                      Type, OperandT >                             >                  , Type>::type&flip(Type& object, const OperandT&){    object.clear();    return object;}//------------------------------------------------------------------------------//- T& flip(T&, c P&) T:{M} P:{M'} total enricher //------------------------------------------------------------------------------#ifdef BOOST_MSVC #pragma warning(push)#pragma warning(disable:4127) // conditional expression is constant#endif                        template<class Type, class OperandT>typename enable_if< mpl::and_< is_total<Type>                             , mpl::not_<absorbs_identities<Type> >                             , is_concept_compatible<is_interval_map,                                                      Type, OperandT >                             >                  , Type>::type&flip(Type& object, const OperandT& operand){    typedef typename Type::codomain_type  codomain_type;    object += operand;    ICL_FORALL(typename Type, it_, object)        (*it_).second = identity_element<codomain_type>::value();    if(mpl::not_<is_interval_splitter<Type> >::value)        icl::join(object);    return object;}#ifdef BOOST_MSVC#pragma warning(pop)#endif//------------------------------------------------------------------------------//- T& flip(T&, c P&) T:{M} P:{M'} partial //------------------------------------------------------------------------------template<class Type, class OperandT>typename enable_if< mpl::and_< mpl::not_<is_total<Type> >                              , is_concept_compatible<is_interval_map,                                                      Type, OperandT >                             >                  , Type>::type&flip(Type& object, const OperandT& operand){    typedef typename OperandT::const_iterator const_iterator;    //CL typedef typename Type::codomain_type  codomain_type;    const_iterator common_lwb, common_upb;    if(!Set::common_range(common_lwb, common_upb, operand, object))        return object += operand;    const_iterator it_ = operand.begin();    // All elements of operand left of the common range are added    while(it_ != common_lwb)        icl::add(object, *it_++);    // All elements of operand in the common range are symmetrically subtracted    while(it_ != common_upb)        icl::flip(object, *it_++);    // All elements of operand right of the common range are added    while(it_ != operand.end())        icl::add(object, *it_++);    return object;}//==============================================================================//= Set selection//==============================================================================template<class Type, class SetT>typename enable_if<is_concept_combinable<is_interval_set, is_interval_map,                                          SetT, Type>, SetT>::type&domain(SetT& result, const Type& object){    typedef typename SetT::iterator set_iterator;    result.clear();     set_iterator prior_ = result.end();    ICL_const_FORALL(typename Type, it_, object)         prior_ = icl::insert(result, prior_, (*it_).first);         return result;}template<class Type, class SetT>typename enable_if<is_concept_combinable<is_interval_set, is_interval_map,                                          SetT, Type>, SetT>::type&between(SetT& in_between, const Type& object){    typedef typename Type::const_iterator const_iterator;    typedef typename SetT::iterator       set_iterator;    in_between.clear();    const_iterator it_ = object.begin(), pred_;    set_iterator   prior_ = in_between.end();    if(it_ != object.end())        pred_ = it_++;    while(it_ != object.end())        prior_ = icl::insert(in_between, prior_,                              between((*pred_++).first, (*it_++).first));        return in_between;}//==============================================================================//= Manipulation by predicates//==============================================================================template<class MapT, class Predicate>typename enable_if<is_interval_map<MapT>, MapT>::type&erase_if(const Predicate& pred, MapT& object){    typename MapT::iterator it_ = object.begin();    while(it_ != object.end())        if(pred(*it_))            object.erase(it_++);         else ++it_;    return object;}template<class MapT, class Predicate>inline typename enable_if<is_interval_map<MapT>, MapT>::type&add_if(const Predicate& pred, MapT& object, const MapT& src){    typename MapT::const_iterator it_ = src.begin();    while(it_ != src.end())        if(pred(*it_))             icl::add(object, *it_++);         return object;}template<class MapT, class Predicate>inline typename enable_if<is_interval_map<MapT>, MapT>::type&assign_if(const Predicate& pred, MapT& object, const MapT& src){    icl::clear(object);    return add_if(object, src, pred);}//==============================================================================//= Morphisms//==============================================================================template<class Type>typename enable_if<mpl::and_< is_interval_map<Type>                            , absorbs_identities<Type> >, Type>::type&absorb_identities(Type& object){    return object;}template<class Type>typename enable_if<mpl::and_< is_interval_map<Type>                            , mpl::not_<absorbs_identities<Type> > >, Type>::type&absorb_identities(Type& object){    typedef typename Type::segment_type segment_type;    return icl::erase_if(content_is_identity_element<segment_type>(), object);}//==============================================================================//= Streaming//==============================================================================template<class CharType, class CharTraits, class Type>typename enable_if<is_interval_map<Type>,                    std::basic_ostream<CharType, CharTraits> >::type& operator << (std::basic_ostream<CharType, CharTraits>& stream, const Type& object){    stream << "{";    ICL_const_FORALL(typename Type, it_, object)        stream << "(" << (*it_).first << "->" << (*it_).second << ")";    return stream << "}";}}} // namespace boost icl#endif
 |