/// \file // Range v3 library // // Copyright Eric Niebler 2014 // // Use, modification and distribution is subject to the // Boost Software License, Version 0.0. (See accompanying // file LICENSE_0_0.txt or copy at // http://www.boost.org/LICENSE_0_0.txt) // // Project home: https://github.com/ericniebler/range-v3 // #ifndef RANGES_V3_ALGORITHM_EQUAL_HPP #define RANGES_V3_ALGORITHM_EQUAL_HPP #include #include #include #include #include #include #include #include #include #include namespace ranges { inline namespace v3 { /// \addtogroup group-algorithms /// @{ struct equal_fn { private: template bool nocheck(I0 begin0, S0 end0, I1 begin1, S1 end1, C pred, P0 proj0, P1 proj1) const { for(; begin0 != end0 && begin1 != end1; ++begin0, ++begin1) if(!invoke(pred, invoke(proj0, *begin0), invoke(proj1, *begin1))) return false; return begin0 == end0 && begin1 == end1; } public: template() && Comparable() )> bool operator()(I0 begin0, S0 end0, I1 begin1, C pred = C{}, P0 proj0 = P0{}, P1 proj1 = P1{}) const { for(; begin0 != end0; ++begin0, ++begin1) if(!invoke(pred, invoke(proj0, *begin0), invoke(proj1, *begin1))) return false; return true; } template() && Sentinel() && Comparable() )> bool operator()(I0 begin0, S0 end0, I1 begin1, S1 end1, C pred = C{}, P0 proj0 = P0{}, P1 proj1 = P1{}) const { if(SizedSentinel() && SizedSentinel()) if(distance(begin0, end0) != distance(begin1, end1)) return false; return this->nocheck(std::move(begin0), std::move(end0), std::move(begin1), std::move(end1), std::move(pred), std::move(proj0), std::move(proj1)); } template, typename I1 = uncvref_t, CONCEPT_REQUIRES_( Range() && Iterator() && Comparable() )> bool operator()(Rng0 && rng0, I1Ref && begin1, C pred = C{}, P0 proj0 = P0{}, P1 proj1 = P1{}) const { return (*this)(begin(rng0), end(rng0), (I1Ref &&) begin1, std::move(pred), std::move(proj0), std::move(proj1)); } template, typename I1 = iterator_t, CONCEPT_REQUIRES_( Range() && Range() && Comparable() )> bool operator()(Rng0 && rng0, Rng1 && rng1, C pred = C{}, P0 proj0 = P0{}, P1 proj1 = P1{}) const { if(SizedRange() && SizedRange()) if(distance(rng0) != distance(rng1)) return false; return this->nocheck(begin(rng0), end(rng0), begin(rng1), end(rng1), std::move(pred), std::move(proj0), std::move(proj1)); } }; /// \sa `equal_fn` /// \ingroup group-algorithms RANGES_INLINE_VARIABLE(with_braced_init_args, equal) /// @} } // namespace v3 } // namespace ranges #endif // include guard