#include "app.hpp" #include "compat/assert.hpp" #include "compat/limits.hpp" #include "entity/metadata.hpp" #include "entity/accessor.hpp" #include using namespace floormat; using namespace floormat::entities; namespace { struct TestAccessors { constexpr int bar() const { return _bar; } constexpr void set_bar(int value) { _bar = value; } int foo; int _bar; int _baz; }; } // namespace namespace floormat::entities { template<> struct entity_accessors { static constexpr auto accessors() { using entity = Entity; constexpr auto r_baz = [](const TestAccessors& x) { return x._baz; }; constexpr auto w_baz = [](TestAccessors& x, int v) { x._baz = v; }; constexpr auto tuple = std::make_tuple( entity::type::field{"foo"_s, &TestAccessors::foo, &TestAccessors::foo}, entity::type::field{"bar"_s, &TestAccessors::bar, &TestAccessors::set_bar}, entity::type::field("baz"_s, r_baz, w_baz) ); return tuple; } }; } // namespace floormat::entities namespace { using entity = Entity; constexpr auto m_foo = entity::type::field{"foo"_s, &TestAccessors::foo, &TestAccessors::foo}; constexpr auto m_bar = entity::type::field{"bar"_s, &TestAccessors::bar, &TestAccessors::set_bar}; constexpr auto r_baz = [](const TestAccessors& x) { return x._baz; }; constexpr auto w_baz = [](TestAccessors& x, int v) { x._baz = v; }; constexpr auto m_baz = entity::type::field("baz"_s, r_baz, w_baz); } // namespace namespace floormat { namespace { constexpr bool test_accessors() { auto x = TestAccessors{111, 222, 333}; { auto a = m_foo.read(x), b = m_bar.read(x), c = m_baz.read(x); fm_assert(a == 111 && b == 222 && c == 333); } { m_foo.write(x, 1111); m_bar.write(x, 2222); m_baz.write(x, 3333); auto a = m_foo.read(x), b = m_bar.read(x), c = m_baz.read(x); fm_assert(a == 1111 && b == 2222 && c == 3333); } return true; } constexpr bool test_visitor() { { auto tuple = std::make_tuple((unsigned char)1, (unsigned short)2, (int)3, (long)4); long ret = 0; visit_tuple([&](auto x) { ret += (long)x; }, tuple); fm_assert(ret == 1 + 2 + 3 + 4); } { int ret = 0; visit_tuple([&] { ret++; }, std::tuple<>{}); fm_assert(ret == 0); } { constexpr auto tuple = std::make_tuple((char)1, (short)2, (long)3); static_assert(find_in_tuple([](auto x) { return x == 3; }, tuple)); static_assert(!find_in_tuple([](auto x) { return x == 5; }, tuple)); } return true; } void test_fun2() { using entity = Entity; static constexpr auto read_fn = [](const TestAccessors& x) constexpr { return x.bar(); }; static constexpr auto write_fn = [](TestAccessors& x, int value) constexpr { x.set_bar(value); }; constexpr auto read_bar = fu2::function_view{read_fn}; constexpr auto write_bar = fu2::function_view{write_fn}; constexpr auto m_bar2 = entity::type::field{"bar"_s, read_bar, write_bar}; auto x = TestAccessors{1, 2, 3}; fm_assert(m_bar2.read(x) == 2); m_bar2.write(x, 22); fm_assert(m_bar2.read(x) == 22); } void test_erasure() { erased_accessor accessors[] = { m_foo.erased(), m_bar.erased(), m_baz.erased(), }; auto obj = TestAccessors{1, 2, 3}; int value = 0; accessors[1].read_fun(&obj, accessors[1].reader, &value); fm_assert(value == 2); int value2 = 22222, value2_ = 0; accessors[1].write_fun(&obj, accessors[1].writer, &value2); accessors[1].read_fun(&obj, accessors[1].reader, &value2_); fm_assert(value2 == value2_); } void test_metadata() { constexpr auto m = entity_metadata(); static_assert(sizeof m == 1); fm_assert(m.class_name == name_of); fm_assert(m.class_name.contains("TestAccessors"_s)); const auto [foo, bar, baz] = m.accessors; const auto [foo2, bar2, baz2] = m.erased_accessors; TestAccessors x{0, 0, 0}; foo.write(x, 1); fm_assert(x.foo == 1); int bar_ = 2; bar2.write_fun(&x, bar2.writer, &bar_); fm_assert(x.bar() == 2); baz2.write(x, 3); fm_assert(baz2.read(x) == 3); } void test_type_name() { struct foobar; constexpr StringView name = name_of; fm_assert(name.contains("foobar"_s)); static_assert(name.data() == name_of.data()); #ifndef __CLION_IDE__ static_assert(name_of != name_of); static_assert(name_of != name_of); #endif using foobar2 = foobar; static_assert(name_of.data() == name_of.data()); } [[maybe_unused]] constexpr void test_null_writer() { using entity = Entity; constexpr auto foo = entity::type::field{"foo"_s, &TestAccessors::foo, nullptr}; static_assert(foo.writer == nullptr); static_assert(!foo.can_write); static_assert(std::get<0>(entity_accessors::accessors()).can_write); } void test_predicate() { using entity = Entity; constexpr TestAccessors x{0, 0, 0}; constexpr auto foo = entity::type::field{"foo"_s, &TestAccessors::foo, &TestAccessors::foo, [](const TestAccessors&) { return field_status::hidden; }}; static_assert(foo.is_enabled(x) == field_status::hidden); fm_assert(foo.erased().is_enabled(&x) == field_status::hidden); foo.erased().do_asserts(); constexpr auto foo2 = entity::type::field{"foo"_s, &TestAccessors::foo, &TestAccessors::foo, [](const TestAccessors&) { return field_status::readonly; }}; static_assert(foo2.is_enabled(x) == field_status::readonly); fm_assert(foo2.erased().is_enabled(&x) == field_status::readonly); constexpr auto foo3 = entity::type::field{"foo"_s, &TestAccessors::foo, &TestAccessors::foo}; static_assert(foo3.is_enabled(x) == field_status::enabled); fm_assert(foo3.erased().is_enabled(&x) == field_status::enabled); } constexpr bool test_names() { constexpr auto m = entity_metadata(); auto [foo1, bar1, baz1] = m.accessors; auto [foo2, bar2, baz2] = m.erased_accessors; fm_assert(foo1.name == "foo"_s); fm_assert(bar1.name == "bar"_s); fm_assert(baz1.name == "baz"_s); fm_assert(foo2.field_name == "foo"_s); fm_assert(bar2.field_name == "bar"_s); fm_assert(baz2.field_name == "baz"_s); return true; } constexpr void test_constraints() { using entity = Entity; constexpr auto x = TestAccessors{}; constexpr auto foo = entity::type::field { "foo"_s, &TestAccessors::foo, &TestAccessors::foo, constantly(constraints::max_length{42}), constantly(constraints::range{37, 42}), }; static_assert(foo.get_range(x) == constraints::range{37, 42}); static_assert(foo.get_max_length(x) == 42); static_assert(m_foo.get_range(x) == constraints::range{}); static_assert(m_foo.get_max_length(x) == (size_t)-1); constexpr auto foo2 = entity::type::field { "foo"_s, &TestAccessors::foo, &TestAccessors::foo, constantly(constraints::max_length {123}), }; static_assert(foo2.get_range(x) == constraints::range{}); static_assert(foo2.get_max_length(x) == 123); } void test_erased_constraints() { using entity = Entity; static constexpr auto foo = entity::type::field{ "foo"_s, &TestAccessors::foo, &TestAccessors::foo, constantly(constraints::max_length{42}), constantly(constraints::range{37, 42}), }; static constexpr auto erased = foo.erased(); const auto x = TestAccessors{}; erased.do_asserts(); fm_assert(erased.get_range(&x) == constraints::range{37, 42}); fm_assert(erased.get_max_length(&x) == 42); } constexpr bool test_range1() { constexpr auto x = TestAccessors{}; constexpr auto r = m_foo.get_range(x); static_assert(r.max == limits::max); return true; } void test_range2() { constexpr auto x = TestAccessors{}; constexpr auto rʹ = m_foo.get_range(x); const auto A = m_foo.erased(); auto r = A.get_range(&x); auto i = r.convert(); fm_assert(i.second == rʹ.max); } constexpr bool test_range3() { constexpr auto f = entity::type::field("vec"_s, constantly(Vector2i(0)), [](auto&&, auto&&) {}); constexpr auto r = f.get_range(f.range, TestAccessors{}); static_assert(r.max == Vector2i(limits::max)); return true; } void test_range4() { constexpr auto x = TestAccessors{}; constexpr auto f = entity::type::field("vec"_s, constantly(Vector2i(0)), [](auto&&, auto&&) {}); constexpr auto rʹ = f.get_range(f.range, TestAccessors{}); const auto A = f.erased(); const auto r = A.get_range(&x).convert(); fm_assert(r.first == rʹ.min); fm_assert(r.second == rʹ.max); } constexpr bool test_enum_range() { enum class E { foo, bar }; constexpr auto x = TestAccessors{}; constexpr auto f = entity::type::field("vec"_s, constantly(E::bar), [](auto&&, auto&&) {}); static_assert(f.read(x) == E::bar); constexpr auto r = f.get_range(x); static_assert(r.max != E{0}); return true; } } // namespace void test_app::test_entity() { static_assert(test_accessors()); static_assert(test_visitor()); test_null_writer(); static_assert(test_names()); test_predicate(); test_fun2(); test_erasure(); test_type_name(); test_metadata(); test_constraints(); test_erased_constraints(); static_assert(test_range1()); test_range2(); static_assert(test_range3()); test_range4(); static_assert(test_enum_range()); } } // namespace floormat