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.

1020 lines
40KB

  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. try:
  20. from PyQt5.QtCore import QTimer
  21. from PyQt5.QtWidgets import QApplication, QMainWindow
  22. except:
  23. from PyQt4.QtCore import QTimer
  24. from PyQt4.QtGui import QApplication, QMainWindow
  25. # ------------------------------------------------------------------------------------------------------------
  26. # Imports (Custom)
  27. import ui_carla_host
  28. from carla_database import *
  29. from carla_settings import *
  30. from carla_widgets import *
  31. # ------------------------------------------------------------------------------------------------------------
  32. # PatchCanvas defines
  33. CANVAS_ANTIALIASING_SMALL = 1
  34. CANVAS_EYECANDY_SMALL = 1
  35. CANVAS_DEFAULT_THEME_NAME = "Modern Dark"
  36. # ------------------------------------------------------------------------------------------------------------
  37. # Session Management support
  38. LADISH_APP_NAME = os.getenv("LADISH_APP_NAME")
  39. NSM_URL = os.getenv("NSM_URL")
  40. # ------------------------------------------------------------------------------------------------------------
  41. # Dummy widget
  42. class CarlaDummyW(object):
  43. def __init__(self, parent):
  44. object.__init__(self)
  45. # -----------------------------------------------------------------
  46. def getPluginCount(self):
  47. return 0
  48. # -----------------------------------------------------------------
  49. def addPlugin(self, pluginId, isProjectLoading):
  50. pass
  51. def removePlugin(self, pluginId):
  52. pass
  53. def renamePlugin(self, pluginId, newName):
  54. pass
  55. def removeAllPlugins(self):
  56. pass
  57. # -----------------------------------------------------------------
  58. def engineStarted(self):
  59. pass
  60. def engineStopped(self):
  61. pass
  62. # -----------------------------------------------------------------
  63. def idleFast(self):
  64. pass
  65. def idleSlow(self):
  66. pass
  67. # -----------------------------------------------------------------
  68. def saveSettings(self, settings):
  69. pass
  70. # ------------------------------------------------------------------------------------------------------------
  71. # Host Window
  72. class HostWindow(QMainWindow):
  73. # signals
  74. DebugCallback = pyqtSignal(int, int, int, float, str)
  75. PluginAddedCallback = pyqtSignal(int)
  76. PluginRemovedCallback = pyqtSignal(int)
  77. PluginRenamedCallback = pyqtSignal(int, str)
  78. ParameterValueChangedCallback = pyqtSignal(int, int, float)
  79. ParameterDefaultChangedCallback = pyqtSignal(int, int, float)
  80. ParameterMidiChannelChangedCallback = pyqtSignal(int, int, int)
  81. ParameterMidiCcChangedCallback = pyqtSignal(int, int, int)
  82. ProgramChangedCallback = pyqtSignal(int, int)
  83. MidiProgramChangedCallback = pyqtSignal(int, int)
  84. NoteOnCallback = pyqtSignal(int, int, int, int)
  85. NoteOffCallback = pyqtSignal(int, int, int)
  86. ShowGuiCallback = pyqtSignal(int, int)
  87. UpdateCallback = pyqtSignal(int)
  88. ReloadInfoCallback = pyqtSignal(int)
  89. ReloadParametersCallback = pyqtSignal(int)
  90. ReloadProgramsCallback = pyqtSignal(int)
  91. ReloadAllCallback = pyqtSignal(int)
  92. PatchbayClientAddedCallback = pyqtSignal(int, int, str)
  93. PatchbayClientRemovedCallback = pyqtSignal(int, str)
  94. PatchbayClientRenamedCallback = pyqtSignal(int, str)
  95. PatchbayPortAddedCallback = pyqtSignal(int, int, int, str)
  96. PatchbayPortRemovedCallback = pyqtSignal(int, int, str)
  97. PatchbayPortRenamedCallback = pyqtSignal(int, int, str)
  98. PatchbayConnectionAddedCallback = pyqtSignal(int, int, int)
  99. PatchbayConnectionRemovedCallback = pyqtSignal(int)
  100. PatchbayIconChangedCallback = pyqtSignal(int, int)
  101. BufferSizeChangedCallback = pyqtSignal(int)
  102. SampleRateChangedCallback = pyqtSignal(float)
  103. ProcessModeChangedCallback = pyqtSignal(int)
  104. EngineStartedCallback = pyqtSignal(str)
  105. EngineStoppedChangedCallback = pyqtSignal()
  106. NSM_AnnounceCallback = pyqtSignal(str)
  107. NSM_OpenCallback = pyqtSignal(str)
  108. NSM_SaveCallback = pyqtSignal()
  109. ErrorCallback = pyqtSignal(str)
  110. QuitCallback = pyqtSignal()
  111. SIGTERM = pyqtSignal()
  112. SIGUSR1 = pyqtSignal()
  113. def __init__(self, parent):
  114. QMainWindow.__init__(self, parent)
  115. self.ui = ui_carla_host.Ui_CarlaHostW()
  116. self.ui.setupUi(self)
  117. if False:
  118. Carla.gui = self
  119. self.fContainer = CarlaDummyW(self)
  120. # -------------------------------------------------------------
  121. # Set callback
  122. Carla.host.set_engine_callback(EngineCallback)
  123. # -------------------------------------------------------------
  124. # Internal stuff
  125. self.fBufferSize = 0
  126. self.fSampleRate = 0.0
  127. self.fIdleTimerFast = 0
  128. self.fIdleTimerSlow = 0
  129. self.fIsProjectLoading = False
  130. self.fProjectFilename = ""
  131. self.fLadspaRdfNeedsUpdate = True
  132. self.fLadspaRdfList = []
  133. self.fLastTransportFrame = 0
  134. self.fLastTransportState = False
  135. self.fSavedSettings = {}
  136. if LADISH_APP_NAME:
  137. self.fClientName = LADISH_APP_NAME
  138. self.fSessionManagerName = "LADISH"
  139. elif NSM_URL:
  140. self.fClientName = "Carla.tmp"
  141. self.fSessionManagerName = "Non Session Manager"
  142. else:
  143. self.fClientName = "Carla"
  144. self.fSessionManagerName = ""
  145. # -------------------------------------------------------------
  146. # Load Settings
  147. self.loadSettings(True)
  148. # -------------------------------------------------------------
  149. # Set up GUI (engine stopped)
  150. if Carla.isPlugin:
  151. self.ui.act_file_new.setEnabled(False)
  152. self.ui.act_file_open.setEnabled(False)
  153. self.ui.act_engine_start.setEnabled(False)
  154. self.ui.act_engine_stop.setEnabled(False)
  155. self.ui.menu_Engine.setEnabled(False)
  156. else:
  157. self.ui.act_engine_start.setEnabled(True)
  158. self.ui.act_file_save.setEnabled(False)
  159. self.ui.act_file_save_as.setEnabled(False)
  160. self.ui.act_engine_stop.setEnabled(False)
  161. self.ui.act_plugin_remove_all.setEnabled(False)
  162. self.ui.menu_PluginMacros.setEnabled(False)
  163. self.ui.menu_Canvas.setEnabled(False)
  164. self.setTransportMenuEnabled(False)
  165. self.setProperWindowTitle()
  166. # -------------------------------------------------------------
  167. # Connect actions to functions
  168. self.ui.act_file_new.triggered.connect(self.slot_fileNew)
  169. self.ui.act_file_open.triggered.connect(self.slot_fileOpen)
  170. self.ui.act_file_save.triggered.connect(self.slot_fileSave)
  171. self.ui.act_file_save_as.triggered.connect(self.slot_fileSaveAs)
  172. self.ui.act_engine_start.triggered.connect(self.slot_engineStart)
  173. self.ui.act_engine_stop.triggered.connect(self.slot_engineStop)
  174. self.ui.act_plugin_add.triggered.connect(self.slot_pluginAdd)
  175. self.ui.act_plugin_add2.triggered.connect(self.slot_pluginAdd)
  176. self.ui.act_plugin_remove_all.triggered.connect(self.slot_pluginRemoveAll)
  177. self.ui.act_transport_play.triggered.connect(self.slot_transportPlayPause)
  178. self.ui.act_transport_stop.triggered.connect(self.slot_transportStop)
  179. self.ui.act_transport_backwards.triggered.connect(self.slot_transportBackwards)
  180. self.ui.act_transport_forwards.triggered.connect(self.slot_transportForwards)
  181. self.ui.act_help_about.triggered.connect(self.slot_aboutCarla)
  182. self.ui.act_help_about_qt.triggered.connect(self.slot_aboutQt)
  183. #self.ui.splitter.splitterMoved.connect(self.slot_splitterMoved)
  184. #self.ui.cb_disk.currentIndexChanged.connect(self.slot_diskFolderChanged)
  185. #self.ui.b_disk_add.clicked.connect(self.slot_diskFolderAdd)
  186. #self.ui.b_disk_remove.clicked.connect(self.slot_diskFolderRemove)
  187. #self.ui.fileTreeView.doubleClicked.connect(self.slot_fileTreeDoubleClicked)
  188. self.DebugCallback.connect(self.slot_handleDebugCallback)
  189. self.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback)
  190. self.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback)
  191. self.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
  192. self.BufferSizeChangedCallback.connect(self.slot_handleBufferSizeChangedCallback)
  193. self.SampleRateChangedCallback.connect(self.slot_handleSampleRateChangedCallback)
  194. self.EngineStartedCallback.connect(self.slot_handleEngineStartedCallback)
  195. self.EngineStoppedChangedCallback.connect(self.slot_handleEngineStoppedCallback)
  196. #self.NSM_AnnounceCallback.connect(self.slot_handleNSM_AnnounceCallback)
  197. #self.NSM_OpenCallback.connect(self.slot_handleNSM_OpenCallback)
  198. #self.NSM_SaveCallback.connect(self.slot_handleNSM_SaveCallback)
  199. #self.ErrorCallback.connect(self.slot_handleErrorCallback)
  200. #self.QuitCallback.connect(self.slot_handleQuitCallback)
  201. self.SIGUSR1.connect(self.slot_handleSIGUSR1)
  202. self.SIGTERM.connect(self.slot_handleSIGTERM)
  203. # -------------------------------------------------------------
  204. # Final setup
  205. if Carla.isPlugin:
  206. QTimer.singleShot(0, self.slot_engineStart)
  207. elif NSM_URL:
  208. Carla.host.nsm_ready()
  209. # -----------------------------------------------------------------
  210. # Called by containers
  211. def openSettingsWindow(self, hasCanvas, hasCanvasGL):
  212. dialog = CarlaSettingsW(self, hasCanvas, hasCanvasGL)
  213. return dialog.exec_()
  214. # -----------------------------------------------------------------
  215. # Internal stuff (files)
  216. def loadProjectNow(self):
  217. if not self.fProjectFilename:
  218. return qCritical("ERROR: loading project without filename set")
  219. # TESTING
  220. if not Carla.host.is_engine_running():
  221. self.slot_engineStart()
  222. self.fIsProjectLoading = True
  223. Carla.host.load_project(self.fProjectFilename)
  224. self.fIsProjectLoading = False
  225. @pyqtSlot()
  226. def slot_loadProjectNow(self):
  227. self.loadProjectNow()
  228. def loadProjectLater(self, filename):
  229. self.fProjectFilename = filename
  230. self.setProperWindowTitle()
  231. QTimer.singleShot(0, self.slot_loadProjectNow)
  232. def saveProjectNow(self):
  233. if not self.fProjectFilename:
  234. return qCritical("ERROR: saving project without filename set")
  235. Carla.host.save_project(self.fProjectFilename)
  236. # -----------------------------------------------------------------
  237. # Internal stuff (engine)
  238. def setEngineSettings(self, settings = None):
  239. if Carla.isPlugin:
  240. return "Plugin"
  241. if settings is None: settings = QSettings()
  242. forceStereo = settings.value("Engine/ForceStereo", CARLA_DEFAULT_FORCE_STEREO, type=bool)
  243. preferPluginBridges = settings.value("Engine/PreferPluginBridges", CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES, type=bool)
  244. preferUiBridges = settings.value("Engine/PreferUiBridges", CARLA_DEFAULT_PREFER_UI_BRIDGES, type=bool)
  245. uisAlwaysOnTop = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UIS_ALWAYS_ON_TOP, type=bool)
  246. uiBridgesTimeout = settings.value("Engine/OscUiTimeout", CARLA_DEFAULT_UI_BRIDGES_TIMEOUT, type=int)
  247. Carla.processMode = settings.value("Engine/ProcessMode", CARLA_DEFAULT_PROCESS_MODE, type=int)
  248. Carla.maxParameters = settings.value("Engine/MaxParameters", CARLA_DEFAULT_MAX_PARAMETERS, type=int)
  249. audioDriver = settings.value("Engine/AudioDriver", CARLA_DEFAULT_AUDIO_DRIVER, type=str)
  250. if audioDriver == "JACK":
  251. #transportMode = settings.value("Engine/TransportMode", TRANSPORT_MODE_JACK, type=int)
  252. transportMode = TRANSPORT_MODE_JACK
  253. else:
  254. transportMode = TRANSPORT_MODE_INTERNAL
  255. audioNumPeriods = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_NUM_PERIODS, type=int)
  256. audioBufferSize = settings.value("Engine/AudioBufferSize", CARLA_DEFAULT_AUDIO_BUFFER_SIZE, type=int)
  257. audioSampleRate = settings.value("Engine/AudioSampleRate", CARLA_DEFAULT_AUDIO_SAMPLE_RATE, type=int)
  258. audioDevice = settings.value("Engine/AudioDevice", "", type=str)
  259. Carla.host.set_engine_option(OPTION_AUDIO_NUM_PERIODS, audioNumPeriods, "")
  260. Carla.host.set_engine_option(OPTION_AUDIO_BUFFER_SIZE, audioBufferSize, "")
  261. Carla.host.set_engine_option(OPTION_AUDIO_SAMPLE_RATE, audioSampleRate, "")
  262. Carla.host.set_engine_option(OPTION_AUDIO_DEVICE, 0, audioDevice)
  263. if Carla.processMode == PROCESS_MODE_CONTINUOUS_RACK:
  264. forceStereo = True
  265. elif Carla.processMode == PROCESS_MODE_MULTIPLE_CLIENTS: # and LADISH_APP_NAME:
  266. print("LADISH detected but using multiple clients (not allowed), forcing single client now")
  267. Carla.processMode = PROCESS_MODE_SINGLE_CLIENT
  268. Carla.host.set_engine_option(OPTION_FORCE_STEREO, forceStereo, "")
  269. Carla.host.set_engine_option(OPTION_PREFER_PLUGIN_BRIDGES, preferPluginBridges, "")
  270. Carla.host.set_engine_option(OPTION_PREFER_UI_BRIDGES, preferUiBridges, "")
  271. Carla.host.set_engine_option(OPTION_UIS_ALWAYS_ON_TOP, uisAlwaysOnTop, "")
  272. Carla.host.set_engine_option(OPTION_UI_BRIDGES_TIMEOUT, uiBridgesTimeout, "")
  273. Carla.host.set_engine_option(OPTION_PROCESS_MODE, Carla.processMode, "")
  274. Carla.host.set_engine_option(OPTION_MAX_PARAMETERS, Carla.maxParameters, "")
  275. Carla.host.set_engine_option(OPTION_TRANSPORT_MODE, transportMode, "")
  276. return audioDriver
  277. def startEngine(self):
  278. audioDriver = self.setEngineSettings()
  279. if not Carla.host.engine_init(audioDriver, self.fClientName):
  280. #if self.fFirstEngineInit:
  281. #self.fFirstEngineInit = False
  282. #return
  283. audioError = cString(Carla.host.get_last_error())
  284. if audioError:
  285. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s', possible reasons:\n%s" % (audioDriver, audioError)))
  286. else:
  287. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s'" % audioDriver))
  288. return
  289. self.fBufferSize = Carla.host.get_buffer_size()
  290. self.fSampleRate = Carla.host.get_sample_rate()
  291. #self.fFirstEngineInit = False
  292. # Peaks and TimeInfo
  293. self.fIdleTimerFast = self.startTimer(self.fSavedSettings["Main/RefreshInterval"])
  294. # LEDs and edit dialog parameters
  295. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  296. def stopEngine(self):
  297. if self.fContainer.getPluginCount() > 0:
  298. 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"
  299. "Do you want to do this now?"),
  300. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  301. if ask != QMessageBox.Yes:
  302. return
  303. self.ui.act_plugin_remove_all.setEnabled(False)
  304. self.fContainer.removeAllPlugins()
  305. if Carla.host.is_engine_running() and not Carla.host.engine_close():
  306. print(cString(Carla.host.get_last_error()))
  307. self.fBufferSize = 0
  308. self.fSampleRate = 0.0
  309. if self.fIdleTimerFast != 0:
  310. self.killTimer(self.fIdleTimerFast)
  311. self.fIdleTimerFast = 0
  312. if self.fIdleTimerSlow != 0:
  313. self.killTimer(self.fIdleTimerSlow)
  314. self.fIdleTimerSlow = 0
  315. # -----------------------------------------------------------------
  316. # Internal stuff (plugins)
  317. def getExtraStuff(self, plugin):
  318. ptype = plugin['type']
  319. if ptype == PLUGIN_LADSPA:
  320. uniqueId = plugin['uniqueId']
  321. self.maybeLoadRDFs()
  322. for rdfItem in self.fLadspaRdfList:
  323. if rdfItem.UniqueID == uniqueId:
  324. return pointer(rdfItem)
  325. elif ptype in (PLUGIN_GIG, PLUGIN_SF2, PLUGIN_SFZ):
  326. if plugin['name'].lower().endswith(" (16 outputs)"):
  327. # return a dummy non-null pointer
  328. INTPOINTER = POINTER(c_int)
  329. int1 = c_int(0x1)
  330. addr = addressof(int1)
  331. return cast(addr, INTPOINTER)
  332. return c_nullptr
  333. def maybeLoadRDFs(self):
  334. if not self.fLadspaRdfNeedsUpdate:
  335. return
  336. self.fLadspaRdfNeedsUpdate = False
  337. self.fLadspaRdfList = []
  338. if not haveLRDF:
  339. return
  340. settingsDir = os.path.join(HOME, ".config", "falkTX")
  341. frLadspaFile = os.path.join(settingsDir, "ladspa_rdf.db")
  342. if os.path.exists(frLadspaFile):
  343. frLadspa = open(frLadspaFile, 'r')
  344. try:
  345. self.fLadspaRdfList = ladspa_rdf.get_c_ladspa_rdfs(json.load(frLadspa))
  346. except:
  347. pass
  348. frLadspa.close()
  349. def setLoadRDFsNeeded(self):
  350. self.fLadspaRdfNeedsUpdate = True
  351. # -----------------------------------------------------------------
  352. # Internal stuff (transport)
  353. def refreshTransport(self, forced = False):
  354. if not Carla.host.is_engine_running():
  355. return
  356. if self.fSampleRate == 0.0:
  357. return
  358. timeInfo = Carla.host.get_transport_info()
  359. playing = bool(timeInfo['playing'])
  360. frame = int(timeInfo['frame'])
  361. if playing != self.fLastTransportState or forced:
  362. if playing:
  363. icon = getIcon("media-playback-pause")
  364. self.ui.act_transport_play.setChecked(True)
  365. self.ui.act_transport_play.setIcon(icon)
  366. self.ui.act_transport_play.setText(self.tr("&Pause"))
  367. else:
  368. icon = getIcon("media-playback-start")
  369. self.ui.act_transport_play.setChecked(False)
  370. self.ui.act_transport_play.setIcon(icon)
  371. self.ui.act_transport_play.setText(self.tr("&Play"))
  372. self.fLastTransportState = playing
  373. if frame != self.fLastTransportFrame or forced:
  374. time = frame / self.fSampleRate
  375. secs = time % 60
  376. mins = (time / 60) % 60
  377. hrs = (time / 3600) % 60
  378. #textTransport = "Transport %s, at %02i:%02i:%02i" % ("playing" if playing else "stopped", hrs, mins, secs)
  379. #self.fInfoLabel.setText("%s | %s" % (self.fInfoText, textTransport))
  380. self.fLastTransportFrame = frame
  381. def setTransportMenuEnabled(self, enabled):
  382. self.ui.act_transport_play.setEnabled(enabled)
  383. self.ui.act_transport_stop.setEnabled(enabled)
  384. self.ui.act_transport_backwards.setEnabled(enabled)
  385. self.ui.act_transport_forwards.setEnabled(enabled)
  386. self.ui.menu_Transport.setEnabled(enabled)
  387. # -----------------------------------------------------------------
  388. # Internal stuff (settings)
  389. def loadSettings(self, doGeometry):
  390. settings = QSettings()
  391. if doGeometry:
  392. self.restoreGeometry(settings.value("Geometry", ""))
  393. showToolbar = settings.value("ShowToolbar", True, type=bool)
  394. self.ui.act_settings_show_toolbar.setChecked(showToolbar)
  395. self.ui.toolBar.setVisible(showToolbar)
  396. #if settings.contains("SplitterState"):
  397. #self.ui.splitter.restoreState(settings.value("SplitterState", ""))
  398. #else:
  399. #self.ui.splitter.setSizes([99999, 210])
  400. #diskFolders = toList(settings.value("DiskFolders", [HOME]))
  401. #self.ui.cb_disk.setItemData(0, HOME)
  402. #for i in range(len(diskFolders)):
  403. #if i == 0: continue
  404. #folder = diskFolders[i]
  405. #self.ui.cb_disk.addItem(os.path.basename(folder), folder)
  406. if MACOS and not settings.value("Main/UseProTheme", True, type=bool):
  407. self.setUnifiedTitleAndToolBarOnMac(True)
  408. # ---------------------------------------------
  409. # plugin checks
  410. if settings.value("Engine/DisableChecks", False, type=bool):
  411. os.environ["CARLA_DISCOVERY_NO_PROCESSING_CHECKS"] = "true"
  412. elif os.getenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS"):
  413. os.environ.pop("CARLA_DISCOVERY_NO_PROCESSING_CHECKS")
  414. # ---------------------------------------------
  415. if not Carla.isPlugin:
  416. # engine
  417. self.setEngineSettings(settings)
  418. # plugin paths
  419. LADSPA_PATH = toList(settings.value("Paths/LADSPA", Carla.DEFAULT_LADSPA_PATH))
  420. DSSI_PATH = toList(settings.value("Paths/DSSI", Carla.DEFAULT_DSSI_PATH))
  421. LV2_PATH = toList(settings.value("Paths/LV2", Carla.DEFAULT_LV2_PATH))
  422. VST_PATH = toList(settings.value("Paths/VST", Carla.DEFAULT_VST_PATH))
  423. AU_PATH = toList(settings.value("Paths/AU", Carla.DEFAULT_AU_PATH))
  424. CSOUND_PATH = toList(settings.value("Paths/CSOUND", Carla.DEFAULT_CSOUND_PATH))
  425. GIG_PATH = toList(settings.value("Paths/GIG", Carla.DEFAULT_GIG_PATH))
  426. SF2_PATH = toList(settings.value("Paths/SF2", Carla.DEFAULT_SF2_PATH))
  427. SFZ_PATH = toList(settings.value("Paths/SFZ", Carla.DEFAULT_SFZ_PATH))
  428. os.environ["LADSPA_PATH"] = splitter.join(LADSPA_PATH)
  429. os.environ["DSSI_PATH"] = splitter.join(DSSI_PATH)
  430. os.environ["LV2_PATH"] = splitter.join(LV2_PATH)
  431. os.environ["VST_PATH"] = splitter.join(VST_PATH)
  432. os.environ["AU_PATH"] = splitter.join(AU_PATH)
  433. os.environ["CSOUND_PATH"] = splitter.join(CSOUND_PATH)
  434. os.environ["GIG_PATH"] = splitter.join(GIG_PATH)
  435. os.environ["SF2_PATH"] = splitter.join(SF2_PATH)
  436. os.environ["SFZ_PATH"] = splitter.join(SFZ_PATH)
  437. # ---------------------------------------------
  438. useCustomMiniCanvasPaint = bool(settings.value("Main/UseProTheme", True, type=bool) and
  439. settings.value("Main/ProThemeColor", "Black", type=str) == "Black")
  440. self.fSavedSettings = {
  441. "Main/DefaultProjectFolder": settings.value("Main/DefaultProjectFolder", HOME, type=str),
  442. "Main/RefreshInterval": settings.value("Main/RefreshInterval", 50, type=int),
  443. "Canvas/Theme": settings.value("Canvas/Theme", CANVAS_DEFAULT_THEME_NAME, type=str),
  444. "Canvas/AutoHideGroups": settings.value("Canvas/AutoHideGroups", False, type=bool),
  445. "Canvas/UseBezierLines": settings.value("Canvas/UseBezierLines", True, type=bool),
  446. "Canvas/EyeCandy": settings.value("Canvas/EyeCandy", CANVAS_EYECANDY_SMALL, type=int),
  447. "Canvas/UseOpenGL": settings.value("Canvas/UseOpenGL", False, type=bool),
  448. "Canvas/Antialiasing": settings.value("Canvas/Antialiasing", CANVAS_ANTIALIASING_SMALL, type=int),
  449. "Canvas/HighQualityAntialiasing": settings.value("Canvas/HighQualityAntialiasing", False, type=bool),
  450. "UseCustomMiniCanvasPaint": useCustomMiniCanvasPaint
  451. }
  452. # ---------------------------------------------
  453. if self.fIdleTimerFast != 0:
  454. self.killTimer(self.fIdleTimerFast)
  455. self.fIdleTimerFast = self.startTimer(self.fSavedSettings["Main/RefreshInterval"])
  456. if self.fIdleTimerSlow != 0:
  457. self.killTimer(self.fIdleTimerSlow)
  458. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  459. def saveSettings(self):
  460. settings = QSettings()
  461. settings.setValue("Geometry", self.saveGeometry())
  462. #settings.setValue("SplitterState", self.ui.splitter.saveState())
  463. settings.setValue("ShowToolbar", self.ui.toolBar.isVisible())
  464. #settings.setValue("HorizontalScrollBarValue", self.ui.graphicsView.horizontalScrollBar().value())
  465. #settings.setValue("VerticalScrollBarValue", self.ui.graphicsView.verticalScrollBar().value())
  466. #diskFolders = []
  467. #for i in range(self.ui.cb_disk.count()):
  468. #diskFolders.append(self.ui.cb_disk.itemData(i))
  469. #settings.setValue("DiskFolders", diskFolders)
  470. self.fContainer.saveSettings(settings)
  471. # -----------------------------------------------------------------
  472. # Internal stuff (gui)
  473. def setProperWindowTitle(self):
  474. title = self.fClientName
  475. if self.fProjectFilename:
  476. title += " - %s" % os.path.basename(self.fProjectFilename)
  477. if self.fSessionManagerName:
  478. title += " (%s)" % self.fSessionManagerName
  479. self.setWindowTitle(title)
  480. # -----------------------------------------------------------------
  481. @pyqtSlot()
  482. def slot_fileNew(self):
  483. self.fContainer.removeAllPlugins()
  484. self.fProjectFilename = ""
  485. self.setProperWindowTitle()
  486. @pyqtSlot()
  487. def slot_fileOpen(self):
  488. fileFilter = self.tr("Carla Project File (*.carxp)")
  489. filenameTry = QFileDialog.getOpenFileName(self, self.tr("Open Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
  490. if not filenameTry:
  491. return
  492. newFile = True
  493. if self.fContainer.getPluginCount() > 0:
  494. ask = QMessageBox.question(self, self.tr("Question"), self.tr("There are some plugins loaded, do you want to remove them now?"),
  495. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  496. newFile = ask == QMessageBox.Yes
  497. if newFile:
  498. self.fContainer.removeAllPlugins()
  499. self.fProjectFilename = filenameTry
  500. self.setProperWindowTitle()
  501. self.loadProjectNow()
  502. else:
  503. filenameOld = self.fProjectFilename
  504. self.fProjectFilename = filenameTry
  505. self.loadProjectNow()
  506. self.fProjectFilename = filenameOld
  507. @pyqtSlot()
  508. def slot_fileSave(self, saveAs=False):
  509. if self.fProjectFilename and not saveAs:
  510. return self.saveProjectNow()
  511. fileFilter = self.tr("Carla Project File (*.carxp)")
  512. filenameTry = QFileDialog.getSaveFileName(self, self.tr("Save Carla Project File"), self.fSavedSettings["Main/DefaultProjectFolder"], filter=fileFilter)[0]
  513. if not filenameTry:
  514. return
  515. if not filenameTry.endswith(".carxp"):
  516. filenameTry += ".carxp"
  517. if self.fProjectFilename != filenameTry:
  518. self.fProjectFilename = filenameTry
  519. self.setProperWindowTitle()
  520. self.saveProjectNow()
  521. @pyqtSlot()
  522. def slot_fileSaveAs(self):
  523. self.slot_fileSave(True)
  524. # -----------------------------------------------------------------
  525. @pyqtSlot()
  526. def slot_engineStart(self, doStart = True):
  527. if doStart: self.startEngine()
  528. check = Carla.host.is_engine_running()
  529. self.ui.menu_PluginMacros.setEnabled(check)
  530. self.ui.menu_Canvas.setEnabled(check)
  531. if not Carla.isPlugin:
  532. self.ui.act_file_save.setEnabled(check)
  533. self.ui.act_engine_start.setEnabled(not check)
  534. self.ui.act_engine_stop.setEnabled(check)
  535. if self.fSessionManagerName != "Non Session Manager":
  536. self.ui.act_file_open.setEnabled(check)
  537. self.ui.act_file_save_as.setEnabled(check)
  538. self.setTransportMenuEnabled(check)
  539. if check:
  540. #self.fInfoText = "Engine running | SampleRate: %g | BufferSize: %i" % (self.fSampleRate, self.fBufferSize)
  541. self.fContainer.engineStarted()
  542. if not Carla.isPlugin:
  543. self.refreshTransport(True)
  544. @pyqtSlot()
  545. def slot_engineStop(self, doStop = True):
  546. if doStop: self.stopEngine()
  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 not check:
  559. self.fContainer.engineStopped()
  560. #self.fInfoText = ""
  561. #self.fInfoLabel.setText("Engine stopped")
  562. # -----------------------------------------------------------------
  563. @pyqtSlot()
  564. def slot_pluginAdd(self):
  565. dialog = PluginDatabaseW(self)
  566. if not dialog.exec_():
  567. return
  568. if not Carla.host.is_engine_running():
  569. QMessageBox.warning(self, self.tr("Warning"), self.tr("Cannot add new plugins while engine is stopped"))
  570. return
  571. btype = dialog.fRetPlugin['build']
  572. ptype = dialog.fRetPlugin['type']
  573. filename = dialog.fRetPlugin['binary']
  574. label = dialog.fRetPlugin['label']
  575. extraStuff = self.getExtraStuff(dialog.fRetPlugin)
  576. if not Carla.host.add_plugin(btype, ptype, filename, None, label, c_nullptr):
  577. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"), self.tr("Failed to load plugin"), cString(Carla.host.get_last_error()), QMessageBox.Ok, QMessageBox.Ok)
  578. return
  579. @pyqtSlot()
  580. def slot_pluginRemoveAll(self):
  581. self.ui.act_plugin_remove_all.setEnabled(False)
  582. self.fContainer.removeAllPlugins()
  583. Carla.host.remove_all_plugins()
  584. # -----------------------------------------------------------------
  585. @pyqtSlot(bool)
  586. def slot_transportPlayPause(self, toggled):
  587. if not Carla.host.is_engine_running():
  588. return
  589. if toggled:
  590. Carla.host.transport_play()
  591. else:
  592. Carla.host.transport_pause()
  593. self.refreshTransport()
  594. @pyqtSlot()
  595. def slot_transportStop(self):
  596. if not Carla.host.is_engine_running():
  597. return
  598. Carla.host.transport_pause()
  599. Carla.host.transport_relocate(0)
  600. self.refreshTransport()
  601. @pyqtSlot()
  602. def slot_transportBackwards(self):
  603. if not Carla.host.is_engine_running():
  604. return
  605. newFrame = Carla.host.get_current_transport_frame() - 100000
  606. if newFrame < 0:
  607. newFrame = 0
  608. Carla.host.transport_relocate(newFrame)
  609. @pyqtSlot()
  610. def slot_transportForwards(self):
  611. if not Carla.host.is_engine_running():
  612. return
  613. newFrame = Carla.host.get_current_transport_frame() + 100000
  614. Carla.host.transport_relocate(newFrame)
  615. # -----------------------------------------------------------------
  616. @pyqtSlot()
  617. def slot_aboutCarla(self):
  618. CarlaAboutW(self).exec_()
  619. @pyqtSlot()
  620. def slot_aboutQt(self):
  621. QApplication.instance().aboutQt()
  622. # -----------------------------------------------------------------
  623. @pyqtSlot(int, int, int, float, str)
  624. def slot_handleDebugCallback(self, pluginId, value1, value2, value3, valueStr):
  625. print("DEBUG:", pluginId, value1, value2, value3, valueStr)
  626. #self.ui.pte_log.appendPlainText(valueStr.replace("", "DEBUG: ").replace("", "ERROR: ").replace("", "").replace("\n", ""))
  627. @pyqtSlot(int)
  628. def slot_handlePluginAddedCallback(self, pluginId):
  629. self.fContainer.addPlugin(pluginId, self.fIsProjectLoading)
  630. if self.fContainer.getPluginCount() == 1:
  631. self.ui.act_plugin_remove_all.setEnabled(True)
  632. @pyqtSlot(int)
  633. def slot_handlePluginRemovedCallback(self, pluginId):
  634. self.fContainer.removePlugin(pluginId)
  635. if self.fContainer.getPluginCount() == 0:
  636. self.ui.act_plugin_remove_all.setEnabled(False)
  637. @pyqtSlot(int, str)
  638. def slot_handlePluginRenamedCallback(self, pluginId, newName):
  639. self.fContainer.renamePlugin(pluginId, newName)
  640. @pyqtSlot(int)
  641. def slot_handleBufferSizeChangedCallback(self, newBufferSize):
  642. self.fBufferSize = newBufferSize
  643. #self.fInfoText = "Engine running | SampleRate: %g | BufferSize: %i" % (self.fSampleRate, self.fBufferSize)
  644. @pyqtSlot(float)
  645. def slot_handleSampleRateChangedCallback(self, newSampleRate):
  646. self.fSampleRate = newSampleRate
  647. #self.fInfoText = "Engine running | SampleRate: %g | BufferSize: %i" % (self.fSampleRate, self.fBufferSize)
  648. # -----------------------------------------------------------------
  649. @pyqtSlot(str)
  650. def slot_handleEngineStartedCallback(self, driverName):
  651. self.fBufferSize = Carla.host.get_buffer_size()
  652. self.fSampleRate = Carla.host.get_sample_rate()
  653. if self.fIdleTimerFast == 0:
  654. self.fIdleTimerFast = self.startTimer(self.fSavedSettings["Main/RefreshInterval"])
  655. if self.fIdleTimerSlow == 0:
  656. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings["Main/RefreshInterval"]*2)
  657. self.slot_engineStart(False)
  658. @pyqtSlot()
  659. def slot_handleEngineStoppedCallback(self):
  660. self.fBufferSize = 0
  661. self.fSampleRate = 0.0
  662. if self.fIdleTimerFast != 0:
  663. self.killTimer(self.fIdleTimerFast)
  664. self.fIdleTimerFast = 0
  665. if self.fIdleTimerSlow != 0:
  666. self.killTimer(self.fIdleTimerSlow)
  667. self.fIdleTimerSlow = 0
  668. if self.fContainer.getPluginCount() > 0:
  669. self.ui.act_plugin_remove_all.setEnabled(False)
  670. self.fContainer.removeAllPlugins() # FIXME?
  671. self.slot_engineStop(False)
  672. # -----------------------------------------------------------------
  673. @pyqtSlot()
  674. def slot_handleSIGUSR1(self):
  675. print("Got SIGUSR1 -> Saving project now")
  676. #QTimer.singleShot(0, self, SLOT("slot_fileSave)
  677. @pyqtSlot()
  678. def slot_handleSIGTERM(self):
  679. print("Got SIGTERM -> Closing now")
  680. self.close()
  681. # -----------------------------------------------------------------
  682. def timerEvent(self, event):
  683. if event.timerId() == self.fIdleTimerFast:
  684. if not Carla.isPlugin:
  685. Carla.host.engine_idle()
  686. self.refreshTransport()
  687. self.fContainer.idleFast()
  688. elif event.timerId() == self.fIdleTimerSlow:
  689. self.fContainer.idleSlow()
  690. QMainWindow.timerEvent(self, event)
  691. def closeEvent(self, event):
  692. if self.fIdleTimerFast != 0:
  693. self.killTimer(self.fIdleTimerFast)
  694. self.fIdleTimerFast = 0
  695. if self.fIdleTimerSlow != 0:
  696. self.killTimer(self.fIdleTimerSlow)
  697. self.fIdleTimerSlow = 0
  698. self.saveSettings()
  699. if Carla.host.is_engine_running():
  700. Carla.host.set_engine_about_to_close()
  701. self.ui.act_plugin_remove_all.setEnabled(False)
  702. self.fContainer.removeAllPlugins()
  703. self.stopEngine()
  704. QMainWindow.closeEvent(self, event)
  705. # ------------------------------------------------------------------------------------------------------------
  706. # Engine callback
  707. def EngineCallback(ptr, action, pluginId, value1, value2, value3, valueStr):
  708. if pluginId < 0 or not Carla.gui:
  709. return
  710. if action == CALLBACK_DEBUG:
  711. Carla.gui.DebugCallback.emit(pluginId, value1, value2, value3, cString(valueStr))
  712. elif action == CALLBACK_PLUGIN_ADDED:
  713. Carla.gui.PluginAddedCallback.emit(pluginId)
  714. elif action == CALLBACK_PLUGIN_REMOVED:
  715. Carla.gui.PluginRemovedCallback.emit(pluginId)
  716. elif action == CALLBACK_PLUGIN_RENAMED:
  717. Carla.gui.PluginRenamedCallback.emit(pluginId, valueStr)
  718. elif action == CALLBACK_PARAMETER_VALUE_CHANGED:
  719. Carla.gui.ParameterValueChangedCallback.emit(pluginId, value1, value3)
  720. elif action == CALLBACK_PARAMETER_DEFAULT_CHANGED:
  721. Carla.gui.ParameterDefaultChangedCallback.emit(pluginId, value1, value3)
  722. elif action == CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED:
  723. Carla.gui.ParameterMidiChannelChangedCallback.emit(pluginId, value1, value2)
  724. elif action == CALLBACK_PARAMETER_MIDI_CC_CHANGED:
  725. Carla.gui.ParameterMidiCcChangedCallback.emit(pluginId, value1, value2)
  726. elif action == CALLBACK_PROGRAM_CHANGED:
  727. Carla.gui.ProgramChangedCallback.emit(pluginId, value1)
  728. elif action == CALLBACK_MIDI_PROGRAM_CHANGED:
  729. Carla.gui.MidiProgramChangedCallback.emit(pluginId, value1)
  730. elif action == CALLBACK_NOTE_ON:
  731. Carla.gui.NoteOnCallback.emit(pluginId, value1, value2, value3)
  732. elif action == CALLBACK_NOTE_OFF:
  733. Carla.gui.NoteOffCallback.emit(pluginId, value1, value2)
  734. elif action == CALLBACK_SHOW_GUI:
  735. Carla.gui.ShowGuiCallback.emit(pluginId, value1)
  736. elif action == CALLBACK_UPDATE:
  737. Carla.gui.UpdateCallback.emit(pluginId)
  738. elif action == CALLBACK_RELOAD_INFO:
  739. Carla.gui.ReloadInfoCallback.emit(pluginId)
  740. elif action == CALLBACK_RELOAD_PARAMETERS:
  741. Carla.gui.ReloadParametersCallback.emit(pluginId)
  742. elif action == CALLBACK_RELOAD_PROGRAMS:
  743. Carla.gui.ReloadProgramsCallback.emit(pluginId)
  744. elif action == CALLBACK_RELOAD_ALL:
  745. Carla.gui.ReloadAllCallback.emit(pluginId)
  746. elif action == CALLBACK_PATCHBAY_CLIENT_ADDED:
  747. Carla.gui.PatchbayClientAddedCallback.emit(value1, value2, cString(valueStr))
  748. elif action == CALLBACK_PATCHBAY_CLIENT_REMOVED:
  749. Carla.gui.PatchbayClientRemovedCallback.emit(value1, cString(valueStr))
  750. elif action == CALLBACK_PATCHBAY_CLIENT_RENAMED:
  751. Carla.gui.PatchbayClientRenamedCallback.emit(value1, cString(valueStr))
  752. elif action == CALLBACK_PATCHBAY_PORT_ADDED:
  753. Carla.gui.PatchbayPortAddedCallback.emit(value1, value2, int(value3), cString(valueStr))
  754. elif action == CALLBACK_PATCHBAY_PORT_REMOVED:
  755. Carla.gui.PatchbayPortRemovedCallback.emit(value1, value2, cString(valueStr))
  756. elif action == CALLBACK_PATCHBAY_PORT_RENAMED:
  757. Carla.gui.PatchbayPortRenamedCallback.emit(value1, value2, cString(valueStr))
  758. elif action == CALLBACK_PATCHBAY_CONNECTION_ADDED:
  759. Carla.gui.PatchbayConnectionAddedCallback.emit(value1, value2, value3)
  760. elif action == CALLBACK_PATCHBAY_CONNECTION_REMOVED:
  761. Carla.gui.PatchbayConnectionRemovedCallback.emit(value1)
  762. elif action == CALLBACK_PATCHBAY_ICON_CHANGED:
  763. Carla.gui.PatchbayIconChangedCallback.emit(value1, value2)
  764. elif action == CALLBACK_BUFFER_SIZE_CHANGED:
  765. Carla.gui.BufferSizeChangedCallback.emit(value1)
  766. elif action == CALLBACK_SAMPLE_RATE_CHANGED:
  767. Carla.gui.SampleRateChangedCallback.emit(value3)
  768. elif action == CALLBACK_PROCESS_MODE_CHANGED:
  769. Carla.gui.ProcessModeChangedCallback.emit(value1)
  770. elif action == CALLBACK_ENGINE_STARTED:
  771. Carla.gui.EngineStartedCallback.emit(cString(valueStr))
  772. elif action == CALLBACK_ENGINE_STOPPED:
  773. Carla.gui.EngineStoppedChangedCallback.emit()
  774. elif action == CALLBACK_NSM_ANNOUNCE:
  775. Carla.gui.NSM_AnnounceCallback.emit(cString(valueStr))
  776. elif action == CALLBACK_NSM_OPEN:
  777. Carla.gui.NSM_OpenCallback.emit(cString(valueStr))
  778. elif action == CALLBACK_NSM_SAVE:
  779. Carla.gui.NSM_SaveCallback.emit()
  780. elif action == CALLBACK_ERROR:
  781. Carla.gui.ErrorCallback.emit(cString(valueStr))
  782. elif action == CALLBACK_QUIT:
  783. Carla.gui.QuitCallback.emit()