OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
min_max.hpp
1#pragma once
2#include <ossia/detail/flat_set.hpp>
3#include <ossia/detail/small_vector.hpp>
5
6namespace ossia
7{
8namespace detail
9{
11
12template <typename T>
14{
15 OSSIA_INLINE ossia::value operator()(const T& value) const
16 {
17 if(value.min)
18 return value::make<typename T::value_type>(*value.min);
19 else
20 return ossia::value{};
21 }
22};
23
24template <>
25struct domain_min_visitor_helper<domain_base<bool>>
26{
27 OSSIA_INLINE ossia::value operator()(const domain_base<bool>& value) const
28 {
29 return value.min;
30 }
31};
32
33template <>
34struct domain_min_visitor_helper<domain_base<impulse>>
35{
36 OSSIA_INLINE ossia::value operator()(const domain_base<impulse>& value) const
37 {
38 return impulse{};
39 }
40};
41
42template <>
43struct domain_min_visitor_helper<domain_base<std::string>>
44{
45 OSSIA_INLINE ossia::value operator()(const domain_base<std::string>& value) const
46 {
47 return {};
48 }
49};
50
51template <>
52struct domain_min_visitor_helper<domain_base<ossia::value>>
53{
54 OSSIA_INLINE ossia::value operator()(const domain_base<ossia::value>& value) const
55 {
56 // TODO for this case, it would maybe be better to
57 // use the empty state of value instead of a boost::optional ?
58 if(value.min)
59 return *value.min;
60 else
61 return ossia::value{};
62 }
63};
64
65template <>
66struct domain_min_visitor_helper<vector_domain>
67{
68 OSSIA_INLINE ossia::value operator()(const vector_domain& value) const
69 {
70 return value.min;
71 }
72};
73
74template <std::size_t N>
75struct domain_min_visitor_helper<vecf_domain<N>>
76{
77 OSSIA_INLINE ossia::value operator()(const vecf_domain<N>& value) const
78 {
79 // TODO for this case, it would maybe be better to
80 // use the empty state of value instead of a boost::optional ?
81 std::array<float, N> arr;
82#if !defined(OSSIA_FREESTANDING)
83 for(auto& val : value.min)
84 if(!val)
85 return ossia::value{};
86
87 for(std::size_t i = 0; i < N; i++)
88 {
89 arr[i] = *value.min[i];
90 }
91#endif
92 return arr;
93 }
94};
95
97template <typename T>
99{
100 OSSIA_INLINE ossia::value operator()(const T& value) const
101 {
102 if(value.max)
103 return value::make<typename T::value_type>(*value.max);
104 else
105 return ossia::value{};
106 }
107};
108
109template <>
110struct domain_max_visitor_helper<domain_base<bool>>
111{
112 OSSIA_INLINE ossia::value operator()(const domain_base<bool>& value) const
113 {
114 return value.max;
115 }
116};
117
118template <>
119struct domain_max_visitor_helper<domain_base<impulse>>
120{
121 OSSIA_INLINE ossia::value operator()(const domain_base<impulse>& value) const
122 {
123 return impulse{};
124 }
125};
126
127template <>
128struct domain_max_visitor_helper<domain_base<std::string>>
129{
130 OSSIA_INLINE ossia::value operator()(const domain_base<std::string>& value) const
131 {
132 return {};
133 }
134};
135
136template <>
137struct domain_max_visitor_helper<domain_base<ossia::value>>
138{
139 OSSIA_INLINE ossia::value operator()(const domain_base<ossia::value>& value) const
140 {
141 // TODO for this case, it would maybe be better to
142 // use the empty state of value instead of a boost::optional ?
143 if(value.max)
144 return *value.max;
145 else
146 return ossia::value{};
147 }
148};
149
150template <>
151struct domain_max_visitor_helper<vector_domain>
152{
153 OSSIA_INLINE ossia::value operator()(const vector_domain& value) const
154 {
155 return value.max;
156 }
157};
158
159template <std::size_t N>
160struct domain_max_visitor_helper<vecf_domain<N>>
161{
162 OSSIA_INLINE ossia::value operator()(const vecf_domain<N>& value) const
163 {
164 std::array<float, N> arr;
165#if !defined(OSSIA_FREESTANDING)
166 for(auto& val : value.max)
167 if(!val)
168 return ossia::value{};
169
170 for(std::size_t i = 0; i < N; i++)
171 {
172 arr[i] = *value.max[i];
173 }
174#endif
175 return arr;
176 }
177};
178
180using float_minmax = std::pair<std::optional<float>, std::optional<float>>;
181template <typename T>
182struct domain_float_minmax_visitor_helper
183{
184 OSSIA_INLINE float_minmax operator()(const T& value) const
185 {
186 float_minmax ret;
187 if(value.min)
188 ret.first = *value.min;
189 if(value.max)
190 ret.second = *value.max;
191 return ret;
192 }
193};
194
195template <>
196struct domain_float_minmax_visitor_helper<domain_base<bool>>
197{
198 OSSIA_INLINE float_minmax operator()(const domain_base<bool>& value) const
199 {
200 return std::make_pair(0.f, 1.f);
201 }
202};
203
204template <>
205struct domain_float_minmax_visitor_helper<domain_base<impulse>>
206{
207 OSSIA_INLINE float_minmax operator()(const domain_base<impulse>& value) const
208 {
209 return {};
210 }
211};
212
213template <>
214struct domain_float_minmax_visitor_helper<domain_base<std::string>>
215{
216 OSSIA_INLINE float_minmax operator()(const domain_base<std::string>& value) const
217 {
218 return {};
219 }
220};
221
222template <>
223struct domain_float_minmax_visitor_helper<domain_base<ossia::value>>
224{
225 OSSIA_INLINE float_minmax operator()(const domain_base<ossia::value>& value) const
226 {
227 // TODO for this case, it would maybe be better to
228 // use the empty state of value instead of a boost::optional ?
229
230 float_minmax ret;
231 if(value.min)
232 ret.first = ossia::convert<float>(*value.min);
233 if(value.max)
234 ret.second = ossia::convert<float>(*value.max);
235 return ret;
236 }
237};
238
239template <>
240struct domain_float_minmax_visitor_helper<vector_domain>
241{
242 OSSIA_INLINE float_minmax operator()(const vector_domain& value) const { return {}; }
243};
244
245template <std::size_t N>
246struct domain_float_minmax_visitor_helper<vecf_domain<N>>
247{
248 OSSIA_INLINE float_minmax operator()(const vecf_domain<N>& value) const { return {}; }
249};
250
251}
252
253struct domain_min_visitor
254{
255 template <typename T>
256 OSSIA_INLINE ossia::value operator()(const domain_base<T>& value) const
257 {
258 return detail::domain_min_visitor_helper<domain_base<T>>{}(value);
259 }
260
261 OSSIA_INLINE ossia::value operator()(const vector_domain& value) const
262 {
263 return detail::domain_min_visitor_helper<vector_domain>{}(value);
264 }
265
266 template <std::size_t N>
267 OSSIA_INLINE ossia::value operator()(const vecf_domain<N>& value) const
268 {
269 return detail::domain_min_visitor_helper<vecf_domain<N>>{}(value);
270 }
271
272 template <typename... T>
273 OSSIA_INLINE ossia::value operator()(const T&...) const
274 {
275 return ossia::value{};
276 }
277};
278
279struct domain_max_visitor
280{
281 template <typename T>
282 OSSIA_INLINE ossia::value operator()(const domain_base<T>& value) const
283 {
284 return detail::domain_max_visitor_helper<domain_base<T>>{}(value);
285 }
286
287 OSSIA_INLINE ossia::value operator()(const vector_domain& value) const
288 {
289 return detail::domain_max_visitor_helper<vector_domain>{}(value);
290 }
291
292 template <std::size_t N>
293 OSSIA_INLINE ossia::value operator()(const vecf_domain<N>& value) const
294 {
295 return detail::domain_max_visitor_helper<vecf_domain<N>>{}(value);
296 }
297
298 template <typename... T>
299 OSSIA_INLINE ossia::value operator()(const T&...) const
300 {
301 return ossia::value{};
302 }
303};
304
305struct domain_float_minmax_visitor
306{
307 using return_type = detail::float_minmax;
308 template <typename T>
309 OSSIA_INLINE return_type operator()(const domain_base<T>& value) const
310 {
311 return detail::domain_float_minmax_visitor_helper<domain_base<T>>{}(value);
312 }
313
314 OSSIA_INLINE return_type operator()(const vector_domain& value) const
315 {
316 return detail::domain_float_minmax_visitor_helper<vector_domain>{}(value);
317 }
318
319 template <std::size_t N>
320 OSSIA_INLINE return_type operator()(const vecf_domain<N>& value) const
321 {
322 return detail::domain_float_minmax_visitor_helper<vecf_domain<N>>{}(value);
323 }
324
325 template <typename... T>
326 OSSIA_INLINE return_type operator()(const T&...) const
327 {
328 return {};
329 }
330};
331struct domain_set_min_visitor
332{
333 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, int32_t incoming)
334 {
335 domain.min = incoming;
336 }
337 OSSIA_INLINE void operator()(domain_base<float>& domain, float incoming)
338 {
339 domain.min = incoming;
340 }
341 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, float incoming)
342 {
343 domain.min = incoming;
344 }
345 OSSIA_INLINE void operator()(domain_base<float>& domain, int32_t incoming)
346 {
347 domain.min = incoming;
348 }
349 OSSIA_INLINE void operator()(domain_base<bool>& domain, bool incoming) { }
350 OSSIA_INLINE void
351 operator()(vector_domain& domain, const std::vector<ossia::value>& incoming)
352 {
353 domain.min = incoming;
354 }
355
356 template <std::size_t N>
357 OSSIA_INLINE void operator()(vecf_domain<N>& domain, float incoming)
358 {
359#if !defined(OSSIA_FREESTANDING)
360 for(std::size_t i = 0; i < N; i++)
361 domain.min[i] = incoming;
362#endif
363 }
364
365 template <std::size_t N>
366 OSSIA_INLINE void operator()(vecf_domain<N>& domain, int incoming)
367 {
368#if !defined(OSSIA_FREESTANDING)
369 for(std::size_t i = 0; i < N; i++)
370 domain.min[i] = incoming;
371#endif
372 }
373
374 template <std::size_t N>
375 OSSIA_INLINE void
376 operator()(vecf_domain<N>& domain, const std::array<float, N>& incoming)
377 {
378#if !defined(OSSIA_FREESTANDING)
379 for(std::size_t i = 0; i < N; i++)
380 domain.min[i] = incoming[i];
381#endif
382 }
383
384 template <std::size_t N>
385 OSSIA_INLINE void
386 operator()(vecf_domain<N>& domain, const std::vector<ossia::value>& incoming)
387 {
388#if !defined(OSSIA_FREESTANDING)
389 if(incoming.size() == N)
390 {
391 auto conv = ossia::convert<std::array<float, N>>(incoming);
392 for(std::size_t i = 0; i < N; i++)
393 domain.min[i] = conv[i];
394 }
395 else
396 {
397 for(std::size_t i = 0; i < N; i++)
398 domain.min[i] = std::nullopt;
399 }
400#endif
401 }
402
403 template <typename T>
404 OSSIA_INLINE void operator()(domain_base<ossia::value>& domain, const T& incoming)
405 {
406#if !defined(OSSIA_FREESTANDING)
407 domain.min = ossia::value{incoming};
408#endif
409 }
410
411 // Removal cases
412 // Here we could maybe allow a cast or something like this...
413 // for e.g. int -> float
414 template <typename... U>
415 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, U&&...)
416 {
417 domain.min = std::nullopt;
418 }
419 template <typename... U>
420 OSSIA_INLINE void operator()(domain_base<float>& domain, U&&...)
421 {
422 domain.min = std::nullopt;
423 }
424 template <typename... U>
425 OSSIA_INLINE void operator()(domain_base<bool>& domain, U&&...)
426 {
427 }
428 template <typename... U>
429 OSSIA_INLINE void operator()(vector_domain& domain, U&&...)
430 {
431 domain.min.clear();
432 }
433 template <std::size_t N, typename... U>
434 OSSIA_INLINE void operator()(vecf_domain<N>& domain, U&&...)
435 {
436#if !defined(OSSIA_FREESTANDING)
437 for(std::size_t i = 0; i < N; i++)
438 domain.min[i] = std::nullopt;
439#endif
440 }
441 template <typename... U>
442 OSSIA_INLINE void operator()(domain_base<ossia::value>& domain, U&&...)
443 {
444 domain.min = std::nullopt;
445 }
446
447 template <typename T, typename... U>
448 OSSIA_INLINE void operator()(const domain_base<T>& domain, U&&...)
449 {
450 }
451};
452
453struct domain_set_max_visitor
454{
455 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, int32_t incoming)
456 {
457 domain.max = incoming;
458 }
459 OSSIA_INLINE void operator()(domain_base<float>& domain, float incoming)
460 {
461 domain.max = incoming;
462 }
463 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, float incoming)
464 {
465 domain.max = incoming;
466 }
467 OSSIA_INLINE void operator()(domain_base<float>& domain, int32_t incoming)
468 {
469 domain.max = incoming;
470 }
471 OSSIA_INLINE void operator()(domain_base<bool>& domain, bool incoming) { }
472 OSSIA_INLINE void
473 operator()(vector_domain& domain, const std::vector<ossia::value>& incoming)
474 {
475 domain.max = incoming;
476 }
477
478 template <std::size_t N>
479 OSSIA_INLINE void operator()(vecf_domain<N>& domain, float incoming)
480 {
481#if !defined(OSSIA_FREESTANDING)
482 for(std::size_t i = 0; i < N; i++)
483 domain.max[i] = incoming;
484#endif
485 }
486
487 template <std::size_t N>
488 OSSIA_INLINE void operator()(vecf_domain<N>& domain, int incoming)
489 {
490#if !defined(OSSIA_FREESTANDING)
491 for(std::size_t i = 0; i < N; i++)
492 domain.max[i] = incoming;
493#endif
494 }
495
496 template <std::size_t N>
497 OSSIA_INLINE void
498 operator()(vecf_domain<N>& domain, const std::array<float, N>& incoming)
499 {
500#if !defined(OSSIA_FREESTANDING)
501 for(std::size_t i = 0; i < N; i++)
502 domain.max[i] = incoming[i];
503#endif
504 }
505
506 template <std::size_t N>
507 OSSIA_INLINE void
508 operator()(vecf_domain<N>& domain, const std::vector<ossia::value>& incoming)
509 {
510#if !defined(OSSIA_FREESTANDING)
511 if(incoming.size() == N)
512 {
513 auto conv = ossia::convert<std::array<float, N>>(incoming);
514 for(std::size_t i = 0; i < N; i++)
515 domain.max[i] = conv[i];
516 }
517 else
518 {
519 for(std::size_t i = 0; i < N; i++)
520 domain.max[i] = std::nullopt;
521 }
522#endif
523 }
524
525 template <typename T>
526 OSSIA_INLINE void operator()(domain_base<ossia::value>& domain, const T& incoming)
527 {
528#if !defined(OSSIA_FREESTANDING)
529 domain.max = ossia::value{incoming};
530#endif
531 }
532
533 // Removal cases
534 template <typename... U>
535 OSSIA_INLINE void operator()(domain_base<int32_t>& domain, U&&...)
536 {
537 domain.max = std::nullopt;
538 }
539 template <typename... U>
540 OSSIA_INLINE void operator()(domain_base<float>& domain, U&&...)
541 {
542 domain.max = std::nullopt;
543 }
544 template <typename... U>
545 OSSIA_INLINE void operator()(domain_base<bool>& domain, U&&...)
546 {
547 }
548 template <typename... U>
549 OSSIA_INLINE void operator()(vector_domain& domain, U&&...)
550 {
551 domain.max.clear();
552 }
553 template <std::size_t N, typename... U>
554 OSSIA_INLINE void operator()(vecf_domain<N>& domain, U&&...)
555 {
556#if !defined(OSSIA_FREESTANDING)
557 for(std::size_t i = 0; i < N; i++)
558 domain.max[i] = std::nullopt;
559#endif
560 }
561 template <typename... U>
562 OSSIA_INLINE void operator()(domain_base<ossia::value>& domain, U&&...)
563 {
564 domain.max = std::nullopt;
565 }
566
567 template <typename T, typename... U>
568 OSSIA_INLINE void operator()(const domain_base<T>& domain, U&&...)
569 {
570 }
571};
572
573struct domain_minmax_creation_visitor
574{
575 template <typename T>
576 OSSIA_INLINE domain operator()(T min, T max)
577 {
578 return domain_base<T>(min, max);
579 }
580
581 OSSIA_INLINE domain operator()(bool min, bool max) { return domain_base<bool>{}; }
582
583 template <std::size_t N>
584 OSSIA_INLINE domain
585 operator()(const std::array<float, N>& lhs, const std::array<float, N>& rhs)
586 {
587 return vecf_domain<N>(lhs, rhs);
588 }
589
590 OSSIA_INLINE domain
591 operator()(const std::vector<ossia::value>& min, const std::vector<ossia::value>& max)
592 {
593 return vector_domain(min, max);
594 }
595 OSSIA_INLINE domain
596 operator()(std::vector<ossia::value>&& min, std::vector<ossia::value>&& max)
597 {
598 return vector_domain(std::move(min), std::move(max));
599 }
600
601 OSSIA_INLINE domain operator()(impulse, impulse) { return domain_base<impulse>{}; }
602
603 OSSIA_INLINE domain operator()(const std::string&, const std::string&)
604 {
605 return domain_base<std::string>();
606 }
607 OSSIA_INLINE domain operator()(const value_map_type&, const value_map_type&)
608 {
609 return domain{};
610 }
611
612 // Cases where there is no possible domain
613 template <typename T, typename U>
614 OSSIA_INLINE domain operator()(const T&, const U&)
615 {
616 return domain{};
617 }
618};
619
620struct domain_min_creation_visitor
621{
622 template <typename T>
623 OSSIA_INLINE domain operator()(T min)
624 {
625 domain_base<T> dom;
626 dom.min = std::move(min);
627 return dom;
628 }
629
630 OSSIA_INLINE domain operator()(bool min) { return domain_base<bool>{}; }
631
632 template <std::size_t N>
633 OSSIA_INLINE domain operator()(const std::array<float, N>& min)
634 {
635 vecf_domain<N> dom;
636#if !defined(OSSIA_FREESTANDING)
637 for(std::size_t i = 0; i < N; i++)
638 {
639 dom.min[i] = min[i];
640 }
641#endif
642 return dom;
643 }
644
645 OSSIA_INLINE domain operator()(const std::vector<ossia::value>& min)
646 {
647 vector_domain dom;
648 dom.min = min;
649 return dom;
650 }
651 OSSIA_INLINE domain operator()(const value_map_type& min) { return {}; }
652 OSSIA_INLINE domain operator()(std::vector<ossia::value>&& min)
653 {
654 return vector_domain(std::move(min), std::vector<ossia::value>(min.size()));
655 }
656
657 OSSIA_INLINE domain operator()(impulse) { return domain_base<impulse>{}; }
658
659 OSSIA_INLINE domain operator()(const std::string&)
660 {
661 return domain_base<std::string>();
662 }
663};
664
665struct domain_max_creation_visitor
666{
667 template <typename T>
668 OSSIA_INLINE domain operator()(T max)
669 {
670 domain_base<T> dom;
671 dom.max = std::move(max);
672 return dom;
673 }
674
675 OSSIA_INLINE domain operator()(bool max) { return domain_base<bool>{}; }
676
677 template <std::size_t N>
678 OSSIA_INLINE domain operator()(const std::array<float, N>& max)
679 {
680 vecf_domain<N> dom;
681#if !defined(OSSIA_FREESTANDING)
682 for(std::size_t i = 0; i < N; i++)
683 {
684 dom.max[i] = max[i];
685 }
686#endif
687 return dom;
688 }
689
690 OSSIA_INLINE domain operator()(const std::vector<ossia::value>& max)
691 {
692 vector_domain dom;
693 dom.max = max;
694 return dom;
695 }
696 OSSIA_INLINE domain operator()(std::vector<ossia::value>&& max)
697 {
698 return vector_domain(std::move(max), std::vector<ossia::value>(max.size()));
699 }
700
701 OSSIA_INLINE domain operator()(const value_map_type& min) { return {}; }
702 OSSIA_INLINE domain operator()(impulse) { return domain_base<impulse>{}; }
703
704 OSSIA_INLINE domain operator()(const std::string&)
705 {
706 return domain_base<std::string>();
707 }
708};
709
717{
718 const ossia::small_vector<ossia::value, 2>& values;
719 template <typename T>
720 domain operator()(const T&)
721 {
722 // Cases where there is no possible domain
723 domain_base<T> dom;
724 for(auto& value : values)
725 {
726 if(auto r = value.target<T>())
727 dom.values.push_back(*r);
728 }
729 return dom;
730 }
731
732 domain operator()(bool) { return domain_base<bool>{}; }
733
734 domain operator()(const std::vector<ossia::value>& orig)
735 {
736 vector_domain res;
737 ossia::flat_set<ossia::value> vals;
738
739 for(const auto& value : values)
740 vals.insert(value);
741
742 res.values
743 = std::vector<ossia::flat_set<ossia::value>>(orig.size(), std::move(vals));
744
745 return res;
746 }
747
748 OSSIA_INLINE domain operator()(const value_map_type& min) { return {}; }
749
750 template <std::size_t N>
751 domain operator()(const std::array<float, N>&)
752 {
753 vecf_domain<N> res;
754#if !defined(OSSIA_FREESTANDING)
755 ossia::flat_set<float> vals;
756
757 for(const auto& value : values)
758 if(auto r = value.target<float>())
759 vals.insert(*r);
760
761 for(std::size_t i = 0; i < N - 1; i++)
762 res.values[i] = vals;
763 res.values[N - 1] = std::move(vals);
764#endif
765 return res;
766 }
767
768 domain operator()()
769 {
770 // TODO
771 return {};
772 }
773
774 domain operator()(const impulse&) { return domain{}; }
775};
776}
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 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
Max ///.
Definition min_max.hpp:99
Min ///.
Definition min_max.hpp:14
The domain_value_set_creation_visitor struct.
Definition min_max.hpp:717
domain A domain of values
Definition domain_base.hpp:23