#ifndef SPROUT_ITERATOR_VALUED_ITERATOR_HPP #define SPROUT_ITERATOR_VALUED_ITERATOR_HPP #include #include #include #include #include #include #include #include namespace sprout { // // valued_iterator // template class valued_iterator : public std::iterator< typename std::iterator_traits::iterator_category, typename std::iterator_traits::value_type, typename std::iterator_traits::difference_type, typename std::iterator_traits::pointer, typename std::iterator_traits::reference > { public: typedef Iterator iterator_type; typedef typename std::iterator_traits::iterator_category iterator_category; typedef typename std::iterator_traits::value_type value_type; typedef typename std::iterator_traits::difference_type difference_type; typedef typename std::iterator_traits::pointer pointer; typedef typename std::iterator_traits::reference reference; typedef T content_type; typedef sprout::value_holder holder_type; typedef typename holder_type::param_type param_type; typedef typename holder_type::mutable_or_const_reference get_type; protected: iterator_type current; holder_type holder; public: valued_iterator() = default; SPROUT_CONSTEXPR valued_iterator(valued_iterator const& other) : current(other.current) , holder(other.holder) {} explicit SPROUT_CONSTEXPR valued_iterator(iterator_type it) : current(it) , holder() {} SPROUT_CONSTEXPR valued_iterator(iterator_type it, holder_type const& r) : current(it) , holder(r) {} SPROUT_CONSTEXPR valued_iterator(iterator_type it, param_type p) : current(it) , holder(p) {} template SPROUT_CONSTEXPR valued_iterator(valued_iterator const& it) : current(it.base()) , holder(it.get()) {} template valued_iterator& operator=(valued_iterator const& it) { valued_iterator temp(it); temp.swap(*this); return *this; } SPROUT_CONSTEXPR iterator_type base() const { return current; } SPROUT_CONSTEXPR get_type get() const { return holder.get(); } SPROUT_CONSTEXPR reference operator*() const { return *current; } SPROUT_CONSTEXPR pointer operator->() const { return &*current; } valued_iterator& operator++() { ++current; return *this; } valued_iterator operator++(int) { valued_iterator result(*this); ++current; return result; } valued_iterator& operator--() { --current; return *this; } valued_iterator operator--(int) { valued_iterator temp(*this); --current; return temp; } SPROUT_CONSTEXPR valued_iterator operator+(difference_type n) const { return valued_iterator(current + n, holder); } SPROUT_CONSTEXPR valued_iterator operator-(difference_type n) const { return valued_iterator(current - n, holder); } valued_iterator& operator+=(difference_type n) { valued_iterator temp(current + n, holder); temp.swap(*this); return *this; } valued_iterator& operator-=(difference_type n) { valued_iterator temp(current - n, holder); temp.swap(*this); return *this; } SPROUT_CONSTEXPR reference operator[](difference_type n) const { return *(current + n); } SPROUT_CONSTEXPR valued_iterator next() const { return valued_iterator(sprout::next(current), holder); } SPROUT_CONSTEXPR valued_iterator prev() const { return valued_iterator(sprout::prev(current), holder); } void swap(valued_iterator& other) SPROUT_NOEXCEPT_EXPR( SPROUT_NOEXCEPT_EXPR(swap(current, other.current)) && SPROUT_NOEXCEPT_EXPR(swap(holder, other.holder)) ) { swap(current, other.current); swap(holder, other.holder); } }; template inline SPROUT_CONSTEXPR bool operator==(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return lhs.base() == rhs.base(); } template inline SPROUT_CONSTEXPR bool operator!=(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return !(lhs == rhs); } template inline SPROUT_CONSTEXPR bool operator<(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return lhs.base() < rhs.base(); } template inline SPROUT_CONSTEXPR bool operator>(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return rhs < lhs; } template inline SPROUT_CONSTEXPR bool operator<=(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return !(rhs < lhs); } template inline SPROUT_CONSTEXPR bool operator>=(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return !(lhs < rhs); } template inline SPROUT_CONSTEXPR decltype(std::declval() - std::declval()) operator-(sprout::valued_iterator const& lhs, sprout::valued_iterator const& rhs) { return lhs.base() - rhs.base(); } template inline SPROUT_CONSTEXPR sprout::valued_iterator operator+( typename sprout::valued_iterator::difference_type n, sprout::valued_iterator const& it ) { return it + n; } // // make_valued_iterator // template inline SPROUT_CONSTEXPR sprout::valued_iterator make_valued_iterator(Iterator it, T const& param) { return sprout::valued_iterator(it, param); } // // swap // template inline void swap(sprout::valued_iterator& lhs, sprout::valued_iterator& rhs) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } // // iterator_next // template inline SPROUT_CONSTEXPR sprout::valued_iterator iterator_next(sprout::valued_iterator const& it) { return it.next(); } // // iterator_prev // template inline SPROUT_CONSTEXPR sprout::valued_iterator iterator_prev(sprout::valued_iterator const& it) { return it.prev(); } } // namespace sprout #endif // #ifndef SPROUT_ITERATOR_VALUED_ITERATOR_HPP