/*============================================================================= Copyright (c) 2011-2013 Bolero MURAKAMI https://github.com/bolero-MURAKAMI/Sprout Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) =============================================================================*/ #ifndef SPROUT_ITERATOR_SET_UNION_ITERATOR_HPP #define SPROUT_ITERATOR_SET_UNION_ITERATOR_HPP #include #include #include #include #include #include #include #include #include #include namespace sprout { // // set_union_iterator // template< typename LIterator, typename RIterator, typename Compare = sprout::less<> > class set_union_iterator : public std::iterator< typename sprout::min_iterator_category< typename sprout::common_iterator_category::type, std::forward_iterator_tag >::type, typename sprout::common_iterator_value_type::type, typename sprout::common_iterator_difference_type::type, typename sprout::common_iterator_pointer::type, typename sprout::common_iterator_reference::type > { public: typedef LIterator iterator_type; typedef RIterator iterator2_type; typedef Compare compare_type; typedef typename sprout::min_iterator_category< typename sprout::common_iterator_category::type, std::forward_iterator_tag >::type iterator_category; typedef typename sprout::common_iterator_value_type::type value_type; typedef typename sprout::common_iterator_difference_type::type difference_type; typedef typename sprout::common_iterator_pointer::type pointer; typedef typename sprout::common_iterator_reference::type reference; protected: typedef sprout::pair pair_type; private: static SPROUT_CONSTEXPR bool check_in_left( iterator_type it1, iterator_type lst1, iterator2_type it2, iterator2_type lst2, Compare comp ) { return it1 != lst1 ? (it2 != lst2 ? comp(*it1, *it2) || !comp(*it2, *it1) : true) : !(it2 != lst2) ; } protected: pair_type current; iterator_type lst1; iterator2_type lst2; Compare comp; private: bool in_left; private: SPROUT_CONSTEXPR set_union_iterator(set_union_iterator const& other, pair_type const& next) : current(next) , lst1(other.lst1), lst2(other.lst2) , comp(other.comp) , in_left(check_in_left(next.first, other.lst1, next.second, other.lst2, other.comp)) {} public: SPROUT_CONSTEXPR set_union_iterator() : current(), lst1(), lst2(), comp(), in_left(true) {} set_union_iterator(set_union_iterator const&) = default; SPROUT_CONSTEXPR set_union_iterator( iterator_type it1, iterator_type lst1, iterator2_type it2, iterator2_type lst2, Compare comp = Compare() ) : current(it1, it2) , lst1(lst1), lst2(lst2) , comp(comp) , in_left(check_in_left(it1, lst1, it2, lst2, comp)) {} template SPROUT_CONSTEXPR set_union_iterator(set_union_iterator const& it) : current(it.base(), it.base2()) , lst1(it.last1()), lst2(it.last2()) , comp(it.compare()) , in_left(it.is_in_left()) {} template set_union_iterator& operator=(set_union_iterator const& it) { set_union_iterator temp(it); temp.swap(*this); return *this; } SPROUT_CONSTEXPR iterator_type base() const { return current.first; } SPROUT_CONSTEXPR iterator_type last1() const { return lst1; } SPROUT_CONSTEXPR iterator2_type base2() const { return current.second; } SPROUT_CONSTEXPR iterator2_type last2() const { return lst2; } SPROUT_CONSTEXPR Compare compare() const { return comp; } SPROUT_CONSTEXPR bool is_in_left() const { return in_left; } SPROUT_CONSTEXPR reference operator*() const { return is_in_left() ? *current.first : *current.second; } SPROUT_CONSTEXPR pointer operator->() const { return &*(*this); } set_union_iterator& operator++() { current = sprout::next_union(current.first, lst1, current.second, lst2, comp); in_left = check_in_left(current.first, lst1, current.second, lst2, comp); return *this; } set_union_iterator operator++(int) { set_union_iterator result(*this); current = sprout::next_union(current.first, lst1, current.second, lst2, comp); in_left = check_in_left(current.first, lst1, current.second, lst2, comp); return result; } SPROUT_CONSTEXPR set_union_iterator next() const { return set_union_iterator( *this, sprout::next_union(current.first, lst1, current.second, lst2, comp) ); } void swap(set_union_iterator& other) SPROUT_NOEXCEPT_EXPR( SPROUT_NOEXCEPT_EXPR(sprout::swap(current, other.current)) && SPROUT_NOEXCEPT_EXPR(sprout::swap(lst1, other.lst1)) && SPROUT_NOEXCEPT_EXPR(sprout::swap(lst2, other.lst2)) && SPROUT_NOEXCEPT_EXPR(sprout::swap(comp, other.comp)) ) { sprout::swap(current, other.current); sprout::swap(lst1, other.lst1); sprout::swap(lst2, other.lst2); sprout::swap(comp, other.comp); sprout::swap(in_left, other.in_left); } }; template< typename LIterator1, typename RIterator1, typename Compare1, typename LIterator2, typename RIterator2, typename Compare2 > inline SPROUT_CONSTEXPR bool operator==( sprout::set_union_iterator const& lhs, sprout::set_union_iterator const& rhs ) { return lhs.base() == rhs.base() && lhs.base2() == rhs.base2(); } template< typename LIterator1, typename RIterator1, typename Compare1, typename LIterator2, typename RIterator2, typename Compare2 > inline SPROUT_CONSTEXPR bool operator!=( sprout::set_union_iterator const& lhs, sprout::set_union_iterator const& rhs ) { return !(lhs == rhs); } // // make_set_union_iterator // template inline SPROUT_CONSTEXPR sprout::set_union_iterator make_set_union_iterator(LIterator it1, LIterator lst1, RIterator it2, RIterator lst2, Compare comp) { return sprout::set_union_iterator(it1, lst1, it2, lst2, comp); } template inline SPROUT_CONSTEXPR sprout::set_union_iterator make_set_union_iterator(LIterator it1, LIterator lst1, RIterator it2, RIterator lst2) { return sprout::set_union_iterator(it1, lst1, it2, lst2); } // // swap // template inline void swap( sprout::set_union_iterator& lhs, sprout::set_union_iterator& rhs ) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } // // iterator_next // template inline SPROUT_CONSTEXPR sprout::set_union_iterator iterator_next(sprout::set_union_iterator const& it) { return it.next(); } } // namespace sprout #endif // #ifndef SPROUT_ITERATOR_SET_UNION_ITERATOR_HPP