qxLib
concepts.h
Go to the documentation of this file.
1 /**
2 
3  @file concepts.h
4  @author Khrapov
5  @date 2.11.2021
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <type_traits>
12 #include <utility>
13 
14 namespace qx
15 {
16 
17 template<class T>
18 concept is_enumeration = std::is_enum_v<T>;
19 
20 template<class T>
21 concept enumeration_c = is_enumeration<T>;
22 
23 template<class T>
24 concept is_iterator = requires(T) { typename T::iterator_category; };
25 
26 template<class T>
27 concept is_random_access_iterator =
28  is_iterator<T> && std::derived_from<typename T::iterator_category, std::random_access_iterator_tag>;
29 
30 template<class T, class return_t, class... args_t>
31 concept callable_c = requires(T t, args_t&&... args) {
32  {
33  t(std::forward<args_t>(args)...)
34  } -> std::convertible_to<return_t>;
35 };
36 
37 // won't work for base_args_t containing template template or type template parameters
38 template<class derived_t, template<class... base_args_t> class base_t, class... partial_specialization_args_t>
39 concept derived_from_template_c = requires(const derived_t& derived) {
40  []<class... base_args_t>(const base_t<partial_specialization_args_t..., base_args_t...>&)
41  {
42  }(derived);
43 };
44 
45 namespace details
46 {
47 
48 template<class range_t, class T>
49 concept is_iterator_arrow_operator_of_t_type = requires(range_t t) {
50  {
51  t.begin().operator->()
52  } -> std::convertible_to<const T*>;
53  {
54  t.end().operator->()
55  } -> std::convertible_to<const T*>;
56 };
57 
58 template<class range_t, class T>
59 concept is_iterator_t_pointer = requires(range_t t) {
60  {
61  t.begin()
62  } -> std::convertible_to<const T*>;
63  {
64  t.end()
65  } -> std::convertible_to<const T*>;
66 };
67 
68 } // namespace details
69 
70 template<class range_t, class T>
71 concept range_of_t_c =
72  details::is_iterator_arrow_operator_of_t_type<range_t, T> || details::is_iterator_t_pointer<range_t, T>;
73 
74 } // namespace qx