qxLib
rtti_cast.h
Go to the documentation of this file.
1 /**
2 
3  @file rtti_cast.h
4  @author Khrapov
5  @date 9.09.2021
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/meta/type_traits.h>
12 
13 #include <memory>
14 
15 namespace qx
16 {
17 
18 namespace details
19 {
20 
21 template<class T, class from_t>
22 concept has_is_derived_from = requires(T* t) { t->template is_derived_from<from_t>(); };
23 
24 } // namespace details
25 
26 /**
27  @brief Returns Y* if X or element_type of X is inherited from Y, otherwise nullptr
28  @details Pointer class should contain QX_RTTI_CLASS
29  @tparam Y - type to cast to
30  @tparam X - smart pointer type or some class to test type
31  @param value - smart pointer object or some class to test object
32  @retval - Y* if X or element_type of X is inherited from Y, otherwise nullptr
33 **/
34 template<class Y, class X>
35 Y* rtti_cast(X& value)
36 {
37  using T = std::decay_t<X>;
39  {
40  using smart_pointer_t = typename std::remove_reference_t<decltype(value)>::element_type;
41  if constexpr (details::has_is_derived_from<smart_pointer_t, Y>)
42  if (value && value->template is_derived_from<Y>())
43  return static_cast<Y*>(value.get());
44  }
45  else
46  {
47  if constexpr (details::has_is_derived_from<T, Y>)
48  if (value.template is_derived_from<Y>())
49  return static_cast<Y*>(&value);
50  }
51 
52  return nullptr;
53 }
54 
55 /**
56  @brief Returns Y* if X is inherited from Y, otherwise nullptr
57  @details Pointer class should contain QX_RTTI_CLASS
58  @tparam Y - type to cast to
59  @tparam X - type to cast from
60  @param pointer - raw pointer
61  @retval - Y* if X is inherited from pointer class, otherwise nullptr
62 **/
63 template<class Y, class X>
64 Y* rtti_cast(X* pointer)
65 {
66  if constexpr (details::has_is_derived_from<X, Y>)
67  if (pointer && pointer->template is_derived_from<Y>())
68  return static_cast<Y*>(pointer);
69 
70  return nullptr;
71 }
72 
73 } // namespace qx
Y * rtti_cast(X &value)
Returns Y* if X or element_type of X is inherited from Y, otherwise nullptr.
Definition: rtti_cast.h:35