Mercurial > hg > pub > prymula > com
comparison DPF-Prymula-audioplugins/dpf/distrho/src/DistrhoUILV2.cpp @ 3:84e66ea83026
DPF-Prymula-audioplugins-0.231015-2
author | prymula <prymula76@outlook.com> |
---|---|
date | Mon, 16 Oct 2023 21:53:34 +0200 (15 months ago) |
parents | |
children |
comparison
equal
deleted
inserted
replaced
2:cf2cb71d31dd | 3:84e66ea83026 |
---|---|
1 /* | |
2 * DISTRHO Plugin Framework (DPF) | |
3 * Copyright (C) 2012-2021 Filipe Coelho <falktx@falktx.com> | |
4 * | |
5 * Permission to use, copy, modify, and/or distribute this software for any purpose with | |
6 * or without fee is hereby granted, provided that the above copyright notice and this | |
7 * permission notice appear in all copies. | |
8 * | |
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD | |
10 * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN | |
11 * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL | |
12 * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER | |
13 * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN | |
14 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |
15 */ | |
16 | |
17 #include "DistrhoUIInternal.hpp" | |
18 | |
19 #include "../extra/String.hpp" | |
20 | |
21 #include "lv2/atom.h" | |
22 #include "lv2/atom-util.h" | |
23 #include "lv2/data-access.h" | |
24 #include "lv2/instance-access.h" | |
25 #include "lv2/midi.h" | |
26 #include "lv2/options.h" | |
27 #include "lv2/parameters.h" | |
28 #include "lv2/patch.h" | |
29 #include "lv2/ui.h" | |
30 #include "lv2/urid.h" | |
31 #include "lv2/lv2_kxstudio_properties.h" | |
32 #include "lv2/lv2_programs.h" | |
33 | |
34 #ifndef DISTRHO_PLUGIN_LV2_STATE_PREFIX | |
35 # define DISTRHO_PLUGIN_LV2_STATE_PREFIX "urn:distrho:" | |
36 #endif | |
37 | |
38 START_NAMESPACE_DISTRHO | |
39 | |
40 typedef struct _LV2_Atom_MidiEvent { | |
41 LV2_Atom atom; /**< Atom header. */ | |
42 uint8_t data[3]; /**< MIDI data (body). */ | |
43 } LV2_Atom_MidiEvent; | |
44 | |
45 #if ! DISTRHO_PLUGIN_WANT_STATE | |
46 static constexpr const setStateFunc setStateCallback = nullptr; | |
47 #endif | |
48 #if ! DISTRHO_PLUGIN_WANT_MIDI_INPUT | |
49 static constexpr const sendNoteFunc sendNoteCallback = nullptr; | |
50 #endif | |
51 | |
52 // ----------------------------------------------------------------------- | |
53 | |
54 template <class LV2F> | |
55 static const LV2F* getLv2Feature(const LV2_Feature* const* features, const char* const uri) | |
56 { | |
57 for (int i=0; features[i] != nullptr; ++i) | |
58 { | |
59 if (std::strcmp(features[i]->URI, uri) == 0) | |
60 return (const LV2F*)features[i]->data; | |
61 } | |
62 | |
63 return nullptr; | |
64 } | |
65 | |
66 class UiLv2 | |
67 { | |
68 public: | |
69 UiLv2(const char* const bundlePath, | |
70 const intptr_t winId, | |
71 const LV2_Options_Option* options, | |
72 const LV2_URID_Map* const uridMap, | |
73 const LV2_Feature* const* const features, | |
74 const LV2UI_Controller controller, | |
75 const LV2UI_Write_Function writeFunc, | |
76 LV2UI_Widget* const widget, | |
77 void* const dspPtr, | |
78 const float sampleRate, | |
79 const float scaleFactor, | |
80 const uint32_t bgColor, | |
81 const uint32_t fgColor, | |
82 const char* const appClassName) | |
83 : fUridMap(uridMap), | |
84 fUridUnmap(getLv2Feature<LV2_URID_Unmap>(features, LV2_URID__unmap)), | |
85 fUiPortMap(getLv2Feature<LV2UI_Port_Map>(features, LV2_UI__portMap)), | |
86 fUiRequestValue(getLv2Feature<LV2UI_Request_Value>(features, LV2_UI__requestValue)), | |
87 fUiTouch(getLv2Feature<LV2UI_Touch>(features, LV2_UI__touch)), | |
88 fController(controller), | |
89 fWriteFunction(writeFunc), | |
90 fURIDs(uridMap), | |
91 fBypassParameterIndex(fUiPortMap != nullptr ? fUiPortMap->port_index(fUiPortMap->handle, "lv2_enabled") | |
92 : LV2UI_INVALID_PORT_INDEX), | |
93 fWinIdWasNull(winId == 0), | |
94 fUI(this, winId, sampleRate, | |
95 editParameterCallback, | |
96 setParameterCallback, | |
97 setStateCallback, | |
98 sendNoteCallback, | |
99 nullptr, // resize is very messy, hosts can do it without extensions | |
100 fileRequestCallback, | |
101 bundlePath, dspPtr, scaleFactor, bgColor, fgColor, appClassName) | |
102 { | |
103 if (widget != nullptr) | |
104 *widget = (LV2UI_Widget)fUI.getNativeWindowHandle(); | |
105 | |
106 #if DISTRHO_PLUGIN_WANT_STATE | |
107 // tell the DSP we're ready to receive msgs | |
108 setState("__dpf_ui_data__", ""); | |
109 #endif | |
110 | |
111 if (winId != 0) | |
112 return; | |
113 | |
114 // if winId == 0 then options must not be null | |
115 DISTRHO_SAFE_ASSERT_RETURN(options != nullptr,); | |
116 | |
117 #ifndef __EMSCRIPTEN__ | |
118 const LV2_URID uridWindowTitle = uridMap->map(uridMap->handle, LV2_UI__windowTitle); | |
119 const LV2_URID uridTransientWinId = uridMap->map(uridMap->handle, LV2_KXSTUDIO_PROPERTIES__TransientWindowId); | |
120 | |
121 const char* windowTitle = nullptr; | |
122 | |
123 for (int i=0; options[i].key != 0; ++i) | |
124 { | |
125 if (options[i].key == uridTransientWinId) | |
126 { | |
127 if (options[i].type == fURIDs.atomLong) | |
128 { | |
129 if (const int64_t transientWinId = *(const int64_t*)options[i].value) | |
130 fUI.setWindowTransientWinId(static_cast<intptr_t>(transientWinId)); | |
131 } | |
132 else | |
133 d_stderr("Host provides transientWinId but has wrong value type"); | |
134 } | |
135 else if (options[i].key == uridWindowTitle) | |
136 { | |
137 if (options[i].type == fURIDs.atomString) | |
138 { | |
139 windowTitle = (const char*)options[i].value; | |
140 } | |
141 else | |
142 d_stderr("Host provides windowTitle but has wrong value type"); | |
143 } | |
144 } | |
145 | |
146 if (windowTitle == nullptr) | |
147 windowTitle = DISTRHO_PLUGIN_NAME; | |
148 | |
149 fUI.setWindowTitle(windowTitle); | |
150 #endif | |
151 } | |
152 | |
153 // ------------------------------------------------------------------- | |
154 | |
155 void lv2ui_port_event(const uint32_t rindex, const uint32_t bufferSize, const uint32_t format, const void* const buffer) | |
156 { | |
157 if (format == 0) | |
158 { | |
159 const uint32_t parameterOffset = fUI.getParameterOffset(); | |
160 | |
161 if (rindex < parameterOffset) | |
162 return; | |
163 | |
164 DISTRHO_SAFE_ASSERT_RETURN(bufferSize == sizeof(float),) | |
165 | |
166 float value = *(const float*)buffer; | |
167 | |
168 if (rindex == fBypassParameterIndex) | |
169 value = 1.0f - value; | |
170 | |
171 fUI.parameterChanged(rindex-parameterOffset, value); | |
172 } | |
173 #if DISTRHO_PLUGIN_WANT_STATE | |
174 else if (format == fURIDs.atomEventTransfer) | |
175 { | |
176 const LV2_Atom* const atom = (const LV2_Atom*)buffer; | |
177 | |
178 if (atom->type == fURIDs.dpfKeyValue) | |
179 { | |
180 const char* const key = (const char*)LV2_ATOM_BODY_CONST(atom); | |
181 const char* const value = key+(std::strlen(key)+1); | |
182 | |
183 fUI.stateChanged(key, value); | |
184 } | |
185 else if (atom->type == fURIDs.atomObject && fUridUnmap != nullptr) | |
186 { | |
187 const LV2_Atom_Object* const obj = (const LV2_Atom_Object*)atom; | |
188 | |
189 const LV2_Atom* property = nullptr; | |
190 const LV2_Atom* atomvalue = nullptr; | |
191 lv2_atom_object_get(obj, fURIDs.patchProperty, &property, fURIDs.patchValue, &atomvalue, 0); | |
192 | |
193 DISTRHO_SAFE_ASSERT_RETURN(property != nullptr,); | |
194 DISTRHO_SAFE_ASSERT_RETURN(atomvalue != nullptr,); | |
195 | |
196 DISTRHO_SAFE_ASSERT_RETURN(property->type == fURIDs.atomURID,); | |
197 DISTRHO_SAFE_ASSERT_RETURN(atomvalue->type == fURIDs.atomPath || atomvalue->type == fURIDs.atomString,); | |
198 | |
199 if (property != nullptr && property->type == fURIDs.atomURID && | |
200 atomvalue != nullptr && (atomvalue->type == fURIDs.atomPath || atomvalue->type == fURIDs.atomString)) | |
201 { | |
202 const LV2_URID dpf_lv2_urid = ((const LV2_Atom_URID*)property)->body; | |
203 DISTRHO_SAFE_ASSERT_RETURN(dpf_lv2_urid != 0,); | |
204 | |
205 const char* const dpf_lv2_key = fUridUnmap->unmap(fUridUnmap->handle, dpf_lv2_urid); | |
206 DISTRHO_SAFE_ASSERT_RETURN(dpf_lv2_key != nullptr,); | |
207 | |
208 /*constexpr*/ const size_t reqLen = std::strlen(DISTRHO_PLUGIN_URI "#"); | |
209 DISTRHO_SAFE_ASSERT_RETURN(std::strlen(dpf_lv2_key) > reqLen,); | |
210 | |
211 const char* const key = dpf_lv2_key + reqLen; | |
212 const char* const value = (const char*)LV2_ATOM_BODY_CONST(atomvalue); | |
213 | |
214 fUI.stateChanged(key, value); | |
215 } | |
216 } | |
217 else if (atom->type == fURIDs.midiEvent) | |
218 { | |
219 // ignore | |
220 } | |
221 else | |
222 { | |
223 d_stdout("DPF :: received atom not handled :: %s", | |
224 fUridUnmap != nullptr ? fUridUnmap->unmap(fUridUnmap->handle, atom->type) : "(null)"); | |
225 } | |
226 } | |
227 #endif | |
228 } | |
229 | |
230 // ------------------------------------------------------------------- | |
231 | |
232 int lv2ui_idle() | |
233 { | |
234 if (fWinIdWasNull) | |
235 return (fUI.plugin_idle() && fUI.isVisible()) ? 0 : 1; | |
236 | |
237 return fUI.plugin_idle() ? 0 : 1; | |
238 } | |
239 | |
240 int lv2ui_show() | |
241 { | |
242 return fUI.setWindowVisible(true) ? 0 : 1; | |
243 } | |
244 | |
245 int lv2ui_hide() | |
246 { | |
247 return fUI.setWindowVisible(false) ? 0 : 1; | |
248 } | |
249 | |
250 // ------------------------------------------------------------------- | |
251 | |
252 uint32_t lv2_get_options(LV2_Options_Option* const /*options*/) | |
253 { | |
254 // currently unused | |
255 return LV2_OPTIONS_ERR_UNKNOWN; | |
256 } | |
257 | |
258 uint32_t lv2_set_options(const LV2_Options_Option* const options) | |
259 { | |
260 for (int i=0; options[i].key != 0; ++i) | |
261 { | |
262 if (options[i].key == fURIDs.paramSampleRate) | |
263 { | |
264 if (options[i].type == fURIDs.atomFloat) | |
265 { | |
266 const float sampleRate = *(const float*)options[i].value; | |
267 fUI.setSampleRate(sampleRate, true); | |
268 continue; | |
269 } | |
270 else | |
271 { | |
272 d_stderr("Host changed UI sample-rate but with wrong value type"); | |
273 continue; | |
274 } | |
275 } | |
276 } | |
277 | |
278 return LV2_OPTIONS_SUCCESS; | |
279 } | |
280 | |
281 // ------------------------------------------------------------------- | |
282 | |
283 #if DISTRHO_PLUGIN_WANT_PROGRAMS | |
284 void lv2ui_select_program(const uint32_t bank, const uint32_t program) | |
285 { | |
286 const uint32_t realProgram = bank * 128 + program; | |
287 | |
288 fUI.programLoaded(realProgram); | |
289 } | |
290 #endif | |
291 | |
292 // ------------------------------------------------------------------- | |
293 | |
294 private: | |
295 // LV2 features | |
296 const LV2_URID_Map* const fUridMap; | |
297 const LV2_URID_Unmap* const fUridUnmap; | |
298 const LV2UI_Port_Map* const fUiPortMap; | |
299 const LV2UI_Request_Value* const fUiRequestValue; | |
300 const LV2UI_Touch* const fUiTouch; | |
301 | |
302 // LV2 UI stuff | |
303 const LV2UI_Controller fController; | |
304 const LV2UI_Write_Function fWriteFunction; | |
305 | |
306 // LV2 URIDs | |
307 const struct URIDs { | |
308 const LV2_URID_Map* _uridMap; | |
309 const LV2_URID dpfKeyValue; | |
310 const LV2_URID atomEventTransfer; | |
311 const LV2_URID atomFloat; | |
312 const LV2_URID atomLong; | |
313 const LV2_URID atomObject; | |
314 const LV2_URID atomPath; | |
315 const LV2_URID atomString; | |
316 const LV2_URID atomURID; | |
317 const LV2_URID midiEvent; | |
318 const LV2_URID paramSampleRate; | |
319 const LV2_URID patchProperty; | |
320 const LV2_URID patchSet; | |
321 const LV2_URID patchValue; | |
322 | |
323 URIDs(const LV2_URID_Map* const uridMap) | |
324 : _uridMap(uridMap), | |
325 dpfKeyValue(map(DISTRHO_PLUGIN_LV2_STATE_PREFIX "KeyValueState")), | |
326 atomEventTransfer(map(LV2_ATOM__eventTransfer)), | |
327 atomFloat(map(LV2_ATOM__Float)), | |
328 atomLong(map(LV2_ATOM__Long)), | |
329 atomObject(map(LV2_ATOM__Object)), | |
330 atomPath(map(LV2_ATOM__Path)), | |
331 atomString(map(LV2_ATOM__String)), | |
332 atomURID(map(LV2_ATOM__URID)), | |
333 midiEvent(map(LV2_MIDI__MidiEvent)), | |
334 paramSampleRate(map(LV2_PARAMETERS__sampleRate)), | |
335 patchProperty(map(LV2_PATCH__property)), | |
336 patchSet(map(LV2_PATCH__Set)), | |
337 patchValue(map(LV2_PATCH__value)) {} | |
338 | |
339 inline LV2_URID map(const char* const uri) const | |
340 { | |
341 return _uridMap->map(_uridMap->handle, uri); | |
342 } | |
343 } fURIDs; | |
344 | |
345 // index of bypass parameter, if present | |
346 const uint32_t fBypassParameterIndex; | |
347 | |
348 // using ui:showInterface if true | |
349 const bool fWinIdWasNull; | |
350 | |
351 // Plugin UI (after LV2 stuff so the UI can call into us during its constructor) | |
352 UIExporter fUI; | |
353 | |
354 // ---------------------------------------------------------------------------------------------------------------- | |
355 // DPF callbacks | |
356 | |
357 void editParameterValue(const uint32_t rindex, const bool started) | |
358 { | |
359 if (fUiTouch != nullptr && fUiTouch->touch != nullptr) | |
360 fUiTouch->touch(fUiTouch->handle, rindex, started); | |
361 } | |
362 | |
363 static void editParameterCallback(void* const ptr, const uint32_t rindex, const bool started) | |
364 { | |
365 static_cast<UiLv2*>(ptr)->editParameterValue(rindex, started); | |
366 } | |
367 | |
368 void setParameterValue(const uint32_t rindex, float value) | |
369 { | |
370 DISTRHO_SAFE_ASSERT_RETURN(fWriteFunction != nullptr,); | |
371 | |
372 if (rindex == fBypassParameterIndex) | |
373 value = 1.0f - value; | |
374 | |
375 fWriteFunction(fController, rindex, sizeof(float), 0, &value); | |
376 } | |
377 | |
378 static void setParameterCallback(void* const ptr, const uint32_t rindex, const float value) | |
379 { | |
380 static_cast<UiLv2*>(ptr)->setParameterValue(rindex, value); | |
381 } | |
382 | |
383 #if DISTRHO_PLUGIN_WANT_STATE | |
384 void setState(const char* const key, const char* const value) | |
385 { | |
386 DISTRHO_SAFE_ASSERT_RETURN(fWriteFunction != nullptr,); | |
387 | |
388 const uint32_t eventInPortIndex = DISTRHO_PLUGIN_NUM_INPUTS + DISTRHO_PLUGIN_NUM_OUTPUTS; | |
389 | |
390 // join key and value | |
391 String tmpStr; | |
392 tmpStr += key; | |
393 tmpStr += "\xff"; | |
394 tmpStr += value; | |
395 | |
396 tmpStr[std::strlen(key)] = '\0'; | |
397 | |
398 // set msg size (key + separator + value + null terminator) | |
399 const uint32_t msgSize = static_cast<uint32_t>(tmpStr.length()) + 1U; | |
400 | |
401 // reserve atom space | |
402 const uint32_t atomSize = sizeof(LV2_Atom) + msgSize; | |
403 char* const atomBuf = (char*)malloc(atomSize); | |
404 DISTRHO_SAFE_ASSERT_RETURN(atomBuf != nullptr,); | |
405 | |
406 std::memset(atomBuf, 0, atomSize); | |
407 | |
408 // set atom info | |
409 LV2_Atom* const atom = (LV2_Atom*)atomBuf; | |
410 atom->size = msgSize; | |
411 atom->type = fURIDs.dpfKeyValue; | |
412 | |
413 // set atom data | |
414 std::memcpy(atomBuf + sizeof(LV2_Atom), tmpStr.buffer(), msgSize); | |
415 | |
416 // send to DSP side | |
417 fWriteFunction(fController, eventInPortIndex, atomSize, fURIDs.atomEventTransfer, atom); | |
418 | |
419 // free atom space | |
420 free(atomBuf); | |
421 } | |
422 | |
423 static void setStateCallback(void* const ptr, const char* const key, const char* const value) | |
424 { | |
425 static_cast<UiLv2*>(ptr)->setState(key, value); | |
426 } | |
427 #endif | |
428 | |
429 #if DISTRHO_PLUGIN_WANT_MIDI_INPUT | |
430 void sendNote(const uint8_t channel, const uint8_t note, const uint8_t velocity) | |
431 { | |
432 DISTRHO_SAFE_ASSERT_RETURN(fWriteFunction != nullptr,); | |
433 | |
434 if (channel > 0xF) | |
435 return; | |
436 | |
437 const uint32_t eventInPortIndex = DISTRHO_PLUGIN_NUM_INPUTS + DISTRHO_PLUGIN_NUM_OUTPUTS; | |
438 | |
439 LV2_Atom_MidiEvent atomMidiEvent; | |
440 atomMidiEvent.atom.size = 3; | |
441 atomMidiEvent.atom.type = fURIDs.midiEvent; | |
442 | |
443 atomMidiEvent.data[0] = channel + (velocity != 0 ? 0x90 : 0x80); | |
444 atomMidiEvent.data[1] = note; | |
445 atomMidiEvent.data[2] = velocity; | |
446 | |
447 // send to DSP side | |
448 fWriteFunction(fController, eventInPortIndex, lv2_atom_total_size(&atomMidiEvent.atom), | |
449 fURIDs.atomEventTransfer, &atomMidiEvent); | |
450 } | |
451 | |
452 static void sendNoteCallback(void* const ptr, const uint8_t channel, const uint8_t note, const uint8_t velocity) | |
453 { | |
454 static_cast<UiLv2*>(ptr)->sendNote(channel, note, velocity); | |
455 } | |
456 #endif | |
457 | |
458 bool fileRequest(const char* const key) | |
459 { | |
460 d_stdout("UI file request %s %p", key, fUiRequestValue); | |
461 | |
462 if (fUiRequestValue == nullptr) | |
463 return false; | |
464 | |
465 String dpf_lv2_key(DISTRHO_PLUGIN_URI "#"); | |
466 dpf_lv2_key += key; | |
467 | |
468 const int r = fUiRequestValue->request(fUiRequestValue->handle, | |
469 fUridMap->map(fUridMap->handle, dpf_lv2_key.buffer()), | |
470 fURIDs.atomPath, | |
471 nullptr); | |
472 | |
473 d_stdout("UI file request %s %p => %s %i", key, fUiRequestValue, dpf_lv2_key.buffer(), r); | |
474 return r == LV2UI_REQUEST_VALUE_SUCCESS; | |
475 } | |
476 | |
477 static bool fileRequestCallback(void* ptr, const char* key) | |
478 { | |
479 return static_cast<UiLv2*>(ptr)->fileRequest(key); | |
480 } | |
481 }; | |
482 | |
483 // ----------------------------------------------------------------------- | |
484 | |
485 static LV2UI_Handle lv2ui_instantiate(const LV2UI_Descriptor*, | |
486 const char* const uri, | |
487 const char* const bundlePath, | |
488 const LV2UI_Write_Function writeFunction, | |
489 const LV2UI_Controller controller, | |
490 LV2UI_Widget* const widget, | |
491 const LV2_Feature* const* const features) | |
492 { | |
493 if (uri == nullptr || std::strcmp(uri, DISTRHO_PLUGIN_URI) != 0) | |
494 { | |
495 d_stderr("Invalid plugin URI"); | |
496 return nullptr; | |
497 } | |
498 | |
499 const LV2_Options_Option* options = nullptr; | |
500 const LV2_URID_Map* uridMap = nullptr; | |
501 void* parentId = nullptr; | |
502 void* instance = nullptr; | |
503 | |
504 #if DISTRHO_PLUGIN_WANT_DIRECT_ACCESS | |
505 struct LV2_DirectAccess_Interface { | |
506 void* (*get_instance_pointer)(LV2_Handle handle); | |
507 }; | |
508 const LV2_Extension_Data_Feature* extData = nullptr; | |
509 #endif | |
510 | |
511 for (int i=0; features[i] != nullptr; ++i) | |
512 { | |
513 /**/ if (std::strcmp(features[i]->URI, LV2_OPTIONS__options) == 0) | |
514 options = (const LV2_Options_Option*)features[i]->data; | |
515 else if (std::strcmp(features[i]->URI, LV2_URID__map) == 0) | |
516 uridMap = (const LV2_URID_Map*)features[i]->data; | |
517 else if (std::strcmp(features[i]->URI, LV2_UI__parent) == 0) | |
518 parentId = features[i]->data; | |
519 #if DISTRHO_PLUGIN_WANT_DIRECT_ACCESS | |
520 else if (std::strcmp(features[i]->URI, LV2_DATA_ACCESS_URI) == 0) | |
521 extData = (const LV2_Extension_Data_Feature*)features[i]->data; | |
522 else if (std::strcmp(features[i]->URI, LV2_INSTANCE_ACCESS_URI) == 0) | |
523 instance = features[i]->data; | |
524 #endif | |
525 } | |
526 | |
527 if (options == nullptr && parentId == nullptr) | |
528 { | |
529 d_stderr("Options feature missing (needed for show-interface), cannot continue!"); | |
530 return nullptr; | |
531 } | |
532 | |
533 if (uridMap == nullptr) | |
534 { | |
535 d_stderr("URID Map feature missing, cannot continue!"); | |
536 return nullptr; | |
537 } | |
538 | |
539 if (parentId == nullptr) | |
540 { | |
541 d_stdout("Parent Window Id missing, host should be using ui:showInterface..."); | |
542 } | |
543 | |
544 #if DISTRHO_PLUGIN_WANT_DIRECT_ACCESS | |
545 if (extData == nullptr || instance == nullptr) | |
546 { | |
547 d_stderr("Data or instance access missing, cannot continue!"); | |
548 return nullptr; | |
549 } | |
550 | |
551 if (const LV2_DirectAccess_Interface* const directAccess = (const LV2_DirectAccess_Interface*)extData->data_access(DISTRHO_PLUGIN_LV2_STATE_PREFIX "direct-access")) | |
552 instance = directAccess->get_instance_pointer(instance); | |
553 else | |
554 instance = nullptr; | |
555 | |
556 if (instance == nullptr) | |
557 { | |
558 d_stderr("Failed to get direct access, cannot continue!"); | |
559 return nullptr; | |
560 } | |
561 #endif | |
562 | |
563 const intptr_t winId = (intptr_t)parentId; | |
564 float sampleRate = 0.0f; | |
565 float scaleFactor = 0.0f; | |
566 uint32_t bgColor = 0; | |
567 uint32_t fgColor = 0xffffffff; | |
568 const char* appClassName = nullptr; | |
569 | |
570 if (options != nullptr) | |
571 { | |
572 const LV2_URID uridAtomInt = uridMap->map(uridMap->handle, LV2_ATOM__Int); | |
573 const LV2_URID uridAtomFloat = uridMap->map(uridMap->handle, LV2_ATOM__Float); | |
574 const LV2_URID uridAtomString = uridMap->map(uridMap->handle, LV2_ATOM__String); | |
575 const LV2_URID uridSampleRate = uridMap->map(uridMap->handle, LV2_PARAMETERS__sampleRate); | |
576 const LV2_URID uridBgColor = uridMap->map(uridMap->handle, LV2_UI__backgroundColor); | |
577 const LV2_URID uridFgColor = uridMap->map(uridMap->handle, LV2_UI__foregroundColor); | |
578 #ifndef DISTRHO_OS_MAC | |
579 const LV2_URID uridScaleFactor = uridMap->map(uridMap->handle, LV2_UI__scaleFactor); | |
580 #endif | |
581 const LV2_URID uridClassName = uridMap->map(uridMap->handle, "urn:distrho:className"); | |
582 | |
583 for (int i=0; options[i].key != 0; ++i) | |
584 { | |
585 /**/ if (options[i].key == uridSampleRate) | |
586 { | |
587 if (options[i].type == uridAtomFloat) | |
588 sampleRate = *(const float*)options[i].value; | |
589 else | |
590 d_stderr("Host provides UI sample-rate but has wrong value type"); | |
591 } | |
592 else if (options[i].key == uridBgColor) | |
593 { | |
594 if (options[i].type == uridAtomInt) | |
595 bgColor = (uint32_t)*(const int32_t*)options[i].value; | |
596 else | |
597 d_stderr("Host provides UI background color but has wrong value type"); | |
598 } | |
599 else if (options[i].key == uridFgColor) | |
600 { | |
601 if (options[i].type == uridAtomInt) | |
602 fgColor = (uint32_t)*(const int32_t*)options[i].value; | |
603 else | |
604 d_stderr("Host provides UI foreground color but has wrong value type"); | |
605 } | |
606 #ifndef DISTRHO_OS_MAC | |
607 else if (options[i].key == uridScaleFactor) | |
608 { | |
609 if (options[i].type == uridAtomFloat) | |
610 scaleFactor = *(const float*)options[i].value; | |
611 else | |
612 d_stderr("Host provides UI scale factor but has wrong value type"); | |
613 } | |
614 #endif | |
615 else if (options[i].key == uridClassName) | |
616 { | |
617 if (options[i].type == uridAtomString) | |
618 appClassName = (const char*)options[i].value; | |
619 else | |
620 d_stderr("Host provides UI scale factor but has wrong value type"); | |
621 } | |
622 } | |
623 } | |
624 | |
625 if (sampleRate < 1.0) | |
626 { | |
627 d_stdout("WARNING: this host does not send sample-rate information for LV2 UIs, using 44100 as fallback (this could be wrong)"); | |
628 sampleRate = 44100.0; | |
629 } | |
630 | |
631 return new UiLv2(bundlePath, winId, options, uridMap, features, | |
632 controller, writeFunction, widget, instance, | |
633 sampleRate, scaleFactor, bgColor, fgColor, appClassName); | |
634 } | |
635 | |
636 #define uiPtr ((UiLv2*)ui) | |
637 | |
638 static void lv2ui_cleanup(LV2UI_Handle ui) | |
639 { | |
640 delete uiPtr; | |
641 } | |
642 | |
643 static void lv2ui_port_event(LV2UI_Handle ui, uint32_t portIndex, uint32_t bufferSize, uint32_t format, const void* buffer) | |
644 { | |
645 uiPtr->lv2ui_port_event(portIndex, bufferSize, format, buffer); | |
646 } | |
647 | |
648 // ----------------------------------------------------------------------- | |
649 | |
650 static int lv2ui_idle(LV2UI_Handle ui) | |
651 { | |
652 return uiPtr->lv2ui_idle(); | |
653 } | |
654 | |
655 static int lv2ui_show(LV2UI_Handle ui) | |
656 { | |
657 return uiPtr->lv2ui_show(); | |
658 } | |
659 | |
660 static int lv2ui_hide(LV2UI_Handle ui) | |
661 { | |
662 return uiPtr->lv2ui_hide(); | |
663 } | |
664 | |
665 // ----------------------------------------------------------------------- | |
666 | |
667 static uint32_t lv2_get_options(LV2UI_Handle ui, LV2_Options_Option* options) | |
668 { | |
669 return uiPtr->lv2_get_options(options); | |
670 } | |
671 | |
672 static uint32_t lv2_set_options(LV2UI_Handle ui, const LV2_Options_Option* options) | |
673 { | |
674 return uiPtr->lv2_set_options(options); | |
675 } | |
676 | |
677 // ----------------------------------------------------------------------- | |
678 | |
679 #if DISTRHO_PLUGIN_WANT_PROGRAMS | |
680 static void lv2ui_select_program(LV2UI_Handle ui, uint32_t bank, uint32_t program) | |
681 { | |
682 uiPtr->lv2ui_select_program(bank, program); | |
683 } | |
684 #endif | |
685 | |
686 // ----------------------------------------------------------------------- | |
687 | |
688 static const void* lv2ui_extension_data(const char* uri) | |
689 { | |
690 static const LV2_Options_Interface options = { lv2_get_options, lv2_set_options }; | |
691 static const LV2UI_Idle_Interface uiIdle = { lv2ui_idle }; | |
692 static const LV2UI_Show_Interface uiShow = { lv2ui_show, lv2ui_hide }; | |
693 | |
694 if (std::strcmp(uri, LV2_OPTIONS__interface) == 0) | |
695 return &options; | |
696 if (std::strcmp(uri, LV2_UI__idleInterface) == 0) | |
697 return &uiIdle; | |
698 if (std::strcmp(uri, LV2_UI__showInterface) == 0) | |
699 return &uiShow; | |
700 | |
701 #if DISTRHO_PLUGIN_WANT_PROGRAMS | |
702 static const LV2_Programs_UI_Interface uiPrograms = { lv2ui_select_program }; | |
703 | |
704 if (std::strcmp(uri, LV2_PROGRAMS__UIInterface) == 0) | |
705 return &uiPrograms; | |
706 #endif | |
707 | |
708 return nullptr; | |
709 } | |
710 | |
711 #undef instancePtr | |
712 | |
713 // ----------------------------------------------------------------------- | |
714 | |
715 static const LV2UI_Descriptor sLv2UiDescriptor = { | |
716 DISTRHO_UI_URI, | |
717 lv2ui_instantiate, | |
718 lv2ui_cleanup, | |
719 lv2ui_port_event, | |
720 lv2ui_extension_data | |
721 }; | |
722 | |
723 // ----------------------------------------------------------------------- | |
724 | |
725 END_NAMESPACE_DISTRHO | |
726 | |
727 DISTRHO_PLUGIN_EXPORT | |
728 const LV2UI_Descriptor* lv2ui_descriptor(uint32_t index) | |
729 { | |
730 USE_NAMESPACE_DISTRHO | |
731 return (index == 0) ? &sLv2UiDescriptor : nullptr; | |
732 } | |
733 | |
734 #if defined(__MOD_DEVICES__) && defined(__EMSCRIPTEN__) | |
735 #include <emscripten/html5.h> | |
736 #include <string> | |
737 | |
738 typedef void (*_custom_param_set)(uint32_t port_index, float value); | |
739 typedef void (*_custom_patch_set)(const char* uri, const char* value); | |
740 | |
741 struct ModguiHandle { | |
742 LV2UI_Handle handle; | |
743 long loop_id; | |
744 _custom_param_set param_set; | |
745 _custom_patch_set patch_set; | |
746 }; | |
747 | |
748 enum URIs { | |
749 kUriNull, | |
750 kUriAtomEventTransfer, | |
751 kUriDpfKeyValue, | |
752 }; | |
753 | |
754 static std::vector<std::string> kURIs; | |
755 | |
756 static LV2_URID lv2_urid_map(LV2_URID_Map_Handle, const char* const uri) | |
757 { | |
758 for (size_t i=0, size=kURIs.size(); i<size; ++i) | |
759 { | |
760 if (kURIs[i] == uri) | |
761 return i; | |
762 } | |
763 | |
764 kURIs.push_back(uri); | |
765 return kURIs.size() - 1u; | |
766 } | |
767 | |
768 static const char* lv2_urid_unmap(LV2_URID_Map_Handle, const LV2_URID urid) | |
769 { | |
770 return kURIs[urid].c_str(); | |
771 } | |
772 | |
773 static void lv2ui_write_function(LV2UI_Controller controller, | |
774 uint32_t port_index, | |
775 uint32_t buffer_size, | |
776 uint32_t port_protocol, | |
777 const void* buffer) | |
778 { | |
779 DISTRHO_SAFE_ASSERT_RETURN(buffer_size >= 1,); | |
780 | |
781 // d_stdout("lv2ui_write_function %p %u %u %u %p", controller, port_index, buffer_size, port_protocol, buffer); | |
782 ModguiHandle* const mhandle = static_cast<ModguiHandle*>(controller); | |
783 | |
784 switch (port_protocol) | |
785 { | |
786 case kUriNull: | |
787 mhandle->param_set(port_index, *static_cast<const float*>(buffer)); | |
788 break; | |
789 case kUriAtomEventTransfer: | |
790 if (const LV2_Atom* const atom = static_cast<const LV2_Atom*>(buffer)) | |
791 { | |
792 // d_stdout("lv2ui_write_function %u %u:%s", atom->size, atom->type, kURIs[atom->type].c_str()); | |
793 | |
794 // if (kURIs[atom->type] == "urn:distrho:KeyValueState") | |
795 { | |
796 const char* const key = (const char*)(atom + 1); | |
797 const char* const value = key + (std::strlen(key) + 1U); | |
798 // d_stdout("lv2ui_write_function %s %s", key, value); | |
799 | |
800 String urikey; | |
801 urikey = DISTRHO_PLUGIN_URI "#"; | |
802 urikey += key; | |
803 | |
804 mhandle->patch_set(urikey, value); | |
805 } | |
806 } | |
807 break; | |
808 } | |
809 } | |
810 | |
811 static void app_idle(void* const handle) | |
812 { | |
813 static_cast<UiLv2*>(handle)->lv2ui_idle(); | |
814 } | |
815 | |
816 DISTRHO_PLUGIN_EXPORT | |
817 LV2UI_Handle modgui_init(const char* const className, _custom_param_set param_set, _custom_patch_set patch_set) | |
818 { | |
819 d_stdout("init \"%s\"", className); | |
820 DISTRHO_SAFE_ASSERT_RETURN(className != nullptr, nullptr); | |
821 | |
822 static LV2_URID_Map uridMap = { nullptr, lv2_urid_map }; | |
823 static LV2_URID_Unmap uridUnmap = { nullptr, lv2_urid_unmap }; | |
824 | |
825 // known first URIDs, matching URIs | |
826 if (kURIs.empty()) | |
827 { | |
828 kURIs.push_back(""); | |
829 kURIs.push_back("http://lv2plug.in/ns/ext/atom#eventTransfer"); | |
830 kURIs.push_back(DISTRHO_PLUGIN_LV2_STATE_PREFIX "KeyValueState"); | |
831 } | |
832 | |
833 static float sampleRateValue = 48000.f; | |
834 static LV2_Options_Option options[3] = { | |
835 { | |
836 LV2_OPTIONS_INSTANCE, | |
837 0, | |
838 uridMap.map(uridMap.handle, LV2_PARAMETERS__sampleRate), | |
839 sizeof(float), | |
840 uridMap.map(uridMap.handle, LV2_ATOM__Float), | |
841 &sampleRateValue | |
842 }, | |
843 { | |
844 LV2_OPTIONS_INSTANCE, | |
845 0, | |
846 uridMap.map(uridMap.handle, "urn:distrho:className"), | |
847 std::strlen(className) + 1, | |
848 uridMap.map(uridMap.handle, LV2_ATOM__String), | |
849 className | |
850 }, | |
851 {} | |
852 }; | |
853 | |
854 static const LV2_Feature optionsFt = { LV2_OPTIONS__options, static_cast<void*>(options) }; | |
855 static const LV2_Feature uridMapFt = { LV2_URID__map, static_cast<void*>(&uridMap) }; | |
856 static const LV2_Feature uridUnmapFt = { LV2_URID__unmap, static_cast<void*>(&uridUnmap) }; | |
857 | |
858 static const LV2_Feature* features[] = { | |
859 &optionsFt, | |
860 &uridMapFt, | |
861 &uridUnmapFt, | |
862 nullptr | |
863 }; | |
864 | |
865 ModguiHandle* const mhandle = new ModguiHandle; | |
866 mhandle->handle = nullptr; | |
867 mhandle->loop_id = 0; | |
868 mhandle->param_set = param_set; | |
869 mhandle->patch_set = patch_set; | |
870 | |
871 LV2UI_Widget widget; | |
872 const LV2UI_Handle handle = lv2ui_instantiate(&sLv2UiDescriptor, | |
873 DISTRHO_PLUGIN_URI, | |
874 "", // bundlePath | |
875 lv2ui_write_function, | |
876 mhandle, | |
877 &widget, | |
878 features); | |
879 mhandle->handle = handle; | |
880 | |
881 static_cast<UiLv2*>(handle)->lv2ui_show(); | |
882 mhandle->loop_id = emscripten_set_interval(app_idle, 1000.0/60, handle); | |
883 | |
884 return mhandle; | |
885 } | |
886 | |
887 DISTRHO_PLUGIN_EXPORT | |
888 void modgui_param_set(const LV2UI_Handle handle, const uint32_t index, const float value) | |
889 { | |
890 lv2ui_port_event(static_cast<ModguiHandle*>(handle)->handle, index, sizeof(float), kUriNull, &value); | |
891 } | |
892 | |
893 DISTRHO_PLUGIN_EXPORT | |
894 void modgui_patch_set(const LV2UI_Handle handle, const char* const uri, const char* const value) | |
895 { | |
896 static const constexpr uint32_t URI_PREFIX_LEN = sizeof(DISTRHO_PLUGIN_URI); | |
897 DISTRHO_SAFE_ASSERT_RETURN(std::strncmp(uri, DISTRHO_PLUGIN_URI "#", URI_PREFIX_LEN) == 0,); | |
898 | |
899 const uint32_t keySize = std::strlen(uri + URI_PREFIX_LEN) + 1; | |
900 const uint32_t valueSize = std::strlen(value) + 1; | |
901 const uint32_t atomSize = sizeof(LV2_Atom) + keySize + valueSize; | |
902 | |
903 LV2_Atom* const atom = static_cast<LV2_Atom*>(std::malloc(atomSize)); | |
904 atom->size = atomSize; | |
905 atom->type = kUriDpfKeyValue; | |
906 | |
907 std::memcpy(static_cast<uint8_t*>(static_cast<void*>(atom + 1)), uri + URI_PREFIX_LEN, keySize); | |
908 std::memcpy(static_cast<uint8_t*>(static_cast<void*>(atom + 1)) + keySize, value, valueSize); | |
909 | |
910 lv2ui_port_event(static_cast<ModguiHandle*>(handle)->handle, | |
911 DISTRHO_PLUGIN_NUM_INPUTS + DISTRHO_PLUGIN_NUM_OUTPUTS, // events input port | |
912 atomSize, kUriAtomEventTransfer, atom); | |
913 | |
914 std::free(atom); | |
915 } | |
916 | |
917 DISTRHO_PLUGIN_EXPORT | |
918 void modgui_cleanup(const LV2UI_Handle handle) | |
919 { | |
920 d_stdout("cleanup"); | |
921 ModguiHandle* const mhandle = static_cast<ModguiHandle*>(handle); | |
922 if (mhandle->loop_id != 0) | |
923 emscripten_clear_interval(mhandle->loop_id); | |
924 lv2ui_cleanup(mhandle->handle); | |
925 delete mhandle; | |
926 } | |
927 #endif | |
928 | |
929 // ----------------------------------------------------------------------- |