#ifndef SPROUT_DETAIL_ITERATOR_HPP #define SPROUT_DETAIL_ITERATOR_HPP #include #include #include HDR_ITERATOR_SSCRISK_CEL_OR_SPROUT_DETAIL namespace sprout { namespace detail { // // distance // template SPROUT_CONSTEXPR typename std::iterator_traits::difference_type distance( Iterator first, Iterator last ) { return first == last ? 0 : 1 + sprout::detail::distance(first + 1, last); } template SPROUT_CONSTEXPR typename std::iterator_traits::difference_type bidirectional_distance_impl( Iterator first1, Iterator first2, Iterator last, typename std::iterator_traits::difference_type current = 1 ) { return first1 == last ? current : first2 == last ? -current : sprout::detail::bidirectional_distance_impl(first1 + 1, first2 - 1, last, current + 1) ; } // // bidirectional_distance // template SPROUT_CONSTEXPR typename std::iterator_traits::difference_type bidirectional_distance( Iterator first, Iterator last ) { return first == last ? 0 : sprout::detail::bidirectional_distance_impl(first + 1, first - 1, last); } // // reverse_iterator // template class reverse_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; protected: iterator_type current; private: iterator_type deref_tmp; public: reverse_iterator() = default; reverse_iterator(reverse_iterator const&) = default; SPROUT_CONSTEXPR explicit reverse_iterator(iterator_type it) : current(it) , deref_tmp(it - 1) {} template SPROUT_CONSTEXPR reverse_iterator(reverse_iterator const& it) : current(it) , deref_tmp(it - 1) {} template reverse_iterator& operator=(reverse_iterator const& it) { reverse_iterator temp(it); temp.swap(this); return this; } SPROUT_CONSTEXPR iterator_type base() const { return current; } SPROUT_CONSTEXPR reference operator*() const { return *deref_tmp; } SPROUT_CONSTEXPR pointer operator->() const { return &*deref_tmp; } reverse_iterator& operator++() { --current; --deref_tmp; return this; } reverse_iterator operator++(int) { reverse_iterator result(this); --current; --deref_tmp; return result; } reverse_iterator& operator--() { ++current; ++deref_tmp; return this; } reverse_iterator operator--(int) { reverse_iterator temp(this); ++current; ++deref_tmp; return temp; } SPROUT_CONSTEXPR reverse_iterator operator+(difference_type n) const { return reverse_iterator(current - n); } SPROUT_CONSTEXPR reverse_iterator operator-(difference_type n) const { return reverse_iterator(current + n); } reverse_iterator& operator+=(difference_type n) { reverse_iterator temp(current - n); temp.swap(this); return this; } reverse_iterator& operator-=(difference_type n) { reverse_iterator temp(current + n); temp.swap(this); return this; } SPROUT_CONSTEXPR reference operator[](difference_type n) const { return *(deref_tmp - n); } void swap(reverse_iterator& other) { using std::swap; swap(current, other.current); swap(deref_tmp, other.deref_tmp); } }; template SPROUT_CONSTEXPR bool operator==( sprout::detail::reverse_iterator const& lhs, sprout::detail::reverse_iterator const& rhs ) { return lhs.base() == rhs.base(); } template SPROUT_CONSTEXPR bool operator!=( sprout::detail::reverse_iterator const& lhs, sprout::detail::reverse_iterator const& rhs ) { return !(lhs == rhs); } template SPROUT_CONSTEXPR sprout::detail::reverse_iterator operator+( typename sprout::detail::reverse_iterator::difference_type n, sprout::detail::reverse_iterator const& it ) { return it + n; } // // swap // template void swap(sprout::detail::reverse_iterator& lhs, sprout::detail::reverse_iterator& rhs) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } } // namespace detail } // namespace sprout #endif // #ifndef SPROUT_DETAIL_ITERATOR_HPP