7 #if !defined(CNL_IMPL_WRAPPER_BINARY_OPERATOR_H)
8 #define CNL_IMPL_WRAPPER_BINARY_OPERATOR_H
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/is_same_tag_family.h"
12 #include "../custom_operator/native_tag.h"
13 #include "../custom_operator/op.h"
14 #include "../custom_operator/overloads.h"
15 #include "../num_traits/set_rep.h"
16 #include "../num_traits/set_tag.h"
17 #include "is_wrapper.h"
18 #include "make_wrapper.h"
20 #include "operator_helpers.h"
23 #include <type_traits>
28 template<_impl::binary_arithmetic_op Operator, std::
floating_po
int Lhs, _impl::any_wrapper Rhs>
29 struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
30 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
32 return Operator()(lhs,
static_cast<Lhs
>(rhs));
37 template<_impl::binary_arithmetic_op Operator, _impl::any_wrapper Lhs, std::
floating_po
int Rhs>
38 struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
39 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
41 return Operator()(
static_cast<Rhs
>(lhs), rhs);
46 template<_impl::binary_arithmetic_op Operator,
class Lhs,
class Rhs>
47 requires _impl::number_can_wrap<Rhs, Lhs>::value
struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
48 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
50 return Operator()(_impl::from_value<Rhs>(lhs), rhs);
55 template<_impl::binary_arithmetic_op Operator,
class Lhs,
class Rhs>
56 requires _impl::number_can_wrap<Lhs, Rhs>::value
struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
57 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
59 return Operator()(lhs, from_value<Lhs, Rhs>{}(rhs));
63 template<_impl::binary_arithmetic_op Operator, _impl::any_wrapper Lhs, _impl::any_wrapper Rhs>
64 requires(_impl::is_same_tag_family<_impl::tag_of_t<Lhs>, _impl::tag_of_t<Rhs>>::value)
struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
65 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
67 auto const lhs_rep{_impl::to_rep(lhs)};
68 auto const rhs_rep{_impl::to_rep(rhs)};
69 using rep_operator = custom_operator<
71 op_value<std::remove_const_t<decltype(lhs_rep)>, _impl::tag_of_t<Lhs>>,
72 op_value<std::remove_const_t<decltype(rhs_rep)>, _impl::tag_of_t<Rhs>>>;
73 auto const result_rep{rep_operator{}(lhs_rep, rhs_rep)};
74 using result_rep_type = std::remove_const_t<decltype(result_rep)>;
75 using result_tag = _impl::op_result<Operator, _impl::tag_of_t<Lhs>, _impl::tag_of_t<Rhs>>;
76 using result_archetype = _impl::set_rep_t<_impl::set_tag_t<Lhs, result_tag>, result_rep_type>;
78 return _impl::from_rep<result_archetype>(result_rep);
83 #endif // CNL_IMPL_WRAPPER_BINARY_OPERATOR_H