OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
port.hpp
1#pragma once
2#include <ossia/dataflow/audio_port.hpp>
3#include <ossia/dataflow/dataflow_fwd.hpp>
4#include <ossia/dataflow/geometry_port.hpp>
5#include <ossia/dataflow/midi_port.hpp>
6#include <ossia/dataflow/texture_port.hpp>
7#include <ossia/dataflow/value_port.hpp>
9#include <ossia/network/common/path.hpp>
10
11namespace ossia
12{
13struct OSSIA_EXPORT port
14{
15 enum scope_t : uint8_t
16 {
17 none = 1 << 0,
18 local = 1 << 1,
19 global = 1 << 2,
20 both = local | global
21 };
22
23 scope_t scope{scope_t::both};
24
25protected:
26 port() = default;
27 port(const port&) = delete;
28 port(port&&) = delete;
29 port& operator=(const port&) = delete;
30 port& operator=(port&&) = delete;
31};
32
33struct value_inlet;
34struct OSSIA_EXPORT inlet : public port
35{
36protected:
37 inlet() noexcept = default;
38
39 inlet(destination_t dest) noexcept
40 : address{std::move(dest)}
41 {
42 }
43
44 inlet(ossia::net::parameter_base& addr) noexcept
45 : address{&addr}
46 {
47 }
48
49 inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
50
51public:
52 virtual ~inlet();
53
54 void connect(graph_edge* e) noexcept
55 {
56 auto it = ossia::find(sources, e);
57 if(it == sources.end())
58 sources.push_back(e);
59 }
60
61 void disconnect(graph_edge* e) noexcept { ossia::remove_erase(sources, e); }
62
63 [[nodiscard]] virtual std::size_t which() const noexcept = 0;
64
65 template <typename T>
66 T* target() noexcept;
67 template <typename T>
68 const T* target() const noexcept;
69 template <typename T>
70 T& cast() noexcept;
71 template <typename T>
72 const T& cast() const noexcept;
73 template <typename T>
74 auto visit(const T& t);
75 template <typename T>
76 auto visit(const T& t) const;
77
78 auto& cables() noexcept { return sources; }
79 [[nodiscard]] auto& cables() const noexcept { return sources; }
80
81 virtual void pre_process();
82 virtual void post_process();
83
84 destination_t address;
85 ossia::small_vector<graph_edge*, 2> sources;
86 ossia::small_vector<value_inlet*, 2> child_inlets;
87
88 friend struct audio_inlet;
89 friend struct value_inlet;
90 friend struct midi_inlet;
91};
92
93struct OSSIA_EXPORT outlet : public port
94{
95protected:
96 outlet() noexcept = default;
97 outlet(destination_t dest) noexcept
98 : address{std::move(dest)}
99 {
100 }
101
102 outlet(ossia::net::parameter_base& addr) noexcept
103 : address{&addr}
104 {
105 }
106
107 outlet(graph_edge& edge) noexcept { targets.push_back(&edge); }
108
109public:
110 virtual ~outlet();
111
112 void connect(graph_edge* e) noexcept
113 {
114 auto it = ossia::find(targets, e);
115 if(it == targets.end())
116 targets.push_back(e);
117 }
118
119 void disconnect(graph_edge* e) noexcept { ossia::remove_erase(targets, e); }
120
121 void write(execution_state& e);
122
123 [[nodiscard]] virtual std::size_t which() const noexcept = 0;
124
125 template <typename T>
126 T* target() noexcept;
127 template <typename T>
128 const T* target() const noexcept;
129 template <typename T>
130 T& cast() noexcept;
131 template <typename T>
132 const T& cast() const noexcept;
133 template <typename T>
134 auto visit(const T& t);
135 template <typename T>
136 auto visit(const T& t) const;
137
138 virtual void pre_process();
139 virtual void post_process();
140
141 auto& cables() noexcept { return targets; }
142 [[nodiscard]] auto& cables() const noexcept { return targets; }
143
144 destination_t address;
145 ossia::small_vector<graph_edge*, 2> targets;
146 ossia::small_vector<value_inlet*, 2> child_inlets;
147
148 friend struct audio_outlet;
149 friend struct value_outlet;
150 friend struct midi_outlet;
151};
152
153struct OSSIA_EXPORT audio_inlet : public ossia::inlet
154{
155 audio_inlet() noexcept = default;
156
157 audio_inlet(destination_t dest) noexcept
158 : inlet{std::move(dest)}
159 {
160 }
161
162 audio_inlet(ossia::net::parameter_base& addr) noexcept
163 : inlet{&addr}
164 {
165 }
166
167 audio_inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
168
169 ~audio_inlet();
170
171 const ossia::audio_port& operator*() const noexcept { return data; }
172 const ossia::audio_port* operator->() const noexcept { return &data; }
173 ossia::audio_port& operator*() noexcept { return data; }
174 ossia::audio_port* operator->() noexcept { return &data; }
175
176 [[nodiscard]] std::size_t which() const noexcept final override
177 {
178 return audio_port::which;
179 }
180
181 ossia::audio_port data;
182};
183
184struct OSSIA_EXPORT midi_inlet : public ossia::inlet
185{
186 midi_inlet() noexcept = default;
187
188 midi_inlet(destination_t dest) noexcept
189 : inlet{std::move(dest)}
190 {
191 }
192
193 midi_inlet(ossia::net::parameter_base& addr) noexcept
194 : inlet{&addr}
195 {
196 }
197
198 midi_inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
199
200 ~midi_inlet();
201
202 const ossia::midi_port& operator*() const noexcept { return data; }
203 const ossia::midi_port* operator->() const noexcept { return &data; }
204 ossia::midi_port& operator*() noexcept { return data; }
205 ossia::midi_port* operator->() noexcept { return &data; }
206
207 [[nodiscard]] std::size_t which() const noexcept final override
208 {
209 return midi_port::which;
210 }
211
212 ossia::midi_port data;
213};
214
215struct OSSIA_EXPORT value_inlet : public ossia::inlet
216{
217 value_inlet() noexcept = default;
218
219 value_inlet(destination_t dest) noexcept
220 : inlet{std::move(dest)}
221 {
222 }
223
224 value_inlet(ossia::net::parameter_base& addr) noexcept
225 : inlet{&addr}
226 {
227 }
228
229 value_inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
230
231 ~value_inlet();
232
233 const ossia::value_port& operator*() const noexcept { return data; }
234 const ossia::value_port* operator->() const noexcept { return &data; }
235 ossia::value_port& operator*() noexcept { return data; }
236 ossia::value_port* operator->() noexcept { return &data; }
237
238 [[nodiscard]] std::size_t which() const noexcept final override
239 {
240 return value_port::which;
241 }
242
243 ossia::value_port data;
244};
245
246struct audio_outlet;
247
248OSSIA_EXPORT
249void process_audio_out_mono(ossia::audio_outlet& audio_out);
250
251OSSIA_EXPORT
252void process_audio_out_general(ossia::audio_outlet& audio_out);
253
254OSSIA_EXPORT
255void process_audio_out_mono(ossia::audio_port& i, ossia::audio_outlet& audio_out);
256
257OSSIA_EXPORT
258void process_audio_out_general(ossia::audio_port& i, ossia::audio_outlet& audio_out);
259
260struct OSSIA_EXPORT audio_outlet : public ossia::outlet
261{
262 audio_outlet() noexcept { init(); }
263
264 audio_outlet(destination_t dest) noexcept
265 : outlet{std::move(dest)}
266 {
267 init();
268 }
269
270 audio_outlet(ossia::net::parameter_base& addr) noexcept
271 : outlet{&addr}
272 {
273 init();
274 }
275
276 audio_outlet(graph_edge& edge) noexcept
277 {
278 targets.push_back(&edge);
279 init();
280 }
281
282 ~audio_outlet();
283
284 const ossia::audio_port& operator*() const noexcept { return data; }
285 const ossia::audio_port* operator->() const noexcept { return &data; }
286 ossia::audio_port& operator*() noexcept { return data; }
287 ossia::audio_port* operator->() noexcept { return &data; }
288
289 [[nodiscard]] std::size_t which() const noexcept final override
290 {
291 return audio_port::which;
292 }
293
294 void post_process() override;
295
296 double gain{1.};
297 pan_weight pan{1., 1.};
298
299 ossia::value_inlet gain_inlet;
300 ossia::value_inlet pan_inlet;
301
302 ossia::audio_port data;
303 bool has_gain{};
304
305private:
306 void init() noexcept
307 {
308 this->child_inlets.resize(2);
309 this->child_inlets[0] = &gain_inlet;
310 this->child_inlets[1] = &pan_inlet;
311 }
312};
313
314#if BOOST_VERSION >= 107200
315static_assert(noexcept(pan_weight{}));
316static_assert(noexcept(value_inlet{}));
317static_assert(noexcept(std::allocator<ossia::outlet>{}));
318static_assert(noexcept(audio_outlet{}));
319#endif
320
321struct OSSIA_EXPORT midi_outlet : public ossia::outlet
322{
323 midi_outlet() noexcept = default;
324
325 midi_outlet(destination_t dest) noexcept
326 : outlet{std::move(dest)}
327 {
328 }
329
330 midi_outlet(ossia::net::parameter_base& addr) noexcept
331 : outlet{&addr}
332 {
333 }
334
335 midi_outlet(graph_edge& edge) noexcept { targets.push_back(&edge); }
336 ~midi_outlet();
337
338 const ossia::midi_port& operator*() const noexcept { return data; }
339 const ossia::midi_port* operator->() const noexcept { return &data; }
340 ossia::midi_port& operator*() noexcept { return data; }
341 ossia::midi_port* operator->() noexcept { return &data; }
342
343 [[nodiscard]] std::size_t which() const noexcept final override
344 {
345 return midi_port::which;
346 }
347
348 ossia::midi_port data;
349};
350
351struct OSSIA_EXPORT value_outlet : public ossia::outlet
352{
353 value_outlet() noexcept = default;
354
355 value_outlet(destination_t dest) noexcept
356 : outlet{std::move(dest)}
357 {
358 }
359
360 value_outlet(ossia::net::parameter_base& addr) noexcept
361 : outlet{&addr}
362 {
363 }
364
365 value_outlet(graph_edge& edge) noexcept { targets.push_back(&edge); }
366 ~value_outlet();
367
368 const ossia::value_port& operator*() const noexcept { return data; }
369 const ossia::value_port* operator->() const noexcept { return &data; }
370 ossia::value_port& operator*() noexcept { return data; }
371 ossia::value_port* operator->() noexcept { return &data; }
372
373 [[nodiscard]] std::size_t which() const noexcept final override
374 {
375 return value_port::which;
376 }
377
378 ossia::value_port data;
379};
380
381struct texture_port
382{
383 static const constexpr int which = 3;
384};
385
386struct OSSIA_EXPORT texture_inlet : public ossia::inlet
387{
388 texture_inlet() noexcept = default;
389
390 texture_inlet(destination_t dest) noexcept
391 : inlet{std::move(dest)}
392 {
393 }
394
395 texture_inlet(ossia::net::parameter_base& addr) noexcept
396 : inlet{&addr}
397 {
398 }
399
400 texture_inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
401
402 ~texture_inlet();
403
404 [[nodiscard]] std::size_t which() const noexcept final override
405 {
406 return texture_port::which;
407 }
408
409 render_target_spec data;
410};
411
412struct OSSIA_EXPORT texture_outlet : public ossia::outlet
413{
414 texture_outlet() noexcept = default;
415
416 texture_outlet(destination_t dest) noexcept
417 : outlet{std::move(dest)}
418 {
419 }
420
421 texture_outlet(ossia::net::parameter_base& addr) noexcept
422 : outlet{&addr}
423 {
424 }
425
426 texture_outlet(graph_edge& edge) noexcept { targets.push_back(&edge); }
427 ~texture_outlet();
428
429 [[nodiscard]] std::size_t which() const noexcept final override
430 {
431 return texture_port::which;
432 }
433};
434
435struct OSSIA_EXPORT geometry_inlet : public ossia::inlet
436{
437 geometry_inlet() noexcept = default;
438
439 geometry_inlet(destination_t dest) noexcept
440 : inlet{std::move(dest)}
441 {
442 }
443
444 geometry_inlet(ossia::net::parameter_base& addr) noexcept
445 : inlet{&addr}
446 {
447 }
448
449 geometry_inlet(graph_edge& edge) noexcept { sources.push_back(&edge); }
450
451 ~geometry_inlet();
452
453 [[nodiscard]] std::size_t which() const noexcept final override
454 {
455 return geometry_port::which;
456 }
457
458 geometry_port data;
459};
460
461struct OSSIA_EXPORT geometry_outlet : public ossia::outlet
462{
463 geometry_outlet() noexcept = default;
464
465 geometry_outlet(destination_t dest) noexcept
466 : outlet{std::move(dest)}
467 {
468 }
469
470 geometry_outlet(ossia::net::parameter_base& addr) noexcept
471 : outlet{&addr}
472 {
473 }
474
475 geometry_outlet(graph_edge& edge) noexcept { targets.push_back(&edge); }
476 ~geometry_outlet();
477
478 [[nodiscard]] std::size_t which() const noexcept final override
479 {
480 return geometry_port::which;
481 }
482
483 geometry_port data;
484};
485
486template <typename T>
487inline T* inlet::target() noexcept
488{
489 if constexpr(std::is_same_v<T, audio_port>)
490 {
491 if(which() == 0)
492 {
493 return &static_cast<audio_inlet*>(this)->data;
494 }
495 else
496 {
497 return (audio_port*)nullptr;
498 }
499 }
500 else if constexpr(std::is_same_v<T, midi_port>)
501 {
502 if(which() == 1)
503 {
504 return &static_cast<midi_inlet*>(this)->data;
505 }
506 else
507 {
508 return (midi_port*)nullptr;
509 }
510 }
511 else if constexpr(std::is_same_v<T, value_port>)
512 {
513 if(which() == 2)
514 {
515 return &static_cast<value_inlet*>(this)->data;
516 }
517 else
518 {
519 return (value_port*)nullptr;
520 }
521 }
522 else if constexpr(std::is_same_v<T, texture_inlet>)
523 {
524 if(which() == 3)
525 {
526 return static_cast<texture_inlet*>(this);
527 }
528 else
529 {
530 return (texture_inlet*)nullptr;
531 }
532 }
533 else if constexpr(std::is_same_v<T, geometry_port>)
534 {
535 if(which() == 4)
536 {
537 return &static_cast<geometry_inlet*>(this)->data;
538 }
539 else
540 {
541 return (geometry_port*)nullptr;
542 }
543 }
544 else
545 {
546 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
547 }
548}
549
550template <typename T>
551inline T* outlet::target() noexcept
552{
553 if constexpr(std::is_same_v<T, audio_port>)
554 {
555 if(which() == 0)
556 {
557 return &static_cast<audio_outlet*>(this)->data;
558 }
559 else
560 {
561 return (audio_port*)nullptr;
562 }
563 }
564 else if constexpr(std::is_same_v<T, midi_port>)
565 {
566 if(which() == 1)
567 {
568 return &static_cast<midi_outlet*>(this)->data;
569 }
570 else
571 {
572 return (midi_port*)nullptr;
573 }
574 }
575 else if constexpr(std::is_same_v<T, value_port>)
576 {
577 if(which() == 2)
578 {
579 return &static_cast<value_outlet*>(this)->data;
580 }
581 else
582 {
583 return (value_port*)nullptr;
584 }
585 }
586 else if constexpr(std::is_same_v<T, texture_outlet>)
587 {
588 if(which() == 3)
589 {
590 return static_cast<texture_outlet*>(this);
591 }
592 else
593 {
594 return (texture_outlet*)nullptr;
595 }
596 }
597 else if constexpr(std::is_same_v<T, geometry_port>)
598 {
599 if(which() == 4)
600 {
601 return &static_cast<geometry_outlet*>(this)->data;
602 }
603 else
604 {
605 return (geometry_port*)nullptr;
606 }
607 }
608 else
609 {
610 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
611 }
612}
613template <typename T>
614inline const T* inlet::target() const noexcept
615{
616 if constexpr(std::is_same_v<T, audio_port>)
617 {
618 if(which() == 0)
619 {
620 return &static_cast<const audio_inlet*>(this)->data;
621 }
622 else
623 {
624 return (const audio_port*)nullptr;
625 }
626 }
627 else if constexpr(std::is_same_v<T, midi_port>)
628 {
629 if(which() == 1)
630 {
631 return &static_cast<const midi_inlet*>(this)->data;
632 }
633 else
634 {
635 return (const midi_port*)nullptr;
636 }
637 }
638 else if constexpr(std::is_same_v<T, value_port>)
639 {
640 if(which() == 2)
641 {
642 return &static_cast<const value_inlet*>(this)->data;
643 }
644 else
645 {
646 return (const value_port*)nullptr;
647 }
648 }
649 else if constexpr(std::is_same_v<T, texture_inlet>)
650 {
651 if(which() == 3)
652 {
653 return static_cast<const texture_inlet*>(this);
654 }
655 else
656 {
657 return (const texture_inlet*)nullptr;
658 }
659 }
660 else if constexpr(std::is_same_v<T, geometry_port>)
661 {
662 if(which() == 4)
663 {
664 return &static_cast<const geometry_inlet*>(this)->data;
665 }
666 else
667 {
668 return (const geometry_port*)nullptr;
669 }
670 }
671 else
672 {
673 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
674 }
675}
676
677template <typename T>
678inline const T* outlet::target() const noexcept
679{
680 if constexpr(std::is_same_v<T, audio_port>)
681 {
682 if(which() == 0)
683 {
684 return &static_cast<const audio_outlet*>(this)->data;
685 }
686 else
687 {
688 return (const audio_port*)nullptr;
689 }
690 }
691 else if constexpr(std::is_same_v<T, midi_port>)
692 {
693 if(which() == 1)
694 {
695 return &static_cast<const midi_outlet*>(this)->data;
696 }
697 else
698 {
699 return (const midi_port*)nullptr;
700 }
701 }
702 else if constexpr(std::is_same_v<T, value_port>)
703 {
704 if(which() == 2)
705 {
706 return &static_cast<const value_outlet*>(this)->data;
707 }
708 else
709 {
710 return (const value_port*)nullptr;
711 }
712 }
713 else if constexpr(std::is_same_v<T, texture_outlet>)
714 {
715 if(which() == 3)
716 {
717 return static_cast<const texture_outlet*>(this);
718 }
719 else
720 {
721 return (const texture_outlet*)nullptr;
722 }
723 }
724 else if constexpr(std::is_same_v<T, geometry_port>)
725 {
726 if(which() == 4)
727 {
728 return &static_cast<const geometry_outlet*>(this)->data;
729 }
730 else
731 {
732 return (const geometry_port*)nullptr;
733 }
734 }
735 else
736 {
737 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
738 }
739}
740
741template <typename T>
742inline T& inlet::cast() noexcept
743{
744 if constexpr(std::is_same_v<T, audio_port>)
745 {
746 return static_cast<audio_inlet*>(this)->data;
747 }
748 else if constexpr(std::is_same_v<T, midi_port>)
749 {
750 return static_cast<midi_inlet*>(this)->data;
751 }
752 else if constexpr(std::is_same_v<T, value_port>)
753 {
754 return static_cast<value_inlet*>(this)->data;
755 }
756 else if constexpr(std::is_same_v<T, texture_inlet>)
757 {
758 return static_cast<texture_inlet&>(*this);
759 }
760 else if constexpr(std::is_same_v<T, geometry_port>)
761 {
762 return static_cast<geometry_inlet*>(this)->data;
763 }
764 else
765 {
766 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
767 }
768}
769
770template <typename T>
771inline T& outlet::cast() noexcept
772{
773 if constexpr(std::is_same_v<T, audio_port>)
774 {
775 return static_cast<audio_outlet*>(this)->data;
776 }
777 else if constexpr(std::is_same_v<T, midi_port>)
778 {
779 return static_cast<midi_outlet*>(this)->data;
780 }
781 else if constexpr(std::is_same_v<T, value_port>)
782 {
783 return static_cast<value_outlet*>(this)->data;
784 }
785 else if constexpr(std::is_same_v<T, texture_outlet>)
786 {
787 return static_cast<texture_outlet&>(*this);
788 }
789 else if constexpr(std::is_same_v<T, geometry_port>)
790 {
791 return static_cast<geometry_outlet*>(this)->data;
792 }
793 else
794 {
795 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
796 }
797}
798template <typename T>
799inline const T& inlet::cast() const noexcept
800{
801 if constexpr(std::is_same_v<T, audio_port>)
802 {
803 return static_cast<const audio_inlet*>(this)->data;
804 }
805 else if constexpr(std::is_same_v<T, midi_port>)
806 {
807 return static_cast<const midi_inlet*>(this)->data;
808 }
809 else if constexpr(std::is_same_v<T, value_port>)
810 {
811 return static_cast<const value_inlet*>(this)->data;
812 }
813 else if constexpr(std::is_same_v<T, texture_inlet>)
814 {
815 return static_cast<const texture_inlet&>(*this);
816 }
817 else if constexpr(std::is_same_v<T, geometry_port>)
818 {
819 return static_cast<const geometry_inlet*>(this)->data;
820 }
821 else
822 {
823 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
824 }
825}
826
827template <typename T>
828inline const T& outlet::cast() const noexcept
829{
830 if constexpr(std::is_same_v<T, audio_port>)
831 {
832 return static_cast<const audio_outlet*>(this)->data;
833 }
834 else if constexpr(std::is_same_v<T, midi_port>)
835 {
836 return static_cast<const midi_outlet*>(this)->data;
837 }
838 else if constexpr(std::is_same_v<T, value_port>)
839 {
840 return static_cast<const value_outlet*>(this)->data;
841 }
842 else if constexpr(std::is_same_v<T, texture_outlet>)
843 {
844 return static_cast<const texture_outlet&>(*this);
845 }
846 else if constexpr(std::is_same_v<T, geometry_port>)
847 {
848 return static_cast<const geometry_outlet*>(this)->data;
849 }
850 else
851 {
852 static_assert(std::is_same_v<T, struct _>, "Invalid type requested");
853 }
854}
855
856template <typename T>
857inline auto inlet::visit(const T& t)
858{
859 switch(which())
860 {
861 case 0:
862 return t(static_cast<audio_inlet*>(this)->data);
863 case 1:
864 return t(static_cast<midi_inlet*>(this)->data);
865 case 2:
866 return t(static_cast<value_inlet*>(this)->data);
867 // case 3: return t(static_cast<texture_inlet&>(*this));
868 case 4:
869 return t(static_cast<geometry_inlet*>(this)->data);
870 }
871
872 if constexpr(std::is_invocable_v<T>)
873 return t();
874}
875
876template <typename T>
877inline auto outlet::visit(const T& t)
878{
879 switch(which())
880 {
881 case 0:
882 return t(static_cast<audio_outlet*>(this)->data);
883 case 1:
884 return t(static_cast<midi_outlet*>(this)->data);
885 case 2:
886 return t(static_cast<value_outlet*>(this)->data);
887 // case 3: return t(static_cast<texture_outlet&>(*this));
888 case 4:
889 return t(static_cast<geometry_outlet*>(this)->data);
890 }
891
892 if constexpr(std::is_invocable_v<T>)
893 return t();
894}
895template <typename T>
896inline auto inlet::visit(const T& t) const
897{
898 switch(which())
899 {
900 case 0:
901 return t(static_cast<const audio_inlet*>(this)->data);
902 case 1:
903 return t(static_cast<const midi_inlet*>(this)->data);
904 case 2:
905 return t(static_cast<const value_inlet*>(this)->data);
906 // case 3: return t(static_cast<const texture_inlet&>(*this));
907 case 4:
908 return t(static_cast<const geometry_inlet*>(this)->data);
909 }
910
911 if constexpr(std::is_invocable_v<T>)
912 return t();
913}
914template <typename T>
915inline auto outlet::visit(const T& t) const
916{
917 switch(which())
918 {
919 case 0:
920 return t(static_cast<const audio_outlet*>(this)->data);
921 case 1:
922 return t(static_cast<const midi_outlet*>(this)->data);
923 case 2:
924 return t(static_cast<const value_outlet*>(this)->data);
925 // case 3: return t(static_cast<const texture_outlet&>(*this));
926 case 4:
927 return t(static_cast<const geometry_outlet*>(this)->data);
928 }
929
930 if constexpr(std::is_invocable_v<T>)
931 return t();
932}
933
934}
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
Definition git_info.h:7