5#include "ncutility/Algorithm.h"
14namespace nc::ecs::detail
36 using index_type = Entity::index_type;
38 explicit SparseSet(
size_t maxCount)
39 : m_sparse(maxCount, Entity::NullIndex) {}
41 auto Insert(Entity entity,
const T& component) -> T&;
42 auto Insert(Entity entity, T&& component) -> T&;
43 auto Remove(Entity entity) -> bool;
44 auto Get(Entity entity) -> T&;
45 auto Get(Entity entity)
const ->
const T&;
46 auto GetParent(
const T* component)
const noexcept -> Entity;
47 auto GetIndex(Entity entity)
const -> index_type;
48 auto GetEntity(index_type index)
const -> Entity;
49 auto Contains(Entity entity)
const noexcept -> bool;
50 auto Size() const noexcept ->
size_t {
return m_dense.size(); }
51 auto MaxSize() const noexcept ->
size_t {
return m_sparse.size(); }
52 void Swap(Entity lhs, Entity rhs);
53 void Reserve(
size_t capacity);
54 auto GetPackedArray() noexcept -> std::span<T> {
return m_dense; }
55 auto GetPackedArray() const noexcept -> std::span<const T> {
return m_dense; }
56 auto GetEntities() const noexcept -> std::span<const Entity> {
return m_entities; }
57 void ClearNonPersistent();
60 template<std::predicate<const T&, const T&> Predicate>
61 void Sort(Predicate&& lessThan);
64 std::vector<index_type> m_sparse;
65 std::vector<T> m_dense;
66 std::vector<Entity> m_entities;
68 void InsertEntity(Entity entity);
72void SparseSet<T>::InsertEntity(Entity entity)
74 auto& denseIndex = m_sparse.at(entity.Index());
76 throw NcError{
"Entity already exists."};
78 denseIndex =
static_cast<index_type
>(m_entities.size());
79 m_entities.push_back(entity);
83auto SparseSet<T>::Insert(Entity entity,
const T& component) -> T&
86 m_dense.push_back(component);
87 return m_dense.back();
91auto SparseSet<T>::Insert(Entity entity, T&& component) -> T&
94 m_dense.push_back(std::move(component));
95 return m_dense.back();
99auto SparseSet<T>::Remove(Entity toRemove) ->
bool
101 if (!Contains(toRemove))
104 const auto toRemoveSparse = toRemove.Index();
105 const auto swappedSparse = m_entities.back().Index();
106 const auto toRemoveDense = m_sparse.at(toRemoveSparse);
108 m_dense.at(toRemoveDense) = std::move(m_dense.back());
110 m_entities.at(toRemoveDense) = m_entities.back();
111 m_entities.pop_back();
112 m_sparse.at(swappedSparse) = toRemoveDense;
118auto SparseSet<T>::Get(Entity entity) -> T&
120 return m_dense.at(m_sparse.at(entity.Index()));
124auto SparseSet<T>::Get(Entity entity)
const ->
const T&
126 return m_dense.at(m_sparse.at(entity.Index()));
130auto SparseSet<T>::GetParent(
const T* component)
const noexcept -> Entity
132 const auto beg = m_dense.data();
133 const auto end = beg + m_dense.size();
134 return component >= beg && component < end ? m_entities[static_cast<size_t>(component - beg)]
139auto SparseSet<T>::GetIndex(Entity entity)
const -> index_type
141 return m_sparse.at(entity.Index());
145auto SparseSet<T>::GetEntity(index_type index)
const -> Entity
147 return m_entities.at(m_sparse.at(index));
151auto SparseSet<T>::Contains(Entity entity)
const noexcept ->
bool
153 const auto i = entity.Index();
158void SparseSet<T>::Swap(Entity lhs, Entity rhs)
163 const auto sparseIndex1 = lhs.Index();
164 const auto sparseIndex2 = rhs.Index();
165 const auto denseIndex1 = m_sparse.at(sparseIndex1);
166 const auto denseIndex2 = m_sparse.at(sparseIndex2);
167 std::swap(m_sparse.at(sparseIndex1), m_sparse.at(sparseIndex2));
168 std::swap(m_entities.at(denseIndex1), m_entities.at(denseIndex2));
172void SparseSet<T>::Reserve(
size_t capacity)
174 m_dense.reserve(capacity);
175 m_entities.reserve(capacity);
179template<std::predicate<const T&, const T&> Predicate>
180void SparseSet<T>::Sort(Predicate&& lessThan)
183 const auto size = m_dense.size();
184 auto permutation = std::vector<uint32_t>(size);
185 const auto beg = permutation.begin();
186 const auto end = permutation.end();
187 std::iota(beg, end, 0u);
190 auto compare = [&pool = m_dense, predicate = std::forward<Predicate>(lessThan)](
const auto lhs,
192 {
return predicate(pool[lhs], pool[rhs]); };
195 for (
auto cur = beg; cur != end; ++cur)
197 const auto pos = std::upper_bound(beg, cur, *cur, compare);
198 std::rotate(pos, cur, cur + 1);
202 for (
auto i = 0u; i < size; ++i)
205 auto next = permutation[cur];
209 const auto i1 = permutation[cur];
210 const auto i2 = permutation[next];
211 std::swap(m_dense[i1], m_dense[i2]);
212 std::swap(m_entities[i1], m_entities[i2]);
213 const auto sparse1 = m_entities[i1].Index();
214 const auto sparse2 = m_entities[i2].Index();
215 m_sparse[sparse1] = i2;
216 m_sparse[sparse2] = i1;
217 permutation[cur] = cur;
219 next = permutation[cur];
225void SparseSet<T>::ClearNonPersistent()
228 auto swapToIndex = 0u;
229 for (
auto [denseIndex, entity] : algo::Enumerate(m_entities))
231 if (!entity.IsPersistent())
235 m_sparse.at(entity.Index()) = swapToIndex;
236 if (swapToIndex != denseIndex)
238 m_dense.at(swapToIndex) = std::move(m_dense.at(denseIndex));
239 m_entities.at(swapToIndex) = entity;
245 m_dense.erase(m_dense.begin() + swapToIndex, m_dense.end());
246 m_dense.shrink_to_fit();
247 m_entities.erase(m_entities.begin() + swapToIndex, m_entities.end());
248 m_entities.shrink_to_fit();
252void SparseSet<T>::Clear()
256 m_dense.shrink_to_fit();
258 m_entities.shrink_to_fit();
static constexpr auto Null() noexcept
Construct a null Entity.
Definition: Entity.h:61
static constexpr index_type NullIndex
Index used for an invalid Entity.
Definition: Entity.h:26