registry: decouple ::storage/::assure

This commit is contained in:
Michele Caini
2021-11-23 12:52:24 +01:00
parent adfdbf138d
commit e4d36e4982
2 changed files with 73 additions and 55 deletions

View File

@@ -58,11 +58,11 @@ class basic_registry {
template<typename Component>
void maybe_valid_if(basic_registry &owner, const Entity entt) {
[[maybe_unused]] const auto cpools = std::forward_as_tuple(owner.storage<Owned>()...);
[[maybe_unused]] const auto cpools = std::forward_as_tuple(owner.assure<Owned>()...);
const auto is_valid = ((std::is_same_v<Component, Owned> || std::get<storage_type<Owned> &>(cpools).contains(entt)) && ...)
&& ((std::is_same_v<Component, Get> || owner.storage<Get>().contains(entt)) && ...)
&& ((std::is_same_v<Component, Exclude> || !owner.storage<Exclude>().contains(entt)) && ...);
&& ((std::is_same_v<Component, Get> || owner.assure<Get>().contains(entt)) && ...)
&& ((std::is_same_v<Component, Exclude> || !owner.assure<Exclude>().contains(entt)) && ...);
if constexpr(sizeof...(Owned) == 0) {
if(is_valid && !current.contains(entt)) {
@@ -80,7 +80,7 @@ class basic_registry {
if constexpr(sizeof...(Owned) == 0) {
current.remove(entt);
} else {
if(const auto cpools = std::forward_as_tuple(owner.storage<Owned>()...); std::get<0>(cpools).contains(entt) && (std::get<0>(cpools).index(entt) < current)) {
if(const auto cpools = std::forward_as_tuple(owner.assure<Owned>()...); std::get<0>(cpools).contains(entt) && (std::get<0>(cpools).index(entt) < current)) {
const auto pos = --current;
(std::get<storage_type<Owned> &>(cpools).swap_elements(std::get<storage_type<Owned> &>(cpools).data()[pos], entt), ...);
}
@@ -96,6 +96,31 @@ class basic_registry {
bool (*exclude)(const id_type) ENTT_NOEXCEPT;
};
template<typename Component>
[[nodiscard]] storage_type<Component> &assure(const id_type id = type_hash<Component>::value()) {
static_assert(std::is_same_v<Component, std::decay_t<Component>>, "Non-decayed types not allowed");
auto &&cpool = pools[id];
if(!cpool) {
cpool.reset(new storage_type<Component>{});
cpool->bind(forward_as_any(*this));
}
return static_cast<storage_type<Component> &>(*cpool);
}
template<typename Component>
[[nodiscard]] const storage_type<Component> &assure(const id_type id = type_hash<Component>::value()) const {
static_assert(std::is_same_v<Component, std::decay_t<Component>>, "Non-decayed types not allowed");
if(const auto it = pools.find(id); it != pools.cend()) {
return static_cast<storage_type<Component> &>(*it->second);
}
static storage_type<Component> placeholder{};
return placeholder;
}
auto generate_identifier(const std::size_t pos) ENTT_NOEXCEPT {
ENTT_ASSERT(pos < entity_traits::to_integral(null), "No entities available");
return entity_traits::combine(static_cast<typename entity_traits::entity_type>(pos), {});
@@ -167,16 +192,8 @@ public:
* @return The container for the given component type.
*/
template<typename Component>
[[nodiscard]] storage_type<Component> &storage(const id_type id = type_hash<Component>::value()) {
static_assert(std::is_same_v<Component, std::decay_t<Component>>, "Non-decayed types not allowed");
auto &&cpool = pools[id];
if(!cpool) {
cpool.reset(new storage_type<Component>{});
cpool->bind(forward_as_any(*this));
}
return static_cast<storage_type<Component> &>(*cpool);
[[nodiscard]] storage_type<Component> &storage(const id_type id = type_hash<std::remove_const_t<Component>>::value()) {
return assure<std::remove_const_t<Component>>(id);
}
/**
@@ -191,15 +208,8 @@ public:
* @return The container for the given component type.
*/
template<typename Component>
[[nodiscard]] const storage_type<Component> &storage(const id_type id = type_hash<Component>::value()) const {
static_assert(std::is_same_v<Component, std::decay_t<Component>>, "Non-decayed types not allowed");
if(const auto it = pools.find(id); it != pools.cend()) {
return static_cast<storage_type<Component> &>(*it->second);
}
static storage_type<Component> placeholder{};
return placeholder;
[[nodiscard]] storage_type<const Component> &storage(const id_type id = type_hash<std::remove_const_t<Component>>::value()) const {
return assure<std::remove_const_t<Component>>(id);
}
/**
@@ -209,7 +219,7 @@ public:
*/
template<typename Component>
[[nodiscard]] size_type size() const {
return storage<std::remove_const_t<Component>>().size();
return assure<std::remove_const_t<Component>>().size();
}
/**
@@ -252,7 +262,7 @@ public:
if constexpr(sizeof...(Component) == 0) {
entities.reserve(cap);
} else {
(storage<Component>().reserve(cap), ...);
(assure<Component>().reserve(cap), ...);
}
}
@@ -263,7 +273,7 @@ public:
*/
template<typename Component>
[[nodiscard]] size_type capacity() const {
return storage<std::remove_const_t<Component>>().capacity();
return assure<std::remove_const_t<Component>>().capacity();
}
/**
@@ -282,7 +292,7 @@ public:
*/
template<typename... Component>
void shrink_to_fit() {
(storage<Component>().shrink_to_fit(), ...);
(assure<Component>().shrink_to_fit(), ...);
}
/**
@@ -301,7 +311,7 @@ public:
if constexpr(sizeof...(Component) == 0) {
return !alive();
} else {
return (storage<std::remove_const_t<Component>>().empty() && ...);
return (assure<std::remove_const_t<Component>>().empty() && ...);
}
}
@@ -580,7 +590,7 @@ public:
template<typename Component, typename... Args>
decltype(auto) emplace(const entity_type entity, Args &&...args) {
ENTT_ASSERT(valid(entity), "Invalid entity");
return storage<Component>().emplace(entity, std::forward<Args>(args)...);
return assure<Component>().emplace(entity, std::forward<Args>(args)...);
}
/**
@@ -597,7 +607,7 @@ public:
template<typename Component, typename It>
void insert(It first, It last, const Component &value = {}) {
ENTT_ASSERT(std::all_of(first, last, [this](const auto entity) { return valid(entity); }), "Invalid entity");
storage<Component>().insert(first, last, value);
assure<Component>().insert(first, last, value);
}
/**
@@ -616,7 +626,7 @@ public:
void insert(EIt first, EIt last, CIt from) {
static_assert(std::is_constructible_v<Component, typename std::iterator_traits<CIt>::value_type>, "Invalid value type");
ENTT_ASSERT(std::all_of(first, last, [this](const auto entity) { return valid(entity); }), "Invalid entity");
storage<Component>().insert(first, last, from);
assure<Component>().insert(first, last, from);
}
/**
@@ -634,7 +644,7 @@ public:
template<typename Component, typename... Args>
decltype(auto) emplace_or_replace(const entity_type entity, Args &&...args) {
ENTT_ASSERT(valid(entity), "Invalid entity");
auto &cpool = storage<Component>();
auto &cpool = assure<Component>();
return cpool.contains(entity)
? cpool.patch(entity, [&args...](auto &...curr) { ((curr = Component{std::forward<Args>(args)...}), ...); })
@@ -668,7 +678,7 @@ public:
template<typename Component, typename... Func>
decltype(auto) patch(const entity_type entity, Func &&...func) {
ENTT_ASSERT(valid(entity), "Invalid entity");
return storage<Component>().patch(entity, std::forward<Func>(func)...);
return assure<Component>().patch(entity, std::forward<Func>(func)...);
}
/**
@@ -691,7 +701,7 @@ public:
template<typename Component, typename... Args>
decltype(auto) replace(const entity_type entity, Args &&...args) {
ENTT_ASSERT(valid(entity), "Invalid entity");
return storage<Component>().patch(entity, [&args...](auto &...curr) { ((curr = Component{std::forward<Args>(args)...}), ...); });
return assure<Component>().patch(entity, [&args...](auto &...curr) { ((curr = Component{std::forward<Args>(args)...}), ...); });
}
/**
@@ -708,7 +718,7 @@ public:
size_type remove(const entity_type entity) {
ENTT_ASSERT(valid(entity), "Invalid entity");
static_assert(sizeof...(Component) > 0, "Provide one or more component types");
return (storage<Component>().remove(entity) + ... + size_type{});
return (assure<Component>().remove(entity) + ... + size_type{});
}
/**
@@ -725,7 +735,7 @@ public:
template<typename... Component, typename It>
size_type remove(It first, It last) {
static_assert(sizeof...(Component) > 0, "Provide one or more component types");
const auto cpools = std::forward_as_tuple(storage<Component>()...);
const auto cpools = std::forward_as_tuple(assure<Component>()...);
size_type count{};
for(; first != last; ++first) {
@@ -751,7 +761,7 @@ public:
void erase(const entity_type entity) {
ENTT_ASSERT(valid(entity), "Invalid entity");
static_assert(sizeof...(Component) > 0, "Provide one or more component types");
(storage<Component>().erase(entity), ...);
(assure<Component>().erase(entity), ...);
}
/**
@@ -767,7 +777,7 @@ public:
template<typename... Component, typename It>
void erase(It first, It last) {
static_assert(sizeof...(Component) > 0, "Provide one or more component types");
const auto cpools = std::forward_as_tuple(storage<Component>()...);
const auto cpools = std::forward_as_tuple(assure<Component>()...);
for(; first != last; ++first) {
const auto entity = *first;
@@ -788,7 +798,7 @@ public:
curr.second->compact();
}
} else {
(storage<Component>().compact(), ...);
(assure<Component>().compact(), ...);
}
}
@@ -805,7 +815,7 @@ public:
template<typename... Component>
[[nodiscard]] bool all_of(const entity_type entity) const {
ENTT_ASSERT(valid(entity), "Invalid entity");
return (storage<std::remove_const_t<Component>>().contains(entity) && ...);
return (assure<std::remove_const_t<Component>>().contains(entity) && ...);
}
/**
@@ -822,7 +832,7 @@ public:
template<typename... Component>
[[nodiscard]] bool any_of(const entity_type entity) const {
ENTT_ASSERT(valid(entity), "Invalid entity");
return (storage<std::remove_const_t<Component>>().contains(entity) || ...);
return (assure<std::remove_const_t<Component>>().contains(entity) || ...);
}
/**
@@ -841,7 +851,7 @@ public:
ENTT_ASSERT(valid(entity), "Invalid entity");
if constexpr(sizeof...(Component) == 1) {
return storage<std::remove_const_t<Component>...>().get(entity);
return assure<std::remove_const_t<Component>...>().get(entity);
} else {
return std::forward_as_tuple(get<Component>(entity)...);
}
@@ -853,7 +863,7 @@ public:
ENTT_ASSERT(valid(entity), "Invalid entity");
if constexpr(sizeof...(Component) == 1) {
return (const_cast<Component &>(storage<std::remove_const_t<Component>>().get(entity)), ...);
return (const_cast<Component &>(assure<std::remove_const_t<Component>>().get(entity)), ...);
} else {
return std::forward_as_tuple(get<Component>(entity)...);
}
@@ -877,7 +887,7 @@ public:
template<typename Component, typename... Args>
[[nodiscard]] decltype(auto) get_or_emplace(const entity_type entity, Args &&...args) {
ENTT_ASSERT(valid(entity), "Invalid entity");
auto &cpool = storage<Component>();
auto &cpool = assure<Component>();
return cpool.contains(entity) ? cpool.get(entity) : cpool.emplace(entity, std::forward<Args>(args)...);
}
@@ -899,7 +909,7 @@ public:
ENTT_ASSERT(valid(entity), "Invalid entity");
if constexpr(sizeof...(Component) == 1) {
const auto &cpool = storage<std::remove_const_t<Component>...>();
const auto &cpool = assure<std::remove_const_t<Component>...>();
return cpool.contains(entity) ? std::addressof(cpool.get(entity)) : nullptr;
} else {
return std::make_tuple(try_get<Component>(entity)...);
@@ -929,7 +939,7 @@ public:
each([this](const auto entity) { release_entity(entity, entity_traits::to_version(entity) + 1u); });
} else {
(storage<Component>().clear(), ...);
(assure<Component>().clear(), ...);
}
}
@@ -1022,7 +1032,7 @@ public:
*/
template<typename Component>
[[nodiscard]] auto on_construct() {
return storage<Component>().on_construct();
return assure<Component>().on_construct();
}
/**
@@ -1045,7 +1055,7 @@ public:
*/
template<typename Component>
[[nodiscard]] auto on_update() {
return storage<Component>().on_update();
return assure<Component>().on_update();
}
/**
@@ -1070,7 +1080,7 @@ public:
*/
template<typename Component>
[[nodiscard]] auto on_destroy() {
return storage<Component>().on_destroy();
return assure<Component>().on_destroy();
}
/**
@@ -1108,14 +1118,14 @@ public:
template<typename... Component, typename... Exclude>
[[nodiscard]] basic_view<Entity, get_t<std::add_const_t<Component>...>, exclude_t<Exclude...>> view(exclude_t<Exclude...> = {}) const {
static_assert(sizeof...(Component) > 0, "Exclusion-only views are not supported");
return {storage<std::remove_const_t<Component>>()..., storage<Exclude>()...};
return {assure<std::remove_const_t<Component>>()..., assure<Exclude>()...};
}
/*! @copydoc view */
template<typename... Component, typename... Exclude>
[[nodiscard]] basic_view<Entity, get_t<Component...>, exclude_t<Exclude...>> view(exclude_t<Exclude...> = {}) {
static_assert(sizeof...(Component) > 0, "Exclusion-only views are not supported");
return {storage<std::remove_const_t<Component>>()..., storage<Exclude>()...};
return {assure<std::remove_const_t<Component>>()..., assure<Exclude>()...};
}
/**
@@ -1201,7 +1211,7 @@ public:
using handler_type = group_handler<exclude_t<Exclude...>, get_t<std::remove_const_t<Get>...>, std::remove_const_t<Owned>...>;
const auto cpools = std::forward_as_tuple(storage<std::remove_const_t<Owned>>()..., storage<std::remove_const_t<Get>>()...);
const auto cpools = std::forward_as_tuple(assure<std::remove_const_t<Owned>>()..., assure<std::remove_const_t<Get>>()...);
constexpr auto size = sizeof...(Owned) + sizeof...(Get) + sizeof...(Exclude);
handler_type *handler = nullptr;
@@ -1289,7 +1299,7 @@ public:
return {};
} else {
using handler_type = group_handler<exclude_t<Exclude...>, get_t<std::remove_const_t<Get>...>, std::remove_const_t<Owned>...>;
return {static_cast<handler_type *>(it->group.get())->current, storage<std::remove_const_t<Owned>>()..., storage<std::remove_const_t<Get>>()...};
return {static_cast<handler_type *>(it->group.get())->current, assure<std::remove_const_t<Owned>>()..., assure<std::remove_const_t<Get>>()...};
}
}
@@ -1371,7 +1381,7 @@ public:
template<typename Component, typename Compare, typename Sort = std_sort, typename... Args>
void sort(Compare compare, Sort algo = Sort{}, Args &&...args) {
ENTT_ASSERT(sortable<Component>(), "Cannot sort owned storage");
auto &cpool = storage<Component>();
auto &cpool = assure<Component>();
if constexpr(std::is_invocable_v<Compare, decltype(cpool.get({})), decltype(cpool.get({}))>) {
auto comp = [&cpool, compare = std::move(compare)](const auto lhs, const auto rhs) { return compare(std::as_const(cpool.get(lhs)), std::as_const(cpool.get(rhs))); };
@@ -1408,7 +1418,7 @@ public:
template<typename To, typename From>
void sort() {
ENTT_ASSERT(sortable<To>(), "Cannot sort owned storage");
storage<To>().respect(storage<From>());
assure<To>().respect(assure<From>());
}
/**

View File

@@ -1939,6 +1939,14 @@ TEST(Registry, RuntimePools) {
auto &storage = registry.storage<empty_type>("other"_hs);
const auto entity = registry.create();
static_assert(std::is_same_v<decltype(registry.storage<empty_type>()), typename entt::storage_traits<entt::entity, empty_type>::storage_type &>);
static_assert(std::is_same_v<decltype(registry.storage<const empty_type>()), const typename entt::storage_traits<entt::entity, empty_type>::storage_type &>);
static_assert(std::is_same_v<decltype(std::as_const(registry).storage<empty_type>()), const typename entt::storage_traits<entt::entity, empty_type>::storage_type &>);
static_assert(std::is_same_v<decltype(std::as_const(registry).storage<const empty_type>()), const typename entt::storage_traits<entt::entity, empty_type>::storage_type &>);
ASSERT_EQ(&storage, &registry.storage<const empty_type>("other"_hs));
ASSERT_EQ(&registry.storage<empty_type>(), &registry.storage<const empty_type>());
ASSERT_FALSE(registry.any_of<empty_type>(entity));
ASSERT_FALSE(storage.contains(entity));