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.

1298 lines
50KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla host code
  4. # Copyright (C) 2011-2014 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 disablePlugin(self, pluginId, errorMsg):
  52. pass
  53. def removeAllPlugins(self):
  54. pass
  55. # -----------------------------------------------------------------
  56. def engineStarted(self):
  57. pass
  58. def engineStopped(self):
  59. pass
  60. def engineChanged(self):
  61. pass
  62. # -----------------------------------------------------------------
  63. def idleFast(self):
  64. pass
  65. def idleSlow(self):
  66. pass
  67. # -----------------------------------------------------------------
  68. def projectLoadingStarted(self):
  69. pass
  70. def projectLoadingFinished(self):
  71. pass
  72. # -----------------------------------------------------------------
  73. def saveSettings(self, settings):
  74. pass
  75. def showEditDialog(self, pluginId):
  76. pass
  77. # ------------------------------------------------------------------------------------------------------------
  78. # Host Window
  79. class HostWindow(QMainWindow):
  80. # signals
  81. DebugCallback = pyqtSignal(int, int, int, float, str)
  82. PluginAddedCallback = pyqtSignal(int, str)
  83. PluginRemovedCallback = pyqtSignal(int)
  84. PluginRenamedCallback = pyqtSignal(int, str)
  85. PluginUnavailableCallback = pyqtSignal(int, str)
  86. ParameterValueChangedCallback = pyqtSignal(int, int, float)
  87. ParameterDefaultChangedCallback = pyqtSignal(int, int, float)
  88. ParameterMidiCcChangedCallback = pyqtSignal(int, int, int)
  89. ParameterMidiChannelChangedCallback = pyqtSignal(int, int, int)
  90. ProgramChangedCallback = pyqtSignal(int, int)
  91. MidiProgramChangedCallback = pyqtSignal(int, int)
  92. UiStateChangedCallback = pyqtSignal(int, int)
  93. NoteOnCallback = pyqtSignal(int, int, int, int)
  94. NoteOffCallback = pyqtSignal(int, int, int)
  95. UpdateCallback = pyqtSignal(int)
  96. ReloadInfoCallback = pyqtSignal(int)
  97. ReloadParametersCallback = pyqtSignal(int)
  98. ReloadProgramsCallback = pyqtSignal(int)
  99. ReloadAllCallback = pyqtSignal(int)
  100. PatchbayClientAddedCallback = pyqtSignal(int, int, int, str)
  101. PatchbayClientRemovedCallback = pyqtSignal(int)
  102. PatchbayClientRenamedCallback = pyqtSignal(int, str)
  103. PatchbayClientDataChangedCallback = pyqtSignal(int, int, int)
  104. PatchbayPortAddedCallback = pyqtSignal(int, int, int, str)
  105. PatchbayPortRemovedCallback = pyqtSignal(int, int)
  106. PatchbayPortRenamedCallback = pyqtSignal(int, int, str)
  107. PatchbayConnectionAddedCallback = pyqtSignal(int, int, int)
  108. PatchbayConnectionRemovedCallback = pyqtSignal(int, int, int)
  109. EngineStartedCallback = pyqtSignal(int, int, str)
  110. EngineStoppedCallback = pyqtSignal()
  111. ProcessModeChangedCallback = pyqtSignal(int)
  112. TransportModeChangedCallback = pyqtSignal(int)
  113. BufferSizeChangedCallback = pyqtSignal(int)
  114. SampleRateChangedCallback = pyqtSignal(float)
  115. InfoCallback = pyqtSignal(str)
  116. ErrorCallback = pyqtSignal(str)
  117. QuitCallback = pyqtSignal()
  118. SIGTERM = pyqtSignal()
  119. SIGUSR1 = pyqtSignal()
  120. def __init__(self, parent):
  121. QMainWindow.__init__(self, parent)
  122. self.ui = ui_carla_host.Ui_CarlaHostW()
  123. self.ui.setupUi(self)
  124. if False:
  125. # kdevelop likes this :)
  126. gCarla.gui = self
  127. gCarla.host = Host("")
  128. self.fContainer = CarlaDummyW(self)
  129. # -------------------------------------------------------------
  130. # Set callback, TODO put somewhere else
  131. if gCarla.host is not None:
  132. gCarla.host.set_engine_callback(engineCallback)
  133. gCarla.host.set_file_callback(fileCallback)
  134. # -------------------------------------------------------------
  135. # Internal stuff
  136. self.fIdleTimerFast = 0
  137. self.fIdleTimerSlow = 0
  138. self.fIsProjectLoading = False
  139. self.fProjectFilename = ""
  140. self.fLadspaRdfNeedsUpdate = True
  141. self.fLadspaRdfList = []
  142. self.fLastTransportFrame = 0
  143. self.fLastTransportState = False
  144. self.fTransportText = ""
  145. # when true, call engineChanged() asap
  146. self.fEngineChanged = False
  147. # first attempt of auto-start engine doesn't show an error
  148. self.fFirstEngineInit = True
  149. self.fSavedSettings = {}
  150. if LADISH_APP_NAME:
  151. self.fClientName = LADISH_APP_NAME
  152. self.fSessionManagerName = "LADISH"
  153. elif NSM_URL:
  154. self.fClientName = "Carla.tmp"
  155. self.fSessionManagerName = "Non Session Manager"
  156. else:
  157. self.fClientName = "Carla"
  158. self.fSessionManagerName = ""
  159. # -------------------------------------------------------------
  160. # Load Settings
  161. self.loadSettings(True)
  162. # -------------------------------------------------------------
  163. # Set up GUI (engine stopped)
  164. if gCarla.isPlugin:
  165. self.ui.act_file_new.setEnabled(False)
  166. self.ui.act_file_open.setEnabled(False)
  167. self.ui.act_engine_start.setEnabled(False)
  168. self.ui.menu_Engine.setEnabled(False)
  169. else:
  170. self.ui.act_engine_start.setEnabled(True)
  171. self.ui.act_file_save.setEnabled(False)
  172. self.ui.act_file_save_as.setEnabled(False)
  173. self.ui.act_engine_stop.setEnabled(False)
  174. self.ui.act_plugin_remove_all.setEnabled(False)
  175. self.ui.menu_PluginMacros.setEnabled(False)
  176. self.ui.menu_Canvas.setEnabled(False)
  177. self.setTransportMenuEnabled(False)
  178. # -------------------------------------------------------------
  179. # Set up GUI (right panel)
  180. self.fDirModel = QFileSystemModel(self)
  181. self.fDirModel.setRootPath(HOME)
  182. if gCarla.host is not None:
  183. self.fDirModel.setNameFilters(gCarla.host.get_supported_file_extensions().split(";"))
  184. self.ui.fileTreeView.setModel(self.fDirModel)
  185. self.ui.fileTreeView.setRootIndex(self.fDirModel.index(HOME))
  186. self.ui.fileTreeView.setColumnHidden(1, True)
  187. self.ui.fileTreeView.setColumnHidden(2, True)
  188. self.ui.fileTreeView.setColumnHidden(3, True)
  189. self.ui.fileTreeView.setHeaderHidden(True)
  190. self.setProperWindowTitle()
  191. # -------------------------------------------------------------
  192. # Connect actions to functions
  193. self.ui.act_file_new.triggered.connect(self.slot_fileNew)
  194. self.ui.act_file_open.triggered.connect(self.slot_fileOpen)
  195. self.ui.act_file_save.triggered.connect(self.slot_fileSave)
  196. self.ui.act_file_save_as.triggered.connect(self.slot_fileSaveAs)
  197. self.ui.act_engine_start.triggered.connect(self.slot_engineStart)
  198. self.ui.act_engine_stop.triggered.connect(self.slot_engineStop)
  199. self.ui.act_plugin_add.triggered.connect(self.slot_pluginAdd)
  200. self.ui.act_plugin_add2.triggered.connect(self.slot_pluginAdd)
  201. self.ui.act_plugin_remove_all.triggered.connect(self.slot_pluginRemoveAll)
  202. self.ui.act_transport_play.triggered.connect(self.slot_transportPlayPause)
  203. self.ui.act_transport_stop.triggered.connect(self.slot_transportStop)
  204. self.ui.act_transport_backwards.triggered.connect(self.slot_transportBackwards)
  205. self.ui.act_transport_forwards.triggered.connect(self.slot_transportForwards)
  206. self.ui.act_help_about.triggered.connect(self.slot_aboutCarla)
  207. self.ui.act_help_about_qt.triggered.connect(self.slot_aboutQt)
  208. self.ui.cb_disk.currentIndexChanged.connect(self.slot_diskFolderChanged)
  209. self.ui.b_disk_add.clicked.connect(self.slot_diskFolderAdd)
  210. self.ui.b_disk_remove.clicked.connect(self.slot_diskFolderRemove)
  211. self.ui.fileTreeView.doubleClicked.connect(self.slot_fileTreeDoubleClicked)
  212. self.DebugCallback.connect(self.slot_handleDebugCallback)
  213. self.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback)
  214. self.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback)
  215. self.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
  216. self.PluginUnavailableCallback.connect(self.slot_handlePluginUnavailableCallback)
  217. # parameter (rack, patchbay)
  218. # program, midi-program, ui-state (rack, patchbay)
  219. # note on, off (rack, patchbay)
  220. # update, reload (rack, patchbay)
  221. # patchbay
  222. self.EngineStartedCallback.connect(self.slot_handleEngineStartedCallback)
  223. self.EngineStoppedCallback.connect(self.slot_handleEngineStoppedCallback)
  224. self.ProcessModeChangedCallback.connect(self.slot_handleProcessModeChangedCallback)
  225. self.TransportModeChangedCallback.connect(self.slot_handleTransportModeChangedCallback)
  226. self.BufferSizeChangedCallback.connect(self.slot_handleBufferSizeChangedCallback)
  227. self.SampleRateChangedCallback.connect(self.slot_handleSampleRateChangedCallback)
  228. self.InfoCallback.connect(self.slot_handleInfoCallback)
  229. self.ErrorCallback.connect(self.slot_handleErrorCallback)
  230. self.QuitCallback.connect(self.slot_handleQuitCallback)
  231. self.SIGUSR1.connect(self.slot_handleSIGUSR1)
  232. self.SIGTERM.connect(self.slot_handleSIGTERM)
  233. # -------------------------------------------------------------
  234. # Final setup
  235. QTimer.singleShot(0, self.slot_engineStart)
  236. #QTimer.singleShot(2000, self.slot_test)
  237. #@pyqtSlot()
  238. #def slot_test(self):
  239. #print("test started")
  240. #if not gCarla.host.add_plugin(BINARY_NATIVE, PLUGIN_JACK, "/usr/bin/zita-rev1", "name of client", "label of client", None):
  241. #print(gCarla.host.get_last_error())
  242. #print("test ended")
  243. # -----------------------------------------------------------------
  244. # Called by containers
  245. def openSettingsWindow(self, hasCanvas, hasCanvasGL):
  246. dialog = CarlaSettingsW(self, hasCanvas, hasCanvasGL)
  247. return dialog.exec_()
  248. def setupContainer(self, showCanvas, canvasThemeData = []):
  249. if showCanvas:
  250. canvasWidth, canvasHeight, canvasBg, canvasBrush, canvasPen = canvasThemeData
  251. self.ui.miniCanvasPreview.setViewTheme(canvasBg, canvasBrush, canvasPen)
  252. self.ui.miniCanvasPreview.init(self.fContainer.scene, canvasWidth, canvasHeight, self.fSavedSettings[CARLA_KEY_CUSTOM_PAINTING])
  253. else:
  254. self.ui.act_canvas_arrange.setVisible(False)
  255. self.ui.act_canvas_print.setVisible(False)
  256. self.ui.act_canvas_refresh.setVisible(False)
  257. self.ui.act_canvas_save_image.setVisible(False)
  258. self.ui.act_canvas_zoom_100.setVisible(False)
  259. self.ui.act_canvas_zoom_fit.setVisible(False)
  260. self.ui.act_canvas_zoom_in.setVisible(False)
  261. self.ui.act_canvas_zoom_out.setVisible(False)
  262. self.ui.act_settings_show_meters.setVisible(False)
  263. self.ui.act_settings_show_keyboard.setVisible(False)
  264. self.ui.menu_Canvas.setEnabled(False)
  265. self.ui.menu_Canvas.setVisible(False)
  266. self.ui.menu_Canvas_Zoom.setEnabled(False)
  267. self.ui.menu_Canvas_Zoom.setVisible(False)
  268. self.ui.miniCanvasPreview.hide()
  269. self.ui.splitter.insertWidget(1, self.fContainer)
  270. def updateContainer(self, canvasThemeData):
  271. canvasWidth, canvasHeight, canvasBg, canvasBrush, canvasPen = canvasThemeData
  272. self.ui.miniCanvasPreview.setViewTheme(canvasBg, canvasBrush, canvasPen)
  273. self.ui.miniCanvasPreview.init(self.fContainer.scene, canvasWidth, canvasHeight, self.fSavedSettings[CARLA_KEY_CUSTOM_PAINTING])
  274. # -----------------------------------------------------------------
  275. # Internal stuff (files)
  276. def loadProjectNow(self):
  277. if not self.fProjectFilename:
  278. return qCritical("ERROR: loading project without filename set")
  279. self.fContainer.projectLoadingStarted()
  280. self.fIsProjectLoading = True
  281. gCarla.host.load_project(self.fProjectFilename)
  282. self.fIsProjectLoading = False
  283. self.fContainer.projectLoadingFinished()
  284. @pyqtSlot()
  285. def slot_loadProjectNow(self):
  286. self.loadProjectNow()
  287. def loadProjectLater(self, filename):
  288. self.fProjectFilename = filename
  289. self.setProperWindowTitle()
  290. QTimer.singleShot(0, self.slot_loadProjectNow)
  291. def saveProjectNow(self):
  292. if not self.fProjectFilename:
  293. return qCritical("ERROR: saving project without filename set")
  294. gCarla.host.save_project(self.fProjectFilename)
  295. # -----------------------------------------------------------------
  296. # Internal stuff (engine)
  297. def setEngineSettings(self, settings = None):
  298. if gCarla.isPlugin:
  299. return "Plugin"
  300. if settings is None: settings = QSettings()
  301. # -------------------------------------------------------------
  302. # read settings
  303. # bool values
  304. try:
  305. forceStereo = settings.value(CARLA_KEY_ENGINE_FORCE_STEREO, CARLA_DEFAULT_FORCE_STEREO, type=bool)
  306. except:
  307. forceStereo = CARLA_DEFAULT_FORCE_STEREO
  308. try:
  309. preferPluginBridges = settings.value(CARLA_KEY_ENGINE_PREFER_PLUGIN_BRIDGES, CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES, type=bool)
  310. except:
  311. preferPluginBridges = CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES
  312. try:
  313. preferUiBridges = settings.value(CARLA_KEY_ENGINE_PREFER_UI_BRIDGES, CARLA_DEFAULT_PREFER_UI_BRIDGES, type=bool)
  314. except:
  315. preferUiBridges = CARLA_DEFAULT_PREFER_UI_BRIDGES
  316. try:
  317. uisAlwaysOnTop = settings.value(CARLA_KEY_ENGINE_UIS_ALWAYS_ON_TOP, CARLA_DEFAULT_UIS_ALWAYS_ON_TOP, type=bool)
  318. except:
  319. uisAlwaysOnTop = CARLA_DEFAULT_UIS_ALWAYS_ON_TOP
  320. # int values
  321. try:
  322. maxParameters = settings.value(CARLA_KEY_ENGINE_MAX_PARAMETERS, CARLA_DEFAULT_MAX_PARAMETERS, type=int)
  323. except:
  324. maxParameters = CARLA_DEFAULT_MAX_PARAMETERS
  325. try:
  326. uiBridgesTimeout = settings.value(CARLA_KEY_ENGINE_UI_BRIDGES_TIMEOUT, CARLA_DEFAULT_UI_BRIDGES_TIMEOUT, type=int)
  327. except:
  328. uiBridgesTimeout = CARLA_DEFAULT_UI_BRIDGES_TIMEOUT
  329. # enums
  330. try:
  331. processMode = settings.value(CARLA_KEY_ENGINE_PROCESS_MODE, CARLA_DEFAULT_PROCESS_MODE, type=int)
  332. except:
  333. processMode = CARLA_DEFAULT_PROCESS_MODE
  334. try:
  335. transportMode = settings.value(CARLA_KEY_ENGINE_TRANSPORT_MODE, CARLA_DEFAULT_TRANSPORT_MODE, type=int)
  336. except:
  337. transportMode = CARLA_DEFAULT_TRANSPORT_MODE
  338. # driver name
  339. try:
  340. audioDriver = settings.value(CARLA_KEY_ENGINE_AUDIO_DRIVER, CARLA_DEFAULT_AUDIO_DRIVER, type=str)
  341. except:
  342. audioDriver = CARLA_DEFAULT_AUDIO_DRIVER
  343. # driver options
  344. try:
  345. audioDevice = settings.value("%s%s/Device" % (CARLA_KEY_ENGINE_DRIVER_PREFIX, audioDriver), "", type=str)
  346. except:
  347. audioDevice = ""
  348. try:
  349. audioNumPeriods = settings.value("%s%s/NumPeriods" % (CARLA_KEY_ENGINE_DRIVER_PREFIX, audioDriver), CARLA_DEFAULT_AUDIO_NUM_PERIODS, type=int)
  350. except:
  351. audioNumPeriods = CARLA_DEFAULT_AUDIO_NUM_PERIODS
  352. try:
  353. audioBufferSize = settings.value("%s%s/BufferSize" % (CARLA_KEY_ENGINE_DRIVER_PREFIX, audioDriver), CARLA_DEFAULT_AUDIO_BUFFER_SIZE, type=int)
  354. except:
  355. audioBufferSize = CARLA_DEFAULT_AUDIO_BUFFER_SIZE
  356. try:
  357. audioSampleRate = settings.value("%s%s/SampleRate" % (CARLA_KEY_ENGINE_DRIVER_PREFIX, audioDriver), CARLA_DEFAULT_AUDIO_SAMPLE_RATE, type=int)
  358. except:
  359. audioSampleRate = CARLA_DEFAULT_AUDIO_SAMPLE_RATE
  360. # -------------------------------------------------------------
  361. # fix things if needed
  362. if processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK:
  363. forceStereo = True
  364. elif processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS and LADISH_APP_NAME:
  365. print("LADISH detected but using multiple clients (not allowed), forcing single client now")
  366. processMode = ENGINE_PROCESS_MODE_SINGLE_CLIENT
  367. if audioDriver != "JACK" and transportMode == ENGINE_TRANSPORT_MODE_JACK:
  368. transportMode = ENGINE_TRANSPORT_MODE_INTERNAL
  369. # -------------------------------------------------------------
  370. # apply to engine
  371. gCarla.host.set_engine_option(ENGINE_OPTION_FORCE_STEREO, forceStereo, "")
  372. gCarla.host.set_engine_option(ENGINE_OPTION_UIS_ALWAYS_ON_TOP, uisAlwaysOnTop, "")
  373. if not gCarla.isPlugin:
  374. gCarla.host.set_engine_option(ENGINE_OPTION_PREFER_PLUGIN_BRIDGES, preferPluginBridges, "")
  375. gCarla.host.set_engine_option(ENGINE_OPTION_PREFER_UI_BRIDGES, preferUiBridges, "")
  376. gCarla.host.set_engine_option(ENGINE_OPTION_MAX_PARAMETERS, maxParameters, "")
  377. gCarla.host.set_engine_option(ENGINE_OPTION_UI_BRIDGES_TIMEOUT, uiBridgesTimeout, "")
  378. gCarla.host.set_engine_option(ENGINE_OPTION_PROCESS_MODE, processMode, "")
  379. gCarla.host.set_engine_option(ENGINE_OPTION_TRANSPORT_MODE, transportMode, "")
  380. gCarla.host.set_engine_option(ENGINE_OPTION_AUDIO_NUM_PERIODS, audioNumPeriods, "")
  381. gCarla.host.set_engine_option(ENGINE_OPTION_AUDIO_BUFFER_SIZE, audioBufferSize, "")
  382. gCarla.host.set_engine_option(ENGINE_OPTION_AUDIO_SAMPLE_RATE, audioSampleRate, "")
  383. gCarla.host.set_engine_option(ENGINE_OPTION_AUDIO_DEVICE, 0, audioDevice)
  384. # save this for later
  385. gCarla.maxParameters = maxParameters
  386. # return selected driver name
  387. return audioDriver
  388. def startEngine(self):
  389. audioDriver = self.setEngineSettings()
  390. if not gCarla.host.engine_init(audioDriver, self.fClientName):
  391. if self.fFirstEngineInit:
  392. self.fFirstEngineInit = False
  393. return
  394. audioError = gCarla.host.get_last_error()
  395. if audioError:
  396. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s', possible reasons:\n%s" % (audioDriver, audioError)))
  397. else:
  398. QMessageBox.critical(self, self.tr("Error"), self.tr("Could not connect to Audio backend '%s'" % audioDriver))
  399. return
  400. self.fFirstEngineInit = False
  401. def stopEngine(self):
  402. if self.fContainer.getPluginCount() > 0:
  403. 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"
  404. "Do you want to do this now?"),
  405. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  406. if ask != QMessageBox.Yes:
  407. return
  408. self.ui.act_plugin_remove_all.setEnabled(False)
  409. self.fContainer.removeAllPlugins()
  410. if gCarla.host.is_engine_running() and not gCarla.host.engine_close():
  411. print(gCarla.host.get_last_error())
  412. # -----------------------------------------------------------------
  413. # Internal stuff (plugins)
  414. def getExtraPtr(self, plugin):
  415. ptype = plugin['type']
  416. if ptype == PLUGIN_LADSPA:
  417. uniqueId = plugin['uniqueId']
  418. self.maybeLoadRDFs()
  419. for rdfItem in self.fLadspaRdfList:
  420. if rdfItem.UniqueID == uniqueId:
  421. return pointer(rdfItem)
  422. elif ptype in (PLUGIN_FILE_GIG, PLUGIN_FILE_SF2):
  423. if plugin['name'].lower().endswith(" (16 outputs)"):
  424. return c_char_p("true".encode("utf-8"))
  425. return None
  426. def maybeLoadRDFs(self):
  427. if not self.fLadspaRdfNeedsUpdate:
  428. return
  429. self.fLadspaRdfNeedsUpdate = False
  430. self.fLadspaRdfList = []
  431. if not haveLRDF:
  432. return
  433. settingsDir = os.path.join(HOME, ".config", "falkTX")
  434. frLadspaFile = os.path.join(settingsDir, "ladspa_rdf.db")
  435. if os.path.exists(frLadspaFile):
  436. frLadspa = open(frLadspaFile, 'r')
  437. try:
  438. self.fLadspaRdfList = ladspa_rdf.get_c_ladspa_rdfs(json.load(frLadspa))
  439. except:
  440. pass
  441. frLadspa.close()
  442. def setLoadRDFsNeeded(self):
  443. self.fLadspaRdfNeedsUpdate = True
  444. # -----------------------------------------------------------------
  445. # Internal stuff (transport)
  446. def refreshTransport(self, forced = False):
  447. if gCarla.sampleRate == 0.0 or not gCarla.host.is_engine_running():
  448. return
  449. timeInfo = gCarla.host.get_transport_info()
  450. playing = bool(timeInfo['playing'])
  451. frame = int(timeInfo['frame'])
  452. if playing != self.fLastTransportState or forced:
  453. if playing:
  454. icon = getIcon("media-playback-pause")
  455. self.ui.act_transport_play.setChecked(True)
  456. self.ui.act_transport_play.setIcon(icon)
  457. self.ui.act_transport_play.setText(self.tr("&Pause"))
  458. else:
  459. icon = getIcon("media-playback-start")
  460. self.ui.act_transport_play.setChecked(False)
  461. self.ui.act_transport_play.setIcon(icon)
  462. self.ui.act_transport_play.setText(self.tr("&Play"))
  463. self.fLastTransportState = playing
  464. if frame != self.fLastTransportFrame or forced:
  465. time = frame / gCarla.sampleRate
  466. secs = time % 60
  467. mins = (time / 60) % 60
  468. hrs = (time / 3600) % 60
  469. self.fTextTransport = "Transport %s, at %02i:%02i:%02i" % ("playing" if playing else "stopped", hrs, mins, secs)
  470. self.fLastTransportFrame = frame
  471. def setTransportMenuEnabled(self, enabled):
  472. self.ui.act_transport_play.setEnabled(enabled)
  473. self.ui.act_transport_stop.setEnabled(enabled)
  474. self.ui.act_transport_backwards.setEnabled(enabled)
  475. self.ui.act_transport_forwards.setEnabled(enabled)
  476. self.ui.menu_Transport.setEnabled(enabled)
  477. # -----------------------------------------------------------------
  478. # Internal stuff (settings)
  479. def loadSettings(self, firstTime):
  480. settings = QSettings()
  481. if firstTime:
  482. self.restoreGeometry(settings.value("Geometry", ""))
  483. showToolbar = settings.value("ShowToolbar", True, type=bool)
  484. self.ui.act_settings_show_toolbar.setChecked(showToolbar)
  485. self.ui.toolBar.setVisible(showToolbar)
  486. if settings.contains("SplitterState"):
  487. self.ui.splitter.restoreState(settings.value("SplitterState", ""))
  488. else:
  489. self.ui.splitter.setSizes([210, 99999])
  490. diskFolders = toList(settings.value("DiskFolders", [HOME]))
  491. self.ui.cb_disk.setItemData(0, HOME)
  492. for i in range(len(diskFolders)):
  493. if i == 0: continue
  494. folder = diskFolders[i]
  495. self.ui.cb_disk.addItem(os.path.basename(folder), folder)
  496. if MACOS and not settings.value(CARLA_KEY_MAIN_USE_PRO_THEME, True, type=bool):
  497. self.setUnifiedTitleAndToolBarOnMac(True)
  498. # ---------------------------------------------
  499. # plugin checks
  500. if settings.value("Engine/DisableChecks", False, type=bool):
  501. os.environ["CARLA_DISCOVERY_NO_PROCESSING_CHECKS"] = "true"
  502. elif os.getenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS"):
  503. os.environ.pop("CARLA_DISCOVERY_NO_PROCESSING_CHECKS")
  504. # ---------------------------------------------
  505. if not gCarla.isPlugin:
  506. # engine
  507. self.setEngineSettings(settings)
  508. # plugin paths
  509. LADSPA_PATH = toList(settings.value("Paths/LADSPA", gCarla.DEFAULT_LADSPA_PATH))
  510. DSSI_PATH = toList(settings.value("Paths/DSSI", gCarla.DEFAULT_DSSI_PATH))
  511. LV2_PATH = toList(settings.value("Paths/LV2", gCarla.DEFAULT_LV2_PATH))
  512. VST_PATH = toList(settings.value("Paths/VST", gCarla.DEFAULT_VST_PATH))
  513. AU_PATH = toList(settings.value("Paths/AU", gCarla.DEFAULT_AU_PATH))
  514. CSOUND_PATH = toList(settings.value("Paths/CSOUND", gCarla.DEFAULT_CSOUND_PATH))
  515. GIG_PATH = toList(settings.value("Paths/GIG", gCarla.DEFAULT_GIG_PATH))
  516. SF2_PATH = toList(settings.value("Paths/SF2", gCarla.DEFAULT_SF2_PATH))
  517. SFZ_PATH = toList(settings.value("Paths/SFZ", gCarla.DEFAULT_SFZ_PATH))
  518. os.environ["LADSPA_PATH"] = splitter.join(LADSPA_PATH)
  519. os.environ["DSSI_PATH"] = splitter.join(DSSI_PATH)
  520. os.environ["LV2_PATH"] = splitter.join(LV2_PATH)
  521. os.environ["VST_PATH"] = splitter.join(VST_PATH)
  522. os.environ["AU_PATH"] = splitter.join(AU_PATH)
  523. os.environ["CSOUND_PATH"] = splitter.join(CSOUND_PATH)
  524. os.environ["GIG_PATH"] = splitter.join(GIG_PATH)
  525. os.environ["SF2_PATH"] = splitter.join(SF2_PATH)
  526. os.environ["SFZ_PATH"] = splitter.join(SFZ_PATH)
  527. # ---------------------------------------------
  528. # TODO
  529. self.fSavedSettings = {
  530. CARLA_KEY_MAIN_PROJECT_FOLDER: settings.value(CARLA_KEY_MAIN_PROJECT_FOLDER, CARLA_DEFAULT_MAIN_PROJECT_FOLDER, type=str),
  531. CARLA_KEY_MAIN_REFRESH_INTERVAL: settings.value(CARLA_KEY_MAIN_REFRESH_INTERVAL, CARLA_DEFAULT_MAIN_REFRESH_INTERVAL, type=int),
  532. CARLA_KEY_CANVAS_THEME: settings.value(CARLA_KEY_CANVAS_THEME, CARLA_DEFAULT_CANVAS_THEME, type=str),
  533. CARLA_KEY_CANVAS_SIZE: settings.value(CARLA_KEY_CANVAS_SIZE, CARLA_DEFAULT_CANVAS_SIZE, type=str),
  534. CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS: settings.value(CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS, CARLA_DEFAULT_CANVAS_AUTO_HIDE_GROUPS, type=bool),
  535. CARLA_KEY_CANVAS_USE_BEZIER_LINES: settings.value(CARLA_KEY_CANVAS_USE_BEZIER_LINES, CARLA_DEFAULT_CANVAS_USE_BEZIER_LINES, type=bool),
  536. CARLA_KEY_CANVAS_EYE_CANDY: settings.value(CARLA_KEY_CANVAS_EYE_CANDY, CARLA_DEFAULT_CANVAS_EYE_CANDY, type=int),
  537. CARLA_KEY_CANVAS_USE_OPENGL: settings.value(CARLA_KEY_CANVAS_USE_OPENGL, CARLA_DEFAULT_CANVAS_USE_OPENGL, type=bool),
  538. CARLA_KEY_CANVAS_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_ANTIALIASING, CARLA_DEFAULT_CANVAS_ANTIALIASING, type=int),
  539. CARLA_KEY_CANVAS_HQ_ANTIALIASING: settings.value(CARLA_KEY_CANVAS_HQ_ANTIALIASING, CARLA_DEFAULT_CANVAS_HQ_ANTIALIASING, type=bool),
  540. CARLA_KEY_CUSTOM_PAINTING: (settings.value(CARLA_KEY_MAIN_USE_PRO_THEME, True, type=bool) and
  541. settings.value(CARLA_KEY_MAIN_PRO_THEME_COLOR, "Black", type=str).lower() == "black")
  542. }
  543. # ---------------------------------------------
  544. if self.fIdleTimerFast != 0:
  545. self.killTimer(self.fIdleTimerFast)
  546. self.fIdleTimerFast = self.startTimer(self.fSavedSettings[CARLA_KEY_MAIN_REFRESH_INTERVAL])
  547. if self.fIdleTimerSlow != 0:
  548. self.killTimer(self.fIdleTimerSlow)
  549. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings[CARLA_KEY_MAIN_REFRESH_INTERVAL]*2)
  550. def saveSettings(self):
  551. settings = QSettings()
  552. settings.setValue("Geometry", self.saveGeometry())
  553. settings.setValue("SplitterState", self.ui.splitter.saveState())
  554. settings.setValue("ShowToolbar", self.ui.toolBar.isVisible())
  555. diskFolders = []
  556. for i in range(self.ui.cb_disk.count()):
  557. diskFolders.append(self.ui.cb_disk.itemData(i))
  558. settings.setValue("DiskFolders", diskFolders)
  559. self.fContainer.saveSettings(settings)
  560. # -----------------------------------------------------------------
  561. # Internal stuff (gui)
  562. def killTimers(self):
  563. if self.fIdleTimerFast != 0:
  564. self.killTimer(self.fIdleTimerFast)
  565. self.fIdleTimerFast = 0
  566. if self.fIdleTimerSlow != 0:
  567. self.killTimer(self.fIdleTimerSlow)
  568. self.fIdleTimerSlow = 0
  569. def setProperWindowTitle(self):
  570. title = self.fClientName
  571. if self.fProjectFilename:
  572. title += " - %s" % os.path.basename(self.fProjectFilename)
  573. if self.fSessionManagerName:
  574. title += " (%s)" % self.fSessionManagerName
  575. self.setWindowTitle(title)
  576. # -----------------------------------------------------------------
  577. @pyqtSlot()
  578. def slot_fileNew(self):
  579. self.fContainer.removeAllPlugins()
  580. self.fProjectFilename = ""
  581. self.setProperWindowTitle()
  582. @pyqtSlot()
  583. def slot_fileOpen(self):
  584. fileFilter = self.tr("Carla Project File (*.carxp)")
  585. filenameTry = QFileDialog.getOpenFileName(self, self.tr("Open Carla Project File"), self.fSavedSettings[CARLA_KEY_MAIN_PROJECT_FOLDER], filter=fileFilter)
  586. if not filenameTry:
  587. return
  588. filename = filenameTry if isinstance(filenameTry, str) else filenameTry[0]
  589. newFile = True
  590. if self.fContainer.getPluginCount() > 0:
  591. ask = QMessageBox.question(self, self.tr("Question"), self.tr("There are some plugins loaded, do you want to remove them now?"),
  592. QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
  593. newFile = (ask == QMessageBox.Yes)
  594. if newFile:
  595. self.fContainer.removeAllPlugins()
  596. self.fProjectFilename = filename
  597. self.setProperWindowTitle()
  598. self.loadProjectNow()
  599. else:
  600. filenameOld = self.fProjectFilename
  601. self.fProjectFilename = filename
  602. self.loadProjectNow()
  603. self.fProjectFilename = filenameOld
  604. @pyqtSlot()
  605. def slot_fileSave(self, saveAs=False):
  606. if self.fProjectFilename and not saveAs:
  607. return self.saveProjectNow()
  608. fileFilter = self.tr("Carla Project File (*.carxp)")
  609. filenameTry = QFileDialog.getSaveFileName(self, self.tr("Save Carla Project File"), self.fSavedSettings[CARLA_KEY_MAIN_PROJECT_FOLDER], filter=fileFilter)
  610. if not filenameTry:
  611. return
  612. filename = filenameTry if isinstance(filenameTry, str) else filenameTry[0]
  613. if not filename.endswith(".carxp"):
  614. filename += ".carxp"
  615. if self.fProjectFilename != filename:
  616. self.fProjectFilename = filename
  617. self.setProperWindowTitle()
  618. self.saveProjectNow()
  619. @pyqtSlot()
  620. def slot_fileSaveAs(self):
  621. self.slot_fileSave(True)
  622. # -----------------------------------------------------------------
  623. @pyqtSlot()
  624. def slot_engineStart(self, doStart = True):
  625. if doStart: self.startEngine()
  626. check = gCarla.host.is_engine_running()
  627. self.ui.menu_PluginMacros.setEnabled(check)
  628. self.ui.menu_Canvas.setEnabled(check)
  629. if not gCarla.isPlugin:
  630. self.ui.act_file_save.setEnabled(check)
  631. self.ui.act_engine_start.setEnabled(not check)
  632. self.ui.act_engine_stop.setEnabled(check)
  633. if self.fSessionManagerName != "Non Session Manager":
  634. self.ui.act_file_open.setEnabled(check)
  635. self.ui.act_file_save_as.setEnabled(check)
  636. self.setTransportMenuEnabled(check)
  637. if check:
  638. if not gCarla.isPlugin:
  639. self.refreshTransport(True)
  640. self.fContainer.engineStarted()
  641. @pyqtSlot()
  642. def slot_engineStop(self, doStop = True):
  643. if doStop: self.stopEngine()
  644. # FIXME?
  645. if self.fContainer.getPluginCount() > 0:
  646. self.ui.act_plugin_remove_all.setEnabled(False)
  647. self.fContainer.removeAllPlugins()
  648. check = gCarla.host.is_engine_running()
  649. self.ui.menu_PluginMacros.setEnabled(check)
  650. self.ui.menu_Canvas.setEnabled(check)
  651. if not gCarla.isPlugin:
  652. self.ui.act_file_save.setEnabled(check)
  653. self.ui.act_engine_start.setEnabled(not check)
  654. self.ui.act_engine_stop.setEnabled(check)
  655. if self.fSessionManagerName != "Non Session Manager":
  656. self.ui.act_file_open.setEnabled(check)
  657. self.ui.act_file_save_as.setEnabled(check)
  658. self.setTransportMenuEnabled(check)
  659. if not check:
  660. self.fTextTransport = ""
  661. self.fContainer.engineStopped()
  662. # -----------------------------------------------------------------
  663. @pyqtSlot()
  664. def slot_pluginAdd(self):
  665. dialog = PluginDatabaseW(self)
  666. if not dialog.exec_():
  667. return
  668. if not gCarla.host.is_engine_running():
  669. QMessageBox.warning(self, self.tr("Warning"), self.tr("Cannot add new plugins while engine is stopped"))
  670. return
  671. btype = dialog.fRetPlugin['build']
  672. ptype = dialog.fRetPlugin['type']
  673. filename = dialog.fRetPlugin['filename']
  674. label = dialog.fRetPlugin['label']
  675. uniqueId = dialog.fRetPlugin['uniqueId']
  676. extraPtr = self.getExtraPtr(dialog.fRetPlugin)
  677. if not gCarla.host.add_plugin(btype, ptype, filename, None, label, uniqueId, extraPtr):
  678. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"), self.tr("Failed to load plugin"), charPtrToString(gCarla.host.get_last_error()), QMessageBox.Ok, QMessageBox.Ok)
  679. return
  680. @pyqtSlot()
  681. def slot_pluginRemoveAll(self):
  682. self.ui.act_plugin_remove_all.setEnabled(False)
  683. count = self.fContainer.getPluginCount()
  684. if count == 0:
  685. return
  686. self.fContainer.projectLoadingStarted()
  687. app = QApplication.instance()
  688. for i in range(count):
  689. app.processEvents()
  690. gCarla.host.remove_plugin(count-i-1)
  691. self.fContainer.projectLoadingFinished()
  692. #self.fContainer.removeAllPlugins()
  693. #gCarla.host.remove_all_plugins()
  694. # -----------------------------------------------------------------
  695. @pyqtSlot(bool)
  696. def slot_transportPlayPause(self, toggled):
  697. if not gCarla.host.is_engine_running():
  698. return
  699. if toggled:
  700. gCarla.host.transport_play()
  701. else:
  702. gCarla.host.transport_pause()
  703. self.refreshTransport()
  704. @pyqtSlot()
  705. def slot_transportStop(self):
  706. if not gCarla.host.is_engine_running():
  707. return
  708. gCarla.host.transport_pause()
  709. gCarla.host.transport_relocate(0)
  710. self.refreshTransport()
  711. @pyqtSlot()
  712. def slot_transportBackwards(self):
  713. if not gCarla.host.is_engine_running():
  714. return
  715. newFrame = gCarla.host.get_current_transport_frame() - 100000
  716. if newFrame < 0:
  717. newFrame = 0
  718. gCarla.host.transport_relocate(newFrame)
  719. @pyqtSlot()
  720. def slot_transportForwards(self):
  721. if not gCarla.host.is_engine_running():
  722. return
  723. newFrame = gCarla.host.get_current_transport_frame() + 100000
  724. gCarla.host.transport_relocate(newFrame)
  725. # -----------------------------------------------------------------
  726. @pyqtSlot()
  727. def slot_aboutCarla(self):
  728. CarlaAboutW(self).exec_()
  729. @pyqtSlot()
  730. def slot_aboutQt(self):
  731. QApplication.instance().aboutQt()
  732. # -----------------------------------------------------------------
  733. @pyqtSlot(int)
  734. def slot_diskFolderChanged(self, index):
  735. if index < 0:
  736. return
  737. elif index == 0:
  738. filename = HOME
  739. self.ui.b_disk_remove.setEnabled(False)
  740. else:
  741. filename = self.ui.cb_disk.itemData(index)
  742. self.ui.b_disk_remove.setEnabled(True)
  743. self.fDirModel.setRootPath(filename)
  744. self.ui.fileTreeView.setRootIndex(self.fDirModel.index(filename))
  745. @pyqtSlot()
  746. def slot_diskFolderAdd(self):
  747. newPath = QFileDialog.getExistingDirectory(self, self.tr("New Folder"), "", QFileDialog.ShowDirsOnly)
  748. if newPath:
  749. if newPath[-1] == os.sep:
  750. newPath = newPath[:-1]
  751. self.ui.cb_disk.addItem(os.path.basename(newPath), newPath)
  752. self.ui.cb_disk.setCurrentIndex(self.ui.cb_disk.count()-1)
  753. self.ui.b_disk_remove.setEnabled(True)
  754. @pyqtSlot()
  755. def slot_diskFolderRemove(self):
  756. index = self.ui.cb_disk.currentIndex()
  757. if index <= 0:
  758. return
  759. self.ui.cb_disk.removeItem(index)
  760. if self.ui.cb_disk.currentIndex() == 0:
  761. self.ui.b_disk_remove.setEnabled(False)
  762. @pyqtSlot(QModelIndex)
  763. def slot_fileTreeDoubleClicked(self, modelIndex):
  764. filename = self.fDirModel.filePath(modelIndex)
  765. if not gCarla.host.load_file(filename):
  766. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
  767. self.tr("Failed to load file"),
  768. gCarla.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  769. # -----------------------------------------------------------------
  770. @pyqtSlot(int, int, int, float, str)
  771. def slot_handleDebugCallback(self, pluginId, value1, value2, value3, valueStr):
  772. print("DEBUG:", pluginId, value1, value2, value3, valueStr)
  773. #self.ui.pte_log.appendPlainText(valueStr.replace("", "DEBUG: ").replace("", "ERROR: ").replace("", "").replace("\n", ""))
  774. # -----------------------------------------------------------------
  775. @pyqtSlot(int, str)
  776. def slot_handlePluginAddedCallback(self, pluginId, pluginName):
  777. self.fContainer.addPlugin(pluginId, self.fIsProjectLoading)
  778. if self.fContainer.getPluginCount() == 1:
  779. self.ui.act_plugin_remove_all.setEnabled(True)
  780. @pyqtSlot(int)
  781. def slot_handlePluginRemovedCallback(self, pluginId):
  782. self.fContainer.removePlugin(pluginId)
  783. if self.fContainer.getPluginCount() == 0:
  784. self.ui.act_plugin_remove_all.setEnabled(False)
  785. @pyqtSlot(int, str)
  786. def slot_handlePluginRenamedCallback(self, pluginId, newName):
  787. self.fContainer.renamePlugin(pluginId, newName)
  788. @pyqtSlot(int, str)
  789. def slot_handlePluginUnavailableCallback(self, pluginId, errorMsg):
  790. self.fContainer.disablePlugin(pluginId, errorMsg)
  791. # -----------------------------------------------------------------
  792. @pyqtSlot(str)
  793. def slot_handleEngineStartedCallback(self, processMode, transportMode, driverName):
  794. gCarla.processMode = processMode
  795. gCarla.transportMode = transportMode
  796. gCarla.bufferSize = gCarla.host.get_buffer_size()
  797. gCarla.sampleRate = gCarla.host.get_sample_rate()
  798. self.slot_engineStart(False)
  799. if self.fIdleTimerFast == 0:
  800. self.fIdleTimerFast = self.startTimer(self.fSavedSettings[CARLA_KEY_MAIN_REFRESH_INTERVAL])
  801. if self.fIdleTimerSlow == 0:
  802. self.fIdleTimerSlow = self.startTimer(self.fSavedSettings[CARLA_KEY_MAIN_REFRESH_INTERVAL]*2)
  803. @pyqtSlot()
  804. def slot_handleEngineStoppedCallback(self):
  805. self.killTimers()
  806. self.slot_engineStop(False)
  807. gCarla.bufferSize = 0
  808. gCarla.sampleRate = 0.0
  809. # -----------------------------------------------------------------
  810. @pyqtSlot(int)
  811. def slot_handleProcessModeChangedCallback(self, newProcessMode):
  812. self.fEngineChanged = True
  813. @pyqtSlot(int)
  814. def slot_handleTransportModeChangedCallback(self, newTransportMode):
  815. self.fEngineChanged = True
  816. # -----------------------------------------------------------------
  817. @pyqtSlot(int)
  818. def slot_handleBufferSizeChangedCallback(self, newBufferSize):
  819. self.fEngineChanged = True
  820. @pyqtSlot(float)
  821. def slot_handleSampleRateChangedCallback(self, newSampleRate):
  822. self.fEngineChanged = True
  823. # -----------------------------------------------------------------
  824. @pyqtSlot(str)
  825. def slot_handleInfoCallback(self, info):
  826. QMessageBox.information(self, "Information", info)
  827. @pyqtSlot(str)
  828. def slot_handleErrorCallback(self, error):
  829. QMessageBox.critical(self, "Error", error)
  830. @pyqtSlot()
  831. def slot_handleQuitCallback(self):
  832. pass
  833. # -----------------------------------------------------------------
  834. @pyqtSlot()
  835. def slot_handleSIGUSR1(self):
  836. print("Got SIGUSR1 -> Saving project now")
  837. QTimer.singleShot(0, self.slot_fileSave)
  838. @pyqtSlot()
  839. def slot_handleSIGTERM(self):
  840. print("Got SIGTERM -> Closing now")
  841. self.close()
  842. # -----------------------------------------------------------------
  843. def timerEvent(self, event):
  844. if event.timerId() == self.fIdleTimerFast:
  845. #if not gCarla.isPlugin:
  846. gCarla.host.engine_idle()
  847. self.refreshTransport()
  848. self.fContainer.idleFast()
  849. elif event.timerId() == self.fIdleTimerSlow:
  850. if self.fEngineChanged:
  851. self.fContainer.engineChanged()
  852. self.fEngineChanged = False
  853. self.fContainer.idleSlow()
  854. QMainWindow.timerEvent(self, event)
  855. def closeEvent(self, event):
  856. self.killTimers()
  857. self.saveSettings()
  858. if gCarla.host.is_engine_running() and not gCarla.isPlugin:
  859. gCarla.host.set_engine_about_to_close()
  860. count = self.fContainer.getPluginCount()
  861. if count > 0:
  862. # simulate project loading, to disable container
  863. self.ui.act_plugin_remove_all.setEnabled(False)
  864. self.fContainer.projectLoadingStarted()
  865. app = QApplication.instance()
  866. for i in range(count):
  867. app.processEvents()
  868. gCarla.host.remove_plugin(count-i-1)
  869. app.processEvents()
  870. #self.fContainer.removeAllPlugins()
  871. #gCarla.host.remove_all_plugins()
  872. self.stopEngine()
  873. QMainWindow.closeEvent(self, event)
  874. # ------------------------------------------------------------------------------------------------------------
  875. # Engine callback
  876. def engineCallback(ptr, action, pluginId, value1, value2, value3, valueStr):
  877. if action == ENGINE_CALLBACK_PROCESS_MODE_CHANGED:
  878. gCarla.processMode = value1
  879. if gCarla.gui is not None:
  880. gCarla.gui.ProcessModeChangedCallback.emit(value1)
  881. return
  882. if action == ENGINE_CALLBACK_TRANSPORT_MODE_CHANGED:
  883. gCarla.transportMode = value1
  884. if gCarla.gui is not None:
  885. gCarla.gui.TransportModeChangedCallback.emit(value1)
  886. return
  887. if action == ENGINE_CALLBACK_BUFFER_SIZE_CHANGED:
  888. gCarla.bufferSize = value1
  889. if gCarla.gui is not None:
  890. gCarla.gui.BufferSizeChangedCallback.emit(value1)
  891. return
  892. if action == ENGINE_CALLBACK_SAMPLE_RATE_CHANGED:
  893. gCarla.sampleRate = value1
  894. if gCarla.gui is not None:
  895. gCarla.gui.SampleRateChangedCallback.emit(value3)
  896. return
  897. if gCarla.gui is None:
  898. print("WARNING: Got engine callback but UI is not ready : ", pluginId, value1, value2, value3, valueStr)
  899. return
  900. valueStr = charPtrToString(valueStr)
  901. if action == ENGINE_CALLBACK_DEBUG:
  902. gCarla.gui.DebugCallback.emit(pluginId, value1, value2, value3, valueStr)
  903. elif action == ENGINE_CALLBACK_PLUGIN_ADDED:
  904. gCarla.gui.PluginAddedCallback.emit(pluginId, valueStr)
  905. elif action == ENGINE_CALLBACK_PLUGIN_REMOVED:
  906. gCarla.gui.PluginRemovedCallback.emit(pluginId)
  907. elif action == ENGINE_CALLBACK_PLUGIN_RENAMED:
  908. gCarla.gui.PluginRenamedCallback.emit(pluginId, valueStr)
  909. elif action == ENGINE_CALLBACK_PLUGIN_UNAVAILABLE:
  910. gCarla.gui.PluginUnavailableCallback.emit(pluginId, valueStr)
  911. elif action == ENGINE_CALLBACK_PARAMETER_VALUE_CHANGED:
  912. gCarla.gui.ParameterValueChangedCallback.emit(pluginId, value1, value3)
  913. elif action == ENGINE_CALLBACK_PARAMETER_DEFAULT_CHANGED:
  914. gCarla.gui.ParameterDefaultChangedCallback.emit(pluginId, value1, value3)
  915. elif action == ENGINE_CALLBACK_PARAMETER_MIDI_CC_CHANGED:
  916. gCarla.gui.ParameterMidiCcChangedCallback.emit(pluginId, value1, value2)
  917. elif action == ENGINE_CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED:
  918. gCarla.gui.ParameterMidiChannelChangedCallback.emit(pluginId, value1, value2)
  919. elif action == ENGINE_CALLBACK_PROGRAM_CHANGED:
  920. gCarla.gui.ProgramChangedCallback.emit(pluginId, value1)
  921. elif action == ENGINE_CALLBACK_MIDI_PROGRAM_CHANGED:
  922. gCarla.gui.MidiProgramChangedCallback.emit(pluginId, value1)
  923. elif action == ENGINE_CALLBACK_UI_STATE_CHANGED:
  924. gCarla.gui.UiStateChangedCallback.emit(pluginId, value1)
  925. elif action == ENGINE_CALLBACK_NOTE_ON:
  926. gCarla.gui.NoteOnCallback.emit(pluginId, value1, value2, int(value3))
  927. elif action == ENGINE_CALLBACK_NOTE_OFF:
  928. gCarla.gui.NoteOffCallback.emit(pluginId, value1, value2)
  929. elif action == ENGINE_CALLBACK_UPDATE:
  930. gCarla.gui.UpdateCallback.emit(pluginId)
  931. elif action == ENGINE_CALLBACK_RELOAD_INFO:
  932. gCarla.gui.ReloadInfoCallback.emit(pluginId)
  933. elif action == ENGINE_CALLBACK_RELOAD_PARAMETERS:
  934. gCarla.gui.ReloadParametersCallback.emit(pluginId)
  935. elif action == ENGINE_CALLBACK_RELOAD_PROGRAMS:
  936. gCarla.gui.ReloadProgramsCallback.emit(pluginId)
  937. elif action == ENGINE_CALLBACK_RELOAD_ALL:
  938. gCarla.gui.ReloadAllCallback.emit(pluginId)
  939. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED:
  940. gCarla.gui.PatchbayClientAddedCallback.emit(pluginId, value1, value2, valueStr)
  941. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_REMOVED:
  942. gCarla.gui.PatchbayClientRemovedCallback.emit(pluginId)
  943. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_RENAMED:
  944. gCarla.gui.PatchbayClientRenamedCallback.emit(pluginId, valueStr)
  945. elif action == ENGINE_CALLBACK_PATCHBAY_CLIENT_DATA_CHANGED:
  946. gCarla.gui.PatchbayClientDataChangedCallback.emit(pluginId, value1, value2)
  947. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_ADDED:
  948. gCarla.gui.PatchbayPortAddedCallback.emit(pluginId, value1, value2, valueStr)
  949. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_REMOVED:
  950. gCarla.gui.PatchbayPortRemovedCallback.emit(pluginId, value1)
  951. elif action == ENGINE_CALLBACK_PATCHBAY_PORT_RENAMED:
  952. gCarla.gui.PatchbayPortRenamedCallback.emit(pluginId, value1, valueStr)
  953. elif action == ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED:
  954. gCarla.gui.PatchbayConnectionAddedCallback.emit(pluginId, value1, value2)
  955. elif action == ENGINE_CALLBACK_PATCHBAY_CONNECTION_REMOVED:
  956. gCarla.gui.PatchbayConnectionRemovedCallback.emit(pluginId, value1, value2)
  957. elif action == ENGINE_CALLBACK_ENGINE_STARTED:
  958. gCarla.gui.EngineStartedCallback.emit(value1, value2, valueStr)
  959. elif action == ENGINE_CALLBACK_ENGINE_STOPPED:
  960. gCarla.gui.killTimers()
  961. gCarla.gui.EngineStoppedCallback.emit()
  962. elif action == ENGINE_CALLBACK_IDLE:
  963. QApplication.instance().processEvents()
  964. elif action == ENGINE_CALLBACK_INFO:
  965. gCarla.gui.InfoCallback.emit(valueStr)
  966. elif action == ENGINE_CALLBACK_ERROR:
  967. gCarla.gui.ErrorCallback.emit(valueStr)
  968. elif action == ENGINE_CALLBACK_QUIT:
  969. gCarla.gui.QuitCallback.emit()
  970. # ------------------------------------------------------------------------------------------------------------
  971. # File callback
  972. def fileCallback(ptr, action, isDir, title, filter):
  973. if gCarla.gui is None:
  974. return None
  975. ret = ""
  976. if action == FILE_CALLBACK_DEBUG:
  977. pass
  978. elif action == FILE_CALLBACK_OPEN:
  979. ret = QFileDialog.getOpenFileName(gCarla.gui, charPtrToString(title), "", charPtrToString(filter) ) #, QFileDialog.ShowDirsOnly if isDir else 0x0)
  980. elif action == FILE_CALLBACK_SAVE:
  981. ret = QFileDialog.getSaveFileName(gCarla.gui, charPtrToString(title), "", charPtrToString(filter), QFileDialog.ShowDirsOnly if isDir else 0x0)
  982. if not ret:
  983. return None
  984. gCarla.gui._fileRet = c_char_p(ret.encode("utf-8"))
  985. retval = cast(byref(gCarla.gui._fileRet), POINTER(c_uintptr))
  986. return retval.contents.value