143 lines
5.3 KiB
C++
143 lines
5.3 KiB
C++
|
/*!
|
||
|
@file
|
||
|
Defines `boost::hana::less`.
|
||
|
|
||
|
@copyright Louis Dionne 2013-2016
|
||
|
Distributed under the Boost Software License, Version 1.0.
|
||
|
(See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
|
||
|
*/
|
||
|
|
||
|
#ifndef BOOST_HANA_LESS_HPP
|
||
|
#define BOOST_HANA_LESS_HPP
|
||
|
|
||
|
#include <boost/hana/fwd/less.hpp>
|
||
|
|
||
|
#include <boost/hana/and.hpp>
|
||
|
#include <boost/hana/bool.hpp>
|
||
|
#include <boost/hana/concept/constant.hpp>
|
||
|
#include <boost/hana/concept/orderable.hpp>
|
||
|
#include <boost/hana/concept/product.hpp>
|
||
|
#include <boost/hana/concept/sequence.hpp>
|
||
|
#include <boost/hana/config.hpp>
|
||
|
#include <boost/hana/core/common.hpp>
|
||
|
#include <boost/hana/core/to.hpp>
|
||
|
#include <boost/hana/core/dispatch.hpp>
|
||
|
#include <boost/hana/detail/concepts.hpp>
|
||
|
#include <boost/hana/detail/has_common_embedding.hpp>
|
||
|
#include <boost/hana/detail/nested_than.hpp> // required by fwd decl
|
||
|
#include <boost/hana/equal.hpp>
|
||
|
#include <boost/hana/first.hpp>
|
||
|
#include <boost/hana/if.hpp>
|
||
|
#include <boost/hana/less_equal.hpp>
|
||
|
#include <boost/hana/lexicographical_compare.hpp>
|
||
|
#include <boost/hana/or.hpp>
|
||
|
#include <boost/hana/second.hpp>
|
||
|
#include <boost/hana/value.hpp>
|
||
|
|
||
|
|
||
|
BOOST_HANA_NAMESPACE_BEGIN
|
||
|
//! @cond
|
||
|
template <typename X, typename Y>
|
||
|
constexpr auto less_t::operator()(X&& x, Y&& y) const {
|
||
|
using T = typename hana::tag_of<X>::type;
|
||
|
using U = typename hana::tag_of<Y>::type;
|
||
|
using Less = BOOST_HANA_DISPATCH_IF(decltype(less_impl<T, U>{}),
|
||
|
hana::Orderable<T>::value &&
|
||
|
hana::Orderable<U>::value &&
|
||
|
!is_default<less_impl<T, U>>::value
|
||
|
);
|
||
|
|
||
|
#ifndef BOOST_HANA_CONFIG_DISABLE_CONCEPT_CHECKS
|
||
|
static_assert(hana::Orderable<T>::value,
|
||
|
"hana::less(x, y) requires 'x' to be Orderable");
|
||
|
|
||
|
static_assert(hana::Orderable<U>::value,
|
||
|
"hana::less(x, y) requires 'y' to be Orderable");
|
||
|
|
||
|
static_assert(!is_default<less_impl<T, U>>::value,
|
||
|
"hana::less(x, y) requires 'x' and 'y' to be embeddable "
|
||
|
"in a common Orderable");
|
||
|
#endif
|
||
|
|
||
|
return Less::apply(static_cast<X&&>(x), static_cast<Y&&>(y));
|
||
|
}
|
||
|
//! @endcond
|
||
|
|
||
|
template <typename T, typename U, bool condition>
|
||
|
struct less_impl<T, U, when<condition>> : default_ {
|
||
|
template <typename ...Args>
|
||
|
static constexpr auto apply(Args&& ...) = delete;
|
||
|
};
|
||
|
|
||
|
// Cross-type overload
|
||
|
template <typename T, typename U>
|
||
|
struct less_impl<T, U, when<
|
||
|
detail::has_nontrivial_common_embedding<Orderable, T, U>::value &&
|
||
|
!detail::LessThanComparable<T, U>::value
|
||
|
>> {
|
||
|
using C = typename hana::common<T, U>::type;
|
||
|
template <typename X, typename Y>
|
||
|
static constexpr decltype(auto) apply(X&& x, Y&& y) {
|
||
|
return hana::less(hana::to<C>(static_cast<X&&>(x)),
|
||
|
hana::to<C>(static_cast<Y&&>(y)));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
// Model for LessThanComparable data types
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
template <typename T, typename U>
|
||
|
struct less_impl<T, U, when<detail::LessThanComparable<T, U>::value>> {
|
||
|
template <typename X, typename Y>
|
||
|
static constexpr decltype(auto) apply(X&& x, Y&& y)
|
||
|
{ return static_cast<X&&>(x) < static_cast<Y&&>(y); }
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
// Model for Constants wrapping an Orderable
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
template <typename C>
|
||
|
struct less_impl<C, C, when<
|
||
|
hana::Constant<C>::value &&
|
||
|
Orderable<typename C::value_type>::value
|
||
|
>> {
|
||
|
template <typename X, typename Y>
|
||
|
static constexpr auto apply(X const&, Y const&) {
|
||
|
constexpr auto less = hana::less(hana::value<X>(), hana::value<Y>());
|
||
|
constexpr bool truth_value = hana::if_(less, true, false);
|
||
|
return hana::bool_c<truth_value>;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
// Model for Products
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
template <typename T, typename U>
|
||
|
struct less_impl<T, U, when<hana::Product<T>::value && hana::Product<U>::value>> {
|
||
|
template <typename X, typename Y>
|
||
|
static constexpr decltype(auto) apply(X const& x, Y const& y) {
|
||
|
return hana::or_(
|
||
|
hana::less(hana::first(x), hana::first(y)),
|
||
|
hana::and_(
|
||
|
hana::less_equal(hana::first(x), hana::first(y)),
|
||
|
hana::less(hana::second(x), hana::second(y))
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
// Model for Sequences
|
||
|
//////////////////////////////////////////////////////////////////////////
|
||
|
template <typename T, typename U>
|
||
|
struct less_impl<T, U, when<
|
||
|
hana::Sequence<T>::value && hana::Sequence<U>::value
|
||
|
>> {
|
||
|
template <typename Xs, typename Ys>
|
||
|
static constexpr auto apply(Xs const& xs, Ys const& ys)
|
||
|
{ return hana::lexicographical_compare(xs, ys); }
|
||
|
};
|
||
|
BOOST_HANA_NAMESPACE_END
|
||
|
|
||
|
#endif // !BOOST_HANA_LESS_HPP
|