| 
							- #!/usr/bin/env python3
 - # -*- coding: utf-8 -*-
 - 
 - # Carla OSC controller
 - # Copyright (C) 2011-2013 Filipe Coelho <falktx@falktx.com>
 - #
 - # 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 any later version.
 - #
 - # This program is distributed in the hope that it will be useful,
 - # but WITHOUT ANY WARRANTY; without even the implied warranty of
 - # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 - # GNU General Public License for more details.
 - #
 - # For a full copy of the GNU General Public License see the GPL.txt file
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # Imports (Global)
 - 
 - from PyQt4.QtCore import QLibrary
 - from PyQt4.QtGui import QApplication, QInputDialog, QMainWindow
 - from liblo import make_method, Address, ServerError, ServerThread
 - from liblo import send as lo_send
 - from liblo import TCP as LO_TCP
 - from liblo import UDP as LO_UDP
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # Imports (Custom)
 - 
 - import ui_carla_control
 - from carla_shared import *
 - 
 - global lo_target, lo_targetName
 - lo_target     = None
 - lo_targetName = ""
 - 
 - Carla.isControl = True
 - Carla.isLocal   = False
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # Helper class
 - 
 - class ControlPluginInfo(object):
 -     __slots__ = [
 -         'pluginInfo',
 -         'pluginRealName',
 -         'audioCountInfo',
 -         'midiCountInfo',
 -         'parameterCountInfo',
 -         'parameterInfoS',
 -         'parameterDataS',
 -         'parameterRangeS',
 -         'parameterValueS',
 -         'programCount',
 -         'programCurrent',
 -         'programNameS',
 -         'midiProgramCount',
 -         'midiProgramCurrent',
 -         'midiProgramDataS',
 -         'peaks'
 -     ]
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # Python Object class compatible to 'Host' on the Carla Backend code
 - 
 - class Host(object):
 -     def __init__(self):
 -         object.__init__(self)
 - 
 -         self.fPluginsInfo = []
 - 
 -     def _add(self, pluginId):
 -         if len(self.fPluginsInfo) != pluginId:
 -             return
 - 
 -         info = ControlPluginInfo()
 -         info.pluginInfo = CarlaPluginInfo
 -         info.pluginRealName = None
 -         info.audioCountInfo = CarlaPortCountInfo
 -         info.midiCountInfo  = CarlaPortCountInfo
 -         info.parameterCountInfo = CarlaPortCountInfo
 -         info.parameterInfoS  = []
 -         info.parameterDataS  = []
 -         info.parameterRangeS = []
 -         info.parameterValueS = []
 -         info.programCount = 0
 -         info.programCurrent = -1
 -         info.programNameS = []
 -         info.midiProgramCount = 0
 -         info.midiProgramCurrent = -1
 -         info.midiProgramDataS = []
 -         info.peaks = [0.0, 0.0, 0.0, 0.0]
 -         self.fPluginsInfo.append(info)
 - 
 -     def _set_pluginInfo(self, index, info):
 -         self.fPluginsInfo[index].pluginInfo = info
 - 
 -     def _set_pluginRealName(self, index, realName):
 -         self.fPluginsInfo[index].pluginRealName = realName
 - 
 -     def _set_audioCountInfo(self, index, info):
 -         self.fPluginsInfo[index].audioCountInfo = info
 - 
 -     def _set_midiCountInfo(self, index, info):
 -         self.fPluginsInfo[index].midiCountInfo = info
 - 
 -     def _set_parameterCountInfo(self, index, info):
 -         self.fPluginsInfo[index].parameterCountInfo = info
 - 
 -         # clear
 -         self.fPluginsInfo[index].parameterInfoS  = []
 -         self.fPluginsInfo[index].parameterDataS  = []
 -         self.fPluginsInfo[index].parameterRangeS = []
 -         self.fPluginsInfo[index].parameterValueS = []
 - 
 -         # add placeholders
 -         for x in range(info['total']):
 -             self.fPluginsInfo[index].parameterInfoS.append(CarlaParameterInfo)
 -             self.fPluginsInfo[index].parameterDataS.append(ParameterData)
 -             self.fPluginsInfo[index].parameterRangeS.append(ParameterRanges)
 -             self.fPluginsInfo[index].parameterValueS.append(0.0)
 - 
 -     def _set_programCount(self, index, count):
 -         self.fPluginsInfo[index].programCount = count
 - 
 -         # clear
 -         self.fPluginsInfo[index].programNameS = []
 - 
 -         # add placeholders
 -         for x in range(count):
 -             self.fPluginsInfo[index].programNameS.append(None)
 - 
 -     def _set_midiProgramCount(self, index, count):
 -         self.fPluginsInfo[index].midiProgramCount = count
 - 
 -         # clear
 -         self.fPluginsInfo[index].midiProgramDataS = []
 - 
 -         # add placeholders
 -         for x in range(count):
 -             self.fPluginsInfo[index].midiProgramDataS.append(MidiProgramData)
 - 
 -     def _set_parameterInfoS(self, index, paramIndex, data):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterInfoS[paramIndex] = data
 - 
 -     def _set_parameterDataS(self, index, paramIndex, data):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterDataS[paramIndex] = data
 - 
 -     def _set_parameterRangeS(self, index, paramIndex, data):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterRangeS[paramIndex] = data
 - 
 -     def _set_parameterValueS(self, index, paramIndex, value):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterValueS[paramIndex] = value
 - 
 -     def _set_parameterDefaultValue(self, index, paramIndex, value):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterRangeS[paramIndex]['def'] = value
 - 
 -     def _set_parameterMidiCC(self, index, paramIndex, cc):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterDataS[paramIndex]['midiCC'] = cc
 - 
 -     def _set_parameterMidiChannel(self, index, paramIndex, channel):
 -         if paramIndex < self.fPluginsInfo[index].parameterCountInfo['total']:
 -             self.fPluginsInfo[index].parameterDataS[paramIndex]['midiChannel'] = channel
 - 
 -     def _set_currentProgram(self, index, pIndex):
 -         self.fPluginsInfo[index].programCurrent = pIndex
 - 
 -     def _set_currentMidiProgram(self, index, mpIndex):
 -         self.fPluginsInfo[index].midiProgramCurrent = mpIndex
 - 
 -     def _set_programNameS(self, index, pIndex, data):
 -         if pIndex < self.fPluginsInfo[index].programCount:
 -             self.fPluginsInfo[index].programNameS[pIndex] = data
 - 
 -     def _set_midiProgramDataS(self, index, mpIndex, data):
 -         if mpIndex < self.fPluginsInfo[index].midiProgramCount:
 -             self.fPluginsInfo[index].midiProgramDataS[mpIndex] = data
 - 
 -     def _set_peaks(self, index, in1, in2, out1, out2):
 -         self.fPluginsInfo[index].peaks = [in1, in2, out1, out2]
 - 
 -     # get_extended_license_text
 -     # get_supported_file_types
 -     # get_engine_driver_count
 -     # get_engine_driver_name
 -     # get_engine_driver_device_names
 -     # get_internal_plugin_count
 -     # get_internal_plugin_info
 -     # engine_init
 -     # engine_close
 -     # engine_idle
 -     # is_engine_running
 -     # set_engine_about_to_close
 -     # set_engine_callback
 -     # set_engine_option
 -     # load_filename
 -     # load_project
 -     # save_project
 -     # patchbay_connect
 -     # patchbay_disconnect
 -     # patchbay_refresh
 -     # transport_play
 -     # transport_pause
 -     # transport_relocate
 -     # get_current_transport_frame
 -     # get_transport_info
 -     # add_plugin
 -     # remove_plugin
 -     # remove_all_plugins
 -     # rename_plugin
 -     # clone_plugin
 -     # replace_plugin
 -     # switch_plugins
 -     # load_plugin_state
 -     # save_plugin_state
 - 
 -     def get_plugin_info(self, pluginId):
 -         return self.fPluginsInfo[pluginId].pluginInfo
 - 
 -     def get_audio_port_count_info(self, pluginId):
 -         return self.fPluginsInfo[pluginId].audioCountInfo
 - 
 -     def get_midi_port_count_info(self, pluginId):
 -         return self.fPluginsInfo[pluginId].midiCountInfo
 - 
 -     def get_parameter_count_info(self, pluginId):
 -         return self.fPluginsInfo[pluginId].parameterCountInfo
 - 
 -     def get_parameter_info(self, pluginId, parameterId):
 -         return self.fPluginsInfo[pluginId].parameterInfoS[parameterId]
 - 
 -     def get_parameter_scalepoint_info(self, pluginId, parameterId, scalepoint_id):
 -         return CarlaScalePointInfo
 - 
 -     def get_parameter_data(self, pluginId, parameterId):
 -         return self.fPluginsInfo[pluginId].parameterDataS[parameterId]
 - 
 -     def get_parameter_ranges(self, pluginId, parameterId):
 -         return self.fPluginsInfo[pluginId].parameterRangeS[parameterId]
 - 
 -     def get_midi_program_data(self, pluginId, midiProgramId):
 -         return self.fPluginsInfo[pluginId].midiProgramDataS[midiProgramId]
 - 
 -     def get_custom_data(self, pluginId, custom_data_id):
 -         return CustomData
 - 
 -     def get_chunk_data(self, pluginId):
 -         return None
 - 
 -     def get_parameter_count(self, pluginId):
 -         return self.fPluginsInfo[pluginId].parameterCountInfo['total']
 - 
 -     def get_program_count(self, pluginId):
 -         return self.fPluginsInfo[pluginId].programCount
 - 
 -     def get_midi_program_count(self, pluginId):
 -         return self.fPluginsInfo[pluginId].midiProgramCount
 - 
 -     def get_custom_data_count(self, pluginId):
 -         return 0
 - 
 -     def get_parameter_text(self, pluginId, parameterId):
 -         return None
 - 
 -     def get_program_name(self, pluginId, programId):
 -         return self.fPluginsInfo[pluginId].programNameS[programId]
 - 
 -     def get_midi_program_name(self, pluginId, midiProgramId):
 -         return self.fPluginsInfo[pluginId].midiProgramDataS[midiProgramId]['label']
 - 
 -     def get_real_plugin_name(self, pluginId):
 -         return self.fPluginsInfo[pluginId].pluginRealName
 - 
 -     def get_current_program_index(self, pluginId):
 -         return self.fPluginsInfo[pluginId].programCurrent
 - 
 -     def get_current_midi_program_index(self, pluginId):
 -         return self.fPluginsInfo[pluginId].midiProgramCurrent
 - 
 -     def get_default_parameter_value(self, pluginId, parameterId):
 -         return self.fPluginsInfo[pluginId].parameterRangeS[parameterId]['def']
 - 
 -     def get_current_parameter_value(self, pluginId, parameterId):
 -         return self.fPluginsInfo[pluginId].parameterValueS[parameterId]
 - 
 -     def get_input_peak_value(self, pluginId, portId):
 -         return self.fPluginsInfo[pluginId].peaks[portId-1]
 - 
 -     def get_output_peak_value(self, pluginId, portId):
 -         return self.fPluginsInfo[pluginId].peaks[portId+1]
 - 
 -     def set_option(self, pluginId, option, yesNo):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_option" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, option, yesNo)
 - 
 -     def set_active(self, pluginId, onoff):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_active" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, 1 if onoff else 0)
 - 
 -     def set_drywet(self, pluginId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_drywet" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, value)
 - 
 -     def set_volume(self, pluginId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_volume" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, value)
 - 
 -     def set_balance_left(self, pluginId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_balance_left" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, value)
 - 
 -     def set_balance_right(self, pluginId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_balance_right" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, value)
 - 
 -     def set_panning(self, pluginId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_panning" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, value)
 - 
 -     def set_ctrl_channel(self, pluginId, channel):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_ctrl_channel" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, channel)
 - 
 -     def set_parameter_value(self, pluginId, parameterId, value):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_parameter_value" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, parameterId, value)
 - 
 -     def set_parameter_midi_cc(self, pluginId, parameterId, midi_cc):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_parameter_midi_cc" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, parameterId, midi_cc)
 - 
 -     def set_parameter_midi_channel(self, pluginId, parameterId, channel):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_parameter_midi_channel" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, parameterId, channel)
 - 
 -     def set_program(self, pluginId, programId):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_program" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, programId)
 - 
 -     def set_midi_program(self, pluginId, midiProgramId):
 -         global to_target, lo_targetName
 -         lo_path = "/%s/%i/set_midi_program" % (lo_targetName, pluginId)
 -         lo_send(lo_target, lo_path, midiProgramId)
 - 
 -     # set_custom_data
 -     # set_chunk_data
 -     # prepare_for_save
 - 
 -     def send_midi_note(self, pluginId, channel, note, velocity):
 -         global to_target, lo_targetName
 -         if velocity:
 -             lo_path = "/%s/%i/note_on" % (lo_targetName, pluginId)
 -             lo_send(lo_target, lo_path, channel, note, velocity)
 -         else:
 -             lo_path = "/%s/%i/note_off" % (lo_targetName, pluginId)
 -             lo_send(lo_target, lo_path, channel, note)
 - 
 -     # show_gui
 -     # get_buffer_size
 -     # get_sample_rate
 -     # get_last_error
 -     # get_host_osc_url
 -     # nsm_announce
 -     # nsm_reply_open
 -     # nsm_reply_save
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # OSC Control server
 - 
 - class ControlServer(ServerThread):
 -     def __init__(self, parent, mode):
 -         ServerThread.__init__(self, 8087, mode)
 - 
 -         self.fParent = parent
 - 
 -     def getFullURL(self):
 -         return "%scarla-control" % self.get_url()
 - 
 -     @make_method('/carla-control/add_plugin_start', 'is')
 -     def add_plugin_start_callback(self, path, args):
 -         pluginId, pluginName = args
 -         self.fParent.emit(SIGNAL("AddPluginStart(int, QString)"), pluginId, pluginName)
 - 
 -     @make_method('/carla-control/add_plugin_end', 'i')
 -     def add_plugin_end_callback(self, path, args):
 -         pluginId, = args
 -         self.fParent.emit(SIGNAL("AddPluginEnd(int)"), pluginId)
 - 
 -     @make_method('/carla-control/remove_plugin', 'i')
 -     def remove_plugin_callback(self, path, args):
 -         pluginId, = args
 -         self.fParent.emit(SIGNAL("RemovePlugin(int)"), pluginId)
 - 
 -     @make_method('/carla-control/set_plugin_data', 'iiiissssh')
 -     def set_plugin_data_callback(self, path, args):
 -         pluginId, ptype, category, hints, realName, label, maker, copyright_, uniqueId = args
 -         self.fParent.emit(SIGNAL("SetPluginData(int, int, int, int, QString, QString, QString, QString, int)"), pluginId, ptype, category, hints, realName, label, maker, copyright_, uniqueId)
 - 
 -     @make_method('/carla-control/set_plugin_ports', 'iiiiiiii')
 -     def set_plugin_ports_callback(self, path, args):
 -         pluginId, audioIns, audioOuts, midiIns, midiOuts, cIns, cOuts, cTotals = args
 -         self.fParent.emit(SIGNAL("SetPluginPorts(int, int, int, int, int, int, int, int)"), pluginId, audioIns, audioOuts, midiIns, midiOuts, cIns, cOuts, cTotals)
 - 
 -     @make_method('/carla-control/set_parameter_data', 'iiiissf')
 -     def set_parameter_data_callback(self, path, args):
 -         pluginId, index, type_, hints, name, label, current = args
 -         self.fParent.emit(SIGNAL("SetParameterData(int, int, int, int, QString, QString, double)"), pluginId, index, type_, hints, name, label, current)
 - 
 -     @make_method('/carla-control/set_parameter_ranges', 'iiffffff')
 -     def set_parameter_ranges_callback(self, path, args):
 -         pluginId, index, min_, max_, def_, step, stepSmall, stepLarge = args
 -         self.fParent.emit(SIGNAL("SetParameterRanges(int, int, double, double, double, double, double, double)"), pluginId, index, min_, max_, def_, step, stepSmall, stepLarge)
 - 
 -     @make_method('/carla-control/set_parameter_midi_cc', 'iii')
 -     def set_parameter_midi_cc_callback(self, path, args):
 -         pluginId, index, cc = args
 -         self.fParent.emit(SIGNAL("SetParameterMidiCC(int, int, int)"), pluginId, index, cc)
 - 
 -     @make_method('/carla-control/set_parameter_midi_channel', 'iii')
 -     def set_parameter_midi_channel_callback(self, path, args):
 -         pluginId, index, channel = args
 -         self.fParent.emit(SIGNAL("SetParameterMidiChannel(int, int, int)"), pluginId, index, channel)
 - 
 -     @make_method('/carla-control/set_parameter_value', 'iif')
 -     def set_parameter_value_callback(self, path, args):
 -         pluginId, index, value = args
 -         self.fParent.emit(SIGNAL("SetParameterValue(int, int, double)"), pluginId, index, value)
 - 
 -     @make_method('/carla-control/set_default_value', 'iif')
 -     def set_default_value_callback(self, path, args):
 -         pluginId, index, value = args
 -         self.fParent.emit(SIGNAL("SetDefaultValue(int, int, double)"), pluginId, index, value)
 - 
 -     @make_method('/carla-control/set_program', 'ii')
 -     def set_program_callback(self, path, args):
 -         pluginId, index = args
 -         self.fParent.emit(SIGNAL("SetProgram(int, int)"), pluginId, index)
 - 
 -     @make_method('/carla-control/set_program_count', 'ii')
 -     def set_program_count_callback(self, path, args):
 -         pluginId, count = args
 -         self.fParent.emit(SIGNAL("SetProgramCount(int, int)"), pluginId, count)
 - 
 -     @make_method('/carla-control/set_program_name', 'iis')
 -     def set_program_name_callback(self, path, args):
 -         pluginId, index, name = args
 -         self.fParent.emit(SIGNAL("SetProgramName(int, int, QString)"), pluginId, index, name)
 - 
 -     @make_method('/carla-control/set_midi_program', 'ii')
 -     def set_midi_program_callback(self, path, args):
 -         pluginId, index = args
 -         self.fParent.emit(SIGNAL("SetMidiProgram(int, int)"), pluginId, index)
 - 
 -     @make_method('/carla-control/set_midi_program_count', 'ii')
 -     def set_midi_program_count_callback(self, path, args):
 -         pluginId, count = args
 -         self.fParent.emit(SIGNAL("SetMidiProgramCount(int, int)"), pluginId, count)
 - 
 -     @make_method('/carla-control/set_midi_program_data', 'iiiis')
 -     def set_midi_program_data_callback(self, path, args):
 -         pluginId, index, bank, program, name = args
 -         self.fParent.emit(SIGNAL("SetMidiProgramData(int, int, int, int, QString)"), pluginId, index, bank, program, name)
 - 
 -     @make_method('/carla-control/note_on', 'iiii')
 -     def note_on_callback(self, path, args):
 -         pluginId, channel, note, velo = args
 -         self.fParent.emit(SIGNAL("NoteOn(int, int, int, int)"), pluginId, channel, note, velo)
 - 
 -     @make_method('/carla-control/note_off', 'iii')
 -     def note_off_callback(self, path, args):
 -         pluginId, channel, note = args
 -         self.fParent.emit(SIGNAL("NoteOff(int, int, int)"), pluginId, channel, note)
 - 
 -     @make_method('/carla-control/set_peaks', 'iffff')
 -     def set_output_peak_value_callback(self, path, args):
 -         pluginId, in1, in2, out1, out2 = args
 -         self.fParent.emit(SIGNAL("SetPeaks(int, double, double, double, double)"), pluginId, in1, in2, out1, out2)
 - 
 -     @make_method('/carla-control/exit', '')
 -     def exit_callback(self, path, args):
 -         self.fParent.emit(SIGNAL("Exit()"))
 - 
 -     @make_method(None, None)
 -     def fallback(self, path, args):
 -         print("ControlServer::fallback(\"%s\") - unknown message, args =" % path, args)
 - 
 - # ------------------------------------------------------------------------------------------------------------
 - # Main Window
 - 
 - class CarlaControlW(QMainWindow):
 -     def __init__(self, oscAddr=None):
 -         QMainWindow.__init__(self, None)
 -         self.ui = ui_carla_control.Ui_CarlaControlW()
 -         self.ui.setupUi(self)
 - 
 -         if MACOS:
 -             self.setUnifiedTitleAndToolBarOnMac(True)
 - 
 -         # -------------------------------------------------------------
 -         # Internal stuff
 - 
 -         self.fProjectFilename = None
 -         self.fProjectLoading  = False
 - 
 -         self.fPluginCount = 0
 -         self.fPluginList  = []
 - 
 -         self.fIdleTimerFast = 0
 -         self.fIdleTimerSlow = 0
 - 
 -         self.fLastPluginName = ""
 - 
 -         self.lo_address = ""
 -         self.lo_server  = None
 - 
 -         # -------------------------------------------------------------
 -         # Load Settings
 - 
 -         self.loadSettings()
 - 
 -         # -------------------------------------------------------------
 -         # Set-up GUI stuff
 - 
 -         self.ui.act_file_refresh.setEnabled(False)
 -         #self.ui.act_plugin_remove_all.setEnabled(False)
 - 
 -         self.resize(self.width(), 0)
 - 
 -         # -------------------------------------------------------------
 -         # Connect actions to functions
 - 
 -         #self.connect(self.ui.act_file_connect, SIGNAL("triggered()"), SLOT("slot_fileConnect()"))
 -         #self.connect(self.ui.act_file_refresh, SIGNAL("triggered()"), SLOT("slot_fileRefresh()"))
 - 
 -         #self.connect(self.ui.act_plugin_add, SIGNAL("triggered()"), SLOT("slot_pluginAdd()"))
 -         #self.connect(self.ui.act_plugin_add2, SIGNAL("triggered()"), SLOT("slot_pluginAdd()"))
 -         #self.connect(self.ui.act_plugin_refresh, SIGNAL("triggered()"), SLOT("slot_pluginRefresh()"))
 -         #self.connect(self.ui.act_plugin_remove_all, SIGNAL("triggered()"), SLOT("slot_pluginRemoveAll()"))
 - 
 -         #self.connect(self.ui.act_settings_show_toolbar, SIGNAL("triggered(bool)"), SLOT("slot_toolbarShown()"))
 -         #self.connect(self.ui.act_settings_configure, SIGNAL("triggered()"), SLOT("slot_configureCarla()"))
 - 
 -         #self.connect(self.ui.act_help_about, SIGNAL("triggered()"), SLOT("slot_aboutCarlaControl()"))
 -         #self.connect(self.ui.act_help_about_qt, SIGNAL("triggered()"), app, SLOT("aboutQt()"))
 - 
 -         #self.connect(self, SIGNAL("SIGTERM()"), SLOT("slot_handleSIGTERM()"))
 - 
 -         #self.connect(self, SIGNAL("AddPluginStart(int, QString)"), SLOT("slot_handleAddPluginStart(int, QString)"))
 -         #self.connect(self, SIGNAL("AddPluginEnd(int)"), SLOT("slot_handleAddPluginEnd(int)"))
 -         #self.connect(self, SIGNAL("RemovePlugin(int)"), SLOT("slot_handleRemovePlugin(int)"))
 -         #self.connect(self, SIGNAL("SetPluginData(int, int, int, int, QString, QString, QString, QString, int)"), SLOT("slot_handleSetPluginData(int, int, int, int, QString, QString, QString, QString, int)"))
 -         #self.connect(self, SIGNAL("SetPluginPorts(int, int, int, int, int, int, int, int)"), SLOT("slot_handleSetPluginPorts(int, int, int, int, int, int, int, int)"))
 -         #self.connect(self, SIGNAL("SetParameterData(int, int, int, int, QString, QString, double)"), SLOT("slot_handleSetParameterData(int, int, int, int, QString, QString, double)"))
 -         #self.connect(self, SIGNAL("SetParameterRanges(int, int, double, double, double, double, double, double)"), SLOT("slot_handleSetParameterRanges(int, int, double, double, double, double, double, double)"))
 -         #self.connect(self, SIGNAL("SetParameterMidiCC(int, int, int)"), SLOT("slot_handleSetParameterMidiCC(int, int, int)"))
 -         #self.connect(self, SIGNAL("SetParameterMidiChannel(int, int, int)"), SLOT("slot_handleSetParameterMidiChannel(int, int, int)"))
 -         #self.connect(self, SIGNAL("SetParameterValue(int, int, double)"), SLOT("slot_handleSetParameterValue(int, int, double)"))
 -         #self.connect(self, SIGNAL("SetDefaultValue(int, int, double)"), SLOT("slot_handleSetDefaultValue(int, int, double)"))
 -         #self.connect(self, SIGNAL("SetProgram(int, int)"), SLOT("slot_handleSetProgram(int, int)"))
 -         #self.connect(self, SIGNAL("SetProgramCount(int, int)"), SLOT("slot_handleSetProgramCount(int, int)"))
 -         #self.connect(self, SIGNAL("SetProgramName(int, int, QString)"), SLOT("slot_handleSetProgramName(int, int, QString)"))
 -         #self.connect(self, SIGNAL("SetMidiProgram(int, int)"), SLOT("slot_handleSetMidiProgram(int, int)"))
 -         #self.connect(self, SIGNAL("SetMidiProgramCount(int, int)"), SLOT("slot_handleSetMidiProgramCount(int, int)"))
 -         #self.connect(self, SIGNAL("SetMidiProgramData(int, int, int, int, QString)"), SLOT("slot_handleSetMidiProgramData(int, int, int, int, QString)"))
 -         #self.connect(self, SIGNAL("NoteOn(int, int, int, int)"), SLOT("slot_handleNoteOn(int, int, int, int)"))
 -         #self.connect(self, SIGNAL("NoteOff(int, int, int)"), SLOT("slot_handleNoteOff(int, int, int)"))
 -         #self.connect(self, SIGNAL("SetPeaks(int, double, double, double, double)"), SLOT("slot_handleSetPeaks(int, double, double, double, double)"))
 -         #self.connect(self, SIGNAL("Exit()"), SLOT("slot_handleExit()"))
 - 
 -         if oscAddr:
 -             self.connectToAddr(oscAddr)
 - 
 -     def connectToAddr(self, addr):
 -         global lo_target, lo_targetName
 - 
 -         self.lo_address = addr
 -         lo_target       = Address(self.lo_address)
 -         lo_targetName   = self.lo_address.rsplit("/", 1)[-1]
 -         print("Connecting to \"%s\" as '%s'..." % (self.lo_address, lo_targetName))
 - 
 -         try:
 -             self.lo_server = ControlServer(self, LO_UDP if self.lo_address.startswith("osc.udp") else LO_TCP)
 -         except: # ServerError, err:
 -             print("Connecting error!")
 -             #print str(err)
 -             QMessageBox.critical(self, self.tr("Error"), self.tr("Failed to connect, operation failed."))
 -             return
 - 
 -         if self.lo_server:
 -             self.lo_server.start()
 -             self.ui.act_file_refresh.setEnabled(True)
 -             lo_send(lo_target, "/register", self.lo_server.getFullURL())
 - 
 -         self.fIdleTimerFast = self.startTimer(60)
 -         self.fIdleTimerSlow = self.startTimer(60*2)
 - 
 -     def removeAll(self):
 -         self.killTimer(self.fIdleTimerFast)
 -         self.killTimer(self.fIdleTimerSlow)
 -         self.fIdleTimerFast = 0
 -         self.fIdleTimerSlow = 0
 - 
 -         for i in range(self.fPluginCount):
 -             pwidget = self.fPluginList[i]
 - 
 -             if pwidget is None:
 -                 break
 - 
 -             pwidget.ui.edit_dialog.close()
 -             pwidget.close()
 -             pwidget.deleteLater()
 -             del pwidget
 - 
 -         self.fPluginCount = 0
 -         self.fPluginList  = []
 -         gCarla.host.fPluginsInfo = []
 - 
 -         self.fIdleTimerFast = self.startTimer(60)
 -         self.fIdleTimerSlow = self.startTimer(60*2)
 - 
 -     @pyqtSlot()
 -     def slot_handleSIGTERM(self):
 -         print("Got SIGTERM -> Closing now")
 -         self.close()
 - 
 -     @pyqtSlot()
 -     def slot_fileConnect(self):
 -         global lo_target, lo_targetName
 - 
 -         if lo_target and self.lo_server:
 -             urlText = self.lo_address
 -         else:
 -             urlText = "osc.tcp://127.0.0.1:19000/Carla"
 - 
 -         askValue = QInputDialog.getText(self, self.tr("Carla Control - Connect"), self.tr("Address"), text=urlText)
 - 
 -         if not askValue[1]:
 -             return
 - 
 -         self.slot_handleExit()
 - 
 -         self.connectToAddr(askValue[0])
 - 
 -     @pyqtSlot()
 -     def slot_fileRefresh(self):
 -         global lo_target
 -         if lo_target and self.lo_server:
 -             self.removeAll()
 -             lo_send(lo_target, "/unregister")
 -             lo_send(lo_target, "/register", self.lo_server.getFullURL())
 - 
 -     @pyqtSlot()
 -     def slot_aboutCarlaControl(self):
 -         CarlaAboutW(self).exec_()
 - 
 -     @pyqtSlot(int, str)
 -     def slot_handleAddPluginStart(self, pluginId, pluginName):
 -         self.fLastPluginName = pluginName
 -         gCarla.host._add(pluginId)
 - 
 -     @pyqtSlot(int)
 -     def slot_handleAddPluginEnd(self, pluginId):
 -         pwidget = PluginWidget(self, pluginId)
 - 
 -         self.ui.w_plugins.layout().addWidget(pwidget)
 - 
 -         self.fPluginCount += 1
 -         self.fPluginList.append(pwidget)
 - 
 -     @pyqtSlot(int)
 -     def slot_handleRemovePlugin(self, pluginId):
 -         if pluginId >= self.fPluginCount:
 -             print("handleRemovePlugin(%i) - invalid plugin id" % pluginId)
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             print("handleRemovePlugin(%i) - invalid plugin" % pluginId)
 -             return
 - 
 -         self.fPluginList.pop(pluginId)
 -         self.fPluginCount -= 1
 - 
 -         self.ui.w_plugins.layout().removeWidget(pwidget)
 - 
 -         pwidget.ui.edit_dialog.close()
 -         pwidget.close()
 -         pwidget.deleteLater()
 -         del pwidget
 - 
 -         # push all plugins 1 slot back
 -         for i in range(pluginId, self.fPluginCount):
 -             self.fPluginList[i].setId(i)
 - 
 -         gCarla.host.fPluginsInfo.pop(pluginId)
 - 
 -     @pyqtSlot(int, int, int, int, str, str, str, str, int)
 -     def slot_handleSetPluginData(self, pluginId, type_, category, hints, realName, label, maker, copyright, uniqueId):
 -         info = deepcopy(CarlaPluginInfo)
 -         info['type']      = type_
 -         info['category']  = category
 -         info['hints']     = hints
 -         info['name']      = self.fLastPluginName
 -         info['label']     = label
 -         info['maker']     = maker
 -         info['copyright'] = copyright
 -         info['uniqueId']  = uniqueId
 -         gCarla.host._set_pluginInfo(pluginId, info)
 -         gCarla.host._set_pluginRealName(pluginId, realName)
 - 
 -     @pyqtSlot(int, int, int, int, int, int, int, int)
 -     def slot_handleSetPluginPorts(self, pluginId, audioIns, audioOuts, midiIns, midiOuts, cIns, cOuts, cTotals):
 -         audioInfo = deepcopy(CarlaPortCountInfo)
 -         midiInfo  = deepcopy(CarlaPortCountInfo)
 -         paramInfo = deepcopy(CarlaPortCountInfo)
 - 
 -         audioInfo['ins']   = audioIns
 -         audioInfo['outs']  = audioOuts
 -         audioInfo['total'] = audioIns + audioOuts
 - 
 -         midiInfo['ins']   = midiIns
 -         midiInfo['outs']  = midiOuts
 -         midiInfo['total'] = midiIns + midiOuts
 - 
 -         paramInfo['ins']   = cIns
 -         paramInfo['outs']  = cOuts
 -         paramInfo['total'] = cTotals
 - 
 -         gCarla.host._set_audioCountInfo(pluginId, audioInfo)
 -         gCarla.host._set_midiCountInfo(pluginId, midiInfo)
 -         gCarla.host._set_parameterCountInfo(pluginId, paramInfo)
 - 
 -     @pyqtSlot(int, int, int, int, str, str, float)
 -     def slot_handleSetParameterData(self, pluginId, index, type_, hints, name, label, current):
 -         # remove hints not possible in bridge mode
 -         hints &= ~(PARAMETER_USES_SCALEPOINTS | PARAMETER_USES_CUSTOM_TEXT)
 - 
 -         data = deepcopy(ParameterData)
 -         data['type']   = type_
 -         data['index']  = index
 -         data['rindex'] = index
 -         data['hints']  = hints
 - 
 -         info = deepcopy(CarlaParameterInfo)
 -         info['name']  = name
 -         info['label'] = label
 - 
 -         gCarla.host._set_parameterDataS(pluginId, index, data)
 -         gCarla.host._set_parameterInfoS(pluginId, index, info)
 -         gCarla.host._set_parameterValueS(pluginId, index, current)
 - 
 -     @pyqtSlot(int, int, float, float, float, float, float, float)
 -     def slot_handleSetParameterRanges(self, pluginId, index, min_, max_, def_, step, stepSmall, stepLarge):
 -         ranges = deepcopy(ParameterRanges)
 -         ranges['min'] = min_
 -         ranges['max'] = max_
 -         ranges['def'] = def_
 -         ranges['step'] = step
 -         ranges['stepSmall'] = stepSmall
 -         ranges['stepLarge'] = stepLarge
 - 
 -         gCarla.host._set_parameterRangeS(pluginId, index, ranges)
 - 
 -     @pyqtSlot(int, int, float)
 -     def slot_handleSetParameterValue(self, pluginId, parameterId, value):
 -         if parameterId >= 0:
 -             gCarla.host._set_parameterValueS(pluginId, parameterId, value)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setParameterValue(parameterId, value)
 - 
 -     @pyqtSlot(int, int, float)
 -     def slot_handleSetDefaultValue(self, pluginId, parameterId, value):
 -         gCarla.host._set_parameterDefaultValue(pluginId, parameterId, value)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setParameterDefault(parameterId, value)
 - 
 -     @pyqtSlot(int, int, int)
 -     def slot_handleSetParameterMidiCC(self, pluginId, index, cc):
 -         gCarla.host._set_parameterMidiCC(pluginId, index, cc)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setParameterMidiControl(index, cc)
 - 
 -     @pyqtSlot(int, int, int)
 -     def slot_handleSetParameterMidiChannel(self, pluginId, index, channel):
 -         gCarla.host._set_parameterMidiChannel(pluginId, index, channel)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setParameterMidiChannel(index, channel)
 - 
 -     @pyqtSlot(int, int)
 -     def slot_handleSetProgram(self, pluginId, index):
 -         gCarla.host._set_currentProgram(pluginId, index)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setProgram(index)
 - 
 -     @pyqtSlot(int, int)
 -     def slot_handleSetProgramCount(self, pluginId, count):
 -         gCarla.host._set_programCount(pluginId, count)
 - 
 -     @pyqtSlot(int, int, str)
 -     def slot_handleSetProgramName(self, pluginId, index, name):
 -         gCarla.host._set_programNameS(pluginId, index, name)
 - 
 -     @pyqtSlot(int, int)
 -     def slot_handleSetMidiProgram(self, pluginId, index):
 -         gCarla.host._set_currentMidiProgram(pluginId, index)
 - 
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.setMidiProgram(index)
 - 
 -     @pyqtSlot(int, int)
 -     def slot_handleSetMidiProgramCount(self, pluginId, count):
 -         gCarla.host._set_midiProgramCount(pluginId, count)
 - 
 -     @pyqtSlot(int, int, int, int, str)
 -     def slot_handleSetMidiProgramData(self, pluginId, index, bank, program, name):
 -         data = deepcopy(MidiProgramData)
 -         data['bank']    = bank
 -         data['program'] = program
 -         data['label']   = name
 -         gCarla.host._set_midiProgramDataS(pluginId, index, data)
 - 
 -     @pyqtSlot(int, int, float)
 -     def slot_handleSetInputPeakValue(self, pluginId, portId, value):
 -         gCarla.host._set_inPeak(pluginId, portId-1, value)
 - 
 -     @pyqtSlot(int, int, float)
 -     def slot_handleSetOutputPeakValue(self, pluginId, portId, value):
 -         gCarla.host._set_outPeak(pluginId, portId-1, value)
 - 
 -     @pyqtSlot(int, int, int, int)
 -     def slot_handleNoteOn(self, pluginId, channel, note, velo):
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.sendNoteOn(channel, note)
 - 
 -     @pyqtSlot(int, int, int)
 -     def slot_handleNoteOff(self, pluginId, channel, note):
 -         if pluginId >= self.fPluginCount:
 -             return
 - 
 -         pwidget = self.fPluginList[pluginId]
 -         if pwidget is None:
 -             return
 - 
 -         pwidget.sendNoteOff(channel, note)
 - 
 -     @pyqtSlot(int, float, float, float, float)
 -     def slot_handleSetPeaks(self, pluginId, in1, in2, out1, out2):
 -         gCarla.host._set_peaks(pluginId, in1, in2, out1, out2)
 - 
 -     @pyqtSlot()
 -     def slot_handleExit(self):
 -         self.removeAll()
 - 
 -         if self.lo_server:
 -             self.lo_server.stop()
 -             self.lo_server = None
 -             self.ui.act_file_refresh.setEnabled(False)
 - 
 -         global lo_target, lo_targetName
 -         lo_target     = None
 -         lo_targetName = ""
 -         self.lo_address = ""
 - 
 -     def saveSettings(self):
 -         settings = QSettings()
 -         settings.setValue("Geometry", self.saveGeometry())
 -         #settings.setValue("ShowToolbar", self.ui.toolBar.isVisible())
 - 
 -     def loadSettings(self):
 -         settings = QSettings()
 -         self.restoreGeometry(settings.value("Geometry", ""))
 - 
 -         #showToolbar = settings.value("ShowToolbar", True, type=bool)
 -         #self.ui.act_settings_show_toolbar.setChecked(showToolbar)
 -         #self.ui.toolBar.setVisible(showToolbar)
 - 
 -     def timerEvent(self, event):
 -         if event.timerId() == self.fIdleTimerFast:
 -             for pwidget in self.fPluginList:
 -                 if pwidget is None:
 -                     break
 -                 pwidget.idleFast()
 - 
 -         elif event.timerId() == self.fIdleTimerSlow:
 -             for pwidget in self.fPluginList:
 -                 if pwidget is None:
 -                     break
 -                 pwidget.idleSlow()
 - 
 -         QMainWindow.timerEvent(self, event)
 - 
 -     def closeEvent(self, event):
 -         self.saveSettings()
 - 
 -         global lo_target
 -         if lo_target and self.lo_server:
 -             lo_send(lo_target, "/unregister")
 - 
 -         QMainWindow.closeEvent(self, event)
 - 
 - #--------------- main ------------------
 - if __name__ == '__main__':
 - 
 -     # App initialization
 -     app = QApplication(sys.argv)
 -     app.setApplicationName("Carla-Control")
 -     app.setApplicationVersion(VERSION)
 -     app.setOrganizationName("Cadence")
 -     app.setWindowIcon(QIcon(":/scalable/carla-control.svg"))
 - 
 -     libPrefix = None
 -     oscAddr = None
 - 
 -     argv = app.arguments()
 -     argc = len(argv)
 - 
 -     for i in range(argc):
 -         if i == 0: continue
 -         argument = argv[i]
 - 
 -         if argument.startswith("--with-libprefix="):
 -             libPrefix = argument.replace("--with-libprefix=", "")
 - 
 -         elif argument.startswith("osc."):
 -             oscAddr = argument
 - 
 -     if libPrefix is not None:
 -         libName = os.path.join(libPrefix, "lib", "carla", carla_libname)
 -     else:
 -         libName = carla_library_filename
 - 
 -     # Init backend (OSC bridge version)
 -     gCarla.host = Host()
 - 
 -     # Create GUI
 -     gCarla.gui = CarlaControlW(oscAddr)
 - 
 -     # Set-up custom signal handling
 -     setUpSignals()
 - 
 -     # Show GUI
 -     gCarla.gui.show()
 - 
 -     # App-Loop
 -     sys.exit(app.exec_())
 
 
  |