NasNas
An intuitive and beginner friendly 2D game framework for C++
Registry.hpp
1 // Created by Modar Nasser on 15/08/2021.
2 
3 #pragma once
4 
5 #include <iostream>
6 #include <memory>
7 #include <queue>
8 #include <string>
9 
10 #include <NasNas/ecs/System.hpp>
11 #include <NasNas/ecs/Storage.hpp>
12 #include <NasNas/ecs/View.hpp>
13 
14 namespace ns::ecs::detail {
15  template <typename TEntity=Entity>
16  class Registry {
17  public:
18  TEntity create() {
19  static TEntity ent_id = 0;
20  if (!m_cemetery.empty()) {
21  m_entities.emplace_back(m_cemetery.front());
22  m_cemetery.pop();
23  }
24  else {
25  m_entities.emplace_back(ent_id++);
26  }
27  return m_entities.back();
28  }
29 
30  void destroy(TEntity ent) {
31  auto it = std::find(m_entities.begin(), m_entities.end(), ent);
32  if (it == m_entities.end())
33  return;
34 
35  for (auto& [id, pool] : m_pools) {
36  pool->remove(ent);
37  }
38 
39  m_cemetery.push(ent);
40  m_entities[it-m_entities.begin()] = m_entities.back();
41  m_entities.pop_back();
42  }
43 
44  template <typename TComp, typename ...Targs>
45  auto attach(TEntity ent, Targs&& ...args) -> TComp& {
46  return getPool<TComp>().add(ent, std::forward<Targs>(args)...);
47  }
48 
49  template <typename TComp>
50  void detach(TEntity ent) {
51  if (has<TComp>(ent)) {
52  getPool<TComp>().remove(ent);
53  }
54  }
55 
56  template <typename TComp>
57  auto all() -> std::vector<TComp>& {
58  return getPool<TComp>().components();
59  }
60 
61  template <typename TComp>
62  auto has(TEntity ent) -> bool {
63  return getPool<TComp>().contains(ent);
64  }
65 
66  template <typename TComp>
67  auto get(TEntity ent) -> TComp& {
68  auto& pool = getPool<TComp>();
69  if (has<TComp>(ent)) {
70  return pool.components().at(pool.index(ent));
71  } else {
72  throw std::runtime_error("Trying to get non existing component from entity "+std::to_string(ent));
73  }
74  }
75 
76  auto count() const -> std::size_t {
77  return m_entities.size();
78  }
79 
80  template <typename... TComps>
81  auto view() const -> components_view<TEntity, TComps...> {
82  return { getPool<TComps>()...};
83  }
84 
85  template <typename... TComps, typename Func>
86  auto run(Func fn) {
87  view<TComps...>().for_each(std::move(fn));
88  }
89 
90  template <typename... TComps>
91  auto run(System<TComps...>& system) {
92  view<TComps...>().for_each(system.m_function);
93  }
94 
95  private:
96  template <typename TComp>
97  auto getPool() const -> components_pool<TEntity, TComp>& {
98  auto comp_id = getTypeId<TComp>();
99 
100  if (m_pools.find(comp_id) == m_pools.end()) {
101  m_pools[comp_id] = std::make_unique<components_pool<TEntity, TComp>>();
102  }
103  return *static_cast<components_pool<TEntity, TComp>*>(m_pools.at(comp_id).get());
104  }
105 
106  std::vector<TEntity> m_entities;
107  std::queue<TEntity> m_cemetery;
108  mutable std::map<UID, std::unique_ptr<sparse_set<TEntity>>> m_pools;
109  };
110 
111 }
112 
113 namespace ns {
114  // can't use inline because of VS2017 bug (https://developercommunity.visualstudio.com/t/inline-static-destructors-are-called-multiple-time/1157794)
116 }