OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
domain_base_impl.hpp
1#pragma once
2#include <ossia/detail/flat_set.hpp>
3#include <ossia/detail/optional.hpp>
6
7#include <type_traits>
8
9namespace ossia
10{
11
12OSSIA_EXPORT ossia::value
13clamp(const ossia::value& val, const ossia::value& min, const ossia::value& max);
14OSSIA_EXPORT ossia::value
15wrap(const ossia::value& val, const ossia::value& min, const ossia::value& max);
16OSSIA_EXPORT ossia::value
17fold(const ossia::value& val, const ossia::value& min, const ossia::value& max);
18OSSIA_EXPORT ossia::value clamp_min(const ossia::value& val, const ossia::value& min);
19OSSIA_EXPORT ossia::value clamp_max(const ossia::value& val, const ossia::value& max);
20
21OSSIA_EXPORT ossia::value
22clamp(ossia::value&& val, const ossia::value& min, const ossia::value& max);
23OSSIA_EXPORT ossia::value
24wrap(ossia::value&& val, const ossia::value& min, const ossia::value& max);
25OSSIA_EXPORT ossia::value
26fold(ossia::value&& val, const ossia::value& min, const ossia::value& max);
27OSSIA_EXPORT ossia::value clamp_min(ossia::value&& val, const ossia::value& min);
28OSSIA_EXPORT ossia::value clamp_max(ossia::value&& val, const ossia::value& max);
29
30template <typename T>
31struct OSSIA_EXPORT domain_base
32{
33 using value_type = typename value_trait<T>::value_type;
34 std::optional<value_type> min;
35 std::optional<value_type> max;
36 std::vector<value_type> values;
37
38 domain_base() noexcept { }
39
40 domain_base(const domain_base& other)
41 : min{other.min}
42 , max{other.max}
43 , values{other.values}
44 {
45 }
46
47 domain_base(domain_base&& other) noexcept
48 : min{std::move(other.min)}
49 , max{std::move(other.max)}
50 , values{std::move(other.values)}
51 {
52 }
53
54 domain_base& operator=(const domain_base& other)
55 {
56 min = other.min;
57 max = other.max;
58 values = other.values;
59 return *this;
60 }
61
62 domain_base& operator=(domain_base&& other) noexcept
63 {
64 min = std::move(other.min);
65 max = std::move(other.max);
66 values = std::move(other.values);
67 return *this;
68 }
69
70 friend bool operator==(const domain_base<T>& lhs, const domain_base<T>& rhs) noexcept
71 {
72 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
73 }
74 friend bool operator!=(const domain_base<T>& lhs, const domain_base<T>& rhs) noexcept
75 {
76 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
77 }
78
79 domain_base(value_type v1, value_type v2) noexcept
80 : min{std::move(v1)}
81 , max{std::move(v2)}
82 {
83 }
84 domain_base(value_type v1, value_type v2, const ossia::flat_set<value_type>& vals)
85 : min{std::move(v1)}
86 , max{std::move(v2)}
87 , values{vals}
88 {
89 }
90 domain_base(value_type v1, value_type v2, ossia::flat_set<value_type>&& vals) noexcept
91 : min{std::move(v1)}
92 , max{std::move(v2)}
93 , values{std::move(vals)}
94 {
95 }
96};
97
98template <>
99struct OSSIA_EXPORT domain_base<impulse>
100{
101 using value_type = ossia::impulse;
102 friend bool
103 operator==(const domain_base<impulse>& lhs, const domain_base<impulse>& rhs)
104 {
105 return true;
106 }
107 friend bool
108 operator!=(const domain_base<impulse>& lhs, const domain_base<impulse>& rhs)
109 {
110 return false;
111 }
112};
113
114template <>
115struct OSSIA_EXPORT domain_base<bool>
116{
117 using value_type = bool;
118 static const constexpr bool min = false;
119 static const constexpr bool max = true;
120 friend bool operator==(const domain_base<bool>& lhs, const domain_base<bool>& rhs)
121 {
122 return true;
123 }
124 friend bool operator!=(const domain_base<bool>& lhs, const domain_base<bool>& rhs)
125 {
126 return false;
127 }
128};
129
130template <>
131struct OSSIA_EXPORT domain_base<std::string>
132{
133 std::vector<std::string> values;
134 friend bool
135 operator==(const domain_base<std::string>& lhs, const domain_base<std::string>& rhs)
136 {
137 return lhs.values == rhs.values;
138 }
139 friend bool
140 operator!=(const domain_base<std::string>& lhs, const domain_base<std::string>& rhs)
141 {
142 return lhs.values != rhs.values;
143 }
144};
145
146struct OSSIA_EXPORT vector_domain
147{
148 // If a value does not have a min / max the value won't be valid
149 using value_type = std::vector<ossia::value>;
150 value_type min;
151 value_type max;
152 std::vector<ossia::flat_set<ossia::value>> values;
153
154 vector_domain() noexcept { }
155 vector_domain(const vector_domain& other) noexcept
156 : min(other.min)
157 , max(other.max)
158 , values(other.values)
159 {
160 }
161
162 vector_domain(vector_domain&& other) noexcept
163 : min(std::move(other.min))
164 , max(std::move(other.max))
165 , values(std::move(other.values))
166 {
167 }
168
169 vector_domain& operator=(const vector_domain& other)
170 {
171 min = other.min;
172 max = other.max;
173 values = other.values;
174 return *this;
175 }
176
177 vector_domain& operator=(vector_domain&& other) noexcept
178 {
179 min = std::move(other.min);
180 max = std::move(other.max);
181 values = std::move(other.values);
182 return *this;
183 }
184
185 vector_domain(std::nullopt_t, std::nullopt_t) { }
186
187 vector_domain(const value_type& v1, const value_type& v2)
188 : min(v1)
189 , max(v2)
190 {
191 }
192 vector_domain(value_type&& v1, value_type&& v2)
193 : min(std::move(v1))
194 , max(std::move(v2))
195 {
196 }
197 vector_domain(
198 const value_type& v1, const value_type& v2,
199 const std::vector<ossia::flat_set<ossia::value>>& vals)
200 : min(v1)
201 , max(v2)
202 , values(vals)
203 {
204 }
205 vector_domain(
206 value_type&& v1, value_type&& v2,
207 std::vector<ossia::flat_set<ossia::value>>&& vals)
208 : min(std::move(v1))
209 , max(std::move(v2))
210 , values(std::move(vals))
211 {
212 }
213
214 friend bool operator==(const vector_domain& lhs, const vector_domain& rhs)
215 {
216 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
217 }
218 friend bool operator!=(const vector_domain& lhs, const vector_domain& rhs)
219 {
220 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
221 }
222};
223
224#if !defined(OSSIA_FREESTANDING)
225template <std::size_t N>
226struct OSSIA_EXPORT vecf_domain
227{
228 using value_type = std::array<float, N>;
229 std::array<std::optional<float>, N> min;
230 std::array<std::optional<float>, N> max;
231 std::array<ossia::flat_set<float>, N> values;
232
233 vecf_domain() noexcept { }
234 vecf_domain(const vecf_domain& other) noexcept
235 : min{std::move(other.min)}
236 , max{std::move(other.max)}
237 , values{std::move(other.values)}
238 {
239 }
240
241 vecf_domain(vecf_domain&& other) noexcept
242 : min{std::move(other.min)}
243 , max{std::move(other.max)}
244 , values{std::move(other.values)}
245 {
246 }
247
248 vecf_domain& operator=(const vecf_domain& other)
249 {
250 min = other.min;
251 max = other.max;
252 values = other.values;
253 return *this;
254 }
255
256 vecf_domain& operator=(vecf_domain&& other) noexcept
257 {
258 min = std::move(other.min);
259 max = std::move(other.max);
260 values = std::move(other.values);
261 return *this;
262 }
263
264 vecf_domain(
265 const std::array<std::optional<float>, N>& v1,
266 const std::array<std::optional<float>, N>& v2)
267 : min{v1}
268 , max{v2}
269 {
270 }
271
272 vecf_domain(const std::array<float, N>& v1, const std::array<float, N>& v2)
273 {
274 for(std::size_t i = 0; i < N; i++)
275 {
276 min[i] = v1[i];
277 max[i] = v2[i];
278 }
279 }
280 vecf_domain(
281 const std::array<std::optional<float>, N>& v1,
282 const std::array<std::optional<float>, N>& v2,
283 const std::array<ossia::flat_set<float>, N>& vals)
284 : min{v1}
285 , max{v2}
286 , values{vals}
287 {
288 }
289 vecf_domain(
290 const std::array<std::optional<float>, N>& v1,
291 const std::array<std::optional<float>, N>& v2,
292 std::array<ossia::flat_set<float>, N>&& vals)
293 : min{v1}
294 , max{v2}
295 , values{std::move(vals)}
296 {
297 }
298
299 friend bool operator==(const vecf_domain& lhs, const vecf_domain& rhs)
300 {
301 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
302 }
303 friend bool operator!=(const vecf_domain& lhs, const vecf_domain& rhs)
304 {
305 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
306 }
307};
308
309template <>
310struct OSSIA_EXPORT domain_base<ossia::value>
311{
312 using value_type = ossia::value;
313 std::optional<value_type> min;
314 std::optional<value_type> max;
315 std::vector<value_type> values;
316
317 domain_base() noexcept { }
318 domain_base(const domain_base& other) noexcept
319 : min{other.min}
320 , max{other.max}
321 , values{other.values}
322 {
323 }
324
325 domain_base(domain_base&& other) noexcept
326 : min{std::move(other.min)}
327 , max{std::move(other.max)}
328 , values{std::move(other.values)}
329 {
330 }
331
332 domain_base& operator=(const domain_base& other)
333 {
334 min = other.min;
335 max = other.max;
336 values = other.values;
337 return *this;
338 }
339
340 domain_base& operator=(domain_base&& other) noexcept
341 {
342 min = std::move(other.min);
343 max = std::move(other.max);
344 values = std::move(other.values);
345 return *this;
346 }
347
348 domain_base(const value_type& v1, const value_type& v2)
349 : min{v1}
350 , max{v2}
351 {
352 }
353 domain_base(value_type&& v1, value_type&& v2)
354 : min{std::move(v1)}
355 , max{std::move(v2)}
356 {
357 }
358 domain_base(
359 const value_type& v1, const value_type& v2, const std::vector<value_type>& vals)
360 : min{v1}
361 , max{v2}
362 , values{vals}
363 {
364 }
365 domain_base(value_type&& v1, value_type&& v2, std::vector<value_type>&& vals)
366 : min{std::move(v1)}
367 , max{std::move(v2)}
368 , values{std::move(vals)}
369 {
370 }
371
372 friend bool
373 operator==(const domain_base<ossia::value>& lhs, const domain_base<ossia::value>& rhs)
374 {
375 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
376 }
377 friend bool
378 operator!=(const domain_base<ossia::value>& lhs, const domain_base<ossia::value>& rhs)
379 {
380 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
381 }
382};
383#else
384
385template <std::size_t N>
386struct OSSIA_EXPORT vecf_domain : ossia::domain_base<float>
387{
388 using ossia::domain_base<float>::domain_base;
389 vecf_domain(const std::array<float, N>& v1, const std::array<float, N>& v2)
390 {
391 min = v1[0];
392 max = v2[0];
393 }
394 vecf_domain(
395 const std::array<std::optional<float>, N>& v1,
396 const std::array<std::optional<float>, N>& v2)
397 {
398 min = v1[0];
399 max = v2[0];
400 }
401 friend bool operator==(const vecf_domain& lhs, const vecf_domain& rhs)
402 {
403 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
404 }
405 friend bool operator!=(const vecf_domain& lhs, const vecf_domain& rhs)
406 {
407 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
408 }
409};
410template <>
411struct OSSIA_EXPORT domain_base<ossia::value> : ossia::domain_base<float>
412{
413 using ossia::domain_base<float>::domain_base;
414 friend bool
415 operator==(const domain_base<ossia::value>& lhs, const domain_base<ossia::value>& rhs)
416 {
417 return lhs.min == rhs.min && lhs.max == rhs.max && lhs.values == rhs.values;
418 }
419 friend bool
420 operator!=(const domain_base<ossia::value>& lhs, const domain_base<ossia::value>& rhs)
421 {
422 return lhs.min != rhs.min || lhs.max != rhs.max || lhs.values != rhs.values;
423 }
424};
425#endif
426}
The value class.
Definition value.hpp:173
Definition git_info.h:7
OSSIA_INLINE constexpr auto min(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
min function tailored for values
Definition math.hpp:125
OSSIA_INLINE constexpr T clamp(T d, const T min, const T max) noexcept
clamp Returns the value bounded by a min and a max
Definition math.hpp:154
OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
max function tailored for values
Definition math.hpp:96