1#include <ossia/detail/json.hpp>
3#include <ossia-max/src/ocue.hpp>
4#include <ossia-max/src/ossia-max.hpp>
5#include <ossia-max/src/utils.hpp>
7#include <boost/algorithm/string.hpp>
9#include <rapidjson/prettywriter.h>
13 bool is_absolute_path(std::string_view v)
15 if (v.starts_with(
'/') || v.starts_with(
'~'))
17 if (v.find(
':') != std::string_view::npos)
21 static std::string fix_url(std::string_view u)
23 if(u.starts_with(
"~"))
25 const char* home = getenv(
"HOME");
27 std::string res = home;
31 return std::string(u);
33 static std::string to_absolute_path(t_patcher* patcher, std::string_view url)
35 std::string full_path;
36 if (is_absolute_path(url))
38 full_path = fix_url(url);
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)
49 if (patcher_path.starts_with(
"Macintosh HD:"))
50 start = strlen(
"Macintosh HD:");
51 full_path = patcher_path.substr(start, end_it - start);
57 prompt_open_filename(std::string_view dialogtitle, std::string_view default_filename)
59 char buffer[MAX_PATH_CHARS] = {};
60 strncpy(buffer, default_filename.data(), default_filename.size());
62 open_promptset(dialogtitle.data());
66 t_fourcc filetype =
'TEXT';
67 if(open_dialog(buffer, &path, &outtype, &filetype, 1) == 0)
70 path_opensysfile(buffer, path, &hdl, e_max_openfile_permissions::PATH_READ_PERM);
72 char full_path[MAX_PATH_CHARS] = {};
73 path_topathname(path, buffer, full_path);
75 char native_path[MAX_PATH_CHARS];
76 path_nameconform(full_path, native_path, PATH_STYLE_NATIVE, PATH_TYPE_BOOT);
84prompt_save_filename(std::string_view dialogtitle, std::string_view default_filename)
86 char buffer[MAX_PATH_CHARS] = {};
87 strncpy(buffer, default_filename.data(), default_filename.size());
89 saveas_promptset(dialogtitle.data());
93 t_fourcc filetype =
'TEXT';
94 if(saveasdialog_extended(buffer, &path, &outtype, &filetype, 1) == 0)
97 path_createsysfile(buffer, path, filetype, &hdl);
99 char full_path[MAX_PATH_CHARS] = {};
100 path_topathname(path, buffer, full_path);
102 char native_path[MAX_PATH_CHARS];
103 path_nameconform(full_path, native_path, PATH_STYLE_NATIVE, PATH_TYPE_BOOT);
111ossia::selection_filters parse_selection_filter(
int argc, t_atom* argv)
113 ossia::selection_filters filt;
123 processing_visibility
127 for(
int i = 0; i < argc; i++)
129 if(argv[i].a_type == A_SYM)
132 = boost::algorithm::to_lower_copy(std::string(argv[i].a_w.w_sym->s_name));
133 if(symb.starts_with(
"@"))
135 if(symb ==
"@valuetype")
137 state = processing_val_type;
140 if(symb ==
"@access")
142 state = processing_access;
145 if(symb ==
"@bounding")
147 state = processing_bounding;
152 state = processing_tags;
155 if(symb ==
"@visibility")
157 state = processing_visibility;
164 case processing_names:
165 filt.selection.push_back(std::string(symb));
167 case processing_val_type: {
168 if(
auto param = ossia::default_parameter_for_type(symb))
174 case processing_access: {
180 filt.access.push_back(ossia::access_mode::BI);
183 case processing_bounding: {
185 filt.bounding.push_back(ossia::bounding_mode::FREE);
198 case processing_tags: {
199 filt.tags.push_back(std::string(symb));
202 case processing_visibility:
204 = symb ==
"visible" || symb ==
"true" || symb ==
"yes" || symb ==
"ok"
205 ? ossia::selection_filters::visible
206 : ossia::selection_filters::invisible;
216using writer_t = rapidjson::PrettyWriter<rapidjson::StringBuffer>;
217struct write_json_preset_value
220 void operator()(ossia::impulse)
const
223 writer.String(
"impulse");
225 void operator()(
int v)
const
228 writer.String(
"int");
232 void operator()(
float v)
const
239 void operator()(
bool v)
const
242 writer.String(
"bool");
247 void operator()(
const std::string& v)
const
250 writer.String(
"string");
255 template <std::
size_t N>
256 void operator()(
const std::array<float, N>& vec)
const
259 writer.String(
"vec" + std::to_string(N) +
"f");
262 for(std::size_t i = 0; i < N; i++)
264 writer.Double(vec[i]);
269 void operator()(
const std::vector<ossia::value>& vec)
const
272 writer.String(
"list");
277 writer.StartObject();
283 void operator()(
const ossia::value_map_type& vec)
const { }
284 void operator()()
const {
throw std::runtime_error(
"value_to_json_value: no type"); }
287static void cue_to_json(writer_t& doc,
const ossia::cue& cu)
295 for(
auto& [k, v] : cu.preset)
301 v.apply(write_json_preset_value{doc});
308static rapidjson::StringBuffer cues_to_string(
const ossia::cues& c)
311 rapidjson::StringBuffer buffer;
312 writer_t doc{buffer};
317 for(
auto& cue : c.m_cues)
319 cue_to_json(doc, cue);
340read_typed_ossia_value_from_json(
const rapidjson::Document::ConstObject& value);
342 const rapidjson::Document::ConstObject& value, std::string_view type)
344 if(type ==
"impulse")
346 return ossia::impulse{};
348 auto it = value.FindMember(
"value");
349 if(it == value.MemberEnd())
354 if(it->value.IsInt())
356 return it->value.GetInt();
359 else if(type ==
"float")
361 if(it->value.IsFloat())
363 return it->value.GetFloat();
366 else if(type ==
"string")
368 if(it->value.IsString())
370 return std::string(it->value.GetString(), it->value.GetStringLength());
373 else if(type ==
"bool")
375 if(it->value.IsBool())
377 return it->value.GetBool();
380 else if(type ==
"vec4f")
382 if(it->value.IsArray())
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(),
392 else if(type ==
"vec3f")
394 if(it->value.IsArray())
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());
402 else if(type ==
"vec2f")
404 if(it->value.IsArray())
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());
411 else if(type ==
"list")
413 if(it->value.IsArray())
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)
420 if(sub_val.IsObject())
421 vec.push_back(read_typed_ossia_value_from_json(sub_val.GetObject()));
430read_typed_ossia_value_from_json(
const rapidjson::Document::ConstObject& value)
432 if(
auto it = value.FindMember(
"type"); it != value.MemberEnd() && it->value.IsString())
434 std::string_view type{it->value.GetString(), it->value.GetStringLength()};
435 return read_typed_ossia_value_from_json(value, type);
440static void read_cue_preset_from_json(
441 const rapidjson::Document::ConstArray& json, ossia::presets::preset& p)
443 for(
const rapidjson::Value& value : json)
445 if(!value.IsObject())
449 if(
auto it = value.FindMember(
"address");
450 it != value.MemberEnd() && it->value.IsString())
452 addr = std::string{it->value.GetString(), it->value.GetStringLength()};
458 ossia::value v = read_typed_ossia_value_from_json(value.GetObject());
460 p.emplace_back(std::move(addr), std::move(v));
464static void read_cues_from_json(
465 const rapidjson::Document::Array& json, std::vector<ossia::cue>& cues)
467 for(
const rapidjson::Value& value : json)
469 if(!value.IsObject())
472 if(
auto it = value.FindMember(
"name");
473 it != value.MemberEnd() && it->value.IsString())
475 c.name = std::string{it->value.GetString(), it->value.GetStringLength()};
478 if(
auto it = value.FindMember(
"preset");
479 it != value.MemberEnd() && it->value.IsArray())
481 read_cue_preset_from_json(it->value.GetArray(), c.preset);
483 cues.push_back(std::move(c));
487static void invoke_mem_fun(
int argc, t_atom* argv,
auto f)
489#define if_possible(F) \
490 if constexpr(requires { F; }) \
498 else if(argc == 1 && argv[0].a_type == A_SYM)
500 if_possible(f(argv[0].a_w.w_sym->s_name));
502 else if(argc == 1 && argv[0].a_type == A_LONG)
504 if_possible(f(argv[0].a_w.w_long));
506 else if(argc == 1 && argv[0].a_type == A_FLOAT)
508 if_possible(f((
int)argv[0].a_w.w_float));
510 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_LONG)
512 if_possible(f(argv[0].a_w.w_long, argv[1].a_w.w_long));
514 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_FLOAT)
516 if_possible(f(argv[0].a_w.w_long, (
int)argv[1].a_w.w_float));
518 else if(argc == 2 && argv[0].a_type == A_FLOAT && argv[1].a_type == A_LONG)
520 if_possible(f((
int)argv[0].a_w.w_float, argv[1].a_w.w_long));
522 else if(argc == 2 && argv[0].a_type == A_FLOAT && argv[1].a_type == A_FLOAT)
524 if_possible(f((
int)argv[0].a_w.w_float, (
int)argv[1].a_w.w_float));
526 else if(argc == 2 && argv[0].a_type == A_LONG && argv[1].a_type == A_SYM)
528 if_possible(f(argv[0].a_w.w_long, argv[1].a_w.w_sym->s_name));
530 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_LONG)
532 if_possible(f(argv[0].a_w.w_sym->s_name, argv[1].a_w.w_long));
534 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_FLOAT)
536 if_possible(f(argv[0].a_w.w_sym->s_name, (
int)argv[1].a_w.w_float));
538 else if(argc == 2 && argv[0].a_type == A_SYM && argv[1].a_type == A_SYM)
540 if_possible(f(argv[0].a_w.w_sym->s_name, argv[1].a_w.w_sym->s_name));
545static void pack_to_atom(t_atom& t,
long long& i)
549static void pack_to_atom(t_atom& t,
long& i)
553static void pack_to_atom(t_atom& t,
int& i)
557static void pack_to_atom(t_atom& t,
float& i)
559 atom_setfloat(&t, i);
561static void pack_to_atom(t_atom& t,
char*& i)
563 atom_setsym(&t, gensym(i));
565static void pack_to_atom(t_atom& t, std::string_view i)
568 atom_setsym(&t, gensym(i.data()));
570 atom_setsym(&t, gensym(
"<invalid>"));
573static std::string_view
574name_from_args(
const ossia::cues& cue, std::string_view args)
noexcept
579static std::string_view name_from_args(
const ossia::cues& cue,
int idx)
noexcept
581 if(idx >= 0 && idx < cue.m_cues.size())
582 return cue.m_cues[idx].name;
587static std::string_view name_from_args(
const ossia::cues& cue)
noexcept
589 return name_from_args(cue, cue.current_index());
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.