/// \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_VIEW_DROP_WHILE_HPP #define RANGES_V3_VIEW_DROP_WHILE_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace ranges { /// \addtogroup group-views /// @{ template struct drop_while_view : view_interface, is_finite::value ? finite : unknown> { private: Rng rng_; RANGES_NO_UNIQUE_ADDRESS semiregular_box_t pred_; detail::non_propagating_cache> begin_; iterator_t get_begin_() { if(!begin_) begin_ = find_if_not(rng_, std::ref(pred_)); return *begin_; } public: drop_while_view() = default; drop_while_view(Rng rng, Pred pred) : rng_(std::move(rng)) , pred_(std::move(pred)) {} iterator_t begin() { return get_begin_(); } sentinel_t end() { return ranges::end(rng_); } Rng base() const { return rng_; } }; // unlike take_while_view, drop_while_view is transparently safe because we only // need the predicate to find begin() template RANGES_INLINE_VAR constexpr bool enable_borrowed_range> = enable_borrowed_range; #if RANGES_CXX_DEDUCTION_GUIDES >= RANGES_CXX_DEDUCTION_GUIDES_17 template(typename Rng, typename Fun)( requires copy_constructible) drop_while_view(Rng &&, Fun) -> drop_while_view, Fun>; #endif template RANGES_INLINE_VAR constexpr bool disable_sized_range> = true; namespace views { struct drop_while_base_fn { template(typename Rng, typename Pred)( requires viewable_range AND input_range AND indirect_unary_predicate>) auto operator()(Rng && rng, Pred pred) const -> drop_while_view, Pred> { return {all(static_cast(rng)), std::move(pred)}; } template(typename Rng, typename Pred, typename Proj)( requires viewable_range AND input_range AND indirect_unary_predicate, iterator_t>) auto operator()(Rng && rng, Pred pred, Proj proj) const -> drop_while_view, composed> { return {all(static_cast(rng)), compose(std::move(pred), std::move(proj))}; } }; struct drop_while_bind_fn { template constexpr auto operator()(Pred pred) const // TODO: underconstrained { return make_view_closure( bind_back(drop_while_base_fn{}, std::move(pred))); } template(typename Pred, typename Proj)( requires (!range)) // TODO: underconstrained constexpr auto operator()(Pred && pred, Proj proj) const { return make_view_closure(bind_back( drop_while_base_fn{}, static_cast(pred), std::move(proj))); } }; struct RANGES_EMPTY_BASES drop_while_fn : drop_while_base_fn, drop_while_bind_fn { using drop_while_base_fn::operator(); using drop_while_bind_fn::operator(); }; /// \relates drop_while_fn /// \ingroup group-views RANGES_INLINE_VARIABLE(drop_while_fn, drop_while) } // namespace views namespace cpp20 { namespace views { using ranges::views::drop_while; } template(typename Rng, typename Pred)( requires viewable_range AND input_range AND indirect_unary_predicate>) using drop_while_view = ranges::drop_while_view; } // namespace cpp20 /// @} } // namespace ranges #include #include RANGES_SATISFY_BOOST_RANGE(::ranges::drop_while_view) #endif