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.

1487 lines
58KB

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