NcEngine
ComponentPool.h
Go to the documentation of this file.
1
5#pragma once
6
7#include "ncengine/ecs/AnyComponent.h"
8#include "ncengine/ecs/detail/PoolUtility.h"
9#include "ncengine/ecs/detail/SparseSet.h"
10#include "ncengine/type/StableAddress.h"
12
13#include "ncutility/NcError.h"
14
15#include <cassert>
16#include <string_view>
17
18namespace nc::ecs
19{
22{
23 public:
24 using entity_iterator = std::span<const Entity>::iterator;
25
26 virtual ~ComponentPoolBase() = default;
27
29 virtual auto Id() const noexcept -> size_t = 0;
30
32 virtual auto Contains(Entity entity) const -> bool = 0;
33
36 virtual auto GetAsAnyComponent(Entity entity) -> AnyComponent = 0;
37
39 virtual auto Remove(Entity entity) -> bool = 0;
40
42 virtual auto Size() const noexcept -> size_t = 0;
43
45 virtual auto StagedSize() const noexcept -> size_t = 0;
46
48 virtual auto TotalSize() const noexcept -> size_t = 0;
49
51 virtual void Reserve(size_t capacity) = 0;
52
54 virtual auto GetEntityPool() const noexcept -> std::span<const Entity> = 0;
55
57 virtual auto GetComponentName() const noexcept -> std::string_view = 0;
58
60 virtual auto HasFactory() const noexcept -> bool = 0;
61
63 virtual auto HasUserData() const noexcept -> bool = 0;
64
66 virtual auto HasSerialize() const noexcept -> bool = 0;
67
69 virtual auto HasDrawUI() const noexcept -> bool = 0;
70
75 virtual auto AddDefault(Entity entity) -> AnyComponent = 0;
76
78 virtual void Serialize(std::ostream& stream, Entity entity, const SerializationContext& ctx) = 0;
79
81 virtual void Deserialize(std::istream& stream, Entity entity, const DeserializationContext& ctx) = 0;
82
85 virtual void ClearNonPersistent() = 0;
86
89 virtual void Clear() = 0;
90
94 virtual void CommitStagedComponents(const std::vector<Entity>& deleted) = 0;
95};
96
98template<PooledComponent T>
99class ComponentPool final : public ComponentPoolBase
100{
101 public:
102 using value_type = T;
103 using iterator = std::vector<T>::iterator;
104 using const_iterator = std::vector<T>::const_iterator;
105 using reverse_iterator = std::vector<T>::reverse_iterator;
106
108 explicit ComponentPool(size_t maxEntities, ComponentHandler<T> handler)
109 : m_storage{maxEntities}, m_handler{std::move(handler)} {}
110
112 template<class... Args>
113 auto Emplace(Entity entity, Args&&... args) -> T&;
114
116 auto Insert(Entity entity, T obj) -> T&;
117
119 auto Get(Entity entity) -> T&;
120
122 auto Get(Entity entity) const -> const T&;
123
125 auto GetParent(const T* component) const -> Entity;
126
128 auto GetComponents() noexcept -> std::span<T> { return m_storage.GetPackedArray(); }
129
131 auto GetComponents() const noexcept -> std::span<const T> { return m_storage.GetPackedArray(); }
132
134 auto Handler() noexcept -> ComponentHandler<T>& { return m_handler; }
135
137 auto OnAdd() noexcept -> Signal<T&>&
138 requires StoragePolicy<T>::EnableOnAddCallbacks
139 {
140 return m_onAdd;
141 }
142
144 auto OnCommit() noexcept -> Signal<T&>&
145 requires StoragePolicy<T>::EnableOnCommitCallbacks
146 {
147 return m_onCommit;
148 }
149
151 auto OnRemove() noexcept -> Signal<Entity>&
152 requires StoragePolicy<T>::EnableOnRemoveCallbacks
153 {
154 return m_onRemove;
155 }
156
158 template<std::predicate<const T&, const T&> Predicate>
159 void Sort(Predicate&& compare);
160
162 auto begin() noexcept { return std::ranges::begin(m_storage.GetPackedArray()); }
163
165 auto begin() const noexcept { return std::ranges::begin(m_storage.GetPackedArray()); }
166
168 auto end() noexcept { return std::ranges::end(m_storage.GetPackedArray()); }
169
171 auto end() const noexcept { return std::ranges::end(m_storage.GetPackedArray()); }
172
174 auto size() const noexcept { return Size(); }
175
177 [[nodiscard]] auto empty() const noexcept { return GetComponents().empty(); }
178
180 auto operator[](size_t pos) noexcept -> T& { return GetComponents()[pos]; }
181
183 auto operator[](size_t pos) const noexcept -> const T& { return GetComponents()[pos]; }
184
186 auto at(size_t pos) -> T& { return GetComponents().at(pos); }
187
189 auto at(size_t pos) const -> const T& { return GetComponents().at(pos); }
190
192 auto data() noexcept { return GetComponents().data(); }
193
195 auto data() const noexcept { return GetComponents().data(); }
196
197 auto Contains(Entity entity) const -> bool override;
198 auto GetAsAnyComponent(Entity entity) -> AnyComponent override;
199 auto Remove(Entity entity) -> bool override;
200 auto Size() const noexcept -> size_t override { return m_storage.Size(); }
201 auto StagedSize() const noexcept -> size_t override { return m_staging.size(); }
202 auto TotalSize() const noexcept -> size_t override { return Size() + StagedSize(); }
203 void Reserve(size_t capacity) override;
204 auto GetEntityPool() const noexcept -> std::span<const Entity> override { return m_storage.GetEntities(); }
205 auto GetComponentName() const noexcept -> std::string_view override { return m_handler.name; }
206 auto HasFactory() const noexcept -> bool override { return m_handler.factory != nullptr; }
207 auto HasUserData() const noexcept -> bool override { return m_handler.userData.has_value(); }
208 auto HasSerialize() const noexcept -> bool override { return m_handler.serialize && m_handler.deserialize; }
209 auto HasDrawUI() const noexcept -> bool override { return m_handler.drawUI != nullptr; }
210 auto Id() const noexcept -> size_t override { return m_handler.id; };
211 auto AddDefault(Entity entity) -> AnyComponent override;
212 void Serialize(std::ostream& stream, Entity entity, const SerializationContext& ctx) override;
213 void Deserialize(std::istream& stream, Entity entity, const DeserializationContext& ctx) override;
214 void ClearNonPersistent() override;
215 void Clear() override;
216 void CommitStagedComponents(const std::vector<Entity>& deleted) override;
217
218 private:
219 ecs::detail::SparseSet<T> m_storage;
220 std::vector<detail::StagedComponent<T>> m_staging;
221 ComponentHandler<T> m_handler;
222 Signal<T&> m_onAdd;
223 Signal<T&> m_onCommit;
224 Signal<Entity> m_onRemove;
225};
226
228template<PooledComponent T>
229template<class... Args>
230auto ComponentPool<T>::Emplace(Entity entity, Args&&... args) -> T&
231{
232 NC_ASSERT(entity.Index() < m_storage.MaxSize() && !Contains(entity), "Bad entity");
233 auto& [_, component] = m_staging.emplace_back(
234 entity,
235 detail::Construct<T>(entity, std::forward<Args>(args)...)
236 );
237
239 m_onAdd.Emit(component);
240
241 return component;
242}
243
244template<PooledComponent T>
245auto ComponentPool<T>::Insert(Entity entity, T obj) -> T&
246{
247 NC_ASSERT(entity.Index() < m_storage.MaxSize() && !Contains(entity), "Bad entity");
248 auto& [_, component] = m_staging.emplace_back(entity, std::move(obj));
249
251 m_onAdd.Emit(component);
252
253 return component;
254}
255
256template<PooledComponent T>
257auto ComponentPool<T>::Remove(Entity entity) -> bool
258{
259 NC_ASSERT(entity.Valid(), "Bad entity");
260 if (!m_storage.Remove(entity) && !detail::EraseUnstable(m_staging, entity))
261 return false;
262
264 m_onRemove.Emit(entity);
265
266 return true;
267}
268
269template<PooledComponent T>
270bool ComponentPool<T>::Contains(Entity entity) const
271{
272 return m_storage.Contains(entity)
273 ? true
274 : std::cend(m_staging) != std::ranges::find(m_staging, entity, &detail::StagedComponent<T>::entity);
275}
276
277template<PooledComponent T>
278auto ComponentPool<T>::Get(Entity entity) -> T&
279{
280 NC_ASSERT(entity.Valid(), "Bad entity");
281 if (m_storage.Contains(entity))
282 return m_storage.Get(entity);
283
284 auto pos = std::ranges::find(m_staging, entity, &detail::StagedComponent<T>::entity);
285 NC_ASSERT(pos != std::cend(m_staging), "Component does not exist");
286 return pos->component;
287}
288
289template<PooledComponent T>
290auto ComponentPool<T>::Get(Entity entity) const -> const T&
291{
292 NC_ASSERT(entity.Valid(), "Bad entity");
293 if (m_storage.Contains(entity))
294 return m_storage.Get(entity);
295
296 auto pos = std::ranges::find(m_staging, entity);
297 NC_ASSERT(pos != std::cend(m_staging), "Component does not exist");
298 return pos->component;
299}
300
301template<PooledComponent T>
302auto ComponentPool<T>::GetAsAnyComponent(Entity entity) -> AnyComponent
303{
304 return Contains(entity) ? AnyComponent{&Get(entity), &m_handler} : AnyComponent{};
305}
306
307template<PooledComponent T>
308auto ComponentPool<T>::GetParent(const T* component) const -> Entity
309{
310 if (auto parent = m_storage.GetParent(component); parent.Valid())
311 return parent;
312
313 const auto pos = std::ranges::find(m_staging, component, [](auto&& staged) { return &staged.component; });
314 return pos != std::cend(m_staging) ? pos->entity : Entity::Null();
315}
316
317template<PooledComponent T>
318template<std::predicate<const T&, const T&> Pred>
319void ComponentPool<T>::Sort(Pred&& compare)
320{
321 m_storage.Sort(std::forward<Pred>(compare));
322}
323
324template<PooledComponent T>
325void ComponentPool<T>::Reserve(size_t capacity)
326{
327 m_storage.Reserve(capacity);
328 const auto existing = TotalSize();
329 if (capacity > existing)
330 {
331 // We don't need to reserve the whole capacity for staging - we just
332 // want to guarantee ptrs remain valid until we exceed capacity.
333 m_staging.reserve(capacity - existing);
334 }
335}
336
337template<PooledComponent T>
338auto ComponentPool<T>::AddDefault(Entity entity) -> AnyComponent
339{
340 if (m_handler.factory)
341 {
342 auto& comp = Insert(entity, m_handler.factory(entity, m_handler.userData));
343 return AnyComponent{&comp, &m_handler};
344 }
345
346 return AnyComponent{};
347}
348
349template<PooledComponent T>
350void ComponentPool<T>::Serialize(std::ostream& stream, Entity entity, const SerializationContext& ctx)
351{
352 NC_ASSERT(HasSerialize() && Contains(entity), "Component does not exist or is not serializable");
353 m_handler.serialize(stream, Get(entity), ctx, m_handler.userData);
354}
355
356template<PooledComponent T>
357void ComponentPool<T>::Deserialize(std::istream& stream, Entity entity, const DeserializationContext& ctx)
358{
359 NC_ASSERT(HasSerialize(), "Component is not serializable");
360 Insert(entity, m_handler.deserialize(stream, ctx, m_handler.userData));
361}
362
363template<PooledComponent T>
364void ComponentPool<T>::CommitStagedComponents(const std::vector<Entity>& deleted)
365{
366 for(auto entity : deleted)
367 Remove(entity);
368
369 for(auto& [entity, component] : m_staging)
370 {
371 [[maybe_unused]] auto& committed = m_storage.Insert(entity, std::move(component));
373 m_onCommit(committed);
374 }
375
376 m_staging.clear();
377}
378
379template<PooledComponent T>
381{
382 m_staging.clear();
383 m_staging.shrink_to_fit();
384 m_storage.ClearNonPersistent();
385}
386
387template<PooledComponent T>
389{
390 m_staging.clear();
391 m_staging.shrink_to_fit();
392 m_storage.Clear();
393}
395} // namespace nc::ecs
Generic interface around component types.
Definition: AnyComponent.h:11
Identifies an object in the registry.
Definition: Entity.h:18
static constexpr auto Null() noexcept
Construct a null Entity.
Definition: Entity.h:61
An event source supporting multiple Connections.
Definition: Signal.h:65
Base class for non-copyable non-movable types.
Definition: StableAddress.h:7
Type-agnostic base class for component pools.
Definition: ComponentPool.h:22
virtual auto AddDefault(Entity entity) -> AnyComponent=0
Factory-construct a component attached to an entity.
virtual auto Contains(Entity entity) const -> bool=0
Check if the pool has a component attached to an entity.
virtual auto GetEntityPool() const noexcept -> std::span< const Entity >=0
Get a span containing the entities with components in the pool.
virtual auto Id() const noexcept -> size_t=0
Get the component's unique id.
virtual auto HasUserData() const noexcept -> bool=0
Check if userData is set in the component's ComponentHandler.
virtual void Clear()=0
Remove all components.
virtual auto Size() const noexcept -> size_t=0
Get the number of components in the pool, excluding those still staged.
virtual auto Remove(Entity entity) -> bool=0
Remove the component attached to an entity.
virtual void CommitStagedComponents(const std::vector< Entity > &deleted)=0
Finalize pending changes by merging staged components and removing data for any entities deleted from...
virtual auto GetAsAnyComponent(Entity entity) -> AnyComponent=0
Get the component attached to an entity as an AnyComponent.
virtual auto TotalSize() const noexcept -> size_t=0
Get the combined number of components in the pool and staged.
virtual void Serialize(std::ostream &stream, Entity entity, const SerializationContext &ctx)=0
Serialize the component attached to an entity to a binary stream.
virtual auto HasSerialize() const noexcept -> bool=0
Check if the serialize and deserialize callbacks are set in the component's ComponentHandler.
virtual void ClearNonPersistent()=0
Remove all components not attached to a persistent entity.
virtual auto HasDrawUI() const noexcept -> bool=0
Check if the drawUI callback set in the the component's ComponentHandler.
virtual auto StagedSize() const noexcept -> size_t=0
Get the number of staged components waiting to be merged into the pool.
virtual void Reserve(size_t capacity)=0
Pre-allocate space for some number of components.
virtual void Deserialize(std::istream &stream, Entity entity, const DeserializationContext &ctx)=0
Deserialize a component from a binary stream and attach it to an entity.
virtual auto GetComponentName() const noexcept -> std::string_view=0
Get the name from the component's ComponentHandler.
virtual auto HasFactory() const noexcept -> bool=0
Check if the factory callback is set in the component's ComponentHandler.
Type-aware implementation for component pools.
Definition: ComponentPool.h:100
auto TotalSize() const noexcept -> size_t override
Get the combined number of components in the pool and staged.
Definition: ComponentPool.h:202
auto HasUserData() const noexcept -> bool override
Check if userData is set in the component's ComponentHandler.
Definition: ComponentPool.h:207
auto GetParent(const T *component) const -> Entity
Get the entity a component is attached to, or a null entity on failure.
ComponentPool(size_t maxEntities, ComponentHandler< T > handler)
Construct a new component pool.
Definition: ComponentPool.h:108
auto end() noexcept
Get an iterator one past the last component in the pool.
Definition: ComponentPool.h:168
void Clear() override
Remove all components.
auto GetAsAnyComponent(Entity entity) -> AnyComponent override
Get the component attached to an entity as an AnyComponent.
void Serialize(std::ostream &stream, Entity entity, const SerializationContext &ctx) override
Serialize the component attached to an entity to a binary stream.
auto operator[](size_t pos) noexcept -> T &
Get a reference to the component at the specified position.
Definition: ComponentPool.h:180
void Deserialize(std::istream &stream, Entity entity, const DeserializationContext &ctx) override
Deserialize a component from a binary stream and attach it to an entity.
auto HasFactory() const noexcept -> bool override
Check if the factory callback is set in the component's ComponentHandler.
Definition: ComponentPool.h:206
auto AddDefault(Entity entity) -> AnyComponent override
Factory-construct a component attached to an entity.
auto Remove(Entity entity) -> bool override
Remove the component attached to an entity.
void ClearNonPersistent() override
Remove all components not attached to a persistent entity.
auto at(size_t pos) -> T &
Get a reference to the component at the specified position with bounds checking.
Definition: ComponentPool.h:186
auto Handler() noexcept -> ComponentHandler< T > &
Get the T's ComponentHandler.
Definition: ComponentPool.h:134
auto GetComponentName() const noexcept -> std::string_view override
Get the name from the component's ComponentHandler.
Definition: ComponentPool.h:205
auto Id() const noexcept -> size_t override
Get the component's unique id.
Definition: ComponentPool.h:210
auto data() const noexcept
Get a pointer to the underlying component array.
Definition: ComponentPool.h:195
void CommitStagedComponents(const std::vector< Entity > &deleted) override
Finalize pending changes by merging staged components and removing data for any entities deleted from...
auto Get(Entity entity) const -> const T &
Get a const pointer to the component attached to an entity, or nullptr if one does not exist.
auto begin() const noexcept
Get a const_iterator the the first component in the pool.
Definition: ComponentPool.h:165
auto HasDrawUI() const noexcept -> bool override
Check if the drawUI callback set in the the component's ComponentHandler.
Definition: ComponentPool.h:209
void Reserve(size_t capacity) override
Pre-allocate space for some number of components.
auto GetComponents() noexcept -> std::span< T >
View all components as a contiguous range.
Definition: ComponentPool.h:128
auto data() noexcept
Get a pointer to the underlying component array.
Definition: ComponentPool.h:192
auto empty() const noexcept
Check if there are no components in the pool.
Definition: ComponentPool.h:177
auto Insert(Entity entity, T obj) -> T &
Insert a component attached to an entity.
auto HasSerialize() const noexcept -> bool override
Check if the serialize and deserialize callbacks are set in the component's ComponentHandler.
Definition: ComponentPool.h:208
auto OnAdd() noexcept -> Signal< T & > &auto begin() noexcept
Get the T's onAdd event Signal.
Definition: ComponentPool.h:162
auto StagedSize() const noexcept -> size_t override
Get the number of staged components waiting to be merged into the pool.
Definition: ComponentPool.h:201
auto Get(Entity entity) -> T &
Get a pointer to the component attached to an entity, or nullptr if one does not exist.
auto Contains(Entity entity) const -> bool override
Check if the pool has a component attached to an entity.
auto GetEntityPool() const noexcept -> std::span< const Entity > override
Get a span containing the entities with components in the pool.
Definition: ComponentPool.h:204
auto at(size_t pos) const -> const T &
Get a reference to the component at the specified position with bounds checking.
Definition: ComponentPool.h:189
auto size() const noexcept
Get the number of components committed to the pool.
Definition: ComponentPool.h:174
auto operator[](size_t pos) const noexcept -> const T &
Get a const reference to the component at the specified position.
Definition: ComponentPool.h:183
auto Emplace(Entity entity, Args &&... args) -> T &
Emplace a component attached to an entity.
auto end() const noexcept
Get a const_iterator one past the last component in the pool.
Definition: ComponentPool.h:171
auto Size() const noexcept -> size_t override
Get the number of components in the pool, excluding those still staged.
Definition: ComponentPool.h:200
auto GetComponents() const noexcept -> std::span< const T >
View all components as a contiguous range.
Definition: ComponentPool.h:131
Requirements for the Registry to recognize a pooled component.
Definition: Component.h:61
Optional data and callbacks for generic component operations.
Definition: Component.h:101
static constexpr bool EnableOnCommitCallbacks
Enable the OnCommit Signal in the component's pool.
Definition: Component.h:75
static constexpr bool EnableOnRemoveCallbacks
Enable the OnRemove Signal in the component's pool.
Definition: Component.h:78
Context object passed to deserialization functions.
Definition: SceneSerialization.h:43
Context object passed to serialization functions.
Definition: SceneSerialization.h:36
Provide a specialization to customize storage options and behavior for a user-defined type.
Definition: Component.h:88