/// \file // Range v3 library // // Copyright Eric Niebler 2013-present // // Use, modification and distribution is subject to 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) // // Project home: https://github.com/ericniebler/range-v3 // #ifndef RANGES_V3_ITERATOR_RANGE_HPP #define RANGES_V3_ITERATOR_RANGE_HPP #include #include #include #include #include #include #include #include #include #include RANGES_DEPRECATED_HEADER( "This header is deprecated. Please switch to subrange in " ".") #include namespace ranges { /// \addtogroup group-views /// @{ template RANGES_INLINE_VAR constexpr bool enable_borrowed_range> = true; template RANGES_INLINE_VAR constexpr bool enable_borrowed_range> = true; template struct RANGES_EMPTY_BASES iterator_range : view_interface, same_as ? infinite : unknown> , compressed_pair { private: template friend struct iterator_range; template friend struct sized_iterator_range; compressed_pair & base() noexcept { return *this; } compressed_pair const & base() const noexcept { return *this; } using compressed_pair::first; using compressed_pair::second; public: using iterator = I; using sentinel = S; /// \cond using const_iterator = I; // Mostly to avoid spurious errors in Boost.Range /// \endcond constexpr I & begin() & { return this->first(); } constexpr I const & begin() const & { return this->first(); } constexpr S & end() & { return this->second(); } constexpr S const & end() const & { return this->second(); } iterator_range() = default; constexpr iterator_range(I first, S last) : compressed_pair{detail::move(first), detail::move(last)} {} template(typename X, typename Y)( requires constructible_from AND constructible_from) constexpr iterator_range(iterator_range rng) : compressed_pair{detail::move(rng.begin()), detail::move(rng.end())} {} template(typename X, typename Y)( requires constructible_from AND constructible_from) constexpr explicit iterator_range(std::pair rng) : compressed_pair{detail::move(rng.first), detail::move(rng.second)} {} template(typename X, typename Y)( requires assignable_from AND assignable_from) iterator_range & operator=(iterator_range rng) { base().first() = std::move(rng.base()).first(); base().second() = std::move(rng.base()).second(); return *this; } template(typename X, typename Y)( requires convertible_to AND convertible_to) constexpr operator std::pair() const { return {base().first(), base().second()}; } constexpr bool empty() const { return base().first() == base().second(); } }; // Like iterator_range, but with a known size. The first and second members // are private to prevent inadvertent violations of the class invariant. // // Class invariant: // distance(begin(), end()) == size() // template struct sized_iterator_range : view_interface, finite> { using size_type = detail::iter_size_t; using iterator = I; using sentinel = S; #ifndef RANGES_DOXYGEN_INVOKED using const_iterator = I; // Mostly to avoid spurious errors in Boost.Range #endif private: template friend struct sized_iterator_range; iterator_range rng_; size_type size_; public: sized_iterator_range() = default; RANGES_NDEBUG_CONSTEXPR sized_iterator_range(I first, S last, size_type size) : rng_{detail::move(first), detail::move(last)} , size_(size) { #ifndef NDEBUG RANGES_ASSERT(!(bool)forward_iterator || static_cast(ranges::distance(rng_)) == size_); #endif } template(typename X, typename Y)( requires constructible_from AND constructible_from) RANGES_NDEBUG_CONSTEXPR sized_iterator_range(std::pair rng, size_type size) : sized_iterator_range{detail::move(rng).first, detail::move(rng).second, size} {} template(typename X, typename Y)( requires constructible_from AND constructible_from) RANGES_NDEBUG_CONSTEXPR sized_iterator_range(iterator_range rng, size_type size) : sized_iterator_range{detail::move(rng).first(), detail::move(rng).second, size} {} template(typename X, typename Y)( requires constructible_from AND constructible_from) RANGES_NDEBUG_CONSTEXPR sized_iterator_range(sized_iterator_range rng) : sized_iterator_range{detail::move(rng).rng_.first(), detail::move(rng).rng_.second, rng.size_} {} template(typename X, typename Y)( requires assignable_from AND assignable_from) sized_iterator_range & operator=(sized_iterator_range rng) { rng_ = detail::move(rng).rng_; size_ = rng.size_; return *this; } I begin() const { return rng_.begin(); } S end() const { return rng_.end(); } size_type size() const noexcept { return size_; } template(typename X, typename Y)( requires convertible_to AND convertible_to) constexpr operator std::pair() const { return rng_; } template(typename X, typename Y)( requires convertible_to AND convertible_to) constexpr operator iterator_range() const { return rng_; } constexpr operator iterator_range const &() const & noexcept { return rng_; } // clang-format off /// Tuple-like access for `sized_iterator_range` template(std::size_t N)( requires (N < 2)) // friend constexpr auto CPP_auto_fun(get)(sized_iterator_range const &p) ( // return ranges::get(p.rng_) return ranges::get(p.*&sized_iterator_range::rng_) // makes clang happy ) // clang-format on /// \overload template(std::size_t N)( requires (N == 2)) // friend constexpr size_type get(sized_iterator_range const & p) noexcept { return p.size(); } }; struct make_iterator_range_fn { /// \return `{first, last}` template(typename I, typename S)( requires sentinel_for) constexpr iterator_range operator()(I first, S last) const { return {detail::move(first), detail::move(last)}; } /// \return `{first, last, size}` template(typename I, typename S)( requires sentinel_for) constexpr sized_iterator_range // operator()(I first, S last, detail::iter_size_t sz) const { return {detail::move(first), detail::move(last), sz}; } }; /// \sa `make_iterator_range_fn` RANGES_INLINE_VARIABLE(make_iterator_range_fn, make_iterator_range) // TODO add specialization of range_cardinality for when we can determine the range is // infinite /// @} } // namespace ranges // The standard is inconsistent about whether these are classes or structs RANGES_DIAGNOSTIC_PUSH RANGES_DIAGNOSTIC_IGNORE_MISMATCHED_TAGS /// \cond namespace std { template struct tuple_size<::ranges::iterator_range> : std::integral_constant {}; template struct tuple_element<0, ::ranges::iterator_range> { using type = I; }; template struct tuple_element<1, ::ranges::iterator_range> { using type = S; }; template struct tuple_size<::ranges::sized_iterator_range> : std::integral_constant {}; template struct tuple_element<0, ::ranges::sized_iterator_range> { using type = I; }; template struct tuple_element<1, ::ranges::sized_iterator_range> { using type = S; }; template struct tuple_element<2, ::ranges::sized_iterator_range> { using type = typename ::ranges::sized_iterator_range::size_type; }; } // namespace std /// \endcond RANGES_DIAGNOSTIC_POP #include #endif