/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ /* Rosegarden A MIDI and audio sequencer and musical notation editor. Copyright 2000-2011 the Rosegarden development team. Other copyrights also apply to some parts of this work. Please see the AUTHORS file and individual file headers for details. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. See the file COPYING included with this distribution for more information. */ #include "AudioPluginOSCGUIManager.h" #include "sound/Midi.h" #include "misc/Debug.h" #include "misc/Strings.h" #include "AudioPluginOSCGUI.h" #include "base/AudioPluginInstance.h" #include "base/Exception.h" #include "base/Instrument.h" #include "base/MidiProgram.h" #include "base/RealTime.h" #include "base/Studio.h" #include "gui/application/RosegardenMainWindow.h" #include "OSCMessage.h" #include "sound/MappedEvent.h" #include "sound/PluginIdentifier.h" #include "StudioControl.h" #include "TimerCallbackAssistant.h" #include #include namespace Rosegarden { static void osc_error(int num, const char *msg, const char *path) { std::cerr << "Rosegarden: ERROR: liblo server error " << num << " in path " << path << ": " << msg << std::endl; } static int osc_message_handler(const char *path, const char *types, lo_arg **argv, int argc, lo_message, void *user_data) { AudioPluginOSCGUIManager *manager = (AudioPluginOSCGUIManager *)user_data; InstrumentId instrument; int position; QString method; if (!manager->parseOSCPath(path, instrument, position, method)) { return 1; } OSCMessage *message = new OSCMessage(); message->setTarget(instrument); message->setTargetData(position); message->setMethod(qstrtostr(method)); int arg = 0; while (types && arg < argc && types[arg]) { message->addArg(types[arg], argv[arg]); ++arg; } manager->postMessage(message); return 0; } AudioPluginOSCGUIManager::AudioPluginOSCGUIManager(RosegardenMainWindow *mainWindow) : m_mainWindow(mainWindow), m_studio(0), m_haveOSCThread(false), m_oscBuffer(1023), m_dispatchTimer(0) {} AudioPluginOSCGUIManager::~AudioPluginOSCGUIManager() { delete m_dispatchTimer; for (TargetGUIMap::iterator i = m_guis.begin(); i != m_guis.end(); ++i) { for (IntGUIMap::iterator j = i->second.begin(); j != i->second.end(); ++j) { delete j->second; } } m_guis.clear(); if (m_haveOSCThread) lo_server_thread_stop(m_serverThread); } void AudioPluginOSCGUIManager::checkOSCThread() { if (m_haveOSCThread) return ; m_serverThread = lo_server_thread_new(NULL, osc_error); lo_server_thread_add_method(m_serverThread, NULL, NULL, osc_message_handler, this); lo_server_thread_start(m_serverThread); RG_DEBUG << "AudioPluginOSCGUIManager: Base OSC URL is " << lo_server_thread_get_url(m_serverThread) << endl; m_dispatchTimer = new TimerCallbackAssistant(20, timerCallback, this); m_haveOSCThread = true; } bool AudioPluginOSCGUIManager::hasGUI(InstrumentId instrument, int position) { PluginContainer *container = 0; container = m_studio->getContainerById(instrument); if (!container) return false; AudioPluginInstance *pluginInstance = container->getPlugin(position); if (!pluginInstance) return false; try { QString filePath = AudioPluginOSCGUI::getGUIFilePath (strtoqstr(pluginInstance->getIdentifier())); return ( !filePath.isEmpty() ); } catch (Exception e) { // that's OK return false; } } void AudioPluginOSCGUIManager::startGUI(InstrumentId instrument, int position) { RG_DEBUG << "AudioPluginOSCGUIManager::startGUI: " << instrument << "," << position << endl; checkOSCThread(); if (m_guis.find(instrument) != m_guis.end() && m_guis[instrument].find(position) != m_guis[instrument].end()) { RG_DEBUG << "stopping GUI first" << endl; stopGUI(instrument, position); } // check the label PluginContainer *container = 0; container = m_studio->getContainerById(instrument); if (!container) { RG_DEBUG << "AudioPluginOSCGUIManager::startGUI: no such instrument or buss as " << instrument << endl; return; } AudioPluginInstance *pluginInstance = container->getPlugin(position); if (!pluginInstance) { RG_DEBUG << "AudioPluginOSCGUIManager::startGUI: no plugin at position " << position << " for instrument " << instrument << endl; return ; } try { AudioPluginOSCGUI *gui = new AudioPluginOSCGUI(pluginInstance, getOSCUrl(instrument, position, strtoqstr(pluginInstance->getIdentifier())), getFriendlyName(instrument, position, strtoqstr(pluginInstance->getIdentifier()))); m_guis[instrument][position] = gui; } catch (Exception e) { RG_DEBUG << "AudioPluginOSCGUIManager::startGUI: failed to start GUI: " << e.getMessage() << endl; } } void AudioPluginOSCGUIManager::showGUI(InstrumentId instrument, int position) { RG_DEBUG << "AudioPluginOSCGUIManager::showGUI: " << instrument << "," << position << endl; if (m_guis.find(instrument) != m_guis.end() && m_guis[instrument].find(position) != m_guis[instrument].end()) { m_guis[instrument][position]->show(); } else { startGUI(instrument, position); } } void AudioPluginOSCGUIManager::stopGUI(InstrumentId instrument, int position) { if (m_guis.find(instrument) != m_guis.end() && m_guis[instrument].find(position) != m_guis[instrument].end()) { delete m_guis[instrument][position]; m_guis[instrument].erase(position); if (m_guis[instrument].empty()) m_guis.erase(instrument); } } void AudioPluginOSCGUIManager::stopAllGUIs() { while (!m_guis.empty()) { while (!m_guis.begin()->second.empty()) { delete (m_guis.begin()->second.begin()->second); m_guis.begin()->second.erase(m_guis.begin()->second.begin()); } m_guis.erase(m_guis.begin()); } } void AudioPluginOSCGUIManager::postMessage(OSCMessage *message) { RG_DEBUG << "AudioPluginOSCGUIManager::postMessage" << endl; m_oscBuffer.write(&message, 1); } void AudioPluginOSCGUIManager::updateProgram(InstrumentId instrument, int position) { RG_DEBUG << "AudioPluginOSCGUIManager::updateProgram(" << instrument << "," << position << ")" << endl; if (m_guis.find(instrument) == m_guis.end() || m_guis[instrument].find(position) == m_guis[instrument].end()) return ; PluginContainer *container = 0; container = m_studio->getContainerById(instrument); if (!container) return; AudioPluginInstance *pluginInstance = container->getPlugin(position); if (!pluginInstance) return; unsigned long rv = StudioControl::getPluginProgram (pluginInstance->getMappedId(), strtoqstr(pluginInstance->getProgram())); int bank = rv >> 16; int program = rv - (bank << 16); RG_DEBUG << "AudioPluginOSCGUIManager::updateProgram(" << instrument << "," << position << "): rv " << rv << ", bank " << bank << ", program " << program << endl; m_guis[instrument][position]->sendProgram(bank, program); } void AudioPluginOSCGUIManager::updatePort(InstrumentId instrument, int position, int port) { RG_DEBUG << "AudioPluginOSCGUIManager::updatePort(" << instrument << "," << position << "," << port << ")" << endl; if (m_guis.find(instrument) == m_guis.end() || m_guis[instrument].find(position) == m_guis[instrument].end()) return ; PluginContainer *container = 0; container = m_studio->getContainerById(instrument); if (!container) return; AudioPluginInstance *pluginInstance = container->getPlugin(position); if (!pluginInstance) return ; PluginPortInstance *porti = pluginInstance->getPort(port); if (!porti) return ; RG_DEBUG << "AudioPluginOSCGUIManager::updatePort(" << instrument << "," << position << "," << port << "): value " << porti->value << endl; m_guis[instrument][position]->sendPortValue(port, porti->value); } void AudioPluginOSCGUIManager::updateConfiguration(InstrumentId instrument, int position, QString key) { RG_DEBUG << "AudioPluginOSCGUIManager::updateConfiguration(" << instrument << "," << position << "," << key << ")" << endl; if (m_guis.find(instrument) == m_guis.end() || m_guis[instrument].find(position) == m_guis[instrument].end()) return ; PluginContainer *container = m_studio->getContainerById(instrument); if (!container) return; AudioPluginInstance *pluginInstance = container->getPlugin(position); if (!pluginInstance) return; QString value = strtoqstr(pluginInstance->getConfigurationValue(qstrtostr(key))); RG_DEBUG << "AudioPluginOSCGUIManager::updatePort(" << instrument << "," << position << "," << key << "): value " << value << endl; m_guis[instrument][position]->sendConfiguration(key, value); } QString AudioPluginOSCGUIManager::getOSCUrl(InstrumentId instrument, int position, QString identifier) { // OSC URL will be of the form // osc.udp://localhost:54343/plugin/dssi///