qxLib
flags.inl
Go to the documentation of this file.
1 /**
2 
3  @file flags.inl
4  @author Khrapov
5  @date 29.09.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<enumeration_c enum_t>
14 constexpr void flags<enum_t>::reverse() noexcept
15 {
16  m_EnumFlags = ~m_EnumFlags;
17 }
18 
19 template<enumeration_c enum_t>
20 constexpr void flags<enum_t>::shift_left(size_t nShift) noexcept
21 {
22  m_EnumFlags << nShift;
23 }
24 
25 template<enumeration_c enum_t>
26 constexpr void flags<enum_t>::shift_right(size_t nShift) noexcept
27 {
28  m_EnumFlags >> nShift;
29 }
30 
31 template<enumeration_c enum_t>
32 constexpr flags<enum_t> flags<enum_t>::operator~() const noexcept
33 {
34  flags result = *this;
35  result.reverse();
36  return result;
37 }
38 
39 template<enumeration_c enum_t>
40 constexpr flags<enum_t>& flags<enum_t>::operator<<=(size_t nShift) noexcept
41 {
42  shift_left(nShift);
43  return *this;
44 }
45 
46 template<enumeration_c enum_t>
47 constexpr flags<enum_t>& flags<enum_t>::operator>>=(size_t nShift) noexcept
48 {
49  shift_right(nShift);
50  return *this;
51 }
52 
53 template<enumeration_c enum_t>
54 constexpr flags<enum_t> flags<enum_t>::operator<<(size_t nShift) const noexcept
55 {
56  flags result = *this;
57  result <<= nShift;
58  return result;
59 }
60 
61 template<enumeration_c enum_t>
62 constexpr flags<enum_t> flags<enum_t>::operator>>(size_t nShift) const noexcept
63 {
64  flags result = *this;
65  result >>= nShift;
66  return result;
67 }
68 
69 template<enumeration_c enum_t>
70 constexpr flags<enum_t>::flags(enum_t eFlag) noexcept : m_EnumFlags(underlying_type(eFlag))
71 {
72 }
73 
74 template<enumeration_c enum_t>
75 template<class... args_t>
76  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
77 constexpr flags<enum_t>::flags(args_t... flags) noexcept
78 {
79  add(flags...);
80 }
81 
82 template<enumeration_c enum_t>
83 template<class... args_t>
84  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
85 constexpr void flags<enum_t>::add(args_t... flags) noexcept
86 {
87  ((m_EnumFlags |= underlying_type(flags)), ...);
88 }
89 
90 template<enumeration_c enum_t>
91 template<class... args_t>
92  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
93 constexpr void flags<enum_t>::remove(args_t... flags) noexcept
94 {
95  ((m_EnumFlags &= ~underlying_type(flags)), ...);
96 }
97 
98 template<enumeration_c enum_t>
99 template<class... args_t>
100  requires(sizeof...(args_t) >= 1 && are_specific_v<enum_t, args_t...>)
101 constexpr void flags<enum_t>::xor_(args_t... flags) noexcept
102 {
103  ((m_EnumFlags ^= underlying_type(flags)), ...);
104 }
105 
106 template<enumeration_c enum_t>
107 constexpr auto flags<enum_t>::operator<=>(enum_t eFlag) const noexcept
108 {
109  return underlying_type(m_EnumFlags) <=> underlying_type(eFlag);
110 }
111 
112 template<enumeration_c enum_t>
113 constexpr flags<enum_t>& flags<enum_t>::operator=(enum_t eFlag) noexcept
114 {
115  m_EnumFlags = underlying_type(eFlag);
116  return *this;
117 }
118 
119 template<enumeration_c enum_t>
120 constexpr flags<enum_t>& flags<enum_t>::operator|=(enum_t eFlag) noexcept
121 {
122  add(eFlag);
123  return *this;
124 }
125 
126 template<enumeration_c enum_t>
127 constexpr flags<enum_t>& flags<enum_t>::operator&=(enum_t eFlag) noexcept
128 {
129  remove(eFlag);
130  return *this;
131 }
132 
133 template<enumeration_c enum_t>
134 constexpr flags<enum_t>& flags<enum_t>::operator^=(enum_t eFlag) noexcept
135 {
136  xor_(eFlag);
137  return *this;
138 }
139 
140 template<enumeration_c enum_t>
141 constexpr flags<enum_t> flags<enum_t>::operator|(enum_t eFlag) const noexcept
142 {
143  flags result = *this;
144  result |= eFlag;
145  return result;
146 }
147 
148 template<enumeration_c enum_t>
149 constexpr flags<enum_t> flags<enum_t>::operator&(enum_t eFlag) const noexcept
150 {
151  flags result = *this;
152  result &= eFlag;
153  return result;
154 }
155 
156 template<enumeration_c enum_t>
157 constexpr flags<enum_t> flags<enum_t>::operator^(enum_t eFlag) const noexcept
158 {
159  flags result = *this;
160  result ^= eFlag;
161  return result;
162 }
163 
164 template<enumeration_c enum_t>
165 constexpr void flags<enum_t>::add(flags flags_) noexcept
166 {
167  m_EnumFlags |= flags_.m_EnumFlags;
168 }
169 
170 template<enumeration_c enum_t>
171 constexpr void flags<enum_t>::remove(flags flags_) noexcept
172 {
173  m_EnumFlags &= flags_.m_EnumFlags;
174 }
175 
176 template<enumeration_c enum_t>
177 constexpr void flags<enum_t>::xor_(flags flags_) noexcept
178 {
179  m_EnumFlags ^= flags_.m_EnumFlags;
180 }
181 
182 template<enumeration_c enum_t>
183 constexpr flags<enum_t>& flags<enum_t>::operator|=(flags flags_) noexcept
184 {
185  add(flags_);
186  return *this;
187 }
188 
189 template<enumeration_c enum_t>
190 constexpr flags<enum_t>& flags<enum_t>::operator&=(flags flags_) noexcept
191 {
192  remove(flags_);
193  return *this;
194 }
195 
196 template<enumeration_c enum_t>
197 constexpr flags<enum_t>& flags<enum_t>::operator^=(flags flags_) noexcept
198 {
199  xor_(flags_);
200  return *this;
201 }
202 
203 template<enumeration_c enum_t>
204 constexpr flags<enum_t> flags<enum_t>::operator|(flags flags_) const noexcept
205 {
206  flags result = *this;
207  result |= flags_;
208  return result;
209 }
210 
211 template<enumeration_c enum_t>
212 constexpr flags<enum_t> flags<enum_t>::operator&(flags flags_) const noexcept
213 {
214  flags result = *this;
215  result &= flags_;
216  return result;
217 }
218 
219 template<enumeration_c enum_t>
220 constexpr flags<enum_t> flags<enum_t>::operator^(flags flags_) const noexcept
221 {
222  flags result = *this;
223  result ^= flags_;
224  return result;
225 }
226 
227 template<enumeration_c enum_t>
228 constexpr bool flags<enum_t>::contains(enum_t eFlag) const noexcept
229 {
230  return m_EnumFlags & underlying_type(eFlag);
231 }
232 
233 template<enumeration_c enum_t>
234 template<class... args_t>
235  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
236 constexpr bool flags<enum_t>::contains_all(args_t... flags) const noexcept
237 {
238  return (contains(flags) && ...);
239 }
240 
241 template<enumeration_c enum_t>
242 template<class... args_t>
243  requires(sizeof...(args_t) >= 2 && are_specific_v<enum_t, args_t...>)
244 constexpr bool flags<enum_t>::contains_any(args_t... flags) const noexcept
245 {
246  return (contains(flags) || ...);
247 }
248 
249 template<enumeration_c enum_t>
250 constexpr typename flags<enum_t>::underlying_type flags<enum_t>::to_integer() const noexcept
251 {
252  return m_EnumFlags;
253 }
254 
255 } // namespace qx
Wrapper for enumerations to be used as a list of flags.
Definition: flags.h:60
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.
constexpr void add(flags flags_) noexcept
Add 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
constexpr underlying_type to_integer() const noexcept
Convert to the corresponding integer value.
Check that tuple type contains T.