Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1089 lines
43KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla host code
  4. # Copyright (C) 2011-2013 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. from PyQt4.QtCore import qCritical, QModelIndex, QTimer
  20. from PyQt4.QtGui import QApplication, QFileSystemModel, QMainWindow, QPalette
  21. # ------------------------------------------------------------------------------------------------------------
  22. # Imports (Custom)
  23. import ui_carla_host
  24. from carla_database import *
  25. from carla_settings import *
  26. from carla_style import *
  27. from carla_widgets import *
  28. # ------------------------------------------------------------------------------------------------------------
  29. # PatchCanvas defines
  30. CANVAS_ANTIALIASING_SMALL = 1
  31. CANVAS_EYECANDY_SMALL = 1
  32. # ------------------------------------------------------------------------------------------------------------
  33. # Session Management support
  34. LADISH_APP_NAME = os.getenv("LADISH_APP_NAME")
  35. NSM_URL = os.getenv("NSM_URL")
  36. # ------------------------------------------------------------------------------------------------------------
  37. # Dummy widget
  38. class CarlaDummyW(object):
  39. def __init__(self, parent):
  40. object.__init__(self)
  41. # -----------------------------------------------------------------
  42. def getPluginCount(self):
  43. return 0
  44. # -----------------------------------------------------------------
  45. def addPlugin(self, pluginId, isProjectLoading):
  46. pass
  47. def removePlugin(self, pluginId):
  48. pass
  49. def renamePlugin(self, pluginId, newName):
  50. pass
  51. def removeAllPlugins(self):
  52. pass
  53. # -----------------------------------------------------------------
  54. def engineStarted(self):
  55. pass
  56. def engineStopped(self):
  57. pass
  58. def engineChanged(self):
  59. pass
  60. # -----------------------------------------------------------------
  61. def idleFast(self):
  62. pass
  63. def idleSlow(self):
  64. pass
  65. # -----------------------------------------------------------------
  66. def saveSettings(self, settings):
  67. pass
  68. # ------------------------------------------------------------------------------------------------------------
  69. # Host Window
  70. class HostWindow(QMainWindow):
  71. # signals
  72. DebugCallback = pyqtSignal(int, int, int, float, str)
  73. PluginAddedCallback = pyqtSignal(int, str)
  74. PluginRemovedCallback = pyqtSignal(int)
  75. PluginRenamedCallback = pyqtSignal(int, str)
  76. PluginUnavailableCallback = pyqtSignal(int, str)
  77. ParameterValueChangedCallback = pyqtSignal(int, int, float)
  78. ParameterDefaultChangedCallback = pyqtSignal(int, int, float)
  79. ParameterMidiCcChangedCallback = pyqtSignal(int, int, int)
  80. ParameterMidiChannelChangedCallback = pyqtSignal(int, int, int)
  81. ProgramChangedCallback = pyqtSignal(int, int)
  82. MidiProgramChangedCallback = pyqtSignal(int, int)
  83. UiStateChangedCallback = pyqtSignal(int, int)
  84. NoteOnCallback = pyqtSignal(int, int, int, int)
  85. NoteOffCallback = pyqtSignal(int, int, int)
  86. UpdateCallback = pyqtSignal(int)
  87. ReloadInfoCallback = pyqtSignal(int)
  88. ReloadParametersCallback = pyqtSignal(int)
  89. ReloadProgramsCallback = pyqtSignal(int)
  90. ReloadAllCallback = pyqtSignal(int)
  91. PatchbayClientAddedCallback = pyqtSignal(int, str)
  92. PatchbayClientRemovedCallback = pyqtSignal(int)
  93. PatchbayClientRenamedCallback = pyqtSignal(int, str)
  94. PatchbayClientIconChangedCallback = pyqtSignal(int, str)
  95. PatchbayPortAddedCallback = pyqtSignal(int, int, int, str)
  96. PatchbayPortRemovedCallback = pyqtSignal(int, int)
  97. PatchbayPortRenamedCallback = pyqtSignal(int, int, str)
  98. PatchbayConnectionAddedCallback = pyqtSignal(int, int, int)
  99. PatchbayConnectionRemovedCallback = pyqtSignal(int, int, int)
  100. EngineStartedCallback = pyqtSignal(int, int, str)
  101. EngineStoppedCallback = pyqtSignal()
  102. ProcessModeChangedCallback = pyqtSignal(int)
  103. TransportModeChangedCallback = pyqtSignal(int)
  104. BufferSizeChangedCallback = pyqtSignal(int)
  105. SampleRateChangedCallback = pyqtSignal(float)
  106. InfoCallback = pyqtSignal(str)
  107. ErrorCallback = pyqtSignal(str)
  108. QuitCallback = pyqtSignal()
  109. SIGTERM = pyqtSignal()
  110. SIGUSR1 = pyqtSignal()
  111. def __init__(self, parent):
  112. QMainWindow.__init__(self, parent)
  113. self.ui = ui_carla_host.Ui_CarlaHostW()
  114. self.ui.setupUi(self)
  115. if False:
  116. Carla.gui = self
  117. self.fContainer = CarlaDummyW(self)
  118. # -------------------------------------------------------------
  119. # Set callback
  120. Carla.host.set_engine_callback(EngineCallback)
  121. # -------------------------------------------------------------
  122. # Internal stuff
  123. self.fBufferSize = 0
  124. self.fSampleRate = 0.0
  125. self.fIdleTimerFast = 0
  126. self.fIdleTimerSlow = 0
  127. self.fIsProjectLoading = False
  128. self.fProjectFilename = ""
  129. self.fLadspaRdfNeedsUpdate = True
  130. self.fLadspaRdfList = []
  131. self.fLastTransportFrame = 0
  132. self.fLastTransportState = False
  133. self.fTransportText = ""
  134. self.fSavedSettings = {}
  135. if LADISH_APP_NAME:
  136. self.fClientName = LADISH_APP_NAME
  137. self.fSessionManagerName = "LADISH"
  138. elif NSM_URL:
  139. self.fClientName = "Carla.tmp"
  140. self.fSessionManagerName = "Non Session Manager"
  141. else:
  142. self.fClientName = "Carla"
  143. self.fSessionManagerName = ""
  144. # -------------------------------------------------------------
  145. # Load Settings
  146. self.loadSettings(True)
  147. # -------------------------------------------------------------
  148. # Set up GUI (engine stopped)
  149. if Carla.isPlugin:
  150. self.ui.act_file_new.setEnabled(False)
  151. self.ui.act_file_open.setEnabled(False)
  152. self.ui.act_engine_start.setEnabled(False)
  153. self.ui.menu_Engine.setEnabled(False)
  154. else:
  155. self.ui.act_engine_start.setEnabled(True)
  156. self.ui.act_file_save.setEnabled(False)
  157. self.ui.act_file_save_as.setEnabled(False)
  158. self.ui.act_engine_stop.setEnabled(False)
  159. self.ui.act_plugin_remove_all.setEnabled(False)
  160. self.ui.menu_PluginMacros.setEnabled(False)
  161. self.ui.menu_Canvas.setEnabled(False)
  162. self.setTransportMenuEnabled(False)
  163. # -------------------------------------------------------------
  164. # Set up GUI (right panel)
  165. self.fDirModel = QFileSystemModel(self)
  166. self.fDirModel.setNameFilters(Carla.host.get_supported_file_extensions().split(";"))
  167. self.fDirModel.setRootPath(HOME)
  168. self.ui.fileTreeView.setModel(self.fDirModel)
  169. self.ui.fileTreeView.setRootIndex(self.fDirModel.index(HOME))
  170. self.ui.fileTreeView.setColumnHidden(1, True)
  171. self.ui.fileTreeView.setColumnHidden(2, True)
  172. self.ui.fileTreeView.setColumnHidden(3, True)
  173. self.ui.fileTreeView.setHeaderHidden(True)
  174. self.setProperWindowTitle()
  175. # -------------------------------------------------------------
  176. # Connect actions to functions
  177. self.ui.act_file_new.triggered.connect(self.slot_fileNew)
  178. self.ui.act_file_open.triggered.connect(self.slot_fileOpen)
  179. self.ui.act_file_save.triggered.connect(self.slot_fileSave)
  180. self.ui.act_file_save_as.triggered.connect(self.slot_fileSaveAs)
  181. self.ui.act_engine_start.triggered.connect(self.slot_engineStart)
  182. self.ui.act_engine_stop.triggered.connect(self.slot_engineStop)
  183. self.ui.act_plugin_add.triggered.connect(self.slot_pluginAdd)
  184. self.ui.act_plugin_add2.triggered.connect(self.slot_pluginAdd)
  185. self.ui.act_plugin_remove_all.triggered.connect(self.slot_pluginRemoveAll)
  186. self.ui.act_transport_play.triggered.connect(self.slot_transportPlayPause)
  187. self.ui.act_transport_stop.triggered.connect(self.slot_transportStop)
  188. self.ui.act_transport_backwards.triggered.connect(self.slot_transportBackwards)
  189. self.ui.act_transport_forwards.triggered.connect(self.slot_transportForwards)
  190. self.ui.act_help_about.triggered.connect(self.slot_aboutCarla)
  191. self.ui.act_help_about_qt.triggered.connect(self.slot_aboutQt)
  192. self.ui.cb_disk.currentIndexChanged.connect(self.slot_diskFolderChanged)
  193. self.ui.b_disk_add.clicked.connect(self.slot_diskFolderAdd)
  194. self.ui.b_disk_remove.clicked.connect(self.slot_diskFolderRemove)
  195. self.ui.fileTreeView.doubleClicked.connect(self.slot_fileTreeDoubleClicked)
  196. self.DebugCallback.connect(self.slot_handleDebugCallback)
  197. self.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback)
  198. self.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback)
  199. self.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
  200. self.BufferSizeChangedCallback.connect(self.slot_handleBufferSizeChangedCallback)
  201. self.SampleRateChangedCallback.connect(self.slot_handleSampleRateChangedCallback)
  202. self.EngineStartedCallback.connect(self.slot_handleEngineStartedCallback)
  203. self.EngineStoppedChangedCallback.connect(self.slot_handleEngineStoppedCallback)
  204. #self.NSM_AnnounceCallback.connect(self.slot_handleNSM_AnnounceCallback)
  205. #self.NSM_OpenCallback.connect(self.slot_handleNSM_OpenCallback)
  206. #self.NSM_SaveCallback.connect(self.slot_handleNSM_SaveCallback)
  207. #self.ErrorCallback.connect(self.slot_handleErrorCallback)
  208. #self.QuitCallback.connect(self.slot_handleQuitCallback)
  209. self.SIGUSR1.connect(self.slot_handleSIGUSR1)
  210. self.SIGTERM.connect(self.slot_handleSIGTERM)
  211. # -------------------------------------------------------------
  212. # Final setup
  213. if Carla.isPlugin:
  214. QTimer.singleShot(0, self.slot_engineStart)
  215. #elif NSM_URL:
  216. #Carla.host.nsm_ready()
  217. # -----------------------------------------------------------------
  218. # Called by containers
  219. def openSettingsWindow(self, hasCanvas, hasCanvasGL):
  220. dialog = CarlaSettingsW(self, hasCanvas, hasCanvasGL)
  221. return dialog.exec_()
  222. def setupContainer(self, showMiniCanvas):
  223. if showMiniCanvas:
  224. CARLA_DEFAULT_CANVAS_WIDTH = 3100
  225. CARLA_DEFAULT_CANVAS_HEIGHT = 2400
  226. self.ui.miniCanvasPreview.setRealParent(self)
  227. #self.ui.miniCanvasPreview.setViewTheme(patchcanvas.canvas.theme.canvas_bg, patchcanvas.canvas.theme.rubberband_brush, patchcanvas.canvas.theme.rubberband_pen.color())
  228. self.ui.miniCanvasPreview.init(self.fContainer.scene, CARLA_DEFAULT_CANVAS_WIDTH, CARLA_DEFAULT_CANVAS_HEIGHT, self.fSavedSettings["UseCustomMiniCanvasPaint"])
  229. else:
  230. self.ui.miniCanvasPreview.hide()
  231. self.ui.mainwidget.hide()
  232. self.ui.splitter.insertWidget(0, self.fContainer)
  233. del self.ui.mainwidget
  234. # -----------------------------------------------------------------
  235. # Internal stuff (files)
  236. def loadProjectNow(self):
  237. if not self.fProjectFilename:
  238. return qCritical("ERROR: loading project without filename set")
  239. # TESTING
  240. if not Carla.host.is_engine_running():
  241. self.slot_engineStart()
  242. self.fIsProjectLoading = True
  243. Carla.host.load_project(self.fProjectFilename)
  244. self.fIsProjectLoading = False
  245. @pyqtSlot()
  246. def slot_loadProjectNow(self):
  247. self.loadProjectNow()
  248. def loadProjectLater(self, filename):
  249. self.fProjectFilename = filename
  250. self.setProperWindowTitle()
  251. QTimer.singleShot(0, self.slot_loadProjectNow)
  252. def saveProjectNow(self):
  253. if not self.fProjectFilename:
  254. return qCritical("ERROR: saving project without filename set")
  255. Carla.host.save_project(self.fProjectFilename)
  256. # -----------------------------------------------------------------
  257. # Internal stuff (engine)
  258. def setEngineSettings(self, settings = None):
  259. if Carla.isPlugin:
  260. return "Plugin"
  261. if settings is None: settings = QSettings()
  262. forceStereo = settings.value("Engine/ForceStereo", CARLA_DEFAULT_FORCE_STEREO, type=bool)
  263. preferPluginBridges = settings.value("Engine/PreferPluginBridges", CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES, type=bool)
  264. preferUiBridges = settings.value("Engine/PreferUiBridges", CARLA_DEFAULT_PREFER_UI_BRIDGES, type=bool)
  265. uisAlwaysOnTop = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UIS_ALWAYS_ON_TOP, type=bool)
  266. uiBridgesTimeout = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UI_BRIDGES_TIMEOUT, type=int)
  267. Carla.maxParameters = settings.value("Engine/MaxParameters", CARLA_DEFAULT_MAX_PARAMETERS, type=int)
  268. audioDriver = settings.value("Engine/AudioDriver", CARLA_DEFAULT_AUDIO_DRIVER, type=str)
  269. if audioDriver == "JACK":
  270. #transportMode = settings.value("Engine/TransportMode", TRANSPORT_MODE_JACK, type=int)
  271. transportMode = ENGINE_TRANSPORT_MODE_JACK
  272. else:
  273. transportMode = ENGINE_TRANSPORT_MODE_INTERNAL
  274. audioNumPeriods = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_NUM_PERIODS, type=int)
  275. audioBufferSize = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_BUFFER_SIZE, type=int)
  276. audioSampleRate = settings.value("Engine/AudioSampleRate", CARLA_DEFAULT_AUDIO_SAMPLE_RATE, type=int)
  277. audioDevice = settings.value("Engine/AudioDevice", "", type=str)
  278. Carla.host.set_engine_option(ENGINE_OPTION_AUDIO_NUM_PERIODS, audioNumPeriods, "")
  279. Carla.host.set_engine_option(ENGINE_OPTION_AUDIO_BUFFER_SIZE, audioBufferSize, "")
  280. Carla.host.set_engine_option(ENGINE_OPTION_AUDIO_SAMPLE_RATE, audioSampleRate, "")
  281. Carla.host.set_engine_option(ENGINE_OPTION_AUDIO_DEVICE, 0, audioDevice)
  282. if Carla.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK:
  283. forceStereo = True
  284. elif Carla.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS and LADISH_APP_NAME:
  285. print("LADISH detected but using multiple clients (not allowed), forcing single client now")
  286. Carla.processMode = ENGINE_PROCESS_MODE_SINGLE_CLIENT
  287. Carla.host.set_engine_option(ENGINE_OPTION_FORCE_STEREO, forceStereo, "")
  288. Carla.host.set_engine_option(ENGINE_OPTION_PREFER_PLUGIN_BRIDGES, preferPluginBridges, "")
  289. Carla.host.set_engine_option(ENGINE_OPTION_PREFER_UI_BRIDGES, preferUiBridges, "")
  290. Carla.host.set_engine_option(ENGINE_OPTION_UIS_ALWAYS_ON_TOP, uisAlwaysOnTop, "")
  291. Carla.host.set_engine_option(ENGINE_OPTION_UI_BRIDGES_TIMEOUT, uiBridgesTimeout, "")
  292. Carla.host.set_engine_option(ENGINE_OPTION_PROCESS_MODE, Carla.processMode, "")
  293. Carla.host.set_engine_option(ENGINE_OPTION_MAX_PARAMETERS, Carla.maxParameters, "")
  294. Carla.host.set_engine_option(ENGINE_OPTION_TRANSPORT_MODE, transportMode, "")
  295. return audioDriver
  296. def startEngine(self):
  297. audioDriver = self.setEngineSettings()
  298. if not Carla.host.engine_init(audioDriver, self.fClientName):
  299. #if self.fFirstEngineInit:
  300. #self.fFirstEngineInit = False
  301. #return
  302. audioError = Carla.host.get_last_error()
  303. if audioError:
  304. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s', possible reasons:\n%s" % (audioDriver, audioError)))
  305. else:
  306. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s'" % audioDriver))
  307. return
  308. self.fBufferSize = Carla.host.get_buffer_size()
  309. self.fSampleRate = Carla.host.get_sample_rate()
  310. #self.fFirstEngineInit = False
  311. # Peaks and TimeInfo
  312. self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
  313. # LEDs and edit dialog parameters
  314. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  315. def stopEngine(self):
  316. if self.fContainer.getPluginCount() > 0:
  317. 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"
  318. "Do you want to do this now?"),
  319. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  320. if ask != QMessageBox.Yes:
  321. return
  322. self.ui.act_plugin_remove_all.setEnabled(False)
  323. self.fContainer.removeAllPlugins()
  324. if Carla.host.is_engine_running() and not Carla.host.engine_close():
  325. print(Carla.host.get_last_error())
  326. self.fBufferSize = 0
  327. self.fSampleRate = 0.0
  328. if self.fIdleTimerFast != 0:
  329. self.killTimer(self.fIdleTimerFast)
  330. self.fIdleTimerFast = 0
  331. if self.fIdleTimerSlow != 0:
  332. self.killTimer(self.fIdleTimerSlow)
  333. self.fIdleTimerSlow = 0
  334. # -----------------------------------------------------------------
  335. # Internal stuff (plugins)
  336. def getExtraPtr(self, plugin):
  337. ptype = plugin['type']
  338. if ptype == PLUGIN_LADSPA:
  339. uniqueId = plugin['uniqueId']
  340. self.maybeLoadRDFs()
  341. for rdfItem in self.fLadspaRdfList:
  342. if rdfItem.UniqueID == uniqueId:
  343. return pointer(rdfItem)
  344. elif ptype in (PLUGIN_GIG, PLUGIN_SF2, PLUGIN_SFZ):
  345. if plugin['name'].lower().endswith(" (16 outputs)"):
  346. # return a dummy non-null pointer
  347. INTPOINTER = POINTER(c_int)
  348. int1 = c_int(0x1)
  349. addr = addressof(int1)
  350. return cast(addr, INTPOINTER)
  351. return None
  352. def maybeLoadRDFs(self):
  353. if not self.fLadspaRdfNeedsUpdate:
  354. return
  355. self.fLadspaRdfNeedsUpdate = False
  356. self.fLadspaRdfList = []
  357. if not haveLRDF:
  358. return
  359. settingsDir = os.path.join(HOME, ".config", "falkTX")
  360. frLadspaFile = os.path.join(settingsDir, "ladspa_rdf.db")
  361. if os.path.exists(frLadspaFile):
  362. frLadspa = open(frLadspaFile, 'r')
  363. try:
  364. self.fLadspaRdfList = ladspa_rdf.get_c_ladspa_rdfs(json.load(frLadspa))
  365. except:
  366. pass
  367. frLadspa.close()
  368. def setLoadRDFsNeeded(self):
  369. self.fLadspaRdfNeedsUpdate = True
  370. # -----------------------------------------------------------------
  371. # Internal stuff (transport)
  372. def refreshTransport(self, forced = False):
  373. if not Carla.host.is_engine_running():
  374. return
  375. if self.fSampleRate == 0.0:
  376. return
  377. timeInfo = Carla.host.get_transport_info()
  378. playing = bool(timeInfo['playing'])
  379. frame = int(timeInfo['frame'])
  380. if playing != self.fLastTransportState or forced:
  381. if playing:
  382. icon = getIcon("media-playback-pause")
  383. self.ui.act_transport_play.setChecked(True)
  384. self.ui.act_transport_play.setIcon(icon)
  385. self.ui.act_transport_play.setText(self.tr("&Pause"))
  386. else:
  387. icon = getIcon("media-playback-start")
  388. self.ui.act_transport_play.setChecked(False)
  389. self.ui.act_transport_play.setIcon(icon)
  390. self.ui.act_transport_play.setText(self.tr("&Play"))
  391. self.fLastTransportState = playing
  392. if frame != self.fLastTransportFrame or forced:
  393. time = frame / self.fSampleRate
  394. secs = time % 60
  395. mins = (time / 60) % 60
  396. hrs = (time / 3600) % 60
  397. self.fTextTransport = "Transport %s, at %02i:%02i:%02i" % ("playing" if playing else "stopped", hrs, mins, secs)
  398. self.fLastTransportFrame = frame
  399. def setTransportMenuEnabled(self, enabled):
  400. self.ui.act_transport_play.setEnabled(enabled)
  401. self.ui.act_transport_stop.setEnabled(enabled)
  402. self.ui.act_transport_backwards.setEnabled(enabled)
  403. self.ui.act_transport_forwards.setEnabled(enabled)
  404. self.ui.menu_Transport.setEnabled(enabled)
  405. # -----------------------------------------------------------------
  406. # Internal stuff (settings)
  407. def loadSettings(self, firstTime):
  408. settings = QSettings()
  409. if firstTime:
  410. self.restoreGeometry(settings.value("Geometry", ""))
  411. showToolbar = settings.value("ShowToolbar", True, type=bool)
  412. self.ui.act_settings_show_toolbar.setChecked(showToolbar)
  413. self.ui.toolBar.setVisible(showToolbar)
  414. #if settings.contains("SplitterState"):
  415. #self.ui.splitter.restoreState(settings.value("SplitterState", ""))
  416. #else:
  417. self.ui.splitter.setSizes([99999, 210])
  418. diskFolders = toList(settings.value("DiskFolders", [HOME]))
  419. self.ui.cb_disk.setItemData(0, HOME)
  420. for i in range(len(diskFolders)):
  421. if i == 0: continue
  422. folder = diskFolders[i]
  423. self.ui.cb_disk.addItem(os.path.basename(folder), folder)
  424. if MACOS and not settings.value("Main/UseProTheme", True, type=bool):
  425. self.setUnifiedTitleAndToolBarOnMac(True)
  426. # ---------------------------------------------
  427. # plugin checks
  428. if settings.value("Engine/DisableChecks", False, type=bool):
  429. os.environ["CARLA_DISCOVERY_NO_PROCESSING_CHECKS"] = "true"
  430. elif os.getenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS"):
  431. os.environ.pop("CARLA_DISCOVERY_NO_PROCESSING_CHECKS")
  432. # ---------------------------------------------
  433. if not Carla.isPlugin:
  434. # engine
  435. self.setEngineSettings(settings)
  436. # plugin paths
  437. LADSPA_PATH = toList(settings.value("Paths/LADSPA", Carla.DEFAULT_LADSPA_PATH))
  438. DSSI_PATH = toList(settings.value("Paths/DSSI", Carla.DEFAULT_DSSI_PATH))
  439. LV2_PATH = toList(settings.value("Paths/LV2", Carla.DEFAULT_LV2_PATH))
  440. VST_PATH = toList(settings.value("Paths/VST", Carla.DEFAULT_VST_PATH))
  441. AU_PATH = toList(settings.value("Paths/AU", Carla.DEFAULT_AU_PATH))
  442. CSOUND_PATH = toList(settings.value("Paths/CSOUND", Carla.DEFAULT_CSOUND_PATH))
  443. GIG_PATH = toList(settings.value("Paths/GIG", Carla.DEFAULT_GIG_PATH))
  444. SF2_PATH = toList(settings.value("Paths/SF2", Carla.DEFAULT_SF2_PATH))
  445. SFZ_PATH = toList(settings.value("Paths/SFZ", Carla.DEFAULT_SFZ_PATH))
  446. os.environ["LADSPA_PATH"] = splitter.join(LADSPA_PATH)
  447. os.environ["DSSI_PATH"] = splitter.join(DSSI_PATH)
  448. os.environ["LV2_PATH"] = splitter.join(LV2_PATH)
  449. os.environ["VST_PATH"] = splitter.join(VST_PATH)
  450. os.environ["AU_PATH"] = splitter.join(AU_PATH)
  451. os.environ["CSOUND_PATH"] = splitter.join(CSOUND_PATH)
  452. os.environ["GIG_PATH"] = splitter.join(GIG_PATH)
  453. os.environ["SF2_PATH"] = splitter.join(SF2_PATH)
  454. os.environ["SFZ_PATH"] = splitter.join(SFZ_PATH)
  455. # ---------------------------------------------
  456. # TODO
  457. self.fSavedSettings = {
  458. "Main/DefaultProjectFolder": settings.value("Main/DefaultProjectFolder", HOME, type=str),
  459. "Main/RefreshInterval": settings.value("Main/RefreshInterval", 50, type=int),
  460. CARLA_KEY_CANVAS_THEME: settings.value(CARLA_KEY_CANVAS_THEME, CARLA_DEFAULT_CANVAS_THEME, type=str),
  461. CARLA_KEY_CANVAS_SIZE: settings.value(CARLA_KEY_CANVAS_SIZE, CARLA_DEFAULT_CANVAS_SIZE, type=str),
  462. CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS: settings.value(CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS, CARLA_DEFAULT_CANVAS_AUTO_HIDE_GROUPS, type=bool),
  463. CARLA_KEY_CANVAS_USE_BEZIER_LINES: settings.value(CARLA_KEY_CANVAS_USE_BEZIER_LINES, CARLA_DEFAULT_CANVAS_USE_BEZIER_LINES, type=bool),
  464. CARLA_KEY_CANVAS_EYE_CANDY: settings.value(CARLA_KEY_CANVAS_EYE_CANDY, CARLA_DEFAULT_CANVAS_EYE_CANDY, type=int),
  465. CARLA_KEY_CANVAS_USE_OPENGL: settings.value(CARLA_KEY_CANVAS_USE_OPENGL, CARLA_DEFAULT_CANVAS_USE_OPENGL, type=bool),
  466. CARLA_KEY_CANVAS_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_ANTIALIASING, CARLA_DEFAULT_CANVAS_ANTIALIASING, type=int),
  467. CARLA_KEY_CANVAS_HQ_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_HQ_ANTIALIASING, CARLA_DEFAULT_CANVAS_HQ_ANTIALIASING, type=bool),
  468. "UseCustomMiniCanvasPaint": (settings.value("Main/UseProTheme", True, type=bool) and
  469. settings.value("Main/ProThemeColor", "Black", type=str).lower() == "black")
  470. }
  471. # ---------------------------------------------
  472. if self.fIdleTimerFast != 0:
  473. self.killTimer(self.fIdleTimerFast)
  474. self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
  475. if self.fIdleTimerSlow != 0:
  476. self.killTimer(self.fIdleTimerSlow)
  477. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  478. def saveSettings(self):
  479. settings = QSettings()
  480. settings.setValue("Geometry", self.saveGeometry())
  481. settings.setValue("SplitterState", self.ui.splitter.saveState())
  482. settings.setValue("ShowToolbar", self.ui.toolBar.isVisible())
  483. #settings.setValue("HorizontalScrollBarValue", self.ui.graphicsView.horizontalScrollBar().value())
  484. #settings.setValue("VerticalScrollBarValue", self.ui.graphicsView.verticalScrollBar().value())
  485. diskFolders = []
  486. for i in range(self.ui.cb_disk.count()):
  487. diskFolders.append(self.ui.cb_disk.itemData(i))
  488. settings.setValue("DiskFolders", diskFolders)
  489. self.fContainer.saveSettings(settings)
  490. # -----------------------------------------------------------------
  491. # Internal stuff (gui)
  492. def setProperWindowTitle(self):
  493. title = self.fClientName
  494. if self.fProjectFilename:
  495. title += " - %s" % os.path.basename(self.fProjectFilename)
  496. if self.fSessionManagerName:
  497. title += " (%s)" % self.fSessionManagerName
  498. self.setWindowTitle(title)
  499. # -----------------------------------------------------------------
  500. @pyqtSlot()
  501. def slot_fileNew(self):
  502. self.fContainer.removeAllPlugins()
  503. self.fProjectFilename = ""
  504. self.setProperWindowTitle()
  505. @pyqtSlot()
  506. def slot_fileOpen(self):
  507. fileFilter = self.tr("Carla Project File (*.carxp)")
  508. filenameTry = QFileDialog.getOpenFileName(self, self.tr("Open Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
  509. if not filenameTry:
  510. return
  511. newFile = True
  512. if self.fContainer.getPluginCount() > 0:
  513. ask = QMessageBox.question(self, self.tr("Question"), self.tr("There are some plugins loaded, do you want to remove them now?"),
  514. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  515. newFile = ask == QMessageBox.Yes
  516. if newFile:
  517. self.fContainer.removeAllPlugins()
  518. self.fProjectFilename = filenameTry
  519. self.setProperWindowTitle()
  520. self.loadProjectNow()
  521. else:
  522. filenameOld = self.fProjectFilename
  523. self.fProjectFilename = filenameTry
  524. self.loadProjectNow()
  525. self.fProjectFilename = filenameOld
  526. @pyqtSlot()
  527. def slot_fileSave(self, saveAs=False):
  528. if self.fProjectFilename and not saveAs:
  529. return self.saveProjectNow()
  530. fileFilter = self.tr("Carla Project File (*.carxp)")
  531. filenameTry = QFileDialog.getSaveFileName(self, self.tr("Save Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
  532. if not filenameTry:
  533. return
  534. if not filenameTry.endswith(".carxp"):
  535. filenameTry += ".carxp"
  536. if self.fProjectFilename != filenameTry:
  537. self.fProjectFilename = filenameTry
  538. self.setProperWindowTitle()
  539. self.saveProjectNow()
  540. @pyqtSlot()
  541. def slot_fileSaveAs(self):
  542. self.slot_fileSave(True)
  543. # -----------------------------------------------------------------
  544. @pyqtSlot()
  545. def slot_engineStart(self, doStart = True):
  546. if doStart: self.startEngine()
  547. check = Carla.host.is_engine_running()
  548. self.ui.menu_PluginMacros.setEnabled(check)
  549. self.ui.menu_Canvas.setEnabled(check)
  550. if not Carla.isPlugin:
  551. self.ui.act_file_save.setEnabled(check)
  552. self.ui.act_engine_start.setEnabled(not check)
  553. self.ui.act_engine_stop.setEnabled(check)
  554. if self.fSessionManagerName != "Non Session Manager":
  555. self.ui.act_file_open.setEnabled(check)
  556. self.ui.act_file_save_as.setEnabled(check)
  557. self.setTransportMenuEnabled(check)
  558. if check:
  559. if not Carla.isPlugin:
  560. self.refreshTransport(True)
  561. self.fContainer.engineStarted()
  562. @pyqtSlot()
  563. def slot_engineStop(self, doStop = True):
  564. if doStop: self.stopEngine()
  565. check = Carla.host.is_engine_running()
  566. self.ui.menu_PluginMacros.setEnabled(check)
  567. self.ui.menu_Canvas.setEnabled(check)
  568. if not Carla.isPlugin:
  569. self.ui.act_file_save.setEnabled(check)
  570. self.ui.act_engine_start.setEnabled(not check)
  571. self.ui.act_engine_stop.setEnabled(check)
  572. if self.fSessionManagerName != "Non Session Manager":
  573. self.ui.act_file_open.setEnabled(check)
  574. self.ui.act_file_save_as.setEnabled(check)
  575. self.setTransportMenuEnabled(check)
  576. if not check:
  577. self.fTextTransport = ""
  578. self.fContainer.engineStopped()
  579. # -----------------------------------------------------------------
  580. @pyqtSlot()
  581. def slot_pluginAdd(self):
  582. dialog = PluginDatabaseW(self)
  583. if not dialog.exec_():
  584. return
  585. if not Carla.host.is_engine_running():
  586. QMessageBox.warning(self, self.tr("Warning"), self.tr("Cannot add new plugins while engine is stopped"))
  587. return
  588. btype = dialog.fRetPlugin['build']
  589. ptype = dialog.fRetPlugin['type']
  590. filename = dialog.fRetPlugin['binary']
  591. label = dialog.fRetPlugin['label']
  592. extraPtr = self.getExtraPtr(dialog.fRetPlugin)
  593. if not Carla.host.add_plugin(btype, ptype, filename, None, label, extraPtr):
  594. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"), self.tr("Failed to load plugin"), charPtrToString(Carla.host.get_last_error()), QMessageBox.Ok, QMessageBox.Ok)
  595. return
  596. @pyqtSlot()
  597. def slot_pluginRemoveAll(self):
  598. self.ui.act_plugin_remove_all.setEnabled(False)
  599. self.fContainer.removeAllPlugins()
  600. Carla.host.remove_all_plugins()
  601. # -----------------------------------------------------------------
  602. @pyqtSlot(bool)
  603. def slot_transportPlayPause(self, toggled):
  604. if not Carla.host.is_engine_running():
  605. return
  606. if toggled:
  607. Carla.host.transport_play()
  608. else:
  609. Carla.host.transport_pause()
  610. self.refreshTransport()
  611. @pyqtSlot()
  612. def slot_transportStop(self):
  613. if not Carla.host.is_engine_running():
  614. return
  615. Carla.host.transport_pause()
  616. Carla.host.transport_relocate(0)
  617. self.refreshTransport()
  618. @pyqtSlot()
  619. def slot_transportBackwards(self):
  620. if not Carla.host.is_engine_running():
  621. return
  622. newFrame = Carla.host.get_current_transport_frame() - 100000
  623. if newFrame < 0:
  624. newFrame = 0
  625. Carla.host.transport_relocate(newFrame)
  626. @pyqtSlot()
  627. def slot_transportForwards(self):
  628. if not Carla.host.is_engine_running():
  629. return
  630. newFrame = Carla.host.get_current_transport_frame() + 100000
  631. Carla.host.transport_relocate(newFrame)
  632. # -----------------------------------------------------------------
  633. @pyqtSlot()
  634. def slot_aboutCarla(self):
  635. CarlaAboutW(self).exec_()
  636. @pyqtSlot()
  637. def slot_aboutQt(self):
  638. QApplication.instance().aboutQt()
  639. # -----------------------------------------------------------------
  640. @pyqtSlot(int)
  641. def slot_diskFolderChanged(self, index):
  642. if index < 0:
  643. return
  644. elif index == 0:
  645. filename = HOME
  646. self.ui.b_disk_remove.setEnabled(False)
  647. else:
  648. filename = self.ui.cb_disk.itemData(index)
  649. self.ui.b_disk_remove.setEnabled(True)
  650. self.fDirModel.setRootPath(filename)
  651. self.ui.fileTreeView.setRootIndex(self.fDirModel.index(filename))
  652. @pyqtSlot()
  653. def slot_diskFolderAdd(self):
  654. newPath = QFileDialog.getExistingDirectory(self, self.tr("New Folder"), "", QFileDialog.ShowDirsOnly)
  655. if newPath:
  656. if newPath[-1] == os.sep:
  657. newPath = newPath[:-1]
  658. self.ui.cb_disk.addItem(os.path.basename(newPath), newPath)
  659. self.ui.cb_disk.setCurrentIndex(self.ui.cb_disk.count()-1)
  660. self.ui.b_disk_remove.setEnabled(True)
  661. @pyqtSlot()
  662. def slot_diskFolderRemove(self):
  663. index = self.ui.cb_disk.currentIndex()
  664. if index <= 0:
  665. return
  666. self.ui.cb_disk.removeItem(index)
  667. if self.ui.cb_disk.currentIndex() == 0:
  668. self.ui.b_disk_remove.setEnabled(False)
  669. @pyqtSlot(QModelIndex)
  670. def slot_fileTreeDoubleClicked(self, modelIndex):
  671. filename = self.fDirModel.filePath(modelIndex)
  672. if not Carla.host.load_filename(filename):
  673. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
  674. self.tr("Failed to load file"),
  675. Carla.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  676. # -----------------------------------------------------------------
  677. @pyqtSlot(int, int, int, float, str)
  678. def slot_handleDebugCallback(self, pluginId, value1, value2, value3, valueStr):
  679. print("DEBUG:", pluginId, value1, value2, value3, valueStr)
  680. #self.ui.pte_log.appendPlainText(valueStr.replace("", "DEBUG: ").replace("", "ERROR: ").replace("", "").replace("\n", ""))
  681. @pyqtSlot(int)
  682. def slot_handlePluginAddedCallback(self, pluginId):
  683. self.fContainer.addPlugin(pluginId, self.fIsProjectLoading)
  684. if self.fContainer.getPluginCount() == 1:
  685. self.ui.act_plugin_remove_all.setEnabled(True)
  686. @pyqtSlot(int)
  687. def slot_handlePluginRemovedCallback(self, pluginId):
  688. self.fContainer.removePlugin(pluginId)
  689. if self.fContainer.getPluginCount() == 0:
  690. self.ui.act_plugin_remove_all.setEnabled(False)
  691. @pyqtSlot(int, str)
  692. def slot_handlePluginRenamedCallback(self, pluginId, newName):
  693. self.fContainer.renamePlugin(pluginId, newName)
  694. @pyqtSlot(int)
  695. def slot_handleBufferSizeChangedCallback(self, newBufferSize):
  696. self.fBufferSize = newBufferSize
  697. self.fContainer.engineChanged()
  698. @pyqtSlot(float)
  699. def slot_handleSampleRateChangedCallback(self, newSampleRate):
  700. self.fSampleRate = newSampleRate
  701. self.fContainer.engineChanged()
  702. # -----------------------------------------------------------------
  703. @pyqtSlot(str)
  704. def slot_handleEngineStartedCallback(self, driverName):
  705. self.fBufferSize = Carla.host.get_buffer_size()
  706. self.fSampleRate = Carla.host.get_sample_rate()
  707. if self.fIdleTimerFast == 0:
  708. self.fIdleTimerFast = self.startTimer(30) #self.fSavedSettings["Main/RefreshInterval"])
  709. if self.fIdleTimerSlow == 0:
  710. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  711. self.slot_engineStart(False)
  712. @pyqtSlot()
  713. def slot_handleEngineStoppedCallback(self):
  714. self.fBufferSize = 0
  715. self.fSampleRate = 0.0
  716. if self.fIdleTimerFast != 0:
  717. self.killTimer(self.fIdleTimerFast)
  718. self.fIdleTimerFast = 0
  719. if self.fIdleTimerSlow != 0:
  720. self.killTimer(self.fIdleTimerSlow)
  721. self.fIdleTimerSlow = 0
  722. if self.fContainer.getPluginCount() > 0:
  723. self.ui.act_plugin_remove_all.setEnabled(False)
  724. self.fContainer.removeAllPlugins() # FIXME?
  725. self.slot_engineStop(False)
  726. # -----------------------------------------------------------------
  727. @pyqtSlot()
  728. def slot_handleSIGUSR1(self):
  729. print("Got SIGUSR1 -> Saving project now")
  730. #QTimer.singleShot(0, self, SLOT("slot_fileSave)
  731. @pyqtSlot()
  732. def slot_handleSIGTERM(self):
  733. print("Got SIGTERM -> Closing now")
  734. self.close()
  735. # -----------------------------------------------------------------
  736. def timerEvent(self, event):
  737. if event.timerId() == self.fIdleTimerFast:
  738. if not Carla.isPlugin:
  739. Carla.host.engine_idle()
  740. self.refreshTransport()
  741. self.fContainer.idleFast()
  742. elif event.timerId() == self.fIdleTimerSlow:
  743. self.fContainer.idleSlow()
  744. QMainWindow.timerEvent(self, event)
  745. def closeEvent(self, event):
  746. if self.fIdleTimerFast != 0:
  747. self.killTimer(self.fIdleTimerFast)
  748. self.fIdleTimerFast = 0
  749. if self.fIdleTimerSlow != 0:
  750. self.killTimer(self.fIdleTimerSlow)
  751. self.fIdleTimerSlow = 0
  752. self.saveSettings()
  753. if Carla.host.is_engine_running():
  754. Carla.host.set_engine_about_to_close()
  755. self.ui.act_plugin_remove_all.setEnabled(False)
  756. self.fContainer.removeAllPlugins()
  757. self.stopEngine()
  758. QMainWindow.closeEvent(self, event)
  759. # ------------------------------------------------------------------------------------------------------------
  760. # Engine callback
  761. def EngineCallback(ptr, action, pluginId, value1, value2, value3, valueStr):
  762. if pluginId < 0 or not Carla.gui:
  763. return
  764. if action == ENGINE_CALLBACK_DEBUG:
  765. Carla.gui.DebugCallback.emit(pluginId, value1, value2, value3, charPtrToString(valueStr))
  766. elif action == ENGINE_CALLBACK_PLUGIN_ADDED:
  767. Carla.gui.PluginAddedCallback.emit(pluginId)
  768. elif action == ENGINE_CALLBACK_PLUGIN_REMOVED:
  769. Carla.gui.PluginRemovedCallback.emit(pluginId)
  770. elif action == ENGINE_CALLBACK_PLUGIN_RENAMED:
  771. Carla.gui.PluginRenamedCallback.emit(pluginId, valueStr)
  772. elif action == ENGINE_CALLBACK_PARAMETER_VALUE_CHANGED:
  773. Carla.gui.ParameterValueChangedCallback.emit(pluginId, value1, value3)
  774. elif action == ENGINE_CALLBACK_PARAMETER_DEFAULT_CHANGED:
  775. Carla.gui.ParameterDefaultChangedCallback.emit(pluginId, value1, value3)
  776. elif action == ENGINE_CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED:
  777. Carla.gui.ParameterMidiChannelChangedCallback.emit(pluginId, value1, value2)
  778. elif action == ENGINE_CALLBACK_PARAMETER_MIDI_CC_CHANGED:
  779. Carla.gui.ParameterMidiCcChangedCallback.emit(pluginId, value1, value2)
  780. elif action == ENGINE_CALLBACK_PROGRAM_CHANGED:
  781. Carla.gui.ProgramChangedCallback.emit(pluginId, value1)
  782. elif action == ENGINE_CALLBACK_MIDI_PROGRAM_CHANGED:
  783. Carla.gui.MidiProgramChangedCallback.emit(pluginId, value1)
  784. elif action == ENGINE_CALLBACK_NOTE_ON:
  785. Carla.gui.NoteOnCallback.emit(pluginId, value1, value2, value3)
  786. elif action == ENGINE_CALLBACK_NOTE_OFF:
  787. Carla.gui.NoteOffCallback.emit(pluginId, value1, value2)
  788. elif action == ENGINE_CALLBACK_UI_STATE_CHANGED:
  789. Carla.gui.ShowGuiCallback.emit(pluginId, value1)
  790. elif action == ENGINE_CALLBACK_UPDATE:
  791. Carla.gui.UpdateCallback.emit(pluginId)
  792. elif action == ENGINE_CALLBACK_RELOAD_INFO:
  793. Carla.gui.ReloadInfoCallback.emit(pluginId)
  794. elif action == ENGINE_CALLBACK_RELOAD_PARAMETERS:
  795. Carla.gui.ReloadParametersCallback.emit(pluginId)
  796. elif action == ENGINE_CALLBACK_RELOAD_PROGRAMS:
  797. Carla.gui.ReloadProgramsCallback.emit(pluginId)
  798. elif action == ENGINE_CALLBACK_RELOAD_ALL:
  799. Carla.gui.ReloadAllCallback.emit(pluginId)
  800. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED:
  801. Carla.gui.PatchbayClientAddedCallback.emit(value1, charPtrToString(valueStr))
  802. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_REMOVED:
  803. Carla.gui.PatchbayClientRemovedCallback.emit(value1, charPtrToString(valueStr))
  804. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_RENAMED:
  805. Carla.gui.PatchbayClientRenamedCallback.emit(value1, charPtrToString(valueStr))
  806. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_ADDED:
  807. Carla.gui.PatchbayPortAddedCallback.emit(value1, value2, int(value3), charPtrToString(valueStr))
  808. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_REMOVED:
  809. Carla.gui.PatchbayPortRemovedCallback.emit(value1, value2, charPtrToString(valueStr))
  810. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_RENAMED:
  811. Carla.gui.PatchbayPortRenamedCallback.emit(value1, value2, charPtrToString(valueStr))
  812. elif action == ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED:
  813. Carla.gui.PatchbayConnectionAddedCallback.emit(value1, value2, value3)
  814. elif action == ENGINE_CALLBACK_PATCHBAY_CONNECTION_REMOVED:
  815. Carla.gui.PatchbayConnectionRemovedCallback.emit(value1)
  816. #elif action == ENGINE_CALLBACK_PATCHBAY_ICON_CHANGED:
  817. #Carla.gui.PatchbayIconChangedCallback.emit(value1, charPtrToString(valueStr))
  818. elif action == ENGINE_CALLBACK_BUFFER_SIZE_CHANGED:
  819. Carla.gui.BufferSizeChangedCallback.emit(value1)
  820. elif action == ENGINE_CALLBACK_SAMPLE_RATE_CHANGED:
  821. Carla.gui.SampleRateChangedCallback.emit(value3)
  822. elif action == ENGINE_CALLBACK_PROCESS_MODE_CHANGED:
  823. Carla.gui.ProcessModeChangedCallback.emit(value1)
  824. elif action == ENGINE_CALLBACK_ENGINE_STARTED:
  825. Carla.gui.EngineStartedCallback.emit(charPtrToString(valueStr))
  826. elif action == ENGINE_CALLBACK_ENGINE_STOPPED:
  827. Carla.gui.EngineStoppedChangedCallback.emit()
  828. #elif action == CALLBACK_NSM_ANNOUNCE:
  829. #Carla.gui.NSM_AnnounceCallback.emit(charPtrToString(valueStr))
  830. #elif action == CALLBACK_NSM_OPEN:
  831. #Carla.gui.NSM_OpenCallback.emit(charPtrToString(valueStr))
  832. #elif action == CALLBACK_NSM_SAVE:
  833. #Carla.gui.NSM_SaveCallback.emit()
  834. elif action == ENGINE_CALLBACK_ERROR:
  835. Carla.gui.ErrorCallback.emit(charPtrToString(valueStr))
  836. elif action == ENGINE_CALLBACK_QUIT:
  837. Carla.gui.QuitCallback.emit()