OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
ossia-max/src/object_base.hpp
1#pragma once
2#include "ext.h"
3#include "ext_critical.h"
4#include "ext_obex.h"
5#include "matcher.hpp"
6
7#include <ossia/detail/config.hpp>
8
10#include <ossia/detail/optional.hpp>
11#include <ossia/detail/safe_vec.hpp>
12#include <ossia/network/base/node.hpp>
13#include <ossia/network/base/osc_address.hpp>
14#include <ossia/network/base/value_callback.hpp>
15#include <ossia/network/common/path.hpp>
16#include <ossia/network/dataspace/dataspace.hpp>
17#include <ossia/network/generic/generic_device.hpp>
18
19#include <iostream>
20#include <map>
21
22#define OSSIA_MAX_MAX_ATTR_SIZE 256
23
24namespace ossia
25{
26namespace max_binding
27{
28
29#pragma mark -
30#pragma mark t_object_base structure declaration
31
32enum class object_class
33{
34 root = 0,
35 param,
36 remote,
37 model,
38 view,
39 device,
40 client,
41 attribute,
42 explorer,
43 search,
44 monitor,
45 fuzzysearch,
46 cue
47};
48
49struct object_base;
50class ossia_max;
51
52struct search_result
53{
54 object_base* object{};
55 t_object* patcher{};
56 t_object* parent{};
57 t_symbol* classname{};
58 int level{}; // relative hierarchy level
59
60 friend bool operator<(search_result a, search_result b) { return a.level < b.level; }
61};
62
63struct object_base
64{
65public:
66 t_object m_object; // the Max object instance.
67 // !!! this member is handled by Max API : that's why there is no place in
68 // our code where it is initialized.
69
70 void** m_inlets{};
71 void* m_data_out{};
72 void* m_set_out{};
73 void* m_dumpout{};
74
75 // flags
76 bool m_dead{false}; // whether this object is being deleted or not;
77 bool m_is_deleted{};
78 bool m_lock{false}; // attribute lock
79 bool m_local_mute{false};
80 bool m_registering{}; // true while registering to prevent recursive registration
81 bool m_registered{}; // true if register_node() have been called at least once
82 ossia::net::address_scope m_addr_scope{};
83 object_class m_otype{};
84
85 void* m_clock{};
86 void* m_highlight_clock{}; // clock to reset color after some amount of time
87 // to highlight the object in the patcher
88
89 float m_rate{10.f};
90
91 std::shared_ptr<ossia::net::generic_device> m_device{};
92 using matcher_vector = std::vector<std::shared_ptr<matcher>>;
93 matcher_vector m_matchers{};
94 std::vector<matcher*> m_node_selection{};
95 std::optional<ossia::traversal::path> m_selection_path{};
96 static void class_setup(t_class* c);
97
98 void fill_selection();
99 void update_path(ossia::net::generic_device* explicit_device = nullptr);
100 void get_hierarchy();
101
102 void set_description();
103 void set_tags();
104 void set_priority();
105 void set_hidden();
106 void set_recall_safe();
107
108 // return the global path of the object with pattern
109 object_base* find_parent_object();
110
111 // return the first parent ossia object, nullptr otherwise
112 object_base*
113 find_parent_object_recursively(t_object* patcher, bool look_for_model_view);
114
115 static void get_description(object_base* x, std::vector<matcher*> nodes);
116 static void get_tags(object_base* x, std::vector<matcher*> nodes);
117 static void get_priority(object_base* x, std::vector<matcher*> nodes);
118 static void get_hidden(object_base* x, std::vector<matcher*> nodes);
119 static void get_zombie(object_base* x, std::vector<matcher*> nodes);
120 static void get_mess_cb(object_base* x, t_symbol* s);
121 static void select_mess_cb(object_base* x, t_symbol* s, int argc, t_atom* argv);
122 static void get_recall_safe(object_base* x, std::vector<matcher*> nodes);
123
124 // default attributes
125 t_symbol* m_name{};
126 t_symbol* m_tags[OSSIA_MAX_MAX_ATTR_SIZE] = {{}};
127 t_symbol* m_description{};
128 float m_priority{};
129 long m_invisible{};
130 long m_defer_set{1};
131 long m_recall_safe{};
132 long m_trim_addr{1};
133 long m_saveget{};
134 long m_saveset{};
135 long m_savebi{1};
136
137 t_object* m_patcher{};
138
139 long m_tags_size{};
140 long m_description_size{};
141 t_jrgba m_color{};
142
143 std::vector<search_result> m_found_parameters{};
144 std::vector<search_result> m_found_models{};
145
146 // constructor
147 object_base();
148
149 // dtor
150 ~object_base();
151
152 bool m_loadbanged{}; // true if object received a loadbang
153
154 std::mutex m_bind_mutex;
155 // TODO check where this is filled
156 std::vector<t_object*> m_patcher_hierarchy; // canvas hierarchy in ascending order, the
157 // last is the root patcher
158
159 static void update_attribute(
160 object_base* x, ossia::string_view attribute, const ossia::net::node_base* node);
161 static t_max_err
162 notify(object_base* x, t_symbol* s, t_symbol* msg, void* sender, void* data);
163 void on_parameter_removing(const ossia::net::parameter_base& n);
164 void on_node_removing(const ossia::net::node_base& n);
165
166 static void defer_set_output(object_base* x, t_symbol* s, int argc, t_atom* argv);
167 static void set(object_base* x, t_symbol* s, int argc, t_atom* argv);
168 static void get_address(object_base* x, std::vector<matcher*> nodes);
169 static void lock_and_touch(object_base* x, t_symbol* s);
170 static void loadbang(object_base* x);
171 void save_children_state();
172 void highlight();
173 void clear_and_init_registration();
174
175 static void reset_color(object_base* x);
176
177 void push_parameter_value(ossia::net::parameter_base* param, const ossia::value& val);
178 void set_matchers_index();
179
180 std::vector<std::string> m_paths{};
181
182 std::vector<std::shared_ptr<matcher>> find_or_create_matchers(ossia::net::generic_device* device_to_use = nullptr);
183
184protected:
185 void remove_matchers(const ossia::net::node_base& m);
186 void remove_matcher(const std::shared_ptr<matcher>& m);
187
188 [[nodiscard]] matcher_vector::iterator remove_matcher(matcher_vector::iterator m);
189
190 std::map<std::string, ossia::value> m_value_map{};
191
192private:
193 std::vector<std::shared_ptr<matcher>> find_parent_nodes();
194 void load_configuration(ossia_max& omax);
195 void create_patcher_hierarchy(ossia_max& omax);
196 unsigned long poly_index();
197 void make_global_paths(const std::string& name);
198};
199
200#pragma mark -
201#pragma mark Utilities
202
203// Typed function switcher to convert ossia::value to t_atom
204struct value2atom
205{
206 std::vector<t_atom>& data;
207 void operator()(impulse) const
208 {
209 t_atom a;
210 atom_setsym(&a, _sym_bang);
211 data.push_back(a);
212 }
213
214 void operator()(int32_t i) const
215 {
216 t_atom a;
217 atom_setlong(&a, i);
218 data.push_back(a);
219 }
220
221 void operator()(int64_t i) const
222 {
223 t_atom a;
224 atom_setlong(&a, i);
225 data.push_back(a);
226 }
227
228 void operator()(float f) const
229 {
230 t_atom a;
231 atom_setfloat(&a, f);
232 data.push_back(a);
233 }
234
235 void operator()(bool b) const
236 {
237 t_atom a;
238 float f = b ? 1. : 0.;
239 atom_setfloat(&a, f);
240 data.push_back(a);
241 }
242
243 void operator()(auto*) const noexcept = delete;
244
245 void operator()(const char* str) const
246 {
247 t_symbol* s = gensym(str);
248 t_atom a;
249 atom_setsym(&a, s);
250 data.push_back(a);
251 }
252
253 void operator()(std::string_view str) const
254 {
255 t_symbol* s = gensym(str.data());
256 t_atom a;
257 atom_setsym(&a, s);
258 data.push_back(a);
259 }
260
261 template <std::size_t N>
262 void operator()(std::array<float, N> vec) const
263 {
264 data.reserve(data.size() + N);
265 for(std::size_t i = 0; i < N; i++)
266 {
267 t_atom a;
268 atom_setfloat(&a, vec[i]);
269 data.push_back(a);
270 }
271 }
272
273 void operator()(const std::vector<ossia::value>& t) const
274 {
275 data.reserve(data.size() + t.size());
276 for(const auto& v : t)
277 v.apply(*this);
278 }
279
280 void operator()(const ossia::value_map_type& t) const { }
281
282 template <typename... T>
283 requires(sizeof...(T) > 1)
284 OSSIA_INLINE void operator()(T&&... t)
285 {
286 ((*this)(t), ...);
287 }
288
289 void operator()() const { }
290};
291
292inline void
293write_message(std::vector<t_atom>& va, void* out, t_symbol* sym, auto&&... args)
294{
295 va.clear();
296 value2atom vm{va};
297 vm(args...);
298 outlet_anything(out, sym, va.size(), va.data());
299}
300
301inline void write_message(
302 std::vector<t_atom>& va, void* out, t_symbol* prefix, t_symbol* sym, auto&&... args)
303{
304 if(!prefix)
305 {
306 write_message(va, out, sym, args...);
307 }
308 else
309 {
310 va.clear();
311 value2atom vm{va};
312 vm(std::string_view(sym->s_name));
313 vm(args...);
314 outlet_anything(out, prefix, va.size(), va.data());
315 }
316}
317// Template typed function switcher to convert t_atom or standard type into
318// ossia::value
319template <typename T>
320struct value_visitor
321{
322 T* x;
323
324 void set_out(t_atom& a) const
325 {
326 if(x->m_set_out)
327 {
328 if(x->m_defer_set)
329 {
330 defer_low((t_object*)x, (method)object_base::defer_set_output, _sym_set, 1, &a);
331 }
332 else
333 {
334 outlet_anything(x->m_set_out, _sym_set, 1, &a);
335 }
336 }
337 }
338
339 void set_out(int N, t_atom* a) const
340 {
341 if(x->m_set_out)
342 {
343 if(x->m_defer_set)
344 {
345 defer_low((t_object*)x, (method)object_base::defer_set_output, _sym_set, N, a);
346 }
347 else
348 {
349 outlet_anything(x->m_set_out, _sym_set, N, a);
350 }
351 }
352 }
353
354 void operator()(impulse) const
355 {
356 outlet_bang(x->m_data_out);
357
358 if(x->m_set_out)
359 outlet_bang(x->m_set_out);
360 }
361
362 void operator()(int32_t i) const
363 {
364 t_atom a;
365 atom_setlong(&a, i);
366 outlet_int(x->m_data_out, i);
367
368 set_out(a);
369 }
370
371 void operator()(float f) const
372 {
373 t_atom a;
374
375 atom_setfloat(&a, f);
376 if(x && x->m_data_out)
377 {
378 outlet_float(x->m_data_out, f);
379 }
380 set_out(a);
381 }
382
383 void operator()(bool b) const { (*this)(b ? 1 : 0); }
384
385 void operator()(const std::string& str) const
386 {
387 t_symbol* s = gensym(str.c_str());
388 t_atom a;
389
390 atom_setsym(&a, s);
391 outlet_anything(x->m_data_out, s, 0, nullptr);
392
393 set_out(a);
394 }
395
396 template <std::size_t N>
397 void operator()(std::array<float, N> vec) const
398 {
399 t_atom a[N];
400
401 for(int i = 0; i < N; i++)
402 atom_setfloat(a + i, vec[i]);
403 outlet_list(x->m_data_out, _sym_list, N, a);
404
405 set_out(N, a);
406 }
407
408 void operator()(const std::vector<ossia::value>& t) const
409 {
410 std::vector<t_atom> va;
411 value2atom vm{va};
412
413 if(t.empty())
414 return;
415
416 for(const auto& v : t)
417 v.apply(vm);
418
419 t_atom* list_ptr = !va.empty() ? va.data() : nullptr;
420 if(x->m_data_out)
421 outlet_list(x->m_data_out, _sym_list, va.size(), list_ptr);
422
423 set_out(va.size(), list_ptr);
424 }
425
426 void operator()(const ossia::value_map_type& t) const { }
427
428 void operator()() const
429 {
430 object_error((t_object*)x, "%s received an invalid data", x->m_name->s_name);
431 }
432};
433
434} // max namespace
435} // ossia namespace
The node_base class.
Definition node.hpp:48
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
The value class.
Definition value.hpp:173
Definition git_info.h:7