qxLib
components.h
Go to the documentation of this file.
1 /**
2 
3  @file components.h
4  @author Khrapov
5  @date 7.03.2021
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/macros/common.h>
12 #include <qx/priority.h>
13 #include <qx/rtti/rtti.h>
14 
15 #include <map>
16 #include <optional>
17 #include <ranges>
18 #include <unordered_map>
19 
20 namespace qx
21 {
22 
23 /**
24 
25  @class components
26  @brief Container for components system
27  @details Stores components and allows them to be accessed by a template argument or a class id.
28  Each component has a priority, which is used to iterate through a sequence of objects of a type (final or any super type)
29  or to use getter methods, where a component with the highest priority will be given.
30  You can use qx::priority::disabled to exclude the component from iteration or getters result.
31  @tparam base_component_t - base component type
32  @author Khrapov
33  @date 9.03.2021
34 
35 **/
36 template<std::derived_from<rtti_pure_base> base_component_t>
38 {
39 public:
40  using pointer_type = std::unique_ptr<base_component_t>;
41 
42 private:
43  struct SClassData
44  {
45  struct SCacheData
46  {
47  base_component_t* pComponent = nullptr;
48  flags<status> statusFlags = status::default_value;
49  };
50 
51  std::unordered_map<class_identificator, std::unique_ptr<SClassData>> derivedClasses;
52  std::vector<pointer_type> components;
53  std::multimap<priority, SCacheData, std::greater<priority>> priorityCache;
54 
55  [[nodiscard]] SClassData& get_or_add_class_data(class_identificator id) noexcept;
56  };
57 
58  static constexpr auto stub_callback = [](auto&&...)
59  {
60  };
61 
62 public:
63  /**
64  @brief Add a component
65  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
66  @param pComponent - component object
67  @param ePriority - initial component priority
68  @param statusFlags - initial component status
69  @retval - raw component pointer which acts like a component's handle
70  **/
71  template<std::derived_from<base_component_t> component_t>
72  [[maybe_unused]] component_t* add(
73  std::unique_ptr<component_t> pComponent,
74  priority ePriority = priority::normal,
75  flags<status> statusFlags = status::default_value) noexcept;
76 
77  /**
78  @brief Remove the component from the container
79  @param pRawComponent - raw component pointer which acts like a component's handle
80  @retval - component object
81  **/
82  [[maybe_unused]] std::unique_ptr<base_component_t> remove(const base_component_t* pRawComponent) noexcept;
83 
84  /**
85  @brief Try to get a component of the given type with the highest priority
86  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
87  @retval - component pointer or nullptr
88  **/
89  template<std::derived_from<base_component_t> component_t>
90  [[nodiscard]] component_t* try_get() noexcept;
91 
92  /**
93  @brief Try to get a component of the given type with the highest priority
94  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
95  @retval - component pointer or nullptr
96  **/
97  template<std::derived_from<base_component_t> component_t>
98  [[nodiscard]] const component_t* try_get() const noexcept;
99 
100  /**
101  @brief Try to get a component of the given type id with the highest priority
102  @param id - type id
103  @retval - component pointer or nullptr
104  **/
105  [[nodiscard]] base_component_t* try_get(class_identificator id) noexcept;
106 
107  /**
108  @brief Try to get a component of the given type id with the highest priority
109  @param id - type id
110  @retval - component pointer or nullptr
111  **/
112  [[nodiscard]] const base_component_t* try_get(class_identificator id) const noexcept;
113 
114  /**
115  @brief Get a component of the given type with the highest priority (no existence checks are performed)
116  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
117  @retval - component reference
118  **/
119  template<std::derived_from<base_component_t> component_t>
120  [[nodiscard]] component_t& get() noexcept;
121 
122  /**
123  @brief Get a component of the given type with the highest priority (no existence checks are performed)
124  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
125  @retval - component reference
126  **/
127  template<std::derived_from<base_component_t> component_t>
128  [[nodiscard]] const component_t& get() const noexcept;
129 
130  /**
131  @brief Get a view which may be used in a ranged-based for loop and consists of components of a given type with decreasing non-disabled priority
132  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
133  @retval - view of components
134  **/
135  template<std::derived_from<base_component_t> component_t = base_component_t>
136  [[nodiscard]] auto view() noexcept;
137 
138  /**
139  @brief Get a view which may be used in a ranged-based for loop and consists of components of a given type with decreasing non-disabled priority
140  @tparam component_t - component type which doesn't have to be a final class type, may be any base or super type
141  @retval - view of components
142  **/
143  template<std::derived_from<base_component_t> component_t = base_component_t>
144  [[nodiscard]] auto view() const noexcept;
145 
146  /**
147  @brief Get a priority of a given component
148  @param pRawComponent - raw component pointer which acts like a component's handle
149  @retval - component's priority or nullopt if can't find the component
150  **/
151  [[nodiscard]] std::optional<priority> get_priority(const base_component_t* pRawComponent) const noexcept;
152 
153  /**
154  @brief Set a priority of a given component
155  @param pRawComponent - raw component pointer which acts like a component's handle
156  @param ePriority - new component's priority
157  @retval - true if the component is found and a priority was set
158  **/
159  [[maybe_unused]] bool set_priority(const base_component_t* pRawComponent, priority ePriority) noexcept;
160 
161  /**
162  @brief Get a status of a given component
163  @param pRawComponent - raw component pointer which acts like a component's handle
164  @retval - status
165  **/
166  [[nodiscard]] std::optional<flags<status>> get_status(const base_component_t* pRawComponent) const noexcept;
167 
168  /**
169  @brief Set a status of a given component
170  @param pRawComponent - raw component pointer which acts like a component's handle
171  @param status - new component's status
172  @retval - true if the component is found and a status was set
173  **/
174  [[maybe_unused]] bool set_status(const base_component_t* pRawComponent, flags<status> status) noexcept;
175 
176  /**
177  @brief Check if container doesn't have any components
178  @retval - true if container doesn't have any components
179  **/
180  [[nodiscard]] bool empty() const noexcept;
181 
182  /**
183  @brief Clear the container, e.g. remove all the components
184  **/
185  void clear() noexcept;
186 
187 private:
188  /**
189  @brief Get or add class data of a given type
190  @tparam component_t - final class type
191  @tparam callable_t - a callable that takes a class data
192  @param iterateClassDataFunction - a function that iterates over a class data from base_component_t + 1 to component_t
193  @retval - or add class data
194  **/
195  template<
196  std::derived_from<base_component_t> component_t,
197  callable_c<void, SClassData&> callable_t = decltype(stub_callback)>
198  SClassData& get_or_add_class_data(callable_t iterateClassDataFunction = stub_callback) noexcept
199  {
200  /*
201  | | rtti_base | c1 | base_component_t | c2 | c3 | component_t |
202  | t1 | <=======================================================> |
203  | t2 | <===============================> |
204  | t3 | <====================> |
205  */
206  using t1 = typename component_t::inheritance_tuple;
207  using t2 = typename base_component_t::inheritance_tuple;
208  using t3 = tuple::remove_t<t1, t2>;
209 
210  SClassData* pClassData = &m_RootClass;
211  iterateClassDataFunction(*pClassData);
212 
213  tuple::iterate<t3>(
214  [&pClassData, &iterateClassDataFunction]<class T, size_t I>()
215  {
216  pClassData = &pClassData->get_or_add_class_data(T::get_class_id_static());
217  iterateClassDataFunction(*pClassData);
218  });
219 
220  return *pClassData;
221  }
222 
223  /**
224  @brief Get or add class data of a given type
225  @tparam component_t - final class type
226  @tparam callable_t - a callable that takes a class data
227  @param iterateClassDataFunction - a function that iterates over a class data from base_component_t + 1 to component_t
228  @retval - or add class data
229  **/
230  template<
231  std::derived_from<base_component_t> component_t,
232  callable_c<void, SClassData&> callable_t = decltype(stub_callback)>
233  const SClassData& get_or_add_class_data(callable_t iterateClassDataFunction = stub_callback) const noexcept
234  {
235  return QX_CONST_CAST_THIS()->template get_or_add_class_data<component_t>(std::move(iterateClassDataFunction));
236  }
237 
238  /**
239  @brief Get or add class data of a given type
240  @tparam callable_t - a callable that takes a class data
241  @param pRawComponent - an object used to identify a class type
242  @param iterateClassDataFunction - a function that iterates over a class data from base_component_t + 1 to component_t
243  @retval - or add class data
244  **/
245  template<callable_c<void, SClassData&> callable_t = decltype(stub_callback)>
246  SClassData& get_or_add_class_data(
247  const base_component_t* pRawComponent,
248  callable_t iterateClassDataFunction = stub_callback) noexcept
249  {
250  SClassData* pClassData = &m_RootClass;
251  iterateClassDataFunction(*pClassData);
252 
253  std::span<const class_identificator> allIds = pRawComponent->get_inheritance_sequence();
254  std::span<const class_identificator> baseClassIds(
255  std::ranges::find(allIds, base_component_t::get_class_id_static()),
256  allIds.end());
257 
258  for (auto it = baseClassIds.begin() + 1; it != baseClassIds.end(); ++it)
259  {
260  pClassData = &pClassData->get_or_add_class_data(*it);
261  iterateClassDataFunction(*pClassData);
262  }
263 
264  return *pClassData;
265  }
266 
267  /**
268  @brief Get or add class data of a given type
269  @tparam callable_t - a callable that takes a class data
270  @param pRawComponent - an object used to identify a class type
271  @param iterateClassDataFunction - a function that iterates over a class data from base_component_t + 1 to component_t
272  @retval - or add class data
273  **/
274  template<callable_c<void, SClassData&> callable_t = decltype(stub_callback)>
275  const SClassData& get_or_add_class_data(
276  const base_component_t* pRawComponent,
277  callable_t iterateClassDataFunction = stub_callback) const noexcept
278  {
279  return QX_CONST_CAST_THIS()->get_or_add_class_data(pRawComponent, std::move(iterateClassDataFunction));
280  }
281 
282 private:
283  SClassData m_RootClass;
284 };
285 
286 } // namespace qx
287 
Container for components system.
Definition: components.h:38
component_t * add(std::unique_ptr< component_t > pComponent, priority ePriority=priority::normal, flags< status > statusFlags=status::default_value) noexcept
Add a component.
Definition: components.inl:28
std::unique_ptr< base_component_t > remove(const base_component_t *pRawComponent) noexcept
Remove the component from the container.
Definition: components.inl:48
bool set_priority(const base_component_t *pRawComponent, priority ePriority) noexcept
Set a priority of a given component.
Definition: components.inl:202
component_t * try_get() noexcept
Try to get a component of the given type with the highest priority.
Definition: components.inl:82
std::optional< priority > get_priority(const base_component_t *pRawComponent) const noexcept
Get a priority of a given component.
Definition: components.inl:185
void clear() noexcept
Clear the container, e.g. remove all the components.
Definition: components.inl:290
auto view() noexcept
Get a view which may be used in a ranged-based for loop and consists of components of a given type wi...
Definition: components.inl:148
bool empty() const noexcept
Check if container doesn't have any components.
Definition: components.inl:284
std::optional< flags< status > > get_status(const base_component_t *pRawComponent) const noexcept
Get a status of a given component.
Definition: components.inl:236
component_t & get() noexcept
Get a component of the given type with the highest priority (no existence checks are performed)
Definition: components.inl:126
bool set_status(const base_component_t *pRawComponent, flags< status > status) noexcept
Set a status of a given component.
Definition: components.inl:254
priority
User may use the predefined values or the custom ones, for ex. "normal - 1", this type is supposed to...
Definition: priority.h:26
RTTI system based on polymorphism.