#define BOOST_YAP_CONVERSION_OPERATOR_TEMPLATE 1 #include #include template using term = boost::yap::terminal; template using ref = boost::yap::expression_ref; namespace yap = boost::yap; namespace bh = boost::hana; namespace user { struct number { explicit operator double () const { return value; } double value; }; // User-defined binary-plus! With weird semantics! template inline auto eval_plus (A a, B b) { return number{a.value - b.value}; } template constexpr auto eval_expression_as ( E const & expr, boost::hana::basic_type, T &&... args) { user::number const x = yap::detail::default_eval_expr(expr, static_cast(args)...); return user::number{x.value + 5.0}; } } TEST(user_eval_expression_as, test_user_eval_expression_as) { term unity{{1.0}}; double d_ = 42.0; term i{{d_}}; yap::expression< yap::expr_kind::plus, bh::tuple< ref& >, term > > expr = unity + std::move(i); yap::expression< yap::expr_kind::plus, bh::tuple< ref& >, yap::expression< yap::expr_kind::plus, bh::tuple< ref& >, term > > > > unevaluated_expr = unity + std::move(expr); { user::number result = unity; EXPECT_EQ(result.value, 6); } { user::number result = expr; EXPECT_EQ(result.value, -36); } { user::number result = unevaluated_expr; EXPECT_EQ(result.value, 47); } { user::number result = evaluate(unity, 5, 6, 7); EXPECT_EQ(result.value, 1); } { double result = (double)evaluate(expr); EXPECT_EQ(result, -41); } { double result = (double)evaluate(unevaluated_expr, std::string("15")); EXPECT_EQ(result, 42); } { user::number result = yap::evaluate_as(unity, 5, 6, 7); EXPECT_EQ(result.value, 6); } { user::number result = yap::evaluate_as(expr); EXPECT_EQ(result.value, -36); } { user::number result = yap::evaluate_as(unevaluated_expr, std::string("15")); EXPECT_EQ(result.value, 47); } }