qxLib
type_traits.h
Go to the documentation of this file.
1 /**
2 
3  @file type_traits.h
4  @author Khrapov
5  @data 10.09.2020
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/meta/concepts.h>
12 
13 namespace qx
14 {
15 
16 //--------------------------------- are_same ---------------------------------
17 
18 template<class...>
19 struct are_same : std::true_type
20 {
21 };
22 
23 template<class first_t, class second_t, class... rest_t>
24 struct are_same<first_t, second_t, rest_t...> : std::false_type
25 {
26 };
27 
28 // check if all of variadic arguments are same type
29 template<class first_t, class... rest_ty>
30 struct are_same<first_t, first_t, rest_ty...> : are_same<first_t, rest_ty...>
31 {
32 };
33 
34 template<class... args_t>
35 constexpr bool are_same_v = are_same<args_t...>::value;
36 
37 
38 
39 //------------------------------- are_specific_v -------------------------------
40 
41 template<class T, class... args_t>
42 static constexpr bool are_specific_v = std::conjunction_v<std::is_same<T, args_t>...>;
43 
44 
45 
46 //------------------------------- iterator_value -------------------------------
47 
48 // get value type of iterator
49 template<class T, class = void>
51 {
52 };
53 
54 // default implementation with ::value_type
55 template<class T>
56 struct iterator_value<T, std::enable_if_t<is_iterator<T>>>
57 {
58  using type = typename T::value_type;
59 };
60 
61 // some iterators may be implemented as pointers
62 template<class T>
63 struct iterator_value<T, std::enable_if_t<std::is_pointer_v<T>>>
64 {
65  using type = std::remove_pointer_t<T>;
66 };
67 
68 template<class T>
69 using iterator_value_t = typename iterator_value<T>::type;
70 
71 
72 
73 //--------------------------- is_specialization_exist --------------------------
74 
75 namespace details
76 {
77 
78 template<class T, std::size_t = sizeof(T)>
79 std::true_type is_specialization_exist_impl(T*);
80 
81 std::false_type is_specialization_exist_impl(...);
82 
83 } // namespace details
84 
85 // decide if a struct/class specialization exist
86 template<class T>
87 using is_specialization_exist = decltype(details::is_specialization_exist_impl(std::declval<T*>()));
88 
89 template<class T>
90 constexpr bool is_specialization_exist_v = is_specialization_exist<T>::value;
91 
92 
93 
94 //---------------------------- is_specialization_of ----------------------------
95 
96 template<class test_t, template<class...> class reference_t>
97 struct is_specialization_of : std::false_type
98 {
99 };
100 
101 template<template<class...> class reference_t, class... args_t>
102 struct is_specialization_of<reference_t<args_t...>, reference_t> : std::true_type
103 {
104 };
105 
106 // ----------------------------- qx::visit_overload ----------------------------
107 
108 /**
109  @struct visit_overload
110  @details This struct allows to visit std::variant with lambdas
111  @note https://www.cppstories.com/2018/09/visit-variants/
112  @tparam args_t - lambda types
113 
114  @code
115  struct Fluid { };
116  struct LightItem { };
117  struct HeavyItem { };
118  struct FragileItem { };
119 
120  int main() {
121  std::variant<Fluid, LightItem, HeavyItem, FragileItem> package;
122 
123  std::visit(qx::visit_overload
124  {
125  [](const Fluid& ) { std::cout << "fluid\n"; },
126  [](const LightItem& ) { std::cout << "light item\n"; },
127  [](const auto&) { std::cout << "default\n"; }
128  },
129  package);
130  }
131  @endcode
132 **/
133 template<class... args_t>
134 struct visit_overload : args_t...
135 {
136  using args_t::operator()...;
137 };
138 
139 template<class... args_t>
140 visit_overload(args_t...) -> visit_overload<args_t...>;
141 
142 } // namespace qx