NcEngine
Type.h
1#pragma once
2
3#include <cstdint>
4#include <tuple>
5#include <type_traits>
6
7namespace nc::type
8{
9template<class T>
10struct Type
11{
12 static constexpr auto name = "Null";
13 static constexpr auto properties = std::make_tuple();
14 static constexpr auto propertyCount = 0u;
15 static constexpr auto isPrimitive = false;
16};
17
18#define REGISTER_TYPE(Class, ...) \
19template<> struct Type<Class> \
20{ \
21 static constexpr const char* name = #Class; \
22 static constexpr auto properties = std::make_tuple(__VA_ARGS__); \
23 static constexpr auto propertyCount = std::tuple_size<decltype(properties)>::value; \
24 static constexpr bool isPrimitive = false; \
25};
26
27#define REGISTER_PRIMITIVE_TYPE(Class) \
28template<> struct Type<Class> \
29{ \
30 static constexpr const char* name = #Class; \
31 static constexpr auto properties = std::make_tuple(); \
32 static constexpr auto propertyCount = 0u; \
33 static constexpr bool isPrimitive = true; \
34 static auto Get(Class obj) -> Class { return obj; } \
35 static auto Set(Class* ptr, Class v) { *ptr = v; } \
36};
37
39{
40 static constexpr uint32_t None = 0b0000;
41 static constexpr uint32_t Nonnegative = 0b0001;
42 static constexpr uint32_t Position = 0b0010;
43 static constexpr uint32_t Scale = 0b0100;
44 static constexpr uint32_t Angles = 0b1000;
45};
46
48{
49 constexpr PropertyBase()
50 : name{"Null"}, flags{PropertyFlags::None}
51 {
52 }
53
54 constexpr PropertyBase(const char* tag, uint32_t flags_)
55 : name{tag}, flags{flags_}
56 {
57 }
58
59 const char* name;
60 uint32_t flags;
61};
62
63template<class T, class P>
64struct Property : public PropertyBase
65{
66 using parent_type = T;
67 using property_type = P;
68
69 constexpr Property(P T::*ptr, const char* tag, uint32_t propertyFlags)
70 : PropertyBase{tag, propertyFlags}, member{ptr}
71 {
72 }
73
74 P T::*member;
75};
76
77#define PROPERTY(Class, Member) \
78Property<Class, decltype(Class::Member)>{&Class::Member, #Member, PropertyFlags::None}
79
80#define PROPERTY_F(Class, Member, Flags) \
81Property<Class, decltype(Class::Member)>{&Class::Member, #Member, Flags}
82
83template<class T, class P>
84auto GetProperty(const T& obj) -> P
85{
86 auto property = std::get<Property<T, P>>(Type<T>::properties);
87 return obj.*(property.member);
88}
89
90template<class T, class Func>
91constexpr void ForEachMember(Func&& func)
92{
93 std::apply([&func](auto&& ... args) { (func(args), ...); }, Type<T>::properties);
94}
95
96REGISTER_PRIMITIVE_TYPE(float);
97REGISTER_PRIMITIVE_TYPE(int);
98REGISTER_PRIMITIVE_TYPE(char);
99REGISTER_PRIMITIVE_TYPE(const char*);
100} // namespace nc::type
Definition: Type.h:48
Definition: Type.h:39
Definition: Type.h:65
Definition: Type.h:11