NcEngine
Ecs.h
Go to the documentation of this file.
1
5#pragma once
6
8
9namespace nc::ecs
10{
16template<FilterBase Base, class... Includes>
18{
19 public:
20 using PolicyType = AccessPolicy<Base, Includes...>;
21 using FilterType = typename PolicyType::FilterType;
22
25 : m_policy{policy} {}
26
28 template<FilterBase TargetBase, class... TargetIncludes>
29 requires PolicyType::template ConvertibleTo<TargetBase, TargetIncludes...>
30 operator EcsInterface<TargetBase, TargetIncludes...>() const noexcept
31 {
32 return EcsInterface<TargetBase, TargetIncludes...>(m_policy);
33 }
34
36 template<std::same_as<Entity> T>
37 requires PolicyType::template HasAccess<Entity, Transform, Tag, Hierarchy>
38 auto Emplace(EntityInfo info = {}) -> Entity
39 {
40 const auto handle = m_policy.template GetPool<Entity>().Add(info.layer, info.flags, info.userData);
41 if (info.parent.Valid())
42 {
43 auto& parentTransform = Get<Transform>(info.parent);
44 Emplace<Transform>(handle, info.position, info.rotation, info.scale, parentTransform.TransformationMatrix());
45 Get<Hierarchy>(info.parent).children.push_back(handle);
46 }
47 else
48 {
49 Emplace<Transform>(handle, info.position, info.rotation, info.scale);
50 }
51
52 Emplace<Hierarchy>(handle, info.parent, std::vector<Entity>{});
53 Emplace<Tag>(handle, std::move(info.tag));
54 return handle;
55 }
56
58 template<Component T, class... Args>
59 requires PolicyType::template HasAccess<T>
60 auto Emplace(Entity entity, Args&&... args) -> T&
61 {
62 NC_ASSERT(Contains<Entity>(entity), "Bad entity");
63 if constexpr (std::derived_from<T, FreeComponent>)
64 {
65 return m_policy.GetFreeComponentPool().template Emplace<T>(entity, std::forward<Args>(args)...);
66 }
67 else
68 {
69 return m_policy.template GetPool<T>().Emplace(entity, std::forward<Args>(args)...);
70 }
71 }
72
74 template<std::same_as<Entity> T>
75 requires PolicyType::template HasAccess<Entity, Transform>
76 auto Remove(Entity entity) -> bool
77 {
78 return Contains<Entity>(entity) ? RemoveNode<true>(entity) : false;
79 }
80
82 template<Component T>
83 requires PolicyType::template HasAccess<T>
84 auto Remove(Entity entity) -> bool
85 {
86 if (!Contains<Entity>(entity))
87 return false;
88
89 if constexpr (std::derived_from<T, FreeComponent>)
90 {
91 return m_policy.GetFreeComponentPool().template Remove<T>(entity);
92 }
93 else
94 {
95 return m_policy.template GetPool<T>().Remove(entity);
96 }
97 }
98
100 template<Component T>
101 requires PolicyType::template HasAccess<T>
102 auto Get(Entity entity) -> T&
103 {
104 if constexpr (std::derived_from<T, FreeComponent>)
105 {
106 return m_policy.GetFreeComponentPool().template Get<T>(entity);
107 }
108 else
109 {
110 return m_policy.template GetPool<T>().Get(entity);
111 }
112 }
113
115 template<Component T>
116 requires PolicyType::template HasAccess<T>
117 auto Get(Entity entity) const -> const T&
118 {
119 if constexpr (std::derived_from<T, FreeComponent>)
120 {
121 return m_policy.GetFreeComponentPool().template Get<T>(entity);
122 }
123 else
124 {
125 return m_policy.template GetPool<T>().Get(entity);
126 }
127 }
128
135 auto GetEntityByTag(std::string_view tagValue) -> Entity
136 requires PolicyType::template HasAccess<Entity>
137 && PolicyType::template HasAccess<Tag>
138 {
139 const auto tags = GetAll<Tag>();
140 const auto pos = std::ranges::find(tags, tagValue, [](const auto& tag) { return tag.value; });
141 NC_ASSERT(pos != std::ranges::end(tags), fmt::format("No Entity found with Tag '{}'", tagValue));
142 return m_policy.template GetPool<Tag>().GetParent(&(*pos));
143 }
144
146 template<class T>
147 requires PolicyType::template HasAccess<T>
148 && (PooledComponent<T> || std::same_as<Entity, T>)
149 auto GetAll() -> std::span<T>
150 {
151 return std::span<T>{GetPool<T>()};
152 }
153
155 template<class T>
156 requires PolicyType::template HasAccess<T>
157 && (PooledComponent<T> || std::same_as<Entity, T>)
158 auto GetAll() const -> std::span<const T>
159 {
160 return std::span<const T>{GetPool<T>()};
161 }
162
164 template<class T>
165 requires PolicyType::template HasAccess<T>
166 && (PooledComponent<T> || std::same_as<Entity, T>)
167 auto GetPool() -> decltype(auto)
168 {
169 return m_policy.template GetPool<T>();
170 }
171
173 template<class T>
174 requires PolicyType::template HasAccess<T>
175 && (PooledComponent<T> || std::same_as<Entity, T>)
176 auto GetPool() const -> decltype(auto)
177 {
178 return m_policy.template GetPool<T>();
179 }
180
182 auto GetComponentPools()
183 requires PolicyType::template BaseContains<FilterBase::All>
184 {
185 return m_policy.GetComponentPools();
186 }
187
189 void SetParent(Entity entity, Entity parent)
190 requires PolicyType::template HasAccess<Hierarchy>
191 {
192 auto& hierarchy = Get<Hierarchy>(entity);
193 const auto oldParent = std::exchange(hierarchy.parent, parent);
194 if (oldParent.Valid())
195 {
196 std::erase(Get<Hierarchy>(oldParent).children, entity);
197 }
198
199 if (parent.Valid())
200 {
201 Get<Hierarchy>(parent).children.push_back(entity);
202 }
203 }
204
206 auto GetRoot(Entity entity) -> Entity
207 requires PolicyType::template HasAccess<Hierarchy>
208 {
209 const auto& hierarchy = Get<Hierarchy>(entity);
210 return !hierarchy.parent.Valid() ? entity : GetRoot(hierarchy.parent);
211 }
212
214 template<PooledComponent T>
215 requires PolicyType::template HasAccess<T>
216 auto GetParent(const T* component) const -> Entity
217 {
218 return m_policy.template GetPool<T>().GetParent(component);
219 }
220
222 template<RegistryType T>
223 requires PolicyType::template HasAccess<T>
224 auto Contains(Entity entity) const -> bool
225 {
226 if constexpr (std::derived_from<T, FreeComponent>)
227 {
228 return m_policy.GetFreeComponentPool().template Contains<T>(entity);
229 }
230 else
231 {
232 return m_policy.template GetPool<T>().Contains(entity);
233 }
234 }
235
236 private:
237 PolicyType m_policy;
238
239 template<bool IsRoot>
240 auto RemoveNode(Entity entity) -> bool
241 {
242 auto& hierarchy = Get<Hierarchy>(entity);
243 if constexpr(IsRoot)
244 {
245 if (hierarchy.parent.Valid())
246 {
247 std::erase(Get<Hierarchy>(hierarchy.parent).children, entity);
248 }
249 }
250
251 for (auto child : hierarchy.children)
252 RemoveNode<false>(child);
253
254 m_policy.template GetPool<Entity>().Remove(entity);
255 return true;
256 }
257};
258
261
263template<class... Includes>
264using BasicEcs = EcsInterface<FilterBase::Basic, Includes...>;
265
267template<class... Includes>
268using ExplicitEcs = EcsInterface<FilterBase::None, Includes...>;
269} // namespace nc::ecs
FilterBase
Filter option for AccessPolicy providing coarse control over type access.
Definition: EcsFilter.h:16
Identifies an object in the registry.
Definition: Entity.h:18
A proxy for ComponentRegistry providing filtered access to data pools.
Definition: AccessPolicy.h:33
std::conditional_t< Base==FilterBase::All, detail::AllFilter, std::conditional_t< Base==FilterBase::Basic, detail::MatchFilter< Entity, Hierarchy, Tag, Transform, Includes... >, detail::MatchFilter< Includes... > > > FilterType
The policy's filter type.
Definition: AccessPolicy.h:43
Interface for higher-level entity and component operations with optional type access restriction.
Definition: Ecs.h:18
EcsInterface(AccessPolicy< Base, Includes... > policy) noexcept
Construct an EcsInterface instance.
Definition: Ecs.h:24
auto GetEntityByTag(std::string_view tagValue) -> Entity
Get the first Entity matching a Tag.
Definition: Ecs.h:135
::template HasAccess< T > auto Emplace(Entity entity, Args &&... args) -> T &
Emplace a component.
Definition: Ecs.h:60
auto GetComponentPools() auto RemoveNode(Entity entity) -> bool
Get a range of pointers to all ComponentPoolBase instances.
Definition: Ecs.h:240
::template HasAccess< Entity, Transform > auto Remove(Entity entity) -> bool
Remove an entity.
Definition: Ecs.h:76
::template HasAccess< T > auto Get(Entity entity) -> T &
Get a component.
Definition: Ecs.h:102
::template HasAccess< T > &&PooledComponent< T > std::same_as< Entity, T > auto GetPool() const -> decltype(auto)
Get the pool for a given type.
Definition: Ecs.h:176
::template HasAccess< T > auto Get(Entity entity) const -> const T &
Get a component.
Definition: Ecs.h:117
::template HasAccess< T > auto Remove(Entity entity) -> bool
Remove a component.
Definition: Ecs.h:84
::template HasAccess< Entity, Transform, Tag, Hierarchy > auto Emplace(EntityInfo info={}) -> Entity
Emplace an entity.
Definition: Ecs.h:38
Requirements for the Registry to recognize a pooled component.
Definition: Component.h:61
Initialization data for Entities.
Definition: Entity.h:111