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

374 lines
12 KiB
C++

/// \file
// Range v3 library
//
// Copyright Andrey Diduh 2019
//
// 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_DROP_LAST_HPP
#define RANGES_V3_VIEW_DROP_LAST_HPP
#include <type_traits>
#include <meta/meta.hpp>
#include <range/v3/functional/bind_back.hpp>
#include <range/v3/iterator/counted_iterator.hpp>
#include <range/v3/iterator/default_sentinel.hpp>
#include <range/v3/iterator/operations.hpp>
#include <range/v3/range/access.hpp>
#include <range/v3/range/concepts.hpp>
#include <range/v3/range/primitives.hpp>
#include <range/v3/utility/optional.hpp>
#include <range/v3/utility/static_const.hpp>
#include <range/v3/view/adaptor.hpp>
#include <range/v3/view/all.hpp>
#include <range/v3/view/interface.hpp>
#include <range/v3/view/view.hpp>
#include <range/v3/detail/prologue.hpp>
namespace ranges
{
/// \addtogroup group-views
/// @{
/// \cond
namespace detail
{
namespace drop_last_view
{
template<typename Rng>
range_size_t<Rng> get_size(Rng & rng, range_difference_t<Rng> n_)
{
RANGES_EXPECT(n_ >= 0);
range_size_t<Rng> const initial_size = ranges::size(rng);
range_size_t<Rng> const n = static_cast<range_size_t<Rng>>(n_);
return initial_size > n ? initial_size - n : 0;
}
template(typename Rng)(
requires random_access_range<Rng> AND sized_range<Rng>)
iterator_t<Rng> get_end(Rng & rng, range_difference_t<Rng> n, int)
{
return begin(rng) + static_cast<range_difference_t<Rng>>(
drop_last_view::get_size(rng, n));
}
template(typename Rng)(
requires bidirectional_range<Rng> AND common_range<Rng>)
iterator_t<Rng> get_end(Rng & rng, range_difference_t<Rng> n, long)
{
return prev(end(rng), n, begin(rng));
}
enum class mode_enum
{
bidi,
forward,
sized,
invalid
};
template<mode_enum Mode>
using mode_t = std::integral_constant<mode_enum, Mode>;
using mode_bidi = mode_t<mode_enum::bidi>;
using mode_forward = mode_t<mode_enum::forward>;
using mode_sized = mode_t<mode_enum::sized>;
using mode_invalid = mode_t<mode_enum::invalid>;
template<typename Rng>
constexpr mode_enum get_mode() noexcept
{
// keep range bound
// Sized Bidi O(N)
return (random_access_range<Rng> && view_<Rng> && sized_range<Rng>) ||
(bidirectional_range<Rng> && view_<Rng> &&
common_range<Rng>)
? mode_enum::bidi //
: sized_range<Rng> && view_<Rng> //
? mode_enum::sized //
: forward_range<Rng> && view_<Rng> //
? mode_enum::forward //
: mode_enum::invalid; //
// max performance
// Sized Bidi O(1)
// Sized Bidi use mode::sized instead of mode::bidi - thus become unbound.
/*return (random_access_range<Rng> && view_<Rng> && sized_range<Rng> &&
view_<Rng>) || (bidirectional_range<Rng> && view_<Rng> &&
common_range<Rng> && view_<Rng>) ? mode::bidi : sized_range<Rng> &&
view_<Rng> ? mode::sized : bidirectional_range<Rng> && view_<Rng> &&
common_range<Rng> && view_<Rng> ? mode::bidi : forward_range<Rng> &&
view_<Rng> ? mode::forward : mode::invalid;*/
}
template<typename Rng>
using mode_of = mode_t<drop_last_view::get_mode<Rng>()>;
} // namespace drop_last_view
} // namespace detail
/// \endcond
template<typename Rng, typename = detail::drop_last_view::mode_of<Rng>>
struct drop_last_view
{};
template<typename Rng>
struct drop_last_view<Rng, detail::drop_last_view::mode_bidi>
: view_interface<drop_last_view<Rng, detail::drop_last_view::mode_bidi>,
is_finite<Rng>::value
? finite
: range_cardinality<Rng>::value> // finite at best
{
CPP_assert(
(random_access_range<Rng> && view_<Rng> && sized_range<Rng>) ||
(bidirectional_range<Rng> && view_<Rng> && common_range<Rng>));
private:
friend range_access;
using difference_t = range_difference_t<Rng>;
Rng rng_;
difference_t n_;
detail::non_propagating_cache<iterator_t<Rng>> end_;
public:
drop_last_view() = default;
constexpr drop_last_view(Rng rng, difference_t n)
: rng_(std::move(rng))
, n_(n)
{
RANGES_EXPECT(n >= 0);
}
iterator_t<Rng> begin()
{
return ranges::begin(rng_);
}
sentinel_t<Rng> end()
{
if(!end_)
end_ = detail::drop_last_view::get_end(rng_, n_, 0);
return *end_;
}
template(typename CRng = Rng const)(
requires random_access_range<CRng> AND sized_range<CRng>)
iterator_t<CRng> begin() const
{
return ranges::begin(rng_);
}
template(typename CRng = Rng const)(
requires random_access_range<CRng> AND sized_range<CRng>)
iterator_t<CRng> end() const
{
return detail::drop_last_view::get_end(rng_, n_, 0);
}
CPP_auto_member
auto CPP_fun(size)()(
requires sized_range<Rng>)
{
return detail::drop_last_view::get_size(rng_, n_);
}
CPP_auto_member
auto CPP_fun(size)()(const //
requires sized_range<Rng const>)
{
return detail::drop_last_view::get_size(rng_, n_);
}
Rng & base()
{
return rng_;
}
Rng const & base() const
{
return rng_;
}
};
template<typename Rng>
struct drop_last_view<Rng, detail::drop_last_view::mode_forward>
: view_adaptor<drop_last_view<Rng, detail::drop_last_view::mode_forward>, Rng,
is_finite<Rng>::value
? finite
: range_cardinality<Rng>::value> // finite at best (but
// unknown is expected)
{
CPP_assert(forward_range<Rng> && view_<Rng>);
private:
friend range_access;
using difference_t = range_difference_t<Rng>;
difference_t n_;
detail::non_propagating_cache<iterator_t<Rng>> probe_begin;
struct adaptor : adaptor_base
{
iterator_t<Rng> probe_;
adaptor() = default;
adaptor(iterator_t<Rng> probe_first)
: probe_(std::move(probe_first))
{}
void next(iterator_t<Rng> & it)
{
++it;
++probe_;
}
};
struct sentinel_adaptor : adaptor_base
{
template<typename I, typename S>
bool empty(I const &, adaptor const & ia, S const & s) const
{
return ia.probe_ == s;
}
};
adaptor begin_adaptor()
{
if(!probe_begin)
probe_begin = next(begin(this->base()), n_, end(this->base()));
return {*probe_begin};
}
sentinel_adaptor end_adaptor()
{
return {};
}
public:
drop_last_view() = default;
constexpr drop_last_view(Rng rng, difference_t n)
: drop_last_view::view_adaptor(std::move(rng))
, n_(n)
{
RANGES_EXPECT(n >= 0);
}
CPP_auto_member
auto CPP_fun(size)()(
requires sized_range<Rng>)
{
return detail::drop_last_view::get_size(this->base(), n_);
}
CPP_auto_member
auto CPP_fun(size)()(const //
requires sized_range<Rng const>)
{
return detail::drop_last_view::get_size(this->base(), n_);
}
};
template<typename Rng>
struct drop_last_view<Rng, detail::drop_last_view::mode_sized>
: view_interface<drop_last_view<Rng, detail::drop_last_view::mode_sized>, finite>
{
CPP_assert(sized_range<Rng> && view_<Rng>);
private:
friend range_access;
using difference_t = range_difference_t<Rng>;
Rng rng_;
difference_t n_;
public:
drop_last_view() = default;
constexpr drop_last_view(Rng rng, difference_t n)
: rng_(std::move(rng))
, n_(n)
{
RANGES_EXPECT(n >= 0);
}
counted_iterator<iterator_t<Rng>> begin()
{
return {ranges::begin(rng_), static_cast<difference_t>(size())};
}
template(typename CRng = Rng const)(
requires sized_range<CRng>)
counted_iterator<iterator_t<CRng>> begin() const
{
return {ranges::begin(rng_), static_cast<difference_t>(size())};
}
default_sentinel_t end() const
{
return {};
}
range_size_t<Rng> size()
{
return detail::drop_last_view::get_size(this->base(), n_);
}
CPP_auto_member
auto CPP_fun(size)()(const //
requires sized_range<Rng const>)
{
return detail::drop_last_view::get_size(this->base(), n_);
}
Rng & base()
{
return rng_;
}
Rng const & base() const
{
return rng_;
}
};
template<typename Rng, typename T>
RANGES_INLINE_VAR constexpr bool enable_borrowed_range<drop_last_view<Rng, T>> = //
enable_borrowed_range<Rng>;
#if RANGES_CXX_DEDUCTION_GUIDES >= RANGES_CXX_DEDUCTION_GUIDES_17
template<typename Rng>
drop_last_view(Rng &&, range_difference_t<Rng>)
-> drop_last_view<views::all_t<Rng>>;
#endif
namespace views
{
struct drop_last_base_fn
{
template(typename Rng)(
requires sized_range<Rng> || forward_range<Rng>)
constexpr auto operator()(Rng && rng, range_difference_t<Rng> n) const
-> drop_last_view<all_t<Rng>>
{
return {all(static_cast<Rng &&>(rng)), n};
}
};
struct drop_last_fn : drop_last_base_fn
{
using drop_last_base_fn::operator();
template(typename Int)(
requires detail::integer_like_<Int>)
constexpr auto operator()(Int n) const
{
return make_view_closure(bind_back(drop_last_base_fn{}, n));
}
};
RANGES_INLINE_VARIABLE(drop_last_fn, drop_last)
} // namespace views
/// @}
} // namespace ranges
#include <range/v3/detail/epilogue.hpp>
#include <range/v3/detail/satisfy_boost_range.hpp>
RANGES_SATISFY_BOOST_RANGE(::ranges::drop_last_view)
#endif