axmol/thirdparty/range-v3/include/range/v3/view/subrange.hpp

486 lines
16 KiB
C++

/// \file
// Range v3 library
//
// Copyright Eric Niebler 2013-present
// Copyright Casey Carter 2017
//
// 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_VIEW_SUBRANGE_HPP
#define RANGES_V3_VIEW_SUBRANGE_HPP
#include <tuple>
#include <type_traits>
#include <utility>
#include <meta/meta.hpp>
#include <concepts/concepts.hpp>
#include <range/v3/iterator/operations.hpp>
#include <range/v3/iterator/unreachable_sentinel.hpp>
#include <range/v3/range/access.hpp>
#include <range/v3/range/concepts.hpp>
#include <range/v3/range/dangling.hpp>
#include <range/v3/utility/get.hpp>
#include <range/v3/view/interface.hpp>
#include <range/v3/detail/prologue.hpp>
namespace ranges
{
/// \addtogroup group-views
/// @{
enum class subrange_kind : bool
{
unsized,
sized
};
/// \cond
namespace detail
{
// clang-format off
/// \concept convertible_to_not_slicing_
/// \brief The \c convertible_to_not_slicing_ concept
template<typename From, typename To>
CPP_concept convertible_to_not_slicing_ =
convertible_to<From, To> &&
// A conversion is a slicing conversion if the source and the destination
// are both pointers, and if the pointed-to types differ after removing
// cv qualifiers.
(!(std::is_pointer<decay_t<From>>::value &&
std::is_pointer<decay_t<To>>::value &&
not_same_as_<std::remove_pointer_t<decay_t<From>>,
std::remove_pointer_t<decay_t<To>>>));
template<std::size_t N, typename T>
using tuple_element_fun_t = void (*)(meta::_t<std::tuple_element<N, T>> const &);
/// \concept pair_like_impl_
/// \brief The \c pair_like_impl_ concept
template<typename T>
CPP_requires(pair_like_impl_, //
requires(T t, tuple_element_fun_t<0, T> p0, tuple_element_fun_t<1, T> p1) //
(
p0( get<0>(t) ),
p1( get<1>(t) )
));
/// \concept pair_like_impl_
/// \brief The \c pair_like_impl_ concept
template<typename T>
CPP_concept pair_like_impl_ = CPP_requires_ref(detail::pair_like_impl_, T);
/// \concept is_complete_
/// \brief The \c is_complete_ concept
template(typename T)(
concept (is_complete_)(T),
0 != sizeof(T));
/// \concept is_complete_
/// \brief The \c is_complete_ concept
template<typename T>
CPP_concept is_complete_ = //
CPP_concept_ref(is_complete_, T);
template(typename T)( //
concept (pair_like_)(T), //
is_complete_<std::tuple_size<T>> AND
derived_from<std::tuple_size<T>, meta::size_t<2>> AND
detail::pair_like_impl_<T>);
/// \concept pair_like
/// \brief The \c pair_like concept
template<typename T>
CPP_concept pair_like = //
CPP_concept_ref(detail::pair_like_, T);
// clang-format off
template(typename T, typename U, typename V)( //
concept (pair_like_convertible_from_helper_)(T, U, V), //
convertible_to_not_slicing_<U, meta::_t<std::tuple_element<0, T>>> AND
convertible_to<V, meta::_t<std::tuple_element<1, T>>>);
/// \concept pair_like_convertible_from_helper_
/// \brief The \c pair_like_convertible_from_helper_ concept
template<typename T, typename U, typename V>
CPP_concept pair_like_convertible_from_helper_ = //
CPP_concept_ref(pair_like_convertible_from_helper_, T, U, V);
template(typename T, typename U, typename V)( //
concept (pair_like_convertible_from_impl_)(T, U, V),
(!range<T>) AND
constructible_from<T, U, V> AND
pair_like<uncvref_t<T>> AND
pair_like_convertible_from_helper_<T, U, V>);
/// \concept pair_like_convertible_from_
/// \brief The \c pair_like_convertible_from_ concept
template<typename T, typename U, typename V>
CPP_concept pair_like_convertible_from_ =
CPP_concept_ref(detail::pair_like_convertible_from_impl_, T, U, V);
/// \concept range_convertible_to_impl_
/// \brief The \c range_convertible_to_impl_ concept
template(typename R, typename I, typename S)(
concept (range_convertible_to_impl_)(R, I, S),
convertible_to_not_slicing_<iterator_t<R>, I> AND
convertible_to<sentinel_t<R>, S>);
/// \concept range_convertible_to_
/// \brief The \c range_convertible_to_ concept
template<typename R, typename I, typename S>
CPP_concept range_convertible_to_ =
borrowed_range<R> &&
CPP_concept_ref(detail::range_convertible_to_impl_, R, I, S);
// clang-format on
template(typename S, typename I)(
requires sentinel_for<S, I>)
constexpr bool is_sized_sentinel_() noexcept
{
return (bool)sized_sentinel_for<S, I>;
}
template<subrange_kind K, typename S, typename I>
constexpr bool store_size_() noexcept
{
return K == subrange_kind::sized && !(bool)sized_sentinel_for<S, I>;
}
} // namespace detail
/// \endcond
template< //
typename I, //
typename S = I, //
subrange_kind K = static_cast<subrange_kind>(detail::is_sized_sentinel_<S, I>())>
struct subrange;
template<typename I, typename S, subrange_kind K>
RANGES_INLINE_VAR constexpr bool enable_borrowed_range<subrange<I, S, K>> = true;
/// \cond
namespace _subrange_
{
struct adl_hook
{};
template(std::size_t N, typename I, typename S, subrange_kind K)(
requires (N == 0)) //
constexpr I get(subrange<I, S, K> const & r)
{
return r.begin();
}
template(std::size_t N, typename I, typename S, subrange_kind K)(
requires (N == 1)) //
constexpr S get(subrange<I, S, K> const & r)
{
return r.end();
}
} // namespace _subrange_
/// \endcond
template<typename I, typename S, subrange_kind K>
struct subrange
: view_interface<subrange<I, S, K>,
same_as<S, unreachable_sentinel_t>
? infinite
: K == subrange_kind::sized ? finite : unknown>
, private _subrange_::adl_hook
{
CPP_assert(input_or_output_iterator<I>);
CPP_assert(sentinel_for<S, I>);
CPP_assert(K == subrange_kind::sized || !sized_sentinel_for<S, I>);
CPP_assert(K != subrange_kind::sized || !same_as<S, unreachable_sentinel_t>);
using size_type = detail::iter_size_t<I>;
using iterator = I;
using sentinel = S;
subrange() = default;
template(typename I2)(
requires detail::convertible_to_not_slicing_<I2, I> AND
(!detail::store_size_<K, S, I>())) //
constexpr subrange(I2 && i, S s)
: data_{static_cast<I2 &&>(i), std::move(s)}
{}
template(typename I2)(
requires detail::convertible_to_not_slicing_<I2, I> AND
(detail::store_size_<K, S, I>())) //
constexpr subrange(I2 && i, S s, size_type n)
: data_{static_cast<I2 &&>(i), std::move(s), n}
{
if(RANGES_CONSTEXPR_IF((bool)random_access_iterator<I>))
{
using D = iter_difference_t<I>;
RANGES_EXPECT(n <= (size_type)std::numeric_limits<D>::max());
RANGES_EXPECT(ranges::next(first_(), (D)n) == last_());
}
}
template(typename I2)(
requires detail::convertible_to_not_slicing_<I2, I> AND
sized_sentinel_for<S, I>)
constexpr subrange(I2 && i, S s, size_type n)
: data_{static_cast<I2 &&>(i), std::move(s)}
{
RANGES_EXPECT(static_cast<size_type>(last_() - first_()) == n);
}
template(typename R)(
requires (!same_as<detail::decay_t<R>, subrange>) AND
detail::range_convertible_to_<R, I, S> AND
(!detail::store_size_<K, S, I>()))
constexpr subrange(R && r)
: subrange{ranges::begin(r), ranges::end(r)}
{}
template(typename R)(
requires (!same_as<detail::decay_t<R>, subrange>) AND
detail::range_convertible_to_<R, I, S> AND
(detail::store_size_<K, S, I>()) AND
sized_range<R>)
constexpr subrange(R && r)
: subrange{ranges::begin(r), ranges::end(r), ranges::size(r)}
{}
template(typename R)(
requires (K == subrange_kind::sized) AND
detail::range_convertible_to_<R, I, S>)
constexpr subrange(R && r, size_type n) //
: subrange{ranges::begin(r), ranges::end(r), n}
{
if(RANGES_CONSTEXPR_IF((bool)sized_range<R>))
{
RANGES_EXPECT(n == ranges::size(r));
}
}
template(typename PairLike)(
requires (!same_as<PairLike, subrange>) AND
detail::pair_like_convertible_from_<PairLike, I const &, S const &>)
constexpr operator PairLike() const
{
return PairLike(first_(), last_());
}
constexpr I begin() const noexcept(std::is_nothrow_copy_constructible<I>::value)
{
return first_();
}
constexpr S end() const noexcept(std::is_nothrow_copy_constructible<S>::value)
{
return last_();
}
constexpr bool empty() const
{
return first_() == last_();
}
CPP_member
constexpr auto size() const //
-> CPP_ret(size_type)(
requires (K == subrange_kind::sized))
{
return get_size_();
}
RANGES_NODISCARD
constexpr subrange next(iter_difference_t<I> n = 1) const
{
auto tmp = *this;
tmp.advance(n);
return tmp;
}
CPP_member
RANGES_NODISCARD constexpr auto prev(iter_difference_t<I> n = 1) const
-> CPP_ret(subrange)(
requires bidirectional_iterator<I>)
{
auto tmp = *this;
tmp.advance(-n);
return tmp;
}
constexpr subrange & advance(iter_difference_t<I> n)
{
set_size_(get_size_() -
static_cast<size_type>(n - ranges::advance(first_(), n, last_())));
return *this;
}
private:
using data_t =
meta::conditional_t< //
detail::store_size_<K, S, I>(), //
std::tuple<I, S, size_type>, //
std::tuple<I, S>>;
data_t data_;
constexpr I & first_() noexcept
{
return std::get<0>(data_);
}
constexpr const I & first_() const noexcept
{
return std::get<0>(data_);
}
constexpr S & last_() noexcept
{
return std::get<1>(data_);
}
constexpr const S & last_() const noexcept
{
return std::get<1>(data_);
}
CPP_member
constexpr auto get_size_() const //
-> CPP_ret(size_type)(
requires sized_sentinel_for<S, I>)
{
return static_cast<size_type>(last_() - first_());
}
CPP_member
constexpr auto get_size_() const noexcept //
-> CPP_ret(size_type)(
requires (detail::store_size_<K, S, I>()))
{
return std::get<2>(data_);
}
static constexpr void set_size_(...) noexcept
{}
CPP_member
constexpr auto set_size_(size_type n) noexcept //
-> CPP_ret(void)(
requires (detail::store_size_<K, S, I>()))
{
std::get<2>(data_) = n;
}
};
#if RANGES_CXX_DEDUCTION_GUIDES >= RANGES_CXX_DEDUCTION_GUIDES_17
template<typename I, typename S>
subrange(I, S) //
-> subrange<I, S>;
template(typename I, typename S)(
requires input_or_output_iterator<I> AND sentinel_for<S, I>)
subrange(I, S, detail::iter_size_t<I>)
-> subrange<I, S, subrange_kind::sized>;
template(typename R)(
requires borrowed_range<R>)
subrange(R &&) //
-> subrange<iterator_t<R>, sentinel_t<R>,
(sized_range<R> ||
sized_sentinel_for<sentinel_t<R>, iterator_t<R>>)
? subrange_kind::sized
: subrange_kind::unsized>;
template(typename R)(
requires borrowed_range<R>)
subrange(R &&, detail::iter_size_t<iterator_t<R>>)
-> subrange<iterator_t<R>, sentinel_t<R>, subrange_kind::sized>;
#endif
// in lieu of deduction guides, use make_subrange
struct make_subrange_fn
{
template<typename I, typename S>
constexpr subrange<I, S> operator()(I i, S s) const
{
return {i, s};
}
template(typename I, typename S)(
requires input_or_output_iterator<I> AND sentinel_for<S, I>)
constexpr subrange<I, S, subrange_kind::sized> //
operator()(I i, S s, detail::iter_size_t<I> n) const
{
return {i, s, n};
}
template(typename R)(
requires borrowed_range<R>)
constexpr auto operator()(R && r) const
-> subrange<iterator_t<R>, sentinel_t<R>,
(sized_range<R> || sized_sentinel_for<sentinel_t<R>, iterator_t<R>>)
? subrange_kind::sized
: subrange_kind::unsized>
{
return {(R &&) r};
}
template(typename R)(
requires borrowed_range<R>)
constexpr subrange<iterator_t<R>, sentinel_t<R>, subrange_kind::sized> //
operator()(R && r, detail::iter_size_t<iterator_t<R>> n) const
{
return {(R &&) r, n};
}
};
/// \relates make_subrange_fn
/// \ingroup group-views
RANGES_INLINE_VARIABLE(make_subrange_fn, make_subrange)
template<typename R>
using borrowed_subrange_t = detail::maybe_dangling_<R, subrange<iterator_t<R>>>;
template<typename R>
using safe_subrange_t RANGES_DEPRECATED("Use borrowed_subrange_t instead.") =
borrowed_subrange_t<R>;
namespace cpp20
{
using ranges::subrange_kind;
template(typename I, //
typename S = I, //
subrange_kind K = //
static_cast<subrange_kind>( //
detail::is_sized_sentinel_<S, I>()))(
requires input_or_output_iterator<I> AND sentinel_for<S, I> AND
(K == subrange_kind::sized || !sized_sentinel_for<S, I>)) //
using subrange = ranges::subrange<I, S, K>;
using ranges::borrowed_subrange_t;
template<typename R>
using safe_subrange_t RANGES_DEPRECATED("Use borrowed_subrange_t instead.") =
borrowed_subrange_t<R>;
} // namespace cpp20
/// @}
} // namespace ranges
RANGES_DIAGNOSTIC_PUSH
RANGES_DIAGNOSTIC_IGNORE_MISMATCHED_TAGS
namespace std
{
template<typename I, typename S, ::ranges::subrange_kind K>
struct tuple_size<::ranges::subrange<I, S, K>> : std::integral_constant<size_t, 2>
{};
template<typename I, typename S, ::ranges::subrange_kind K>
struct tuple_element<0, ::ranges::subrange<I, S, K>>
{
using type = I;
};
template<typename I, typename S, ::ranges::subrange_kind K>
struct tuple_element<1, ::ranges::subrange<I, S, K>>
{
using type = S;
};
} // namespace std
RANGES_DIAGNOSTIC_POP
#include <range/v3/detail/epilogue.hpp>
#endif