| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- /// \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_REPLACE_HPP
- #define RANGES_V3_VIEW_REPLACE_HPP
- #include <type_traits>
- #include <utility>
- #include <meta/meta.hpp>
- #include <concepts/concepts.hpp>
- #include <range/v3/range_fwd.hpp>
- #include <range/v3/functional/bind_back.hpp>
- #include <range/v3/utility/static_const.hpp>
- #include <range/v3/view/all.hpp>
- #include <range/v3/view/transform.hpp>
- #include <range/v3/view/view.hpp>
- #include <range/v3/detail/prologue.hpp>
- namespace ranges
- {
- /// \cond
- namespace detail
- {
- template<typename Val1, typename Val2>
- struct replacer_fn
- {
- private:
- Val1 old_value_;
- Val2 new_value_;
- public:
- replacer_fn() = default;
- constexpr replacer_fn(Val1 old_value, Val2 new_value)
- : old_value_(std::move(old_value))
- , new_value_(std::move(new_value))
- {}
- template<typename I>
- [[noreturn]] common_type_t<decay_t<unwrap_reference_t<Val2 const &>>,
- iter_value_t<I>> &
- operator()(copy_tag, I const &) const
- {
- RANGES_EXPECT(false);
- }
- template<typename I>
- common_reference_t<unwrap_reference_t<Val2 const &>, iter_reference_t<I>>
- operator()(I const & i) const
- {
- auto && x = *i;
- if(x == unwrap_reference(old_value_))
- return unwrap_reference(new_value_);
- return ((decltype(x) &&)x);
- }
- template<typename I>
- common_reference_t<unwrap_reference_t<Val2 const &>,
- iter_rvalue_reference_t<I>>
- operator()(move_tag, I const & i) const
- {
- auto && x = iter_move(i);
- if(x == unwrap_reference(old_value_))
- return unwrap_reference(new_value_);
- return ((decltype(x) &&)x);
- }
- };
- } // namespace detail
- /// \endcond
- /// \addtogroup group-views
- /// @{
- namespace views
- {
- struct replace_base_fn
- {
- template(typename Rng, typename Val1, typename Val2)(
- requires viewable_range<Rng> AND input_range<Rng> AND
- same_as<
- detail::decay_t<unwrap_reference_t<Val1>>,
- detail::decay_t<unwrap_reference_t<Val2>>> AND
- equality_comparable_with<
- detail::decay_t<unwrap_reference_t<Val1>>,
- range_value_t<Rng>> AND
- common_with<detail::decay_t<unwrap_reference_t<Val2 const &>>,
- range_value_t<Rng>> AND
- common_reference_with<unwrap_reference_t<Val2 const &>,
- range_reference_t<Rng>> AND
- common_reference_with<
- unwrap_reference_t<Val2 const &>,
- range_rvalue_reference_t<Rng>>)
- constexpr replace_view< //
- all_t<Rng>, //
- detail::decay_t<Val1>, //
- detail::decay_t<Val2>> //
- operator()(Rng && rng, Val1 && old_value,
- Val2 && new_value) const //
- {
- return {
- all(static_cast<Rng &&>(rng)),
- {static_cast<Val1 &&>(old_value), static_cast<Val2 &&>(new_value)}};
- }
- };
- struct replace_fn : replace_base_fn
- {
- using replace_base_fn::operator();
- template(typename Val1, typename Val2)(
- requires same_as<detail::decay_t<unwrap_reference_t<Val1>>,
- detail::decay_t<unwrap_reference_t<Val2>>>)
- constexpr auto operator()(Val1 old_value, Val2 new_value) const
- {
- return make_view_closure(bind_back(
- replace_base_fn{}, std::move(old_value), std::move(new_value)));
- }
- };
- /// \relates replace_fn
- /// \ingroup group-views
- RANGES_INLINE_VARIABLE(replace_fn, replace)
- } // namespace views
- /// @}
- } // namespace ranges
- #include <range/v3/detail/epilogue.hpp>
- #endif
|