Files
meta.hpp/develop/untests/meta_states/variable_tests.cpp

265 lines
9.3 KiB
C++

/*******************************************************************************
* This file is part of the "https://github.com/blackmatov/meta.hpp"
* For conditions of distribution and use, see copyright notice in LICENSE.md
* Copyright (C) 2021-2024, by Matvey Cherevko (blackmatov@gmail.com)
******************************************************************************/
#include <meta.hpp/meta_all.hpp>
#include <doctest/doctest.h>
namespace
{
struct unique_int {
int i{42};
unique_int() = default;
explicit unique_int(int ni) : i{ni} {}
unique_int(unique_int&&) = default;
unique_int& operator=(unique_int&&) = default;
unique_int(const unique_int&) = delete;
unique_int& operator=(const unique_int&) = delete;
};
struct clazz_1 {
inline static int int_variable{1};
inline static const int const_int_variable{2};
inline static int& ref_int_variable = int_variable;
inline static const int& const_ref_int_variable = const_int_variable;
inline static unique_int unique_int_variable{};
inline static const unique_int const_unique_int_variable{};
static void reset() {
int_variable = 1;
unique_int_variable = unique_int{};
}
};
}
TEST_CASE("meta/meta_states/variable/_") {
namespace meta = meta_hpp;
meta::class_<clazz_1>()
.variable_("int_variable", &clazz_1::int_variable)
.variable_("const_int_variable", &clazz_1::const_int_variable)
.variable_("ref_int_variable", &clazz_1::ref_int_variable)
.variable_("const_ref_int_variable", &clazz_1::const_ref_int_variable)
// .variable_("unique_int_variable", &clazz_1::unique_int_variable)
.variable_("unique_int_variable_as_ptr", &clazz_1::unique_int_variable, meta::variable_policy::as_pointer)
.variable_("unique_int_variable_as_ref", &clazz_1::unique_int_variable, meta::variable_policy::as_reference_wrapper)
// .variable_("const_unique_int_variable", &clazz_1::const_unique_int_variable)
.variable_("const_unique_int_variable_as_ptr", &clazz_1::const_unique_int_variable, meta::variable_policy::as_pointer)
.variable_("const_unique_int_variable_as_ref", &clazz_1::const_unique_int_variable, meta::variable_policy::as_reference_wrapper);
}
TEST_CASE("meta/meta_states/variable") {
namespace meta = meta_hpp;
clazz_1::reset();
const meta::class_type clazz_1_type = meta::resolve_type<clazz_1>();
REQUIRE(clazz_1_type);
SUBCASE("") {
const meta::variable variable;
CHECK_FALSE(variable);
CHECK_FALSE(variable.is_valid());
CHECK(variable == clazz_1_type.get_variable("non-existent-variable"));
}
SUBCASE("operators") {
meta::variable int_variable_v = clazz_1_type.get_variable("int_variable");
meta::variable const_int_variable_v = clazz_1_type.get_variable("const_int_variable");
CHECK(int_variable_v == int_variable_v);
CHECK(int_variable_v != const_int_variable_v);
CHECK((int_variable_v < const_int_variable_v || const_int_variable_v < int_variable_v));
}
SUBCASE("pointers") {
CHECK(clazz_1_type.get_variable("int_variable").get_pointer().as<decltype(&clazz_1::int_variable)>() == &clazz_1::int_variable);
CHECK(clazz_1_type.get_variable("const_int_variable").get_pointer().as<decltype(&clazz_1::const_int_variable)>() == &clazz_1::const_int_variable);
}
SUBCASE("int") {
meta::variable vm = clazz_1_type.get_variable("int_variable");
REQUIRE(vm);
CHECK(vm.get_type() == meta::resolve_type(&clazz_1::int_variable));
CHECK(vm.get_name() == "int_variable");
CHECK(vm.get().as<int>() == 1);
CHECK(vm().as<int>() == 1);
CHECK(vm.is_settable_with<int>());
CHECK(vm.is_settable_with<int&&>());
CHECK(vm.is_settable_with<const int&>());
CHECK(vm.is_settable_with(1));
CHECK_FALSE(vm.is_settable_with<float>());
CHECK_FALSE(vm.is_settable_with<float&&>());
CHECK_FALSE(vm.is_settable_with<const float&>());
CHECK_FALSE(vm.is_settable_with(1.0));
vm.set(10); CHECK(vm.get().as<int>() == 10);
vm(11); CHECK(vm().as<int>() == 11);
}
SUBCASE("const int") {
meta::variable vm = clazz_1_type.get_variable("const_int_variable");
REQUIRE(vm);
CHECK(vm.get_type() == meta::resolve_type(&clazz_1::const_int_variable));
CHECK(vm.get_name() == "const_int_variable");
CHECK(vm.get().as<int>() == 2);
CHECK(vm().as<int>() == 2);
CHECK_FALSE(vm.is_settable_with<int>());
CHECK_FALSE(vm.is_settable_with<int&&>());
CHECK_FALSE(vm.is_settable_with<const int&>());
CHECK_FALSE(vm.is_settable_with(1));
CHECK_FALSE(vm.is_settable_with<float>());
CHECK_FALSE(vm.is_settable_with<float&&>());
CHECK_FALSE(vm.is_settable_with<const float&>());
CHECK_FALSE(vm.is_settable_with(1.0));
CHECK_FALSE(vm.try_set(10)); CHECK(vm.get().as<int>() == 2);
}
SUBCASE("ref int") {
meta::variable vm = clazz_1_type.get_variable("ref_int_variable");
REQUIRE(vm);
CHECK(vm.get_type() == meta::resolve_type(&clazz_1::ref_int_variable));
CHECK(vm.get_name() == "ref_int_variable");
CHECK(vm.get().as<int>() == 1);
CHECK(vm().as<int>() == 1);
CHECK(vm.is_settable_with<int>());
CHECK(vm.is_settable_with<int&&>());
CHECK(vm.is_settable_with<const int&>());
CHECK(vm.is_settable_with(1));
CHECK_FALSE(vm.is_settable_with<float>());
CHECK_FALSE(vm.is_settable_with<float&&>());
CHECK_FALSE(vm.is_settable_with<const float&>());
CHECK_FALSE(vm.is_settable_with(1.0));
vm.set(20); CHECK(vm.get().as<int>() == 20);
vm(21); CHECK(vm().as<int>() == 21);
}
SUBCASE("const ref int") {
meta::variable vm = clazz_1_type.get_variable("const_ref_int_variable");
REQUIRE(vm);
CHECK(vm.get_type() == meta::resolve_type(&clazz_1::const_ref_int_variable));
CHECK(vm.get_name() == "const_ref_int_variable");
CHECK(vm.get().as<int>() == 2);
CHECK(vm().as<int>() == 2);
CHECK_FALSE(vm.is_settable_with<int>());
CHECK_FALSE(vm.is_settable_with<int&&>());
CHECK_FALSE(vm.is_settable_with<const int&>());
CHECK_FALSE(vm.is_settable_with(1));
CHECK_FALSE(vm.is_settable_with<float>());
CHECK_FALSE(vm.is_settable_with<float&&>());
CHECK_FALSE(vm.is_settable_with<const float&>());
CHECK_FALSE(vm.is_settable_with(1.0));
CHECK_FALSE(vm.try_set(10)); CHECK(vm.get().as<int>() == 2);
}
SUBCASE("unique_int_variable_as_ptr") {
meta::variable vm = clazz_1_type.get_variable("unique_int_variable_as_ptr");
REQUIRE(vm);
CHECK(vm.get().get_type() == meta::resolve_type<unique_int*>());
CHECK(vm.get().as<unique_int*>() == std::addressof(clazz_1::unique_int_variable));
{
unique_int nv{11};
vm.set(std::move(nv));
CHECK(clazz_1::unique_int_variable.i == 11);
}
{
unique_int nv{12};
CHECK_FALSE(vm.try_set(nv));
CHECK(clazz_1::unique_int_variable.i == 11);
}
}
SUBCASE("unique_int_variable_as_ref") {
meta::variable vm = clazz_1_type.get_variable("unique_int_variable_as_ref");
REQUIRE(vm);
using ref_t = std::reference_wrapper<unique_int>;
CHECK(vm.get().get_type() == meta::resolve_type<ref_t>());
CHECK(&vm.get().as<ref_t>().get() == &clazz_1::unique_int_variable);
{
unique_int nv{13};
vm.set(std::move(nv));
CHECK(clazz_1::unique_int_variable.i == 13);
}
{
unique_int nv{14};
CHECK_FALSE(vm.try_set(nv));
CHECK(clazz_1::unique_int_variable.i == 13);
}
}
SUBCASE("const_unique_int_variable_as_ptr") {
meta::variable vm = clazz_1_type.get_variable("const_unique_int_variable_as_ptr");
REQUIRE(vm);
CHECK(vm.get().get_type() == meta::resolve_type<const unique_int*>());
CHECK(vm.get().as<const unique_int*>() == std::addressof(clazz_1::const_unique_int_variable));
{
unique_int nv{11};
CHECK_FALSE(vm.try_set(nv));
CHECK_FALSE(vm.try_set(std::move(nv)));
CHECK(clazz_1::const_unique_int_variable.i == 42);
}
}
SUBCASE("const_unique_int_variable_as_ref") {
meta::variable vm = clazz_1_type.get_variable("const_unique_int_variable_as_ref");
REQUIRE(vm);
using ref_t = std::reference_wrapper<const unique_int>;
CHECK(vm.get().get_type() == meta::resolve_type<ref_t>());
CHECK(&vm.get().as<ref_t>().get() == &clazz_1::const_unique_int_variable);
{
unique_int nv{12};
CHECK_FALSE(vm.try_set(nv));
CHECK_FALSE(vm.try_set(std::move(nv)));
CHECK(clazz_1::const_unique_int_variable.i == 42);
}
}
SUBCASE("try_set") {
meta::variable vm = clazz_1_type.get_variable("int_variable");
REQUIRE(vm);
CHECK(vm.try_set(10));
CHECK(vm.get().as<int>() == 10);
CHECK_FALSE(vm.try_set(unique_int{}));
CHECK(vm.get().as<int>() == 10);
}
}