// Copyright (C) 2023 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QCOMPARE_H #error "Do not include qcomparehelpers.h directly. Use qcompare.h instead." #endif #ifndef QCOMPAREHELPERS_H #define QCOMPAREHELPERS_H #if 0 #pragma qt_no_master_include #pragma qt_sync_skip_header_check #pragma qt_sync_stop_processing #endif #include #include #include #ifdef __cpp_lib_three_way_comparison #include #endif #include #include // std::less QT_BEGIN_NAMESPACE class QPartialOrdering; namespace QtOrderingPrivate { #ifdef __cpp_lib_three_way_comparison template struct StdOrdering; template struct QtOrdering; #define QT_STD_MAP(x) \ template <> struct StdOrdering< Qt::x##_ordering> : q20::type_identity {};\ template <> struct StdOrdering : q20::type_identity {};\ template <> struct QtOrdering : q20::type_identity< Qt::x##_ordering> {};\ template <> struct QtOrdering< Qt::x##_ordering> : q20::type_identity< Qt::x##_ordering> {};\ /* end */ QT_STD_MAP(partial) QT_STD_MAP(weak) QT_STD_MAP(strong) #undef QT_STD_MAP template <> struct StdOrdering : q20::type_identity {}; template <> struct QtOrdering : q20::type_identity< Qt::partial_ordering> {}; template constexpr auto to_std(In in) noexcept -> typename QtOrderingPrivate::StdOrdering::type { return in; } template constexpr auto to_Qt(In in) noexcept -> typename QtOrderingPrivate::QtOrdering::type { return in; } #endif // __cpp_lib_three_way_comparison } // namespace QtOrderingPrivate /* For all the macros these parameter names are used: * LeftType - the type of the left operand of the comparison * RightType - the type of the right operand of the comparison * Constexpr - must be either constexpr or empty. Defines whether the operator is constexpr or not * Attributes - an optional list of attributes. For example, pass \c QT_ASCII_CAST_WARN when defining comparisons between C-style string and an encoding-aware string type. The macros require two helper functions. For operators to be constexpr, these must be constexpr, too. Additionally, other attributes (like Q__EXPORT, Q_DECL_CONST_FUNCTION, etc) can be applied to them. Aside from that, their declaration should match: bool comparesEqual(LeftType, RightType) noexcept; ReturnType compareThreeWay(LeftType, RightType) noexcept; The ReturnType can be one of Qt::{partial,weak,strong}_ordering. The actual type depends on the macro being used. It makes sense to define the helper functions as hidden friends of the class, so that they could be found via ADL, and don't participate in unintended implicit conversions. */ // Seems that qdoc uses C++20 even when Qt is compiled in C++17 mode. // Or at least it defines __cpp_lib_three_way_comparison. // Let qdoc see only the C++17 operators for now, because that's what our docs // currently describe. #if defined(__cpp_lib_three_way_comparison) && !defined(Q_QDOC) // C++20 - provide operator==() for equality, and operator<=>() for ordering #define QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr bool operator==(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(comparesEqual(lhs, rhs))) \ { return comparesEqual(lhs, rhs); } #define QT_DECLARE_3WAY_HELPER_STRONG(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::strong_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_3WAY_HELPER_WEAK(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::weak_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_3WAY_HELPER_PARTIAL(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::partial_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_ORDERING_OPERATORS_HELPER(OrderingType, LeftType, RightType, Constexpr, \ Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_3WAY_HELPER_ ## OrderingType (LeftType, RightType, Constexpr, Attributes) #ifdef Q_COMPILER_LACKS_THREE_WAY_COMPARE_SYMMETRY // define reversed versions of the operators manually, because buggy MSVC versions do not do it #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr bool operator==(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(comparesEqual(rhs, lhs))) \ { return comparesEqual(rhs, lhs); } #define QT_DECLARE_REVERSED_3WAY_HELPER_STRONG(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::strong_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { \ const auto r = compareThreeWay(rhs, lhs); \ if (r > 0) return std::strong_ordering::less; \ if (r < 0) return std::strong_ordering::greater; \ return r; \ } #define QT_DECLARE_REVERSED_3WAY_HELPER_WEAK(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::weak_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { \ const auto r = compareThreeWay(rhs, lhs); \ if (r > 0) return std::weak_ordering::less; \ if (r < 0) return std::weak_ordering::greater; \ return r; \ } #define QT_DECLARE_REVERSED_3WAY_HELPER_PARTIAL(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr std::partial_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { \ const auto r = compareThreeWay(rhs, lhs); \ if (r > 0) return std::partial_ordering::less; \ if (r < 0) return std::partial_ordering::greater; \ return r; \ } #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_REVERSED_3WAY_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, Attributes) #else // dummy macros for C++17 compatibility, reversed operators are generated by the compiler #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Attributes) #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Attributes) #endif // Q_COMPILER_LACKS_THREE_WAY_COMPARE_SYMMETRY #else // C++17 - provide operator==() and operator!=() for equality, // and all 4 comparison operators for ordering #define QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr bool operator==(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(comparesEqual(lhs, rhs))) \ { return comparesEqual(lhs, rhs); } \ Attributes \ friend Constexpr bool operator!=(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(comparesEqual(lhs, rhs))) \ { return !comparesEqual(lhs, rhs); } // Helpers for reversed comparison, using the existing comparesEqual() function. #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Attributes) \ Attributes \ friend Constexpr bool operator==(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(comparesEqual(rhs, lhs))) \ { return comparesEqual(rhs, lhs); } \ Attributes \ friend Constexpr bool operator!=(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(comparesEqual(rhs, lhs))) \ { return !comparesEqual(rhs, lhs); } #define QT_DECLARE_ORDERING_HELPER_TEMPLATE(OrderingType, LeftType, RightType, Constexpr, \ Attributes) \ Attributes \ friend Constexpr bool operator<(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { return compareThreeWay(lhs, rhs) < 0; } \ Attributes \ friend Constexpr bool operator>(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { return compareThreeWay(lhs, rhs) > 0; } \ Attributes \ friend Constexpr bool operator<=(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { return compareThreeWay(lhs, rhs) <= 0; } \ Attributes \ friend Constexpr bool operator>=(LeftType const &lhs, RightType const &rhs) \ noexcept(noexcept(compareThreeWay(lhs, rhs))) \ { return compareThreeWay(lhs, rhs) >= 0; } #define QT_DECLARE_ORDERING_HELPER_PARTIAL(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::partial_ordering, LeftType, RightType, Constexpr, \ Attributes) #define QT_DECLARE_ORDERING_HELPER_WEAK(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::weak_ordering, LeftType, RightType, Constexpr, \ Attributes) #define QT_DECLARE_ORDERING_HELPER_STRONG(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::strong_ordering, LeftType, RightType, Constexpr, \ Attributes) #define QT_DECLARE_ORDERING_OPERATORS_HELPER(OrderingString, LeftType, RightType, Constexpr, \ Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_ORDERING_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, Attributes) // Helpers for reversed ordering, using the existing compareThreeWay() function. #define QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(OrderingType, LeftType, RightType, Constexpr, \ Attributes) \ Attributes \ friend Constexpr bool operator<(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { return compareThreeWay(rhs, lhs) > 0; } \ Attributes \ friend Constexpr bool operator>(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { return compareThreeWay(rhs, lhs) < 0; } \ Attributes \ friend Constexpr bool operator<=(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { return compareThreeWay(rhs, lhs) >= 0; } \ Attributes \ friend Constexpr bool operator>=(RightType const &lhs, LeftType const &rhs) \ noexcept(noexcept(compareThreeWay(rhs, lhs))) \ { return compareThreeWay(rhs, lhs) <= 0; } #define QT_DECLARE_REVERSED_ORDERING_HELPER_PARTIAL(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::partial_ordering, LeftType, RightType, \ Constexpr, Attributes) #define QT_DECLARE_REVERSED_ORDERING_HELPER_WEAK(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::weak_ordering, LeftType, RightType, \ Constexpr, Attributes) #define QT_DECLARE_REVERSED_ORDERING_HELPER_STRONG(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::strong_ordering, LeftType, RightType, \ Constexpr, Attributes) #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, \ Attributes) #endif // __cpp_lib_three_way_comparison /* Public API starts here */ // Equality operators #define QT_DECLARE_EQUALITY_COMPARABLE_1(Type) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(Type, Type, /* non-constexpr */, /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_2(LeftType, RightType) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, \ /* no attributes */) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_3(LeftType, RightType, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ Attributes) #define Q_DECLARE_EQUALITY_COMPARABLE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_EQUALITY_COMPARABLE, __VA_ARGS__) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_1(Type) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(Type, Type, constexpr, /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, constexpr, /* no attributes */) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, constexpr, \ /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, constexpr, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, constexpr, Attributes) #define Q_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE, __VA_ARGS__) // Partial ordering operators #define QT_DECLARE_PARTIALLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, Type, Type, /* non-constexpr */, \ /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, Attributes) #define Q_DECLARE_PARTIALLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_PARTIALLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, Type, Type, constexpr, /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, constexpr, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, constexpr, \ /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, constexpr, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, constexpr, \ Attributes) #define Q_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) // Weak ordering operators #define QT_DECLARE_WEAKLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, Type, Type, /* non-constexpr */, /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ Attributes) #define Q_DECLARE_WEAKLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_WEAKLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, Type, Type, constexpr, /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, constexpr, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, constexpr, \ /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, constexpr, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, constexpr, \ Attributes) #define Q_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) // Strong ordering operators #define QT_DECLARE_STRONGLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, Type, Type, /* non-constexpr */, \ /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, Attributes) #define Q_DECLARE_STRONGLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_STRONGLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, Type, Type, constexpr, /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, constexpr, \ /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, constexpr, \ /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, constexpr, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, constexpr, \ Attributes) #define Q_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) namespace QtPrivate { template constexpr bool IsIntegralType_v = std::numeric_limits>::is_specialized && std::numeric_limits>::is_integer; template constexpr bool IsFloatType_v = std::is_floating_point_v; #if QFLOAT16_IS_NATIVE template <> constexpr bool IsFloatType_v = true; #endif } // namespace QtPrivate namespace Qt { template using if_integral = std::enable_if_t, bool>; template using if_floating_point = std::enable_if_t, bool>; template using if_compatible_pointers = std::enable_if_t, std::is_base_of, std::is_base_of>, bool>; template using if_enum = std::enable_if_t, bool>; template = true, if_integral = true> constexpr Qt::strong_ordering compareThreeWay(LeftInt lhs, RightInt rhs) noexcept { static_assert(std::is_signed_v == std::is_signed_v, "Qt::compareThreeWay() does not allow mixed-sign comparison."); #ifdef __cpp_lib_three_way_comparison return lhs <=> rhs; #else if (lhs == rhs) return Qt::strong_ordering::equivalent; else if (lhs < rhs) return Qt::strong_ordering::less; else return Qt::strong_ordering::greater; #endif // __cpp_lib_three_way_comparison } template = true, if_floating_point = true> constexpr Qt::partial_ordering compareThreeWay(LeftFloat lhs, RightFloat rhs) noexcept { QT_WARNING_PUSH QT_WARNING_DISABLE_FLOAT_COMPARE #ifdef __cpp_lib_three_way_comparison return lhs <=> rhs; #else if (lhs < rhs) return Qt::partial_ordering::less; else if (lhs > rhs) return Qt::partial_ordering::greater; else if (lhs == rhs) return Qt::partial_ordering::equivalent; else return Qt::partial_ordering::unordered; #endif // __cpp_lib_three_way_comparison QT_WARNING_POP } template = true, if_floating_point = true> constexpr Qt::partial_ordering compareThreeWay(IntType lhs, FloatType rhs) noexcept { return compareThreeWay(FloatType(lhs), rhs); } template = true, if_integral = true> constexpr Qt::partial_ordering compareThreeWay(FloatType lhs, IntType rhs) noexcept { return compareThreeWay(lhs, FloatType(rhs)); } template = true> constexpr Qt::strong_ordering compareThreeWay(const LeftType *lhs, const RightType *rhs) noexcept { #ifdef __cpp_lib_three_way_comparison return std::compare_three_way{}(lhs, rhs); #else if (lhs == rhs) return Qt::strong_ordering::equivalent; else if (std::less<>{}(lhs, rhs)) return Qt::strong_ordering::less; else return Qt::strong_ordering::greater; #endif // __cpp_lib_three_way_comparison } template constexpr Qt::strong_ordering compareThreeWay(const T *lhs, std::nullptr_t rhs) noexcept { return compareThreeWay(lhs, static_cast(rhs)); } template constexpr Qt::strong_ordering compareThreeWay(std::nullptr_t lhs, const T *rhs) noexcept { return compareThreeWay(static_cast(lhs), rhs); } template = true> constexpr Qt::strong_ordering compareThreeWay(Enum lhs, Enum rhs) noexcept { return compareThreeWay(qToUnderlying(lhs), qToUnderlying(rhs)); } } // namespace Qt QT_END_NAMESPACE #endif // QCOMPAREHELPERS_H