qxLib
flags.h
Go to the documentation of this file.
1 /**
2 
3  @file flags.h
4  @author Khrapov
5  @date 01.10.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/meta/concepts.h>
13 #include "qx/meta/type_traits.h"
14 
15 /**
16  @def QX_FLAGS_ENUM_CLASS
17  @brief Define to let to use this enum class in different binary operations returning qx::flags
18  @param enumName - enum class name
19 **/
20 #define QX_FLAGS_ENUM_CLASS(enumName) \
21  inline qx::flags<enumName> operator|(enumName left, enumName right) \
22  { \
23  return qx::flags(left) | right; \
24  } \
25  inline qx::flags<enumName> operator&(enumName left, enumName right) \
26  { \
27  return qx::flags(left) & right; \
28  } \
29  inline qx::flags<enumName> operator^(enumName left, enumName right) \
30  { \
31  return qx::flags(left) ^ right; \
32  }
33 
34 namespace qx
35 {
36 
37 /**
38 
39  @class flags
40  @brief Wrapper for enumerations to be used as a list of flags
41  @tparam enum_t - enum flags type
42  @code
43 
44  enum class EFlags
45  {
46  None = 0,
47  First = 1 << 0,
48  Second = 1 << 1,
49  Third = 1 << 2,
50  Fourth = 1 << 3,
51  };
52 
53  qx::flags<EFlags> flags;
54 
55  @endcode
56 
57 **/
58 template<enumeration_c enum_t>
59 class flags
60 {
61  using underlying_type = std::underlying_type_t<enum_t>;
62 
63 public:
64  // ------------------------------------------------ common operators -----------------------------------------------
65 
66  constexpr flags() noexcept = default;
67 
68  /**
69  @brief Reverse (or logically negotiate) all bits
70  **/
71  constexpr void reverse() noexcept;
72 
73  /**
74  @brief Shift bits to the left
75  @param nShift - num of positions to shift
76  **/
77  constexpr void shift_left(size_t nShift) noexcept;
78 
79  /**
80  @brief Shift bits to the right
81  @param nShift - num of positions to shift
82  **/
83  constexpr void shift_right(size_t nShift) noexcept;
84 
85  constexpr flags operator~() const noexcept;
86 
87  constexpr flags& operator<<=(size_t nShift) noexcept;
88  constexpr flags& operator>>=(size_t nShift) noexcept;
89 
90  constexpr flags operator<<(size_t nShift) const noexcept;
91  constexpr flags operator>>(size_t nShift) const noexcept;
92 
93  // ------------------------------------------------ enum_t operators -----------------------------------------------
94 
95  /**
96  @brief flags object constructor
97  @details Separate method for 1 arg to allow type deduction
98  @param eFlag -
99  **/
100  constexpr flags(enum_t eFlag) noexcept;
101 
102  /**
103  @brief flags object constructor
104  @tparam args_t... - template parameters pack, all types should be same as enum_t
105  @param flags - flags to add at construction
106  **/
107  template<class... args_t>
108  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
109  constexpr flags(args_t... flags) noexcept;
110 
111  /**
112  @brief Add all the flags specified
113  @tparam args_t... - template parameters pack, all types should be same as enum_t
114  @param flags - flags to add
115  **/
116  template<class... args_t>
117  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
118  constexpr void add(args_t... flags) noexcept;
119 
120  /**
121  @brief Remove all the flags specified
122  @tparam args_t... - template parameters pack, all types should be same as enum_t
123  @param flags - flags to remove
124  **/
125  template<class... args_t>
126  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
127  constexpr void remove(args_t... flags) noexcept;
128 
129  /**
130  @brief Apply a bitwise XOR to a flags value
131  @tparam args_t... - template parameters pack, all types should be same as enum_t
132  @param flags - flags to apply
133  **/
134  template<class... args_t>
135  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
136  constexpr void xor_(args_t... flags) noexcept;
137 
138  constexpr auto operator<=>(enum_t eFlag) const noexcept;
139  constexpr flags& operator=(enum_t eFlag) noexcept;
140 
141  constexpr flags& operator|=(enum_t eFlag) noexcept;
142  constexpr flags& operator&=(enum_t eFlag) noexcept;
143  constexpr flags& operator^=(enum_t eFlag) noexcept;
144 
145  constexpr flags operator|(enum_t eFlag) const noexcept;
146  constexpr flags operator&(enum_t eFlag) const noexcept;
147  constexpr flags operator^(enum_t eFlag) const noexcept;
148 
149  // ------------------------------------------------ flags operators ------------------------------------------------
150 
151  QX_COPYMOVABLE(flags);
152 
153  /**
154  @brief Add all the flags specified
155  @param flags_ - flags to add
156  **/
157  constexpr void add(flags flags_) noexcept;
158 
159  /**
160  @brief Remove all the flags specified
161  @param flags_ - flags to remove
162  **/
163  constexpr void remove(flags flags_) noexcept;
164 
165  /**
166  @brief Apply a bitwise XOR to a flags value
167  @param flags_ - flags to apply
168  **/
169  constexpr void xor_(flags flags_) noexcept;
170 
171  constexpr auto operator<=>(const flags&) const noexcept = default;
172 
173  constexpr flags& operator|=(flags flags_) noexcept;
174  constexpr flags& operator&=(flags flags_) noexcept;
175  constexpr flags& operator^=(flags flags_) noexcept;
176 
177  constexpr flags operator|(flags flags_) const noexcept;
178  constexpr flags operator&(flags flags_) const noexcept;
179  constexpr flags operator^(flags flags_) const noexcept;
180 
181  // --------------------------------------------------- algorithms --------------------------------------------------
182 
183  /**
184  @brief Check if flag is present
185  @param eFlag - flag to check
186  @retval - true if flag is present
187  **/
188  constexpr bool contains(enum_t eFlag) const noexcept;
189 
190  /**
191  @brief Check if all the specified flags are present
192  @tparam args_t... - template parameters pack, all types should be same as enum_t
193  @param flags - flags to check
194  @retval true if all the specified flags are present
195  **/
196  template<class... args_t>
197  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
198  constexpr bool contains_all(args_t... flags) const noexcept;
199 
200  /**
201  @brief Check if any of the specified flags is present
202  @tparam args_t... - template parameters pack, all types should be same as enum_t
203  @param flags - flags to check
204  @retval - true if any of the specified flags is present
205  **/
206  template<class... args_t>
207  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
208  constexpr bool contains_any(args_t... flags) const noexcept;
209 
210  /**
211  @brief Convert to the corresponding integer value
212  @retval - integer value
213  **/
214  constexpr underlying_type to_integer() const noexcept;
215 
216 private:
217  underlying_type m_EnumFlags = 0;
218 };
219 
220 } // namespace qx
221 
222 #include <qx/containers/flags.inl>
Wrapper for enumerations to be used as a list of flags.
Definition: flags.h:60
requires(sizeof...(args_t) >=1 &&are_specific_v< enum_t, args_t... >) const expr void add(args_t... flags) noexcept
Add all the flags specified.
requires(sizeof...(args_t) >=2 &&are_specific_v< enum_t, args_t... >) const expr flags(args_t... flags) noexcept
flags object constructor
constexpr void remove(flags flags_) noexcept
Remove all the flags specified.
constexpr void shift_left(size_t nShift) noexcept
Shift bits to the left.
Definition: flags.inl:20
constexpr bool contains(enum_t eFlag) const noexcept
Check if flag is present.
constexpr void xor_(flags flags_) noexcept
Apply a bitwise XOR to a flags value.
requires(sizeof...(args_t) >=1 &&are_specific_v< enum_t, args_t... >) const expr void xor_(args_t... flags) noexcept
Apply a bitwise XOR to a flags value.
constexpr void add(flags flags_) noexcept
Add all the flags specified.
requires(sizeof...(args_t) >=1 &&are_specific_v< enum_t, args_t... >) const expr void remove(args_t... flags) noexcept
Remove all the flags specified.
constexpr void reverse() noexcept
Reverse (or logically negotiate) all bits.
Definition: flags.inl:14
constexpr void shift_right(size_t nShift) noexcept
Shift bits to the right.
Definition: flags.inl:26
requires(sizeof...(args_t) >=2 &&are_specific_v< enum_t, args_t... >) const expr bool contains_all(args_t... flags) const noexcept
Check if all the specified flags are present.
requires(sizeof...(args_t) >=2 &&are_specific_v< enum_t, args_t... >) const expr bool contains_any(args_t... flags) const noexcept
Check if any of the specified flags is present.
constexpr underlying_type to_integer() const noexcept
Convert to the corresponding integer value.
constexpr bool contains_any(where_fwd_it_t itWhereBegin, where_fwd_it_t itWhereEnd, what_fwd_it_t itWhatBegin, what_fwd_it_t itWhatEnd)
Check that at least one element from the first range is equal to at least one element from the second...
Definition: contains.h:85