NcEngine
SparseSet.h
1#pragma once
2
4
5#include "ncutility/Algorithm.h"
6
7#include <algorithm>
8#include <numeric>
9#include <span>
10#include <utility>
11#include <vector>
12
14namespace nc::ecs::detail
15{
32template<class T>
33class SparseSet
34{
35 public:
36 using index_type = Entity::index_type;
37
38 explicit SparseSet(size_t maxCount)
39 : m_sparse(maxCount, Entity::NullIndex) {}
40
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();
58 void Clear();
59
60 template<std::predicate<const T&, const T&> Predicate>
61 void Sort(Predicate&& lessThan);
62
63 private:
64 std::vector<index_type> m_sparse;
65 std::vector<T> m_dense;
66 std::vector<Entity> m_entities;
67
68 void InsertEntity(Entity entity);
69};
70
71template<class T>
72void SparseSet<T>::InsertEntity(Entity entity)
73{
74 auto& denseIndex = m_sparse.at(entity.Index());
75 if (denseIndex != Entity::NullIndex)
76 throw NcError{"Entity already exists."};
77
78 denseIndex = static_cast<index_type>(m_entities.size());
79 m_entities.push_back(entity);
80}
81
82template<class T>
83auto SparseSet<T>::Insert(Entity entity, const T& component) -> T&
84{
85 InsertEntity(entity);
86 m_dense.push_back(component);
87 return m_dense.back();
88}
89
90template<class T>
91auto SparseSet<T>::Insert(Entity entity, T&& component) -> T&
92{
93 InsertEntity(entity);
94 m_dense.push_back(std::move(component));
95 return m_dense.back();
96}
97
98template<class T>
99auto SparseSet<T>::Remove(Entity toRemove) -> bool
100{
101 if (!Contains(toRemove))
102 return false;
103
104 const auto toRemoveSparse = toRemove.Index();
105 const auto swappedSparse = m_entities.back().Index();
106 const auto toRemoveDense = m_sparse.at(toRemoveSparse);
107
108 m_dense.at(toRemoveDense) = std::move(m_dense.back());
109 m_dense.pop_back();
110 m_entities.at(toRemoveDense) = m_entities.back();
111 m_entities.pop_back();
112 m_sparse.at(swappedSparse) = toRemoveDense;
113 m_sparse.at(toRemoveSparse) = Entity::NullIndex;
114 return true;
115}
116
117template<class T>
118auto SparseSet<T>::Get(Entity entity) -> T&
119{
120 return m_dense.at(m_sparse.at(entity.Index()));
121}
122
123template<class T>
124auto SparseSet<T>::Get(Entity entity) const -> const T&
125{
126 return m_dense.at(m_sparse.at(entity.Index()));
127}
128
129template<class T>
130auto SparseSet<T>::GetParent(const T* component) const noexcept -> Entity
131{
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)]
135 : Entity::Null();
136}
137
138template<class T>
139auto SparseSet<T>::GetIndex(Entity entity) const -> index_type
140{
141 return m_sparse.at(entity.Index());
142}
143
144template<class T>
145auto SparseSet<T>::GetEntity(index_type index) const -> Entity
146{
147 return m_entities.at(m_sparse.at(index));
148}
149
150template<class T>
151auto SparseSet<T>::Contains(Entity entity) const noexcept -> bool
152{
153 const auto i = entity.Index();
154 return i < m_sparse.size() ? m_sparse[i] != Entity::NullIndex : false;
155}
156
157template<class T>
158void SparseSet<T>::Swap(Entity lhs, Entity rhs)
159{
160 if (lhs == rhs)
161 return;
162
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));
169}
170
171template<class T>
172void SparseSet<T>::Reserve(size_t capacity)
173{
174 m_dense.reserve(capacity);
175 m_entities.reserve(capacity);
176}
177
178template<class T>
179template<std::predicate<const T&, const T&> Predicate>
180void SparseSet<T>::Sort(Predicate&& lessThan)
181{
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);
188
190 auto compare = [&pool = m_dense, predicate = std::forward<Predicate>(lessThan)](const auto lhs,
191 const auto rhs)
192 { return predicate(pool[lhs], pool[rhs]); };
193
195 for (auto cur = beg; cur != end; ++cur)
196 {
197 const auto pos = std::upper_bound(beg, cur, *cur, compare);
198 std::rotate(pos, cur, cur + 1);
199 }
200
202 for (auto i = 0u; i < size; ++i)
203 {
204 auto cur = i;
205 auto next = permutation[cur];
206
207 while (cur != next)
208 {
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;
218 cur = next;
219 next = permutation[cur];
220 }
221 }
222}
223
224template<class T>
225void SparseSet<T>::ClearNonPersistent()
226{
227 std::ranges::fill(m_sparse, Entity::NullIndex);
228 auto swapToIndex = 0u;
229 for (auto [denseIndex, entity] : algo::Enumerate(m_entities))
230 {
231 if (!entity.IsPersistent())
232 continue;
233
234 // push persistent entities/components to the front
235 m_sparse.at(entity.Index()) = swapToIndex;
236 if (swapToIndex != denseIndex) // don't self move!
237 {
238 m_dense.at(swapToIndex) = std::move(m_dense.at(denseIndex));
239 m_entities.at(swapToIndex) = entity;
240 }
241
242 ++swapToIndex;
243 }
244
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();
249}
250
251template<class T>
252void SparseSet<T>::Clear()
253{
254 std::ranges::fill(m_sparse, Entity::NullIndex);
255 m_dense.clear();
256 m_dense.shrink_to_fit();
257 m_entities.clear();
258 m_entities.shrink_to_fit();
259}
260} // namespace nc::ecs::detail
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