registration_impl.h 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. /************************************************************************************
  2. * *
  3. * Copyright (c) 2014 - 2018 Axel Menzel <info@rttr.org> *
  4. * *
  5. * This file is part of RTTR (Run Time Type Reflection) *
  6. * License: MIT License *
  7. * *
  8. * Permission is hereby granted, free of charge, to any person obtaining *
  9. * a copy of this software and associated documentation files (the "Software"), *
  10. * to deal in the Software without restriction, including without limitation *
  11. * the rights to use, copy, modify, merge, publish, distribute, sublicense, *
  12. * and/or sell copies of the Software, and to permit persons to whom the *
  13. * Software is furnished to do so, subject to the following conditions: *
  14. * *
  15. * The above copyright notice and this permission notice shall be included in *
  16. * all copies or substantial portions of the Software. *
  17. * *
  18. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR *
  19. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, *
  20. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE *
  21. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER *
  22. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, *
  23. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE *
  24. * SOFTWARE. *
  25. * *
  26. *************************************************************************************/
  27. #ifndef RTTR_REGISTRATION_IMPL_H_
  28. #define RTTR_REGISTRATION_IMPL_H_
  29. #include "rttr/detail/base/core_prerequisites.h"
  30. #include "rttr/detail/misc/argument_extractor.h"
  31. #include "rttr/detail/constructor/constructor_wrapper.h"
  32. #include "rttr/detail/constructor/constructor_wrapper_defaults.h"
  33. #include "rttr/detail/destructor/destructor_wrapper.h"
  34. #include "rttr/detail/enumeration/enumeration_wrapper.h"
  35. #include "rttr/detail/method/method_wrapper.h"
  36. #include "rttr/detail/property/property_wrapper.h"
  37. #include "rttr/detail/type/accessor_type.h"
  38. #include "rttr/detail/misc/misc_type_traits.h"
  39. #include "rttr/detail/misc/utility.h"
  40. #include "rttr/detail/type/type_register.h"
  41. #include "rttr/detail/registration/bind_impl.h"
  42. #include "rttr/detail/registration/registration_state_saver.h"
  43. #include "rttr/policy.h"
  44. #include "rttr/enumeration.h"
  45. #include "rttr/detail/visitor/create_type_visitor_func.h"
  46. #include <string>
  47. #include <vector>
  48. namespace rttr
  49. {
  50. /////////////////////////////////////////////////////////////////////////////////////////
  51. RTTR_INLINE detail::metadata metadata(variant key, variant value)
  52. {
  53. return detail::metadata{std::move(key), std::move(value)};
  54. }
  55. /////////////////////////////////////////////////////////////////////////////////////////
  56. template<typename Enum_Type>
  57. RTTR_INLINE detail::enum_data<Enum_Type> value(string_view name, Enum_Type value)
  58. {
  59. return detail::enum_data<Enum_Type>(name, value);
  60. }
  61. /////////////////////////////////////////////////////////////////////////////////////////
  62. template<typename...TArgs>
  63. RTTR_INLINE detail::default_args<TArgs...> default_arguments(TArgs&&...args)
  64. {
  65. static_assert((sizeof...(TArgs) > 0), "Please provide at least one default argument!");
  66. return { std::forward<TArgs>(args)... };
  67. }
  68. /////////////////////////////////////////////////////////////////////////////////////////
  69. template<typename...TArgs>
  70. RTTR_INLINE detail::parameter_names<detail::decay_t<TArgs>...> parameter_names(TArgs&&...args)
  71. {
  72. using namespace detail;
  73. static_assert(static_all_of<is_string_literal<raw_type_t<TArgs>>::value...>::value, "Please use this function only with string literals!");
  74. return { static_cast<detail::decay_t<TArgs>>(std::forward<TArgs>(args))...};
  75. }
  76. /////////////////////////////////////////////////////////////////////////////////////////
  77. /////////////////////////////////////////////////////////////////////////////////////////
  78. /////////////////////////////////////////////////////////////////////////////////////////
  79. template<typename Class_Type, typename Visitor_List>
  80. registration::class_<Class_Type, Visitor_List>::class_(string_view name)
  81. {
  82. auto t = type::get<Class_Type>();
  83. detail::type_register::custom_name(t, name);
  84. detail::type_register::register_visit_type_func(t, &detail::visit_type<Class_Type, Visitor_List>);
  85. }
  86. /////////////////////////////////////////////////////////////////////////////////////////
  87. template<typename Class_Type, typename Visitor_List>
  88. registration::class_<Class_Type, Visitor_List>::class_(const std::shared_ptr<detail::registration_executer>& reg_exec)
  89. : m_reg_exec(reg_exec)
  90. {
  91. }
  92. /////////////////////////////////////////////////////////////////////////////////////////
  93. template<typename Class_Type, typename Visitor_List>
  94. registration::class_<Class_Type, Visitor_List>::~class_()
  95. {
  96. // make sure that all base classes are registered
  97. detail::base_classes<Class_Type>::get_types();
  98. }
  99. /////////////////////////////////////////////////////////////////////////////////////////
  100. template<typename Class_Type, typename Visitor_List>
  101. template<typename...Args>
  102. registration::class_<Class_Type, Visitor_List>& registration::class_<Class_Type, Visitor_List>::operator()(Args&&...args)
  103. {
  104. detail::type_register::metadata(type::get<Class_Type>(), detail::forward_to_vector<detail::metadata>(std::forward<Args>(args)...));
  105. return *this;
  106. }
  107. /////////////////////////////////////////////////////////////////////////////////////////
  108. template<typename Class_Type, typename Visitor_List>
  109. template<typename... Args, typename acc_level, typename Tp>
  110. registration::bind<detail::ctor, Class_Type, acc_level, Visitor_List, Args...> registration::class_<Class_Type, Visitor_List>::constructor(acc_level level)
  111. {
  112. return {create_if_empty(m_reg_exec)};
  113. }
  114. /////////////////////////////////////////////////////////////////////////////////////////
  115. template<typename Class_Type, typename Visitor_List>
  116. template<typename F, typename acc_level, typename Tp>
  117. registration::bind<detail::ctor_func, Class_Type, F, acc_level, Visitor_List> registration::class_<Class_Type, Visitor_List>::constructor(F func, acc_level level)
  118. {
  119. using namespace detail;
  120. static_assert(is_functor<F>::value,
  121. "No valid accessor for invoking the constructor provided!");
  122. static_assert(std::is_same<return_func, typename method_type<F>::type>::value,
  123. "For creating this 'class type', please provide a function pointer or std::function with a return value.");
  124. return {create_if_empty(m_reg_exec), func};
  125. }
  126. /////////////////////////////////////////////////////////////////////////////////////////
  127. template<typename Class_Type, typename Visitor_List>
  128. template<typename A, typename acc_level, typename Tp>
  129. registration::bind<detail::prop, Class_Type, A, acc_level, Visitor_List> registration::class_<Class_Type, Visitor_List>::property(string_view name, A acc, acc_level level)
  130. {
  131. using namespace detail;
  132. static_assert(contains<acc_level, access_levels_list>::value, "The given type of 'level' is not a type of 'rttr::access_levels.'");
  133. static_assert(std::is_member_object_pointer<A>::value || std::is_pointer<A>::value,
  134. "No valid property accessor provided!");
  135. return {create_if_empty(m_reg_exec), name, acc};
  136. }
  137. /////////////////////////////////////////////////////////////////////////////////////////
  138. template<typename Class_Type, typename Visitor_List>
  139. template<typename A, typename acc_level, typename Tp>
  140. registration::bind<detail::prop_readonly, Class_Type, A, acc_level, Visitor_List> registration::class_<Class_Type, Visitor_List>::property_readonly(string_view name, A acc, acc_level level)
  141. {
  142. using namespace detail;
  143. static_assert(contains<acc_level, access_levels_list>::value, "The given type of 'level' is not a type of 'rttr::access_levels.'");
  144. static_assert(std::is_pointer<A>::value ||
  145. std::is_member_object_pointer<A>::value || std::is_member_function_pointer<A>::value || is_functor<A>::value,
  146. "No valid property accessor provided!");
  147. return {create_if_empty(m_reg_exec), name, acc};
  148. }
  149. /////////////////////////////////////////////////////////////////////////////////////////
  150. template<typename Class_Type, typename Visitor_List>
  151. template<typename A1, typename A2, typename acc_level, typename Tp>
  152. registration::bind<detail::prop, Class_Type, A1, A2, acc_level, Visitor_List> registration::class_<Class_Type, Visitor_List>::property(string_view name, A1 getter, A2 setter, acc_level level)
  153. {
  154. using namespace detail;
  155. static_assert(contains<acc_level, access_levels_list>::value, "The given type of 'level' is not a type of 'rttr::access_levels.'");
  156. static_assert(std::is_member_function_pointer<A1>::value || std::is_member_function_pointer<A2>::value ||
  157. is_functor<A1>::value || is_functor<A2>::value, "No valid property accessor provided!");
  158. static_assert(function_traits<A1>::arg_count == 0, "Invalid number of arguments, please provide as first accessor a getter-member-function without arguments.");
  159. static_assert(function_traits<A2>::arg_count == 1, "Invalid number of arguments, please provide as second argument a setter-member-function with exactly one argument.");
  160. using return_type = typename function_traits<A1>::return_type;
  161. using arg_type = typename param_types<A2, 0>::type;
  162. static_assert(std::is_same<return_type, arg_type>::value, "Please provide the same signature (data type) for getter and setter!");
  163. return {create_if_empty(m_reg_exec), name, getter, setter};
  164. }
  165. /////////////////////////////////////////////////////////////////////////////////////////
  166. template<typename Class_Type, typename Visitor_List>
  167. template<typename F, typename acc_level>
  168. registration::bind<detail::meth, Class_Type, F, acc_level, Visitor_List> registration::class_<Class_Type, Visitor_List>::method(string_view name, F f, acc_level level)
  169. {
  170. using namespace detail;
  171. static_assert(contains<acc_level, access_levels_list>::value, "The given type of 'level' is not a type of 'rttr::access_levels.'");
  172. static_assert(std::is_member_function_pointer<F>::value || is_functor<F>::value, "No valid method accessor provided!");
  173. return {create_if_empty(m_reg_exec), name, f};
  174. }
  175. /////////////////////////////////////////////////////////////////////////////////////////
  176. template<typename Class_Type, typename Visitor_List>
  177. template<typename Enum_Type>
  178. registration::bind<detail::enum_, Class_Type, Enum_Type> registration::class_<Class_Type, Visitor_List>::enumeration(string_view name)
  179. {
  180. using namespace detail;
  181. static_assert(std::is_enum<Enum_Type>::value, "No enum type provided, please call this method with an enum type!");
  182. return {create_if_empty(m_reg_exec), name};
  183. }
  184. /////////////////////////////////////////////////////////////////////////////////////////
  185. /////////////////////////////////////////////////////////////////////////////////////////
  186. /////////////////////////////////////////////////////////////////////////////////////////
  187. template<typename A, typename Visitor_List>
  188. registration::bind<detail::prop, detail::invalid_type, A, detail::public_access, Visitor_List> registration::property(string_view name, A acc)
  189. {
  190. using namespace detail;
  191. static_assert(std::is_pointer<A>::value, "No valid property accessor provided!");
  192. return {std::make_shared<registration_executer>(), name, acc};
  193. }
  194. /////////////////////////////////////////////////////////////////////////////////////////
  195. template<typename A, typename Visitor_List>
  196. registration::bind<detail::prop_readonly, detail::invalid_type, A, detail::public_access, Visitor_List> registration::property_readonly(string_view name, A acc)
  197. {
  198. using namespace detail;
  199. static_assert(std::is_pointer<A>::value || is_functor<A>::value,
  200. "No valid property accessor provided!");
  201. return {std::make_shared<registration_executer>(), name, acc};
  202. }
  203. /////////////////////////////////////////////////////////////////////////////////////////
  204. template<typename A1, typename A2, typename Visitor_List>
  205. registration::bind<detail::prop, detail::invalid_type, A1, A2, detail::public_access, Visitor_List> registration::property(string_view name, A1 getter, A2 setter)
  206. {
  207. using namespace detail;
  208. static_assert(is_functor<A1>::value || is_functor<A2>::value,
  209. "No valid property accessor provided!");
  210. return {std::make_shared<registration_executer>(), name, getter, setter};
  211. }
  212. /////////////////////////////////////////////////////////////////////////////////////////
  213. template<typename F, typename Visitor_List>
  214. registration::bind<detail::meth, detail::invalid_type, F, detail::public_access, Visitor_List> registration::method(string_view name, F f)
  215. {
  216. using namespace detail;
  217. static_assert(is_functor<F>::value, "No valid property accessor provided!");
  218. return {std::make_shared<registration_executer>(), name, f};
  219. }
  220. /////////////////////////////////////////////////////////////////////////////////////////
  221. template<typename Enum_Type>
  222. registration::bind<detail::enum_, detail::invalid_type, Enum_Type> registration::enumeration(string_view name)
  223. {
  224. using namespace detail;
  225. static_assert(std::is_enum<Enum_Type>::value, "No enum type provided, please call this method with an enum type!");
  226. return {std::make_shared<registration_executer>(), name};
  227. }
  228. /////////////////////////////////////////////////////////////////////////////////////////
  229. } // end namespace rttr
  230. #define RTTR_REGISTRATION \
  231. static void rttr_auto_register_reflection_function_(); \
  232. namespace \
  233. { \
  234. struct rttr__auto__register__ \
  235. { \
  236. rttr__auto__register__() \
  237. { \
  238. rttr_auto_register_reflection_function_(); \
  239. } \
  240. }; \
  241. } \
  242. static const rttr__auto__register__ RTTR_CAT(auto_register__, __LINE__); \
  243. static void rttr_auto_register_reflection_function_()
  244. #if RTTR_COMPILER == RTTR_COMPILER_MSVC
  245. #define RTTR_PLUGIN_REGISTRATION RTTR_REGISTRATION
  246. #else
  247. #define RTTR_PLUGIN_REGISTRATION \
  248. static void rttr_auto_register_reflection_function_() RTTR_DECLARE_PLUGIN_CTOR; \
  249. static void rttr_auto_unregister_reflection_function() RTTR_DECLARE_PLUGIN_DTOR; \
  250. \
  251. static void rttr_auto_unregister_reflection_function() \
  252. { \
  253. rttr::detail::get_registration_manager().unregister(); \
  254. } \
  255. static void rttr_auto_register_reflection_function_()
  256. #endif
  257. #endif // RTTR_REGISTRATION_IMPL_H_