qxLib
reflection_creator.h
Go to the documentation of this file.
1 /**
2 
3  @file reflection_creator.h
4  @brief Contains reflection creator functionality
5  @details The creator allows you to create instances
6  of the class using its name or ID
7  @author Khrapov
8  @date 9.09.2021
9  @copyright © Nick Khrapov, 2021. All right reserved.
10 
11 **/
12 #pragma once
13 
15 #include <qx/macros/common.h>
17 
18 #include <functional>
19 #include <map>
20 #include <memory>
21 
22 namespace qx
23 {
24 
25 /**
26 
27  @class reflection_creator
28  @brief Reflection creator
29  @details Performs registration of classes inherited from base_class_t
30  and creates their instances by their ID and name.
31  Allocates memory.
32  @tparam base_class_t - base class type
33  @tparam smart_ptr_t - smart pointer class type
34  @tparam args_t - args for type creation
35  @author Khrapov
36  @date 7.09.2021
37 
38 **/
39 template<class base_class_t, template<class> class smart_ptr_t, class... args_t>
41 {
42 public:
43  using factory = std::function<smart_ptr_t<base_class_t>(args_t...)>;
44 
45 public:
46  /**
47  @brief Create object based on class id
48  @param id - class id
49  @param args - template parameter pack
50  @retval - class instance or nullptr if can't find factory or can't create
51  **/
52  [[nodiscard]] static smart_ptr_t<base_class_t> create_object(class_identificator id, args_t&&... args)
53  {
54  if (auto it = m_FactoriesById.find(id); it != m_FactoriesById.end())
55  return it->second(std::forward<args_t>(args)...);
56 
57  return nullptr;
58  }
59 
60  /**
61  @brief Create object based on class name
62  @param svClassName - class name
63  @param args - template parameter pack
64  @retval - class instance or nullptr if can't find factory or can't create
65  **/
66  [[nodiscard]] static smart_ptr_t<base_class_t> create_object(string_view svClassName, args_t&&... args)
67  {
68  if (auto it = m_FactoriesByName.find(svClassName); it != m_FactoriesByName.end())
69  {
70  return it->second(std::forward<args_t>(args)...);
71  }
72 
73  return nullptr;
74  }
75 
76  /**
77  @brief Register class so it can be constructed by creator
78  @param factory - class factory
79  @param id - class id
80  @param svClassName - class name
81  @retval - true if registered
82  **/
83  static bool _register_class(factory factory, class_identificator id, string_view svClassName)
84  {
85  if (factory && !svClassName.empty())
86  {
87  m_FactoriesById[id] = factory;
88  m_FactoriesByName[svClassName] = factory;
89  return true;
90  }
91  else
92  {
93  return false;
94  }
95  }
96 
97 private:
98  static inline std::map<class_identificator, factory> m_FactoriesById;
99  static inline std::map<string_view, factory> m_FactoriesByName;
100 };
101 
102 namespace details
103 {
104 
105 template<class base_class_t, class T, class... args_t>
106 static std::unique_ptr<base_class_t> create_unique(args_t&&... args)
107 {
108  if constexpr (std::is_constructible_v<T, args_t...>)
109  return std::make_unique<T>(std::forward<args_t>(args)...);
110  else
111  return nullptr;
112 }
113 
114 template<class base_class_t, class T, class... args_t>
115 static std::shared_ptr<base_class_t> create_shared(args_t&&... args)
116 {
117  if constexpr (std::is_constructible_v<T, args_t...>)
118  return std::make_shared<T>(std::forward<args_t>(args)...);
119  else
120  return nullptr;
121 }
122 
123 } // namespace details
124 
125 /**
126  @def QX_REGISTER_UNIQUE_CREATOR
127  @brief Macro for base class. Use YourClass::Creator::create_object
128  @details std::unique_ptr version
129  @param ... - constructor args types
130 **/
131 #define QX_REGISTER_UNIQUE_CREATOR(...) \
132  using CreatorRoot = this_class_type; \
133  using Creator = qx::reflection_creator<CreatorRoot, std::unique_ptr, __VA_ARGS__>
134 
135 /**
136  @def QX_REGISTER_UNIQUE_CONSTRUCTOR
137  @brief Macro for all classes inherited from base class
138  @details std::unique_ptr version
139  @param ... - constructor args types
140 **/
141 #define QX_REGISTER_UNIQUE_CONSTRUCTOR(...) \
142 private: \
143  static inline volatile bool QX_LINE_NAME(s_bRegistered) = Creator::_register_class( \
144  qx::details::create_unique<CreatorRoot, this_class_type, __VA_ARGS__>, \
145  get_class_id_static(), \
146  get_class_name_static())
147 
148 /**
149  @def QX_REGISTER_SHARED_CREATOR
150  @brief Macro for base class. Use YourClass::Creator::create_object
151  @details std::shared_ptr version
152  @param ... - constructor args types
153 **/
154 #define QX_REGISTER_SHARED_CREATOR(...) \
155 protected: \
156  using CreatorRoot = this_class_type; \
157  using Creator = qx::reflection_creator<CreatorRoot, std::shared_ptr, __VA_ARGS__>
158 
159 /**
160  @def QX_REGISTER_SHARED_CONSTRUCTOR
161  @brief Macro for all classes inherited from base class
162  @details std::shared_ptr version
163  @param ... - constructor args types
164 **/
165 #define QX_REGISTER_SHARED_CONSTRUCTOR(...) \
166 private: \
167  static inline volatile bool QX_LINE_NAME(s_bRegistered) = Creator::_register_class( \
168  qx::details::create_shared<CreatorRoot, this_class_type, __VA_ARGS__>, \
169  get_class_id_static(), \
170  get_class_name_static())
171 
172 #define QX_REGISTER_CREATOR QX_REGISTER_UNIQUE_CREATOR
173 #define QX_REGISTER_CONSTRUCTOR QX_REGISTER_UNIQUE_CONSTRUCTOR
174 
175 } // namespace qx
Reflection creator.
static smart_ptr_t< base_class_t > create_object(class_identificator id, args_t &&... args)
Create object based on class id.
static smart_ptr_t< base_class_t > create_object(string_view svClassName, args_t &&... args)
Create object based on class name.
static bool _register_class(factory factory, class_identificator id, string_view svClassName)
Register class so it can be constructed by creator.