|
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094 |
- #!/usr/bin/env python3
- # -*- coding: utf-8 -*-
-
- # Carla host code
- # 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 doc/GPL.txt file.
-
- # ------------------------------------------------------------------------------------------------------------
- # Imports (Global)
-
- try:
- from PyQt5.QtCore import QModelIndex, QTimer
- from PyQt5.QtGui import QPalette
- from PyQt5.QtWidgets import QApplication, QFileSystemModel, QMainWindow
- except:
- from PyQt4.QtCore import QModelIndex, QTimer
- from PyQt4.QtGui import QApplication, QFileSystemModel, QMainWindow, QPalette
-
- # ------------------------------------------------------------------------------------------------------------
- # Imports (Custom)
-
- import ui_carla_host
-
- from carla_database import *
- from carla_settings import *
- from carla_style import *
- from carla_widgets import *
-
- # ------------------------------------------------------------------------------------------------------------
- # PatchCanvas defines
-
- CANVAS_ANTIALIASING_SMALL = 1
- CANVAS_EYECANDY_SMALL = 1
-
- # ------------------------------------------------------------------------------------------------------------
- # Session Management support
-
- LADISH_APP_NAME = os.getenv("LADISH_APP_NAME")
- NSM_URL = os.getenv("NSM_URL")
-
- # ------------------------------------------------------------------------------------------------------------
- # Dummy widget
-
- class CarlaDummyW(object):
- def __init__(self, parent):
- object.__init__(self)
-
- # -----------------------------------------------------------------
-
- def getPluginCount(self):
- return 0
-
- # -----------------------------------------------------------------
-
- def addPlugin(self, pluginId, isProjectLoading):
- pass
-
- def removePlugin(self, pluginId):
- pass
-
- def renamePlugin(self, pluginId, newName):
- pass
-
- def removeAllPlugins(self):
- pass
-
- # -----------------------------------------------------------------
-
- def engineStarted(self):
- pass
-
- def engineStopped(self):
- pass
-
- def engineChanged(self):
- pass
-
- # -----------------------------------------------------------------
-
- def idleFast(self):
- pass
-
- def idleSlow(self):
- pass
-
- # -----------------------------------------------------------------
-
- def saveSettings(self, settings):
- pass
-
- # ------------------------------------------------------------------------------------------------------------
- # Host Window
-
- class HostWindow(QMainWindow):
- # signals
- DebugCallback = pyqtSignal(int, int, int, float, str)
- PluginAddedCallback = pyqtSignal(int)
- PluginRemovedCallback = pyqtSignal(int)
- PluginRenamedCallback = pyqtSignal(int, str)
- ParameterValueChangedCallback = pyqtSignal(int, int, float)
- ParameterDefaultChangedCallback = pyqtSignal(int, int, float)
- ParameterMidiChannelChangedCallback = pyqtSignal(int, int, int)
- ParameterMidiCcChangedCallback = pyqtSignal(int, int, int)
- ProgramChangedCallback = pyqtSignal(int, int)
- MidiProgramChangedCallback = pyqtSignal(int, int)
- NoteOnCallback = pyqtSignal(int, int, int, int)
- NoteOffCallback = pyqtSignal(int, int, int)
- ShowGuiCallback = pyqtSignal(int, int)
- UpdateCallback = pyqtSignal(int)
- ReloadInfoCallback = pyqtSignal(int)
- ReloadParametersCallback = pyqtSignal(int)
- ReloadProgramsCallback = pyqtSignal(int)
- ReloadAllCallback = pyqtSignal(int)
- PatchbayClientAddedCallback = pyqtSignal(int, int, str)
- PatchbayClientRemovedCallback = pyqtSignal(int, str)
- PatchbayClientRenamedCallback = pyqtSignal(int, str)
- PatchbayPortAddedCallback = pyqtSignal(int, int, int, str)
- PatchbayPortRemovedCallback = pyqtSignal(int, int, str)
- PatchbayPortRenamedCallback = pyqtSignal(int, int, str)
- PatchbayConnectionAddedCallback = pyqtSignal(int, int, int)
- PatchbayConnectionRemovedCallback = pyqtSignal(int)
- PatchbayIconChangedCallback = pyqtSignal(int, int)
- BufferSizeChangedCallback = pyqtSignal(int)
- SampleRateChangedCallback = pyqtSignal(float)
- ProcessModeChangedCallback = pyqtSignal(int)
- EngineStartedCallback = pyqtSignal(str)
- EngineStoppedChangedCallback = pyqtSignal()
- NSM_AnnounceCallback = pyqtSignal(str)
- NSM_OpenCallback = pyqtSignal(str)
- NSM_SaveCallback = pyqtSignal()
- ErrorCallback = pyqtSignal(str)
- QuitCallback = pyqtSignal()
- SIGTERM = pyqtSignal()
- SIGUSR1 = pyqtSignal()
-
- def __init__(self, parent):
- QMainWindow.__init__(self, parent)
- self.ui = ui_carla_host.Ui_CarlaHostW()
- self.ui.setupUi(self)
-
- if False:
- Carla.gui = self
- self.fContainer = CarlaDummyW(self)
-
- # -------------------------------------------------------------
- # Set callback
-
- Carla.host.set_engine_callback(EngineCallback)
-
- # -------------------------------------------------------------
- # Internal stuff
-
- self.fBufferSize = 0
- self.fSampleRate = 0.0
-
- self.fIdleTimerFast = 0
- self.fIdleTimerSlow = 0
-
- self.fIsProjectLoading = False
- self.fProjectFilename = ""
-
- self.fLadspaRdfNeedsUpdate = True
- self.fLadspaRdfList = []
-
- self.fLastTransportFrame = 0
- self.fLastTransportState = False
- self.fTextTransport = ""
-
- self.fSavedSettings = {}
-
- if LADISH_APP_NAME:
- self.fClientName = LADISH_APP_NAME
- self.fSessionManagerName = "LADISH"
- elif NSM_URL:
- self.fClientName = "Carla.tmp"
- self.fSessionManagerName = "Non Session Manager"
- else:
- self.fClientName = "Carla"
- self.fSessionManagerName = ""
-
- # -------------------------------------------------------------
- # Load Settings
-
- self.loadSettings(True)
-
- # -------------------------------------------------------------
- # Set up GUI (engine stopped)
-
- if Carla.isPlugin:
- self.ui.act_file_new.setEnabled(False)
- self.ui.act_file_open.setEnabled(False)
- self.ui.act_engine_start.setEnabled(False)
- self.ui.act_engine_stop.setEnabled(False)
- self.ui.menu_Engine.setEnabled(False)
- else:
- self.ui.act_engine_start.setEnabled(True)
-
- self.ui.act_file_save.setEnabled(False)
- self.ui.act_file_save_as.setEnabled(False)
- self.ui.act_engine_stop.setEnabled(False)
- self.ui.act_plugin_remove_all.setEnabled(False)
-
- self.ui.menu_PluginMacros.setEnabled(False)
- self.ui.menu_Canvas.setEnabled(False)
-
- self.setTransportMenuEnabled(False)
-
- # -------------------------------------------------------------
- # Set up GUI (right panel)
-
- self.fDirModel = QFileSystemModel(self)
- self.fDirModel.setNameFilters(cString(Carla.host.get_supported_file_types()).split(";"))
- self.fDirModel.setRootPath(HOME)
-
- self.ui.fileTreeView.setModel(self.fDirModel)
- self.ui.fileTreeView.setRootIndex(self.fDirModel.index(HOME))
- self.ui.fileTreeView.setColumnHidden(1, True)
- self.ui.fileTreeView.setColumnHidden(2, True)
- self.ui.fileTreeView.setColumnHidden(3, True)
- self.ui.fileTreeView.setHeaderHidden(True)
-
- self.setProperWindowTitle()
-
- # -------------------------------------------------------------
- # Connect actions to functions
-
- self.ui.act_file_new.triggered.connect(self.slot_fileNew)
- self.ui.act_file_open.triggered.connect(self.slot_fileOpen)
- self.ui.act_file_save.triggered.connect(self.slot_fileSave)
- self.ui.act_file_save_as.triggered.connect(self.slot_fileSaveAs)
-
- self.ui.act_engine_start.triggered.connect(self.slot_engineStart)
- self.ui.act_engine_stop.triggered.connect(self.slot_engineStop)
-
- self.ui.act_plugin_add.triggered.connect(self.slot_pluginAdd)
- self.ui.act_plugin_add2.triggered.connect(self.slot_pluginAdd)
- self.ui.act_plugin_remove_all.triggered.connect(self.slot_pluginRemoveAll)
-
- self.ui.act_transport_play.triggered.connect(self.slot_transportPlayPause)
- self.ui.act_transport_stop.triggered.connect(self.slot_transportStop)
- self.ui.act_transport_backwards.triggered.connect(self.slot_transportBackwards)
- self.ui.act_transport_forwards.triggered.connect(self.slot_transportForwards)
-
- self.ui.act_help_about.triggered.connect(self.slot_aboutCarla)
- self.ui.act_help_about_qt.triggered.connect(self.slot_aboutQt)
-
- self.ui.cb_disk.currentIndexChanged.connect(self.slot_diskFolderChanged)
- self.ui.b_disk_add.clicked.connect(self.slot_diskFolderAdd)
- self.ui.b_disk_remove.clicked.connect(self.slot_diskFolderRemove)
- self.ui.fileTreeView.doubleClicked.connect(self.slot_fileTreeDoubleClicked)
-
- self.DebugCallback.connect(self.slot_handleDebugCallback)
- self.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback)
- self.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback)
- self.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
- self.BufferSizeChangedCallback.connect(self.slot_handleBufferSizeChangedCallback)
- self.SampleRateChangedCallback.connect(self.slot_handleSampleRateChangedCallback)
- self.EngineStartedCallback.connect(self.slot_handleEngineStartedCallback)
- self.EngineStoppedChangedCallback.connect(self.slot_handleEngineStoppedCallback)
- #self.NSM_AnnounceCallback.connect(self.slot_handleNSM_AnnounceCallback)
- #self.NSM_OpenCallback.connect(self.slot_handleNSM_OpenCallback)
- #self.NSM_SaveCallback.connect(self.slot_handleNSM_SaveCallback)
- #self.ErrorCallback.connect(self.slot_handleErrorCallback)
- #self.QuitCallback.connect(self.slot_handleQuitCallback)
-
- self.SIGUSR1.connect(self.slot_handleSIGUSR1)
- self.SIGTERM.connect(self.slot_handleSIGTERM)
-
- # -------------------------------------------------------------
- # Final setup
-
- if Carla.isPlugin:
- QTimer.singleShot(0, self.slot_engineStart)
- elif NSM_URL:
- Carla.host.nsm_ready()
-
- # -----------------------------------------------------------------
- # Called by containers
-
- def openSettingsWindow(self, hasCanvas, hasCanvasGL):
- dialog = CarlaSettingsW(self, hasCanvas, hasCanvasGL)
- return dialog.exec_()
-
- def setupContainer(self, showMiniCanvas):
- if showMiniCanvas:
- CARLA_DEFAULT_CANVAS_WIDTH = 3100
- CARLA_DEFAULT_CANVAS_HEIGHT = 2400
- self.ui.miniCanvasPreview.setRealParent(self)
- #self.ui.miniCanvasPreview.setViewTheme(patchcanvas.canvas.theme.canvas_bg, patchcanvas.canvas.theme.rubberband_brush, patchcanvas.canvas.theme.rubberband_pen.color())
- self.ui.miniCanvasPreview.init(self.fContainer.scene, CARLA_DEFAULT_CANVAS_WIDTH, CARLA_DEFAULT_CANVAS_HEIGHT, self.fSavedSettings["UseCustomMiniCanvasPaint"])
- else:
- self.ui.miniCanvasPreview.hide()
-
- self.ui.mainwidget.hide()
- self.ui.splitter.insertWidget(0, self.fContainer)
- del self.ui.mainwidget
-
- # -----------------------------------------------------------------
- # Internal stuff (files)
-
- def loadProjectNow(self):
- if not self.fProjectFilename:
- return qCritical("ERROR: loading project without filename set")
-
- # TESTING
- if not Carla.host.is_engine_running():
- self.slot_engineStart()
-
- self.fIsProjectLoading = True
- Carla.host.load_project(self.fProjectFilename)
- self.fIsProjectLoading = False
-
- @pyqtSlot()
- def slot_loadProjectNow(self):
- self.loadProjectNow()
-
- def loadProjectLater(self, filename):
- self.fProjectFilename = filename
- self.setProperWindowTitle()
- QTimer.singleShot(0, self.slot_loadProjectNow)
-
- def saveProjectNow(self):
- if not self.fProjectFilename:
- return qCritical("ERROR: saving project without filename set")
-
- Carla.host.save_project(self.fProjectFilename)
-
- # -----------------------------------------------------------------
- # Internal stuff (engine)
-
- def setEngineSettings(self, settings = None):
- if Carla.isPlugin:
- return "Plugin"
-
- if settings is None: settings = QSettings()
-
- forceStereo = settings.value("Engine/ForceStereo", CARLA_DEFAULT_FORCE_STEREO, type=bool)
- preferPluginBridges = settings.value("Engine/PreferPluginBridges", CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES, type=bool)
- preferUiBridges = settings.value("Engine/PreferUiBridges", CARLA_DEFAULT_PREFER_UI_BRIDGES, type=bool)
- uisAlwaysOnTop = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UIS_ALWAYS_ON_TOP, type=bool)
- uiBridgesTimeout = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UI_BRIDGES_TIMEOUT, type=int)
-
- Carla.maxParameters = settings.value("Engine/MaxParameters", CARLA_DEFAULT_MAX_PARAMETERS, type=int)
-
- audioDriver = settings.value("Engine/AudioDriver", CARLA_DEFAULT_AUDIO_DRIVER, type=str)
-
- if audioDriver == "JACK":
- #transportMode = settings.value("Engine/TransportMode", TRANSPORT_MODE_JACK, type=int)
- transportMode = TRANSPORT_MODE_JACK
- else:
- transportMode = TRANSPORT_MODE_INTERNAL
- audioNumPeriods = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_NUM_PERIODS, type=int)
- audioBufferSize = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_BUFFER_SIZE, type=int)
- audioSampleRate = settings.value("Engine/AudioSampleRate", CARLA_DEFAULT_AUDIO_SAMPLE_RATE, type=int)
- audioDevice = settings.value("Engine/AudioDevice", "", type=str)
-
- Carla.host.set_engine_option(OPTION_AUDIO_NUM_PERIODS, audioNumPeriods, "")
- Carla.host.set_engine_option(OPTION_AUDIO_BUFFER_SIZE, audioBufferSize, "")
- Carla.host.set_engine_option(OPTION_AUDIO_SAMPLE_RATE, audioSampleRate, "")
- Carla.host.set_engine_option(OPTION_AUDIO_DEVICE, 0, audioDevice)
-
- if Carla.processMode == PROCESS_MODE_CONTINUOUS_RACK:
- forceStereo = True
- elif Carla.processMode == PROCESS_MODE_MULTIPLE_CLIENTS and LADISH_APP_NAME:
- print("LADISH detected but using multiple clients (not allowed), forcing single client now")
- Carla.processMode = PROCESS_MODE_SINGLE_CLIENT
-
- Carla.host.set_engine_option(OPTION_FORCE_STEREO, forceStereo, "")
- Carla.host.set_engine_option(OPTION_PREFER_PLUGIN_BRIDGES, preferPluginBridges, "")
- Carla.host.set_engine_option(OPTION_PREFER_UI_BRIDGES, preferUiBridges, "")
- Carla.host.set_engine_option(OPTION_UIS_ALWAYS_ON_TOP, uisAlwaysOnTop, "")
- Carla.host.set_engine_option(OPTION_UI_BRIDGES_TIMEOUT, uiBridgesTimeout, "")
- Carla.host.set_engine_option(OPTION_PROCESS_MODE, Carla.processMode, "")
- Carla.host.set_engine_option(OPTION_MAX_PARAMETERS, Carla.maxParameters, "")
- Carla.host.set_engine_option(OPTION_TRANSPORT_MODE, transportMode, "")
-
- return audioDriver
-
- def startEngine(self):
- audioDriver = self.setEngineSettings()
-
- if not Carla.host.engine_init(audioDriver, self.fClientName):
- #if self.fFirstEngineInit:
- #self.fFirstEngineInit = False
- #return
-
- audioError = cString(Carla.host.get_last_error())
-
- if audioError:
- QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s', possible reasons:\n%s" % (audioDriver, audioError)))
- else:
- QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s'" % audioDriver))
- return
-
- self.fBufferSize = Carla.host.get_buffer_size()
- self.fSampleRate = Carla.host.get_sample_rate()
-
- #self.fFirstEngineInit = False
-
- # Peaks and TimeInfo
- self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
- # LEDs and edit dialog parameters
- self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
-
- def stopEngine(self):
- if self.fContainer.getPluginCount() > 0:
- ask = QMessageBox.question(self, self.tr("Warning"), self.tr("There are still some plugins loaded, you need to remove them to stop the engine.\n"
- "Do you want to do this now?"),
- QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
- if ask != QMessageBox.Yes:
- return
-
- self.ui.act_plugin_remove_all.setEnabled(False)
- self.fContainer.removeAllPlugins()
-
- if Carla.host.is_engine_running() and not Carla.host.engine_close():
- print(cString(Carla.host.get_last_error()))
-
- self.fBufferSize = 0
- self.fSampleRate = 0.0
-
- if self.fIdleTimerFast != 0:
- self.killTimer(self.fIdleTimerFast)
- self.fIdleTimerFast = 0
-
- if self.fIdleTimerSlow != 0:
- self.killTimer(self.fIdleTimerSlow)
- self.fIdleTimerSlow = 0
-
- # -----------------------------------------------------------------
- # Internal stuff (plugins)
-
- def getExtraStuff(self, plugin):
- ptype = plugin['type']
-
- if ptype == PLUGIN_LADSPA:
- uniqueId = plugin['uniqueId']
-
- self.maybeLoadRDFs()
-
- for rdfItem in self.fLadspaRdfList:
- if rdfItem.UniqueID == uniqueId:
- return pointer(rdfItem)
-
- elif ptype in (PLUGIN_GIG, PLUGIN_SF2, PLUGIN_SFZ):
- if plugin['name'].lower().endswith(" (16 outputs)"):
- # return a dummy non-null pointer
- INTPOINTER = POINTER(c_int)
- int1 = c_int(0x1)
- addr = addressof(int1)
- return cast(addr, INTPOINTER)
-
- return c_nullptr
-
- def maybeLoadRDFs(self):
- if not self.fLadspaRdfNeedsUpdate:
- return
-
- self.fLadspaRdfNeedsUpdate = False
- self.fLadspaRdfList = []
-
- if not haveLRDF:
- return
-
- settingsDir = os.path.join(HOME, ".config", "falkTX")
- frLadspaFile = os.path.join(settingsDir, "ladspa_rdf.db")
-
- if os.path.exists(frLadspaFile):
- frLadspa = open(frLadspaFile, 'r')
-
- try:
- self.fLadspaRdfList = ladspa_rdf.get_c_ladspa_rdfs(json.load(frLadspa))
- except:
- pass
-
- frLadspa.close()
-
- def setLoadRDFsNeeded(self):
- self.fLadspaRdfNeedsUpdate = True
-
- # -----------------------------------------------------------------
- # Internal stuff (transport)
-
- def refreshTransport(self, forced = False):
- if not Carla.host.is_engine_running():
- return
- if self.fSampleRate == 0.0:
- return
-
- timeInfo = Carla.host.get_transport_info()
- playing = bool(timeInfo['playing'])
- frame = int(timeInfo['frame'])
-
- if playing != self.fLastTransportState or forced:
- if playing:
- icon = getIcon("media-playback-pause")
- self.ui.act_transport_play.setChecked(True)
- self.ui.act_transport_play.setIcon(icon)
- self.ui.act_transport_play.setText(self.tr("&Pause"))
- else:
- icon = getIcon("media-playback-start")
- self.ui.act_transport_play.setChecked(False)
- self.ui.act_transport_play.setIcon(icon)
- self.ui.act_transport_play.setText(self.tr("&Play"))
-
- self.fLastTransportState = playing
-
- if frame != self.fLastTransportFrame or forced:
- time = frame / self.fSampleRate
- secs = time % 60
- mins = (time / 60) % 60
- hrs = (time / 3600) % 60
-
- self.fTextTransport = "Transport %s, at %02i:%02i:%02i" % ("playing" if playing else "stopped", hrs, mins, secs)
- self.fLastTransportFrame = frame
-
- def setTransportMenuEnabled(self, enabled):
- self.ui.act_transport_play.setEnabled(enabled)
- self.ui.act_transport_stop.setEnabled(enabled)
- self.ui.act_transport_backwards.setEnabled(enabled)
- self.ui.act_transport_forwards.setEnabled(enabled)
- self.ui.menu_Transport.setEnabled(enabled)
-
- # -----------------------------------------------------------------
- # Internal stuff (settings)
-
- def loadSettings(self, firstTime):
- settings = QSettings()
-
- if firstTime:
- 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)
-
- #if settings.contains("SplitterState"):
- #self.ui.splitter.restoreState(settings.value("SplitterState", ""))
- #else:
- self.ui.splitter.setSizes([99999, 210])
-
- diskFolders = toList(settings.value("DiskFolders", [HOME]))
-
- self.ui.cb_disk.setItemData(0, HOME)
-
- for i in range(len(diskFolders)):
- if i == 0: continue
- folder = diskFolders[i]
- self.ui.cb_disk.addItem(os.path.basename(folder), folder)
-
- if MACOS and not settings.value("Main/UseProTheme", True, type=bool):
- self.setUnifiedTitleAndToolBarOnMac(True)
-
- # ---------------------------------------------
- # plugin checks
-
- if settings.value("Engine/DisableChecks", False, type=bool):
- os.environ["CARLA_DISCOVERY_NO_PROCESSING_CHECKS"] = "true"
-
- elif os.getenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS"):
- os.environ.pop("CARLA_DISCOVERY_NO_PROCESSING_CHECKS")
-
- # ---------------------------------------------
-
- if not Carla.isPlugin:
- # engine
- self.setEngineSettings(settings)
-
- # plugin paths
- LADSPA_PATH = toList(settings.value("Paths/LADSPA", Carla.DEFAULT_LADSPA_PATH))
- DSSI_PATH = toList(settings.value("Paths/DSSI", Carla.DEFAULT_DSSI_PATH))
- LV2_PATH = toList(settings.value("Paths/LV2", Carla.DEFAULT_LV2_PATH))
- VST_PATH = toList(settings.value("Paths/VST", Carla.DEFAULT_VST_PATH))
- AU_PATH = toList(settings.value("Paths/AU", Carla.DEFAULT_AU_PATH))
- CSOUND_PATH = toList(settings.value("Paths/CSOUND", Carla.DEFAULT_CSOUND_PATH))
- GIG_PATH = toList(settings.value("Paths/GIG", Carla.DEFAULT_GIG_PATH))
- SF2_PATH = toList(settings.value("Paths/SF2", Carla.DEFAULT_SF2_PATH))
- SFZ_PATH = toList(settings.value("Paths/SFZ", Carla.DEFAULT_SFZ_PATH))
-
- os.environ["LADSPA_PATH"] = splitter.join(LADSPA_PATH)
- os.environ["DSSI_PATH"] = splitter.join(DSSI_PATH)
- os.environ["LV2_PATH"] = splitter.join(LV2_PATH)
- os.environ["VST_PATH"] = splitter.join(VST_PATH)
- os.environ["AU_PATH"] = splitter.join(AU_PATH)
- os.environ["CSOUND_PATH"] = splitter.join(CSOUND_PATH)
- os.environ["GIG_PATH"] = splitter.join(GIG_PATH)
- os.environ["SF2_PATH"] = splitter.join(SF2_PATH)
- os.environ["SFZ_PATH"] = splitter.join(SFZ_PATH)
-
- # ---------------------------------------------
-
- # TODO
-
- self.fSavedSettings = {
- "Main/DefaultProjectFolder": settings.value("Main/DefaultProjectFolder", HOME, type=str),
- "Main/RefreshInterval": settings.value("Main/RefreshInterval", 50, type=int),
- CARLA_KEY_CANVAS_THEME: settings.value(CARLA_KEY_CANVAS_THEME, CARLA_DEFAULT_CANVAS_THEME, type=str),
- CARLA_KEY_CANVAS_SIZE: settings.value(CARLA_KEY_CANVAS_SIZE, CARLA_DEFAULT_CANVAS_SIZE, type=str),
- CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS: settings.value(CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS, CARLA_DEFAULT_CANVAS_AUTO_HIDE_GROUPS, type=bool),
- CARLA_KEY_CANVAS_USE_BEZIER_LINES: settings.value(CARLA_KEY_CANVAS_USE_BEZIER_LINES, CARLA_DEFAULT_CANVAS_USE_BEZIER_LINES, type=bool),
- CARLA_KEY_CANVAS_EYE_CANDY: settings.value(CARLA_KEY_CANVAS_EYE_CANDY, CARLA_DEFAULT_CANVAS_EYE_CANDY, type=int),
- CARLA_KEY_CANVAS_USE_OPENGL: settings.value(CARLA_KEY_CANVAS_USE_OPENGL, CARLA_DEFAULT_CANVAS_USE_OPENGL, type=bool),
- CARLA_KEY_CANVAS_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_ANTIALIASING, CARLA_DEFAULT_CANVAS_ANTIALIASING, type=int),
- CARLA_KEY_CANVAS_HQ_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_HQ_ANTIALIASING, CARLA_DEFAULT_CANVAS_HQ_ANTIALIASING, type=bool),
- "UseCustomMiniCanvasPaint": (settings.value("Main/UseProTheme", True, type=bool) and
- settings.value("Main/ProThemeColor", "Black", type=str).lower() == "black")
- }
-
- # ---------------------------------------------
-
- if self.fIdleTimerFast != 0:
- self.killTimer(self.fIdleTimerFast)
- self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
-
- if self.fIdleTimerSlow != 0:
- self.killTimer(self.fIdleTimerSlow)
- self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
-
- def saveSettings(self):
- settings = QSettings()
-
- settings.setValue("Geometry", self.saveGeometry())
- settings.setValue("SplitterState", self.ui.splitter.saveState())
- settings.setValue("ShowToolbar", self.ui.toolBar.isVisible())
- #settings.setValue("HorizontalScrollBarValue", self.ui.graphicsView.horizontalScrollBar().value())
- #settings.setValue("VerticalScrollBarValue", self.ui.graphicsView.verticalScrollBar().value())
-
- diskFolders = []
-
- for i in range(self.ui.cb_disk.count()):
- diskFolders.append(self.ui.cb_disk.itemData(i))
-
- settings.setValue("DiskFolders", diskFolders)
-
- self.fContainer.saveSettings(settings)
-
- # -----------------------------------------------------------------
- # Internal stuff (gui)
-
- def setProperWindowTitle(self):
- title = self.fClientName
-
- if self.fProjectFilename:
- title += " - %s" % os.path.basename(self.fProjectFilename)
- if self.fSessionManagerName:
- title += " (%s)" % self.fSessionManagerName
-
- self.setWindowTitle(title)
-
- # -----------------------------------------------------------------
-
- @pyqtSlot()
- def slot_fileNew(self):
- self.fContainer.removeAllPlugins()
- self.fProjectFilename = ""
- self.setProperWindowTitle()
-
- @pyqtSlot()
- def slot_fileOpen(self):
- fileFilter = self.tr("Carla Project File (*.carxp)")
- filenameTry = QFileDialog.getOpenFileName(self, self.tr("Open Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
-
- if not filenameTry:
- return
-
- newFile = True
-
- if self.fContainer.getPluginCount() > 0:
- ask = QMessageBox.question(self, self.tr("Question"), self.tr("There are some plugins loaded, do you want to remove them now?"),
- QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
- newFile = ask == QMessageBox.Yes
-
- if newFile:
- self.fContainer.removeAllPlugins()
- self.fProjectFilename = filenameTry
- self.setProperWindowTitle()
- self.loadProjectNow()
- else:
- filenameOld = self.fProjectFilename
- self.fProjectFilename = filenameTry
- self.loadProjectNow()
- self.fProjectFilename = filenameOld
-
- @pyqtSlot()
- def slot_fileSave(self, saveAs=False):
- if self.fProjectFilename and not saveAs:
- return self.saveProjectNow()
-
- fileFilter = self.tr("Carla Project File (*.carxp)")
- filenameTry = QFileDialog.getSaveFileName(self, self.tr("Save Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
-
- if not filenameTry:
- return
-
- if not filenameTry.endswith(".carxp"):
- filenameTry += ".carxp"
-
- if self.fProjectFilename != filenameTry:
- self.fProjectFilename = filenameTry
- self.setProperWindowTitle()
-
- self.saveProjectNow()
-
- @pyqtSlot()
- def slot_fileSaveAs(self):
- self.slot_fileSave(True)
-
- # -----------------------------------------------------------------
-
- @pyqtSlot()
- def slot_engineStart(self, doStart = True):
- if doStart: self.startEngine()
-
- check = Carla.host.is_engine_running()
- self.ui.menu_PluginMacros.setEnabled(check)
- self.ui.menu_Canvas.setEnabled(check)
-
- if not Carla.isPlugin:
- self.ui.act_file_save.setEnabled(check)
- self.ui.act_engine_start.setEnabled(not check)
- self.ui.act_engine_stop.setEnabled(check)
-
- if self.fSessionManagerName != "Non Session Manager":
- self.ui.act_file_open.setEnabled(check)
- self.ui.act_file_save_as.setEnabled(check)
-
- self.setTransportMenuEnabled(check)
-
- if check:
- if not Carla.isPlugin:
- self.refreshTransport(True)
-
- self.fContainer.engineStarted()
-
- @pyqtSlot()
- def slot_engineStop(self, doStop = True):
- if doStop: self.stopEngine()
-
- check = Carla.host.is_engine_running()
- self.ui.menu_PluginMacros.setEnabled(check)
- self.ui.menu_Canvas.setEnabled(check)
-
- if not Carla.isPlugin:
- self.ui.act_file_save.setEnabled(check)
- self.ui.act_engine_start.setEnabled(not check)
- self.ui.act_engine_stop.setEnabled(check)
-
- if self.fSessionManagerName != "Non Session Manager":
- self.ui.act_file_open.setEnabled(check)
- self.ui.act_file_save_as.setEnabled(check)
-
- self.setTransportMenuEnabled(check)
-
- if not check:
- self.fTextTransport = ""
- self.fContainer.engineStopped()
-
- # -----------------------------------------------------------------
-
- @pyqtSlot()
- def slot_pluginAdd(self):
- dialog = PluginDatabaseW(self)
-
- if not dialog.exec_():
- return
-
- if not Carla.host.is_engine_running():
- QMessageBox.warning(self, self.tr("Warning"), self.tr("Cannot add new plugins while engine is stopped"))
- return
-
- btype = dialog.fRetPlugin['build']
- ptype = dialog.fRetPlugin['type']
- filename = dialog.fRetPlugin['binary']
- label = dialog.fRetPlugin['label']
- extraStuff = self.getExtraStuff(dialog.fRetPlugin)
-
- if not Carla.host.add_plugin(btype, ptype, filename, None, label, c_nullptr):
- CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"), self.tr("Failed to load plugin"), cString(Carla.host.get_last_error()), QMessageBox.Ok, QMessageBox.Ok)
- return
-
- @pyqtSlot()
- def slot_pluginRemoveAll(self):
- self.ui.act_plugin_remove_all.setEnabled(False)
- self.fContainer.removeAllPlugins()
- Carla.host.remove_all_plugins()
-
- # -----------------------------------------------------------------
-
- @pyqtSlot(bool)
- def slot_transportPlayPause(self, toggled):
- if not Carla.host.is_engine_running():
- return
-
- if toggled:
- Carla.host.transport_play()
- else:
- Carla.host.transport_pause()
-
- self.refreshTransport()
-
- @pyqtSlot()
- def slot_transportStop(self):
- if not Carla.host.is_engine_running():
- return
-
- Carla.host.transport_pause()
- Carla.host.transport_relocate(0)
-
- self.refreshTransport()
-
- @pyqtSlot()
- def slot_transportBackwards(self):
- if not Carla.host.is_engine_running():
- return
-
- newFrame = Carla.host.get_current_transport_frame() - 100000
-
- if newFrame < 0:
- newFrame = 0
-
- Carla.host.transport_relocate(newFrame)
-
- @pyqtSlot()
- def slot_transportForwards(self):
- if not Carla.host.is_engine_running():
- return
-
- newFrame = Carla.host.get_current_transport_frame() + 100000
- Carla.host.transport_relocate(newFrame)
-
- # -----------------------------------------------------------------
-
- @pyqtSlot()
- def slot_aboutCarla(self):
- CarlaAboutW(self).exec_()
-
- @pyqtSlot()
- def slot_aboutQt(self):
- QApplication.instance().aboutQt()
-
- # -----------------------------------------------------------------
-
- @pyqtSlot(int)
- def slot_diskFolderChanged(self, index):
- if index < 0:
- return
- elif index == 0:
- filename = HOME
- self.ui.b_disk_remove.setEnabled(False)
- else:
- filename = self.ui.cb_disk.itemData(index)
- self.ui.b_disk_remove.setEnabled(True)
-
- self.fDirModel.setRootPath(filename)
- self.ui.fileTreeView.setRootIndex(self.fDirModel.index(filename))
-
- @pyqtSlot()
- def slot_diskFolderAdd(self):
- newPath = QFileDialog.getExistingDirectory(self, self.tr("New Folder"), "", QFileDialog.ShowDirsOnly)
-
- if newPath:
- if newPath[-1] == os.sep:
- newPath = newPath[:-1]
- self.ui.cb_disk.addItem(os.path.basename(newPath), newPath)
- self.ui.cb_disk.setCurrentIndex(self.ui.cb_disk.count()-1)
- self.ui.b_disk_remove.setEnabled(True)
-
- @pyqtSlot()
- def slot_diskFolderRemove(self):
- index = self.ui.cb_disk.currentIndex()
-
- if index <= 0:
- return
-
- self.ui.cb_disk.removeItem(index)
-
- if self.ui.cb_disk.currentIndex() == 0:
- self.ui.b_disk_remove.setEnabled(False)
-
- @pyqtSlot(QModelIndex)
- def slot_fileTreeDoubleClicked(self, modelIndex):
- filename = self.fDirModel.filePath(modelIndex)
-
- if not Carla.host.load_filename(filename):
- CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
- self.tr("Failed to load file"),
- cString(Carla.host.get_last_error()), QMessageBox.Ok, QMessageBox.Ok)
-
- # -----------------------------------------------------------------
-
- @pyqtSlot(int, int, int, float, str)
- def slot_handleDebugCallback(self, pluginId, value1, value2, value3, valueStr):
- print("DEBUG:", pluginId, value1, value2, value3, valueStr)
- #self.ui.pte_log.appendPlainText(valueStr.replace("[30;1m", "DEBUG: ").replace("[31m", "ERROR: ").replace("[0m", "").replace("\n", ""))
-
- @pyqtSlot(int)
- def slot_handlePluginAddedCallback(self, pluginId):
- self.fContainer.addPlugin(pluginId, self.fIsProjectLoading)
-
- if self.fContainer.getPluginCount() == 1:
- self.ui.act_plugin_remove_all.setEnabled(True)
-
- @pyqtSlot(int)
- def slot_handlePluginRemovedCallback(self, pluginId):
- self.fContainer.removePlugin(pluginId)
-
- if self.fContainer.getPluginCount() == 0:
- self.ui.act_plugin_remove_all.setEnabled(False)
-
- @pyqtSlot(int, str)
- def slot_handlePluginRenamedCallback(self, pluginId, newName):
- self.fContainer.renamePlugin(pluginId, newName)
-
- @pyqtSlot(int)
- def slot_handleBufferSizeChangedCallback(self, newBufferSize):
- self.fBufferSize = newBufferSize
- self.fContainer.engineChanged()
-
- @pyqtSlot(float)
- def slot_handleSampleRateChangedCallback(self, newSampleRate):
- self.fSampleRate = newSampleRate
- self.fContainer.engineChanged()
-
- # -----------------------------------------------------------------
-
- @pyqtSlot(str)
- def slot_handleEngineStartedCallback(self, driverName):
- self.fBufferSize = Carla.host.get_buffer_size()
- self.fSampleRate = Carla.host.get_sample_rate()
-
- if self.fIdleTimerFast == 0:
- self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
- if self.fIdleTimerSlow == 0:
- self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
-
- self.slot_engineStart(False)
-
- @pyqtSlot()
- def slot_handleEngineStoppedCallback(self):
- self.fBufferSize = 0
- self.fSampleRate = 0.0
-
- if self.fIdleTimerFast != 0:
- self.killTimer(self.fIdleTimerFast)
- self.fIdleTimerFast = 0
-
- if self.fIdleTimerSlow != 0:
- self.killTimer(self.fIdleTimerSlow)
- self.fIdleTimerSlow = 0
-
- if self.fContainer.getPluginCount() > 0:
- self.ui.act_plugin_remove_all.setEnabled(False)
- self.fContainer.removeAllPlugins() # FIXME?
-
- self.slot_engineStop(False)
-
- # -----------------------------------------------------------------
-
- @pyqtSlot()
- def slot_handleSIGUSR1(self):
- print("Got SIGUSR1 -> Saving project now")
- #QTimer.singleShot(0, self, SLOT("slot_fileSave)
-
- @pyqtSlot()
- def slot_handleSIGTERM(self):
- print("Got SIGTERM -> Closing now")
- self.close()
-
- # -----------------------------------------------------------------
-
- def timerEvent(self, event):
- if event.timerId() == self.fIdleTimerFast:
- if not Carla.isPlugin:
- Carla.host.engine_idle()
- self.refreshTransport()
-
- self.fContainer.idleFast()
-
- elif event.timerId() == self.fIdleTimerSlow:
- self.fContainer.idleSlow()
-
- QMainWindow.timerEvent(self, event)
-
- def closeEvent(self, event):
- if self.fIdleTimerFast != 0:
- self.killTimer(self.fIdleTimerFast)
- self.fIdleTimerFast = 0
-
- if self.fIdleTimerSlow != 0:
- self.killTimer(self.fIdleTimerSlow)
- self.fIdleTimerSlow = 0
-
- self.saveSettings()
-
- if Carla.host.is_engine_running():
- Carla.host.set_engine_about_to_close()
- self.ui.act_plugin_remove_all.setEnabled(False)
- self.fContainer.removeAllPlugins()
- self.stopEngine()
-
- QMainWindow.closeEvent(self, event)
-
- # ------------------------------------------------------------------------------------------------------------
- # Engine callback
-
- def EngineCallback(ptr, action, pluginId, value1, value2, value3, valueStr):
- if pluginId < 0 or not Carla.gui:
- return
-
- if action == CALLBACK_DEBUG:
- Carla.gui.DebugCallback.emit(pluginId, value1, value2, value3, cString(valueStr))
- elif action == CALLBACK_PLUGIN_ADDED:
- Carla.gui.PluginAddedCallback.emit(pluginId)
- elif action == CALLBACK_PLUGIN_REMOVED:
- Carla.gui.PluginRemovedCallback.emit(pluginId)
- elif action == CALLBACK_PLUGIN_RENAMED:
- Carla.gui.PluginRenamedCallback.emit(pluginId, valueStr)
- elif action == CALLBACK_PARAMETER_VALUE_CHANGED:
- Carla.gui.ParameterValueChangedCallback.emit(pluginId, value1, value3)
- elif action == CALLBACK_PARAMETER_DEFAULT_CHANGED:
- Carla.gui.ParameterDefaultChangedCallback.emit(pluginId, value1, value3)
- elif action == CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED:
- Carla.gui.ParameterMidiChannelChangedCallback.emit(pluginId, value1, value2)
- elif action == CALLBACK_PARAMETER_MIDI_CC_CHANGED:
- Carla.gui.ParameterMidiCcChangedCallback.emit(pluginId, value1, value2)
- elif action == CALLBACK_PROGRAM_CHANGED:
- Carla.gui.ProgramChangedCallback.emit(pluginId, value1)
- elif action == CALLBACK_MIDI_PROGRAM_CHANGED:
- Carla.gui.MidiProgramChangedCallback.emit(pluginId, value1)
- elif action == CALLBACK_NOTE_ON:
- Carla.gui.NoteOnCallback.emit(pluginId, value1, value2, value3)
- elif action == CALLBACK_NOTE_OFF:
- Carla.gui.NoteOffCallback.emit(pluginId, value1, value2)
- elif action == CALLBACK_SHOW_GUI:
- Carla.gui.ShowGuiCallback.emit(pluginId, value1)
- elif action == CALLBACK_UPDATE:
- Carla.gui.UpdateCallback.emit(pluginId)
- elif action == CALLBACK_RELOAD_INFO:
- Carla.gui.ReloadInfoCallback.emit(pluginId)
- elif action == CALLBACK_RELOAD_PARAMETERS:
- Carla.gui.ReloadParametersCallback.emit(pluginId)
- elif action == CALLBACK_RELOAD_PROGRAMS:
- Carla.gui.ReloadProgramsCallback.emit(pluginId)
- elif action == CALLBACK_RELOAD_ALL:
- Carla.gui.ReloadAllCallback.emit(pluginId)
- elif action == CALLBACK_PATCHBAY_CLIENT_ADDED:
- Carla.gui.PatchbayClientAddedCallback.emit(value1, value2, cString(valueStr))
- elif action == CALLBACK_PATCHBAY_CLIENT_REMOVED:
- Carla.gui.PatchbayClientRemovedCallback.emit(value1, cString(valueStr))
- elif action == CALLBACK_PATCHBAY_CLIENT_RENAMED:
- Carla.gui.PatchbayClientRenamedCallback.emit(value1, cString(valueStr))
- elif action == CALLBACK_PATCHBAY_PORT_ADDED:
- Carla.gui.PatchbayPortAddedCallback.emit(value1, value2, int(value3), cString(valueStr))
- elif action == CALLBACK_PATCHBAY_PORT_REMOVED:
- Carla.gui.PatchbayPortRemovedCallback.emit(value1, value2, cString(valueStr))
- elif action == CALLBACK_PATCHBAY_PORT_RENAMED:
- Carla.gui.PatchbayPortRenamedCallback.emit(value1, value2, cString(valueStr))
- elif action == CALLBACK_PATCHBAY_CONNECTION_ADDED:
- Carla.gui.PatchbayConnectionAddedCallback.emit(value1, value2, value3)
- elif action == CALLBACK_PATCHBAY_CONNECTION_REMOVED:
- Carla.gui.PatchbayConnectionRemovedCallback.emit(value1)
- elif action == CALLBACK_PATCHBAY_ICON_CHANGED:
- Carla.gui.PatchbayIconChangedCallback.emit(value1, value2)
- elif action == CALLBACK_BUFFER_SIZE_CHANGED:
- Carla.gui.BufferSizeChangedCallback.emit(value1)
- elif action == CALLBACK_SAMPLE_RATE_CHANGED:
- Carla.gui.SampleRateChangedCallback.emit(value3)
- elif action == CALLBACK_PROCESS_MODE_CHANGED:
- Carla.gui.ProcessModeChangedCallback.emit(value1)
- elif action == CALLBACK_ENGINE_STARTED:
- Carla.gui.EngineStartedCallback.emit(cString(valueStr))
- elif action == CALLBACK_ENGINE_STOPPED:
- Carla.gui.EngineStoppedChangedCallback.emit()
- elif action == CALLBACK_NSM_ANNOUNCE:
- Carla.gui.NSM_AnnounceCallback.emit(cString(valueStr))
- elif action == CALLBACK_NSM_OPEN:
- Carla.gui.NSM_OpenCallback.emit(cString(valueStr))
- elif action == CALLBACK_NSM_SAVE:
- Carla.gui.NSM_SaveCallback.emit()
- elif action == CALLBACK_ERROR:
- Carla.gui.ErrorCallback.emit(cString(valueStr))
- elif action == CALLBACK_QUIT:
- Carla.gui.QuitCallback.emit()
|