OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
ocue.utils.cpp
1#include <ossia/detail/json.hpp>
2
3#include <ossia-max/src/ocue.hpp>
4#include <ossia-max/src/ossia-max.hpp>
5#include <ossia-max/src/utils.hpp>
6
7#include <boost/algorithm/string.hpp>
8
9#include <rapidjson/prettywriter.h>
10
11namespace
12{
13 bool is_absolute_path(std::string_view v)
14 {
15 if (v.starts_with('/') || v.starts_with('~'))
16 return true;
17 if (v.find(':') != std::string_view::npos)
18 return true;
19 return false;
20 }
21 static std::string fix_url(std::string_view u)
22 {
23 if(u.starts_with("~"))
24 {
25 const char* home = getenv("HOME");
26 if (!home) home = "";
27 std::string res = home;
28 res += u.substr(1);
29 return res;
30 }
31 return std::string(u);
32 }
33 static std::string to_absolute_path(t_patcher* patcher, std::string_view url)
34 {
35 std::string full_path;
36 if (is_absolute_path(url))
37 {
38 full_path = fix_url(url);
39 }
40 else
41 {
42 std::string_view patcher_path = jpatcher_get_filepath(patcher)->s_name;
43 std::string_view patcher_name = jpatcher_get_filename(patcher)->s_name;
44 auto end_it = patcher_path.rfind(patcher_name);
45 if (end_it <= 0 || end_it > std::string_view::npos)
46 return {};
47
48 int start = 0;
49 if (patcher_path.starts_with("Macintosh HD:"))
50 start = strlen("Macintosh HD:");
51 full_path = patcher_path.substr(start, end_it - start);
52 full_path += url;
53 }
54 return full_path;
55 }
56 static std::string
57 prompt_open_filename(std::string_view dialogtitle, std::string_view default_filename)
58 {
59 char buffer[MAX_PATH_CHARS] = {};
60 strncpy(buffer, default_filename.data(), default_filename.size());
61
62 open_promptset(dialogtitle.data());
63
64 short path{};
65 t_fourcc outtype{};
66 t_fourcc filetype = 'TEXT';
67 if(open_dialog(buffer, &path, &outtype, &filetype, 1) == 0)
68 {
69 t_filehandle hdl{};
70 path_opensysfile(buffer, path, &hdl, e_max_openfile_permissions::PATH_READ_PERM);
71
72 char full_path[MAX_PATH_CHARS] = {};
73 path_topathname(path, buffer, full_path);
74
75 char native_path[MAX_PATH_CHARS];
76 path_nameconform(full_path, native_path, PATH_STYLE_NATIVE, PATH_TYPE_BOOT);
77
78 sysfile_close(hdl);
79 return native_path;
80 }
81 return {};
82 }
83static std::string
84prompt_save_filename(std::string_view dialogtitle, std::string_view default_filename)
85{
86 char buffer[MAX_PATH_CHARS] = {};
87 strncpy(buffer, default_filename.data(), default_filename.size());
88
89 saveas_promptset(dialogtitle.data());
90
91 short path{};
92 t_fourcc outtype{};
93 t_fourcc filetype = 'TEXT';
94 if(saveasdialog_extended(buffer, &path, &outtype, &filetype, 1) == 0)
95 {
96 t_filehandle hdl{};
97 path_createsysfile(buffer, path, filetype, &hdl);
98
99 char full_path[MAX_PATH_CHARS] = {};
100 path_topathname(path, buffer, full_path);
101
102 char native_path[MAX_PATH_CHARS];
103 path_nameconform(full_path, native_path, PATH_STYLE_NATIVE, PATH_TYPE_BOOT);
104
105 sysfile_close(hdl);
106 return native_path;
107 }
108 return {};
109}
110
111ossia::selection_filters parse_selection_filter(int argc, t_atom* argv)
112{
113 ossia::selection_filters filt;
114
115 enum
116 {
117 unknown,
118 processing_names,
119 processing_val_type,
120 processing_access,
121 processing_bounding,
122 processing_tags,
123 processing_visibility
124 } state
125 = processing_names;
126
127 for(int i = 0; i < argc; i++)
128 {
129 if(argv[i].a_type == A_SYM)
130 {
131 std::string symb
132 = boost::algorithm::to_lower_copy(std::string(argv[i].a_w.w_sym->s_name));
133 if(symb.starts_with("@"))
134 {
135 if(symb == "@valuetype")
136 {
137 state = processing_val_type;
138 continue;
139 }
140 if(symb == "@access")
141 {
142 state = processing_access;
143 continue;
144 }
145 if(symb == "@bounding")
146 {
147 state = processing_bounding;
148 continue;
149 }
150 if(symb == "@tags")
151 {
152 state = processing_tags;
153 continue;
154 }
155 if(symb == "@visibility")
156 {
157 state = processing_visibility;
158 continue;
159 }
160 }
161
162 switch(state)
163 {
164 case processing_names:
165 filt.selection.push_back(std::string(symb));
166 break;
167 case processing_val_type: {
168 if(auto param = ossia::default_parameter_for_type(symb))
169 {
170 filt.type.push_back(ossia::underlying_type(param->type));
171 }
172 break;
173 }
174 case processing_access: {
175 if(symb == "get")
176 filt.access.push_back(ossia::access_mode::GET);
177 if(symb == "set")
178 filt.access.push_back(ossia::access_mode::SET);
179 if(symb == "bi")
180 filt.access.push_back(ossia::access_mode::BI);
181 break;
182 }
183 case processing_bounding: {
184 if(symb == "free")
185 filt.bounding.push_back(ossia::bounding_mode::FREE);
186 if(symb == "clip")
187 filt.bounding.push_back(ossia::bounding_mode::CLIP);
188 if(symb == "low")
189 filt.bounding.push_back(ossia::bounding_mode::CLAMP_LOW);
190 if(symb == "high")
191 filt.bounding.push_back(ossia::bounding_mode::CLAMP_HIGH);
192 if(symb == "wrap")
193 filt.bounding.push_back(ossia::bounding_mode::WRAP);
194 if(symb == "fold")
195 filt.bounding.push_back(ossia::bounding_mode::FOLD);
196 break;
197 }
198 case processing_tags: {
199 filt.tags.push_back(std::string(symb));
200 break;
201 }
202 case processing_visibility:
203 filt.visibility
204 = symb == "visible" || symb == "true" || symb == "yes" || symb == "ok"
205 ? ossia::selection_filters::visible
206 : ossia::selection_filters::invisible;
207 break;
208 case unknown:
209 break;
210 }
211 }
212 }
213
214 return filt;
215}
216using writer_t = rapidjson::PrettyWriter<rapidjson::StringBuffer>;
217struct write_json_preset_value
218{
219 writer_t& writer;
220 void operator()(ossia::impulse) const
221 {
222 writer.Key("type");
223 writer.String("impulse");
224 }
225 void operator()(int v) const
226 {
227 writer.Key("type");
228 writer.String("int");
229 writer.Key("value");
230 writer.Int(v);
231 }
232 void operator()(float v) const
233 {
234 writer.Key("type");
235 writer.Key("float");
236 writer.Key("value");
237 writer.Double(v);
238 }
239 void operator()(bool v) const
240 {
241 writer.Key("type");
242 writer.String("bool");
243 writer.Key("value");
244 writer.Bool(v);
245 }
246
247 void operator()(const std::string& v) const
248 {
249 writer.Key("type");
250 writer.String("string");
251 writer.Key("value");
252 writer.String(v);
253 }
254
255 template <std::size_t N>
256 void operator()(const std::array<float, N>& vec) const
257 {
258 writer.Key("type");
259 writer.String("vec" + std::to_string(N) + "f");
260 writer.Key("value");
261 writer.StartArray();
262 for(std::size_t i = 0; i < N; i++)
263 {
264 writer.Double(vec[i]);
265 }
266 writer.EndArray();
267 }
268
269 void operator()(const std::vector<ossia::value>& vec) const
270 {
271 writer.Key("type");
272 writer.String("list");
273 writer.Key("value");
274 writer.StartArray();
275 for(auto& val : vec)
276 {
277 writer.StartObject();
278 val.apply(*this);
279 writer.EndObject();
280 }
281 writer.EndArray();
282 }
283 void operator()(const ossia::value_map_type& vec) const { }
284 void operator()() const { throw std::runtime_error("value_to_json_value: no type"); }
285};
286
287static void cue_to_json(writer_t& doc, const ossia::cue& cu)
288{
289 doc.StartObject();
290 doc.Key("name");
291 doc.String(cu.name);
292
293 doc.Key("preset");
294 doc.StartArray();
295 for(auto& [k, v] : cu.preset)
296 {
297 doc.StartObject();
298 doc.Key("address");
299 doc.String(k);
300
301 v.apply(write_json_preset_value{doc});
302 doc.EndObject();
303 }
304 doc.EndArray();
305 doc.EndObject();
306}
307
308static rapidjson::StringBuffer cues_to_string(const ossia::cues& c)
309{
310
311 rapidjson::StringBuffer buffer;
312 writer_t doc{buffer};
313
314 doc.StartObject();
315 doc.Key("cues");
316 doc.StartArray();
317 for(auto& cue : c.m_cues)
318 {
319 cue_to_json(doc, cue);
320 }
321 doc.EndArray();
322 doc.EndObject();
323
324 return buffer;
325}
326
327// Format of a cue:
328// {
329// "name": "foo",
330// "preset": [
331// { "address": "/foo/bar", "value": 123, type: "int" },
332// { "address": "/baz", "value": 123, type: "int" },
333// ]
334// }
335
336// Format of a cuelist:
337// { "cues": [ ... cue ... ] }
338
339static ossia::value
340read_typed_ossia_value_from_json(const rapidjson::Document::ConstObject& value);
341static ossia::value read_typed_ossia_value_from_json(
342 const rapidjson::Document::ConstObject& value, std::string_view type)
343{
344 if(type == "impulse")
345 {
346 return ossia::impulse{};
347 }
348 auto it = value.FindMember("value");
349 if(it == value.MemberEnd())
350 return {};
351
352 if(type == "int")
353 {
354 if(it->value.IsInt())
355 {
356 return it->value.GetInt();
357 }
358 }
359 else if(type == "float")
360 {
361 if(it->value.IsFloat())
362 {
363 return it->value.GetFloat();
364 }
365 }
366 else if(type == "string")
367 {
368 if(it->value.IsString())
369 {
370 return std::string(it->value.GetString(), it->value.GetStringLength());
371 }
372 }
373 else if(type == "bool")
374 {
375 if(it->value.IsBool())
376 {
377 return it->value.GetBool();
378 }
379 }
380 else if(type == "vec4f")
381 {
382 if(it->value.IsArray())
383 {
384 auto arr = it->value.GetArray();
385 if(arr.Size() == 4 && arr[0].IsDouble() && arr[1].IsDouble() && arr[2].IsDouble()
386 && arr[3].IsDouble())
387 return ossia::make_vec(
388 arr[0].GetDouble(), arr[1].GetDouble(), arr[2].GetDouble(),
389 arr[3].GetDouble());
390 }
391 }
392 else if(type == "vec3f")
393 {
394 if(it->value.IsArray())
395 {
396 auto arr = it->value.GetArray();
397 if(arr.Size() == 3 && arr[0].IsDouble() && arr[1].IsDouble() && arr[2].IsDouble())
398 return ossia::make_vec(
399 arr[0].GetDouble(), arr[1].GetDouble(), arr[2].GetDouble());
400 }
401 }
402 else if(type == "vec2f")
403 {
404 if(it->value.IsArray())
405 {
406 auto arr = it->value.GetArray();
407 if(arr.Size() == 2 && arr[0].IsDouble() && arr[1].IsDouble())
408 return ossia::make_vec(arr[0].GetDouble(), arr[1].GetDouble());
409 }
410 }
411 else if(type == "list")
412 {
413 if(it->value.IsArray())
414 {
415 std::vector<ossia::value> vec;
416 auto arr = it->value.GetArray();
417 vec.reserve(arr.Size());
418 for(const rapidjson::Value& sub_val : arr)
419 {
420 if(sub_val.IsObject())
421 vec.push_back(read_typed_ossia_value_from_json(sub_val.GetObject()));
422 }
423 return vec;
424 }
425 }
426 return {};
427}
428
429static ossia::value
430read_typed_ossia_value_from_json(const rapidjson::Document::ConstObject& value)
431{
432 if(auto it = value.FindMember("type"); it != value.MemberEnd() && it->value.IsString())
433 {
434 std::string_view type{it->value.GetString(), it->value.GetStringLength()};
435 return read_typed_ossia_value_from_json(value, type);
436 }
437 return {};
438}
439
440static void read_cue_preset_from_json(
441 const rapidjson::Document::ConstArray& json, ossia::presets::preset& p)
442{
443 for(const rapidjson::Value& value : json)
444 {
445 if(!value.IsObject())
446 continue;
447 std::string addr;
448
449 if(auto it = value.FindMember("address");
450 it != value.MemberEnd() && it->value.IsString())
451 {
452 addr = std::string{it->value.GetString(), it->value.GetStringLength()};
453
454 if(addr.empty())
455 continue;
456 }
457
458 ossia::value v = read_typed_ossia_value_from_json(value.GetObject());
459 if(v.valid())
460 p.emplace_back(std::move(addr), std::move(v));
461 }
462}
463
464static void read_cues_from_json(
465 const rapidjson::Document::Array& json, std::vector<ossia::cue>& cues)
466{
467 for(const rapidjson::Value& value : json)
468 {
469 if(!value.IsObject())
470 continue;
471 ossia::cue c;
472 if(auto it = value.FindMember("name");
473 it != value.MemberEnd() && it->value.IsString())
474 {
475 c.name = std::string{it->value.GetString(), it->value.GetStringLength()};
476 }
477
478 if(auto it = value.FindMember("preset");
479 it != value.MemberEnd() && it->value.IsArray())
480 {
481 read_cue_preset_from_json(it->value.GetArray(), c.preset);
482 }
483 cues.push_back(std::move(c));
484 }
485}
486
487static void invoke_mem_fun(int argc, t_atom* argv, auto f)
488{
489#define if_possible(F) \
490 if constexpr(requires { F; }) \
491 { \
492 F; \
493 }
494 if(argc == 0)
495 {
496 if_possible(f());
497 }
498 else if(argc == 1 && argv[0].a_type == A_SYM)
499 {
500 if_possible(f(argv[0].a_w.w_sym->s_name));
501 }
502 else if(argc == 1 && argv[0].a_type == A_LONG)
503 {
504 if_possible(f(argv[0].a_w.w_long));
505 }
506 else if(argc == 1 && argv[0].a_type == A_FLOAT)
507 {
508 if_possible(f((int)argv[0].a_w.w_float));
509 }
510 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_LONG)
511 {
512 if_possible(f(argv[0].a_w.w_long, argv[1].a_w.w_long));
513 }
514 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_FLOAT)
515 {
516 if_possible(f(argv[0].a_w.w_long, (int)argv[1].a_w.w_float));
517 }
518 else if(argc == 2 && argv[0].a_type == A_FLOAT && argv[1].a_type == A_LONG)
519 {
520 if_possible(f((int)argv[0].a_w.w_float, argv[1].a_w.w_long));
521 }
522 else if(argc == 2 && argv[0].a_type == A_FLOAT && argv[1].a_type == A_FLOAT)
523 {
524 if_possible(f((int)argv[0].a_w.w_float, (int)argv[1].a_w.w_float));
525 }
526 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_SYM)
527 {
528 if_possible(f(argv[0].a_w.w_long, argv[1].a_w.w_sym->s_name));
529 }
530 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_LONG)
531 {
532 if_possible(f(argv[0].a_w.w_sym->s_name, argv[1].a_w.w_long));
533 }
534 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_FLOAT)
535 {
536 if_possible(f(argv[0].a_w.w_sym->s_name, (int)argv[1].a_w.w_float));
537 }
538 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_SYM)
539 {
540 if_possible(f(argv[0].a_w.w_sym->s_name, argv[1].a_w.w_sym->s_name));
541 }
542#undef if_possible
543}
544
545static void pack_to_atom(t_atom& t, long long& i)
546{
547 atom_setlong(&t, i);
548}
549static void pack_to_atom(t_atom& t, long& i)
550{
551 atom_setlong(&t, i);
552}
553static void pack_to_atom(t_atom& t, int& i)
554{
555 atom_setlong(&t, i);
556}
557static void pack_to_atom(t_atom& t, float& i)
558{
559 atom_setfloat(&t, i);
560}
561static void pack_to_atom(t_atom& t, char*& i)
562{
563 atom_setsym(&t, gensym(i));
564}
565static void pack_to_atom(t_atom& t, std::string_view i)
566{
567 if(!i.empty())
568 atom_setsym(&t, gensym(i.data()));
569 else
570 atom_setsym(&t, gensym("<invalid>"));
571}
572
573static std::string_view
574name_from_args(const ossia::cues& cue, std::string_view args) noexcept
575{
576 return args;
577}
578
579static std::string_view name_from_args(const ossia::cues& cue, int idx) noexcept
580{
581 if(idx >= 0 && idx < cue.m_cues.size())
582 return cue.m_cues[idx].name;
583 else
584 return {};
585}
586
587static std::string_view name_from_args(const ossia::cues& cue) noexcept
588{
589 return name_from_args(cue, cue.current_index());
590}
591
592}
The value class.
Definition value.hpp:173
ossia::val_type underlying_type(const complex_type &t)
Definition complex_type.cpp:40
@ CLIP
The bounds are ignored.
@ GET
The value can be retrieved and changed.
@ SET
The value can be retrieved.