|
|
@@ -19,6 +19,27 @@
|
|
|
|
|
|
namespace entt {
|
|
|
|
|
|
+/**
|
|
|
+ * @cond TURN_OFF_DOXYGEN
|
|
|
+ * Internal details not to be documented.
|
|
|
+ */
|
|
|
+
|
|
|
+namespace internal {
|
|
|
+
|
|
|
+template<typename Type>
|
|
|
+void meta_details_setup(Type &value) {
|
|
|
+ if(!value.details) {
|
|
|
+ value.details = std::make_shared<typename decltype(value.details)::element_type>();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+} // namespace internal
|
|
|
+
|
|
|
+/**
|
|
|
+ * Internal details not to be documented.
|
|
|
+ * @endcond
|
|
|
+ */
|
|
|
+
|
|
|
/**
|
|
|
* @brief Meta factory to be used for reflection purposes.
|
|
|
*
|
|
|
@@ -33,17 +54,17 @@ class meta_factory;
|
|
|
/**
|
|
|
* @brief Extended meta factory to be used for reflection purposes.
|
|
|
* @tparam Type Reflected type for which the factory was created.
|
|
|
- * @tparam Prop Type of container to store properties.
|
|
|
+ * @tparam Node Node for which to create properties, if any.
|
|
|
*/
|
|
|
-template<typename Type, typename Prop>
|
|
|
-class meta_factory<Type, Prop>: public meta_factory<Type> {
|
|
|
+template<typename Type, typename Node>
|
|
|
+class meta_factory<Type, Node>: public meta_factory<Type> {
|
|
|
public:
|
|
|
/**
|
|
|
* @brief Constructs an extended factory from a given node.
|
|
|
- * @param target The underlying container to store properties.
|
|
|
+ * @param target The node for which to store properties, if any.
|
|
|
*/
|
|
|
- meta_factory(Prop &target) noexcept
|
|
|
- : container{&target} {}
|
|
|
+ meta_factory(Node &target) noexcept
|
|
|
+ : node{&target} {}
|
|
|
|
|
|
/**
|
|
|
* @brief Assigns a property to the last meta object created.
|
|
|
@@ -57,7 +78,9 @@ public:
|
|
|
*/
|
|
|
template<typename... Value>
|
|
|
meta_factory prop(id_type key, Value &&...value) {
|
|
|
- (*container)[key] = internal::meta_prop_node{
|
|
|
+ internal::meta_details_setup(*node);
|
|
|
+
|
|
|
+ node->details->prop[key] = internal::meta_prop_node{
|
|
|
&internal::resolve<std::decay_t<Value>>...,
|
|
|
std::forward<Value>(value)...};
|
|
|
|
|
|
@@ -65,7 +88,7 @@ public:
|
|
|
}
|
|
|
|
|
|
private:
|
|
|
- Prop *container;
|
|
|
+ Node *node;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
@@ -89,8 +112,9 @@ class meta_factory<Type> {
|
|
|
+[](meta_handle instance, meta_any value) -> bool { return (meta_setter<Type, value_list_element_v<Index, Setter>>(*instance.operator->(), value.as_ref()) || ...); },
|
|
|
&meta_getter<Type, Getter, Policy>};
|
|
|
|
|
|
- auto it = owner->data.insert_or_assign(id, std::move(node)).first;
|
|
|
- return meta_factory<Type, decltype(internal::meta_data_node::prop)>{it->second.prop};
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
+ auto it = owner->details->data.insert_or_assign(id, std::move(node)).first;
|
|
|
+ return meta_factory<Type, internal::meta_data_node>{it->second};
|
|
|
}
|
|
|
|
|
|
public:
|
|
|
@@ -106,7 +130,7 @@ public:
|
|
|
auto type(const id_type id) noexcept {
|
|
|
ENTT_ASSERT(owner->id == id || !resolve(id), "Duplicate identifier");
|
|
|
owner->id = id;
|
|
|
- return meta_factory<Type, decltype(internal::meta_type_node::prop)>{owner->prop};
|
|
|
+ return meta_factory<Type, internal::meta_type_node>{*owner};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
@@ -120,8 +144,9 @@ public:
|
|
|
template<typename Base>
|
|
|
auto base() noexcept {
|
|
|
static_assert(!std::is_same_v<Type, Base> && std::is_base_of_v<Base, Type>, "Invalid base type");
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
- owner->base[type_id<Base>().hash()] = internal::meta_base_node{
|
|
|
+ owner->details->base[type_id<Base>().hash()] = internal::meta_base_node{
|
|
|
&internal::resolve<Base>,
|
|
|
+[](meta_any other) noexcept -> meta_any {
|
|
|
if(auto *ptr = other.data(); ptr) {
|
|
|
@@ -149,8 +174,9 @@ public:
|
|
|
template<auto Candidate>
|
|
|
auto conv() noexcept {
|
|
|
using conv_type = std::remove_cv_t<std::remove_reference_t<std::invoke_result_t<decltype(Candidate), Type &>>>;
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
- owner->conv[type_id<conv_type>().hash()] = internal::meta_conv_node{
|
|
|
+ owner->details->conv[type_id<conv_type>().hash()] = internal::meta_conv_node{
|
|
|
+[](const meta_any &instance) -> meta_any {
|
|
|
return forward_as_meta(std::invoke(Candidate, *static_cast<const Type *>(instance.data())));
|
|
|
}};
|
|
|
@@ -170,8 +196,9 @@ public:
|
|
|
template<typename To>
|
|
|
auto conv() noexcept {
|
|
|
using conv_type = std::remove_cv_t<std::remove_reference_t<To>>;
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
- owner->conv[type_id<conv_type>().hash()] = internal::meta_conv_node{
|
|
|
+ owner->details->conv[type_id<conv_type>().hash()] = internal::meta_conv_node{
|
|
|
+[](const meta_any &instance) -> meta_any {
|
|
|
return forward_as_meta(static_cast<To>(*static_cast<const Type *>(instance.data())));
|
|
|
}};
|
|
|
@@ -197,8 +224,9 @@ public:
|
|
|
using descriptor = meta_function_helper_t<Type, decltype(Candidate)>;
|
|
|
static_assert(Policy::template value<typename descriptor::return_type>, "Invalid return type for the given policy");
|
|
|
static_assert(std::is_same_v<std::remove_cv_t<std::remove_reference_t<typename descriptor::return_type>>, Type>, "The function doesn't return an object of the required type");
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
- owner->ctor[type_id<typename descriptor::args_type>().hash()] = internal::meta_ctor_node{
|
|
|
+ owner->details->ctor[type_id<typename descriptor::args_type>().hash()] = internal::meta_ctor_node{
|
|
|
descriptor::args_type::size,
|
|
|
&meta_arg<typename descriptor::args_type>,
|
|
|
&meta_construct<Type, Candidate, Policy>};
|
|
|
@@ -219,8 +247,9 @@ public:
|
|
|
template<typename... Args>
|
|
|
auto ctor() noexcept {
|
|
|
using descriptor = meta_function_helper_t<Type, Type (*)(Args...)>;
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
- owner->ctor[type_id<typename descriptor::args_type>().hash()] = internal::meta_ctor_node{
|
|
|
+ owner->details->ctor[type_id<typename descriptor::args_type>().hash()] = internal::meta_ctor_node{
|
|
|
descriptor::args_type::size,
|
|
|
&meta_arg<typename descriptor::args_type>,
|
|
|
&meta_construct<Type, Args...>};
|
|
|
@@ -268,6 +297,8 @@ public:
|
|
|
*/
|
|
|
template<auto Data, typename Policy = as_is_t>
|
|
|
auto data(const id_type id) noexcept {
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
+
|
|
|
if constexpr(std::is_member_object_pointer_v<decltype(Data)>) {
|
|
|
using data_type = std::remove_reference_t<std::invoke_result_t<decltype(Data), Type &>>;
|
|
|
|
|
|
@@ -280,8 +311,8 @@ public:
|
|
|
&meta_setter<Type, Data>,
|
|
|
&meta_getter<Type, Data, Policy>};
|
|
|
|
|
|
- auto it = owner->data.insert_or_assign(id, std::move(node)).first;
|
|
|
- return meta_factory<Type, decltype(internal::meta_data_node::prop)>{it->second.prop};
|
|
|
+ auto it = owner->details->data.insert_or_assign(id, std::move(node)).first;
|
|
|
+ return meta_factory<Type, internal::meta_data_node>{it->second};
|
|
|
} else {
|
|
|
using data_type = std::remove_reference_t<std::remove_pointer_t<decltype(Data)>>;
|
|
|
|
|
|
@@ -293,8 +324,8 @@ public:
|
|
|
&meta_setter<Type, Data>,
|
|
|
&meta_getter<Type, Data, Policy>};
|
|
|
|
|
|
- auto it = owner->data.insert_or_assign(id, std::move(node)).first;
|
|
|
- return meta_factory<Type, decltype(internal::meta_data_node::prop)>{it->second.prop};
|
|
|
+ auto it = owner->details->data.insert_or_assign(id, std::move(node)).first;
|
|
|
+ return meta_factory<Type, internal::meta_data_node>{it->second};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
@@ -322,6 +353,7 @@ public:
|
|
|
auto data(const id_type id) noexcept {
|
|
|
using data_type = std::invoke_result_t<decltype(Getter), Type &>;
|
|
|
static_assert(Policy::template value<data_type>, "Invalid return type for the given policy");
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
if constexpr(std::is_same_v<decltype(Setter), std::nullptr_t>) {
|
|
|
internal::meta_data_node node{
|
|
|
@@ -333,8 +365,8 @@ public:
|
|
|
&meta_setter<Type, Setter>,
|
|
|
&meta_getter<Type, Getter, Policy>};
|
|
|
|
|
|
- auto it = owner->data.insert_or_assign(id, std::move(node)).first;
|
|
|
- return meta_factory<Type, decltype(internal::meta_data_node::prop)>{it->second.prop};
|
|
|
+ auto it = owner->details->data.insert_or_assign(id, std::move(node)).first;
|
|
|
+ return meta_factory<Type, internal::meta_data_node>{it->second};
|
|
|
} else {
|
|
|
using args_type = typename meta_function_helper_t<Type, decltype(Setter)>::args_type;
|
|
|
|
|
|
@@ -347,8 +379,8 @@ public:
|
|
|
&meta_setter<Type, Setter>,
|
|
|
&meta_getter<Type, Getter, Policy>};
|
|
|
|
|
|
- auto it = owner->data.insert_or_assign(id, std::move(node)).first;
|
|
|
- return meta_factory<Type, decltype(internal::meta_data_node::prop)>{it->second.prop};
|
|
|
+ auto it = owner->details->data.insert_or_assign(id, std::move(node)).first;
|
|
|
+ return meta_factory<Type, internal::meta_data_node>{it->second};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
@@ -391,6 +423,7 @@ public:
|
|
|
auto func(const id_type id) noexcept {
|
|
|
using descriptor = meta_function_helper_t<Type, decltype(Candidate)>;
|
|
|
static_assert(Policy::template value<typename descriptor::return_type>, "Invalid return type for the given policy");
|
|
|
+ internal::meta_details_setup(*owner);
|
|
|
|
|
|
internal::meta_func_node node{
|
|
|
(descriptor::is_const ? internal::meta_traits::is_const : internal::meta_traits::is_none) | (descriptor::is_static ? internal::meta_traits::is_static : internal::meta_traits::is_none),
|
|
|
@@ -399,22 +432,22 @@ public:
|
|
|
&meta_arg<typename descriptor::args_type>,
|
|
|
&meta_invoke<Type, Candidate, Policy>};
|
|
|
|
|
|
- if(auto it = owner->func.find(id); it != owner->func.end()) {
|
|
|
+ if(auto it = owner->details->func.find(id); it != owner->details->func.end()) {
|
|
|
for(auto *curr = &it->second; curr; curr = curr->next.get()) {
|
|
|
if(curr->invoke == node.invoke) {
|
|
|
node.next = std::move(curr->next);
|
|
|
*curr = std::move(node);
|
|
|
|
|
|
- return meta_factory<Type, decltype(internal::meta_func_node::prop)>{curr->prop};
|
|
|
+ return meta_factory<Type, internal::meta_func_node>{*curr};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// locally overloaded function
|
|
|
- node.next = std::make_unique<internal::meta_func_node>(std::move(owner->func[id]));
|
|
|
+ node.next = std::make_shared<internal::meta_func_node>(std::move(owner->details->func[id]));
|
|
|
}
|
|
|
|
|
|
- owner->func.insert_or_assign(id, std::move(node));
|
|
|
- return meta_factory<Type, decltype(internal::meta_func_node::prop)>{owner->func[id].prop};
|
|
|
+ owner->details->func.insert_or_assign(id, std::move(node));
|
|
|
+ return meta_factory<Type, internal::meta_func_node>{owner->details->func[id]};
|
|
|
}
|
|
|
|
|
|
private:
|
|
|
@@ -436,7 +469,7 @@ template<typename Type>
|
|
|
[[nodiscard]] auto meta() noexcept {
|
|
|
auto *const node = internal::resolve<Type>();
|
|
|
// extended meta factory to allow assigning properties to opaque meta types
|
|
|
- return meta_factory<Type, decltype(internal::meta_type_node::prop)>{node->prop};
|
|
|
+ return meta_factory<Type, internal::meta_type_node>{*node};
|
|
|
}
|
|
|
|
|
|
/**
|