| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 | // Copyright (c) 2016-2024 Antony Polukhin//// Distributed under 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)#ifndef BOOST_PFR_OPS_HPP#define BOOST_PFR_OPS_HPP#pragma once#include <boost/pfr/detail/config.hpp>#include <boost/pfr/detail/detectors.hpp>#include <boost/pfr/ops_fields.hpp>/// \file boost/pfr/ops.hpp/// Contains comparison and hashing functions./// If type is comparable using its own operator or its conversion operator, then the types operator is used. Otherwise/// the operation is done via corresponding function from boost/pfr/ops.hpp header.////// \b Example:/// \code///     #include <boost/pfr/ops.hpp>///     struct comparable_struct {      // No operators defined for that structure///         int i; short s; char data[7]; bool bl; int a,b,c,d,e,f;///     };///     // ...//////     comparable_struct s1 {0, 1, "Hello", false, 6,7,8,9,10,11};///     comparable_struct s2 {0, 1, "Hello", false, 6,7,8,9,10,11111};///     assert(boost::pfr::lt(s1, s2));/// \endcode////// \podops for other ways to define operators and more details.////// \b Synopsis:namespace boost { namespace pfr {namespace detail {///////////////////// Helper typedefs that are used by all the ops    template <template <class, class> class Detector, class T, class U>    using enable_not_comp_base_t = std::enable_if_t<        not_appliable<Detector, T const&, U const&>::value,        bool    >;    template <template <class, class> class Detector, class T, class U>    using enable_comp_base_t = std::enable_if_t<        !not_appliable<Detector, T const&, U const&>::value,        bool    >;///////////////////// std::enable_if_t like functions that enable only if types do not support operation    template <class T, class U> using enable_not_eq_comp_t = enable_not_comp_base_t<comp_eq_detector, T, U>;    template <class T, class U> using enable_not_ne_comp_t = enable_not_comp_base_t<comp_ne_detector, T, U>;    template <class T, class U> using enable_not_lt_comp_t = enable_not_comp_base_t<comp_lt_detector, T, U>;    template <class T, class U> using enable_not_le_comp_t = enable_not_comp_base_t<comp_le_detector, T, U>;    template <class T, class U> using enable_not_gt_comp_t = enable_not_comp_base_t<comp_gt_detector, T, U>;    template <class T, class U> using enable_not_ge_comp_t = enable_not_comp_base_t<comp_ge_detector, T, U>;    template <class T> using enable_not_hashable_t = std::enable_if_t<        not_appliable<hash_detector, const T&, const T&>::value,        std::size_t    >;///////////////////// std::enable_if_t like functions that enable only if types do support operation    template <class T, class U> using enable_eq_comp_t = enable_comp_base_t<comp_eq_detector, T, U>;    template <class T, class U> using enable_ne_comp_t = enable_comp_base_t<comp_ne_detector, T, U>;    template <class T, class U> using enable_lt_comp_t = enable_comp_base_t<comp_lt_detector, T, U>;    template <class T, class U> using enable_le_comp_t = enable_comp_base_t<comp_le_detector, T, U>;    template <class T, class U> using enable_gt_comp_t = enable_comp_base_t<comp_gt_detector, T, U>;    template <class T, class U> using enable_ge_comp_t = enable_comp_base_t<comp_ge_detector, T, U>;    template <class T> using enable_hashable_t = std::enable_if_t<        !not_appliable<hash_detector, const T&, const T&>::value,        std::size_t    >;} // namespace detail/// \brief Compares lhs and rhs for equality using their own comparison and conversion operators; if no operators available returns \forcedlink{eq_fields}(lhs, rhs).////// \returns true if lhs is equal to rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_eq_comp_t<T, U> eq(const T& lhs, const U& rhs) noexcept {    return boost::pfr::eq_fields(lhs, rhs);}/// \overload eqtemplate <class T, class U>constexpr detail::enable_eq_comp_t<T, U> eq(const T& lhs, const U& rhs) {    return lhs == rhs;}/// \brief Compares lhs and rhs for inequality using their own comparison and conversion operators; if no operators available returns \forcedlink{ne_fields}(lhs, rhs).////// \returns true if lhs is not equal to rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_ne_comp_t<T, U> ne(const T& lhs, const U& rhs) noexcept {    return boost::pfr::ne_fields(lhs, rhs);}/// \overload netemplate <class T, class U>constexpr detail::enable_ne_comp_t<T, U> ne(const T& lhs, const U& rhs) {    return lhs != rhs;}/// \brief Compares lhs and rhs for less-than using their own comparison and conversion operators; if no operators available returns \forcedlink{lt_fields}(lhs, rhs).////// \returns true if lhs is less than rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_lt_comp_t<T, U> lt(const T& lhs, const U& rhs) noexcept {    return boost::pfr::lt_fields(lhs, rhs);}/// \overload lttemplate <class T, class U>constexpr detail::enable_lt_comp_t<T, U> lt(const T& lhs, const U& rhs) {    return lhs < rhs;}/// \brief Compares lhs and rhs for greater-than using their own comparison and conversion operators; if no operators available returns \forcedlink{lt_fields}(lhs, rhs).////// \returns true if lhs is greater than rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_gt_comp_t<T, U> gt(const T& lhs, const U& rhs) noexcept {    return boost::pfr::gt_fields(lhs, rhs);}/// \overload gttemplate <class T, class U>constexpr detail::enable_gt_comp_t<T, U> gt(const T& lhs, const U& rhs) {    return lhs > rhs;}/// \brief Compares lhs and rhs for less-equal using their own comparison and conversion operators; if no operators available returns \forcedlink{le_fields}(lhs, rhs).////// \returns true if lhs is less or equal to rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_le_comp_t<T, U> le(const T& lhs, const U& rhs) noexcept {    return boost::pfr::le_fields(lhs, rhs);}/// \overload letemplate <class T, class U>constexpr detail::enable_le_comp_t<T, U> le(const T& lhs, const U& rhs) {    return lhs <= rhs;}/// \brief Compares lhs and rhs for greater-equal using their own comparison and conversion operators; if no operators available returns \forcedlink{ge_fields}(lhs, rhs).////// \returns true if lhs is greater or equal to rhs; false otherwisetemplate <class T, class U>constexpr detail::enable_not_ge_comp_t<T, U> ge(const T& lhs, const U& rhs) noexcept {    return boost::pfr::ge_fields(lhs, rhs);}/// \overload getemplate <class T, class U>constexpr detail::enable_ge_comp_t<T, U> ge(const T& lhs, const U& rhs) {    return lhs >= rhs;}/// \brief Hashes value using its own std::hash specialization; if no std::hash specialization available returns \forcedlink{hash_fields}(value).////// \returns std::size_t with hash of the valuetemplate <class T>constexpr detail::enable_not_hashable_t<T> hash_value(const T& value) noexcept {    return boost::pfr::hash_fields(value);}/// \overload hash_valuetemplate <class T>constexpr detail::enable_hashable_t<T> hash_value(const T& value) {    return std::hash<T>{}(value);}}} // namespace boost::pfr#endif // BOOST_PFR_OPS_HPP
 |