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"
24 using entity_iterator = std::span<const Entity>::iterator;
29 virtual auto Id() const noexcept ->
size_t = 0;
42 virtual auto
Size() const noexcept ->
size_t = 0;
48 virtual auto
TotalSize() const noexcept ->
size_t = 0;
51 virtual
void Reserve(
size_t capacity) = 0;
69 virtual auto
HasDrawUI() const noexcept ->
bool = 0;
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;
109 : m_storage{maxEntities}, m_handler{std::move(handler)} {}
112 template<
class... Args>
128 auto GetComponents() noexcept -> std::span<T> {
return m_storage.GetPackedArray(); }
131 auto GetComponents() const noexcept -> std::span<const T> {
return m_storage.GetPackedArray(); }
137 auto OnAdd() noexcept ->
Signal<T&>&
144 auto OnCommit() noexcept ->
Signal<T&>&
151 auto OnRemove() noexcept -> Signal<Entity>&
152 requires StoragePolicy<T>::EnableOnRemoveCallbacks
158 template<std::predicate<const T&, const T&> Predicate>
159 void Sort(Predicate&& compare);
162 auto begin() noexcept {
return std::ranges::begin(m_storage.GetPackedArray()); }
165 auto begin() const noexcept {
return std::ranges::begin(m_storage.GetPackedArray()); }
168 auto end() noexcept {
return std::ranges::end(m_storage.GetPackedArray()); }
171 auto end() const noexcept {
return std::ranges::end(m_storage.GetPackedArray()); }
177 [[nodiscard]]
auto empty() const noexcept {
return GetComponents().empty(); }
180 auto operator[](
size_t pos)
noexcept -> T& {
return GetComponents()[pos]; }
183 auto operator[](
size_t pos)
const noexcept ->
const T& {
return GetComponents()[pos]; }
186 auto at(
size_t pos) -> T& {
return GetComponents().at(pos); }
189 auto at(
size_t pos)
const ->
const T& {
return GetComponents().at(pos); }
192 auto data() noexcept {
return GetComponents().data(); }
195 auto data() const noexcept {
return GetComponents().data(); }
200 auto Size() const noexcept ->
size_t override {
return m_storage.Size(); }
201 auto StagedSize() const noexcept ->
size_t override {
return m_staging.size(); }
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; };
219 ecs::detail::SparseSet<T> m_storage;
220 std::vector<detail::StagedComponent<T>> m_staging;
228template<PooledComponent T>
229template<
class... Args>
232 NC_ASSERT(entity.Index() < m_storage.MaxSize() && !
Contains(entity),
"Bad entity");
233 auto& [_, component] = m_staging.emplace_back(
235 detail::Construct<T>(entity, std::forward<Args>(args)...)
239 m_onAdd.Emit(component);
244template<PooledComponent T>
247 NC_ASSERT(entity.Index() < m_storage.MaxSize() && !
Contains(entity),
"Bad entity");
248 auto& [_, component] = m_staging.emplace_back(entity, std::move(obj));
251 m_onAdd.Emit(component);
256template<PooledComponent T>
259 NC_ASSERT(entity.Valid(),
"Bad entity");
260 if (!m_storage.Remove(entity) && !detail::EraseUnstable(m_staging, entity))
264 m_onRemove.Emit(entity);
269template<PooledComponent T>
272 return m_storage.Contains(entity)
274 : std::cend(m_staging) != std::ranges::find(m_staging, entity, &detail::StagedComponent<T>::entity);
277template<PooledComponent T>
280 NC_ASSERT(entity.Valid(),
"Bad entity");
281 if (m_storage.Contains(entity))
282 return m_storage.Get(entity);
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;
289template<PooledComponent T>
292 NC_ASSERT(entity.Valid(),
"Bad entity");
293 if (m_storage.Contains(entity))
294 return m_storage.Get(entity);
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;
301template<PooledComponent T>
304 return Contains(entity) ? AnyComponent{&Get(entity), &m_handler} : AnyComponent{};
307template<PooledComponent T>
310 if (
auto parent = m_storage.GetParent(component); parent.Valid())
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();
317template<PooledComponent T>
318template<std::predicate<const T&, const T&> Pred>
319void ComponentPool<T>::Sort(Pred&& compare)
321 m_storage.Sort(std::forward<Pred>(compare));
324template<PooledComponent T>
327 m_storage.Reserve(capacity);
329 if (capacity > existing)
333 m_staging.reserve(capacity - existing);
337template<PooledComponent T>
340 if (m_handler.factory)
342 auto& comp = Insert(entity, m_handler.factory(entity, m_handler.userData));
343 return AnyComponent{&comp, &m_handler};
346 return AnyComponent{};
349template<PooledComponent T>
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);
356template<PooledComponent T>
359 NC_ASSERT(
HasSerialize(),
"Component is not serializable");
360 Insert(entity, m_handler.deserialize(stream, ctx, m_handler.userData));
363template<PooledComponent T>
366 for(
auto entity : deleted)
369 for(
auto& [entity, component] : m_staging)
371 [[maybe_unused]]
auto& committed = m_storage.Insert(entity, std::move(component));
373 m_onCommit(committed);
379template<PooledComponent T>
383 m_staging.shrink_to_fit();
384 m_storage.ClearNonPersistent();
387template<PooledComponent T>
391 m_staging.shrink_to_fit();
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