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.

696 lines
21KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla rack widget 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 Qt, QSize, QTimer
  20. from PyQt4.QtGui import QAbstractItemView, QApplication, QHBoxLayout, QLabel, QListWidget, QListWidgetItem, QPixmap, QScrollBar
  21. # ------------------------------------------------------------------------------------------------------------
  22. # Imports (Custom Stuff)
  23. from carla_skin import *
  24. # ------------------------------------------------------------------------------------------------------------
  25. # Rack widget item
  26. class CarlaRackItem(QListWidgetItem):
  27. kRackItemType = QListWidgetItem.UserType + 1
  28. def __init__(self, parent, pluginId):
  29. QListWidgetItem.__init__(self, parent, self.kRackItemType)
  30. self.fParent = parent
  31. self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled)
  32. #self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled|Qt.ItemIsDragEnabled|Qt.ItemIsDropEnabled)
  33. self.createWidget(pluginId)
  34. # -----------------------------------------------------------------
  35. def createWidget(self, pluginId):
  36. self.widget = createPluginSlot(self.fParent, pluginId)
  37. self.widget.setFixedHeight(self.widget.getFixedHeight())
  38. self.setSizeHint(QSize(640, self.widget.getFixedHeight()))
  39. self.fParent.setItemWidget(self, self.widget)
  40. # -----------------------------------------------------------------
  41. def close(self):
  42. self.widget.fEditDialog.close()
  43. def reloadAll(self, pluginId):
  44. self.widget.fEditDialog.close()
  45. del self.widget
  46. self.createWidget(pluginId)
  47. # ------------------------------------------------------------------------------------------------------------
  48. # Rack widget list
  49. class CarlaRackList(QListWidget):
  50. def __init__(self, parent):
  51. QListWidget.__init__(self, parent)
  52. exts = gCarla.host.get_supported_file_extensions().split(";") if gCarla.host is not None else ["wav",]
  53. # plugin files
  54. exts.append("dll")
  55. exts.append("so")
  56. if MACOS:
  57. exts.append("dylib")
  58. self.fSupportedExtensions = tuple(i.replace("*.","") for i in exts)
  59. self.fWasLastDragValid = False
  60. self.setMinimumWidth(640+20) # required by zita, 591 was old value
  61. self.setSelectionMode(QAbstractItemView.SingleSelection)
  62. self.setSortingEnabled(False)
  63. #self.setSortingEnabled(True)
  64. self.setDragEnabled(True)
  65. self.setDragDropMode(QAbstractItemView.DropOnly)
  66. self.setDropIndicatorShown(True)
  67. self.viewport().setAcceptDrops(True)
  68. self.fPixmapL = QPixmap(":/bitmaps/rack_interior_left.png")
  69. self.fPixmapR = QPixmap(":/bitmaps/rack_interior_right.png")
  70. self.fPixmapWidth = self.fPixmapL.width()
  71. def isDragEventValid(self, urls):
  72. for url in urls:
  73. filename = url.toLocalFile()
  74. if os.path.isdir(filename):
  75. if os.path.exists(os.path.join(filename, "manifest.ttl")):
  76. return False
  77. elif os.path.isfile(filename):
  78. if filename.lower().endswith(self.fSupportedExtensions):
  79. return True
  80. return False
  81. def dragEnterEvent(self, event):
  82. if self.isDragEventValid(event.mimeData().urls()):
  83. self.fWasLastDragValid = True
  84. event.acceptProposedAction()
  85. return
  86. self.fWasLastDragValid = False
  87. QListWidget.dragEnterEvent(self, event)
  88. def dragMoveEvent(self, event):
  89. if self.fWasLastDragValid:
  90. event.acceptProposedAction()
  91. tryItem = self.itemAt(event.pos())
  92. if tryItem is not None:
  93. self.setCurrentRow(tryItem.widget.getPluginId())
  94. else:
  95. self.setCurrentRow(-1)
  96. return
  97. QListWidget.dragMoveEvent(self, event)
  98. #def dragLeaveEvent(self, event):
  99. #self.fWasLastDragValid = False
  100. #QListWidget.dragLeaveEvent(self, event)
  101. def dropEvent(self, event):
  102. event.acceptProposedAction()
  103. urls = event.mimeData().urls()
  104. if len(urls) == 0:
  105. return
  106. tryItem = self.itemAt(event.pos())
  107. if tryItem is not None:
  108. pluginId = tryItem.widget.getPluginId()
  109. gCarla.host.replace_plugin(pluginId)
  110. for url in urls:
  111. filename = url.toLocalFile()
  112. if not gCarla.host.load_file(filename):
  113. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
  114. self.tr("Failed to load file"),
  115. gCarla.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  116. if tryItem is not None:
  117. gCarla.host.replace_plugin(self.parent().fPluginCount)
  118. tryItem.widget.setActive(True, True, True)
  119. def mousePressEvent(self, event):
  120. if self.itemAt(event.pos()) is None:
  121. event.accept()
  122. self.setCurrentRow(-1)
  123. return
  124. QListWidget.mousePressEvent(self, event)
  125. def paintEvent(self, event):
  126. painter = QPainter(self.viewport())
  127. painter.drawTiledPixmap(0, 0, self.fPixmapWidth, self.height(), self.fPixmapL)
  128. painter.drawTiledPixmap(self.width()-self.fPixmapWidth-2, 0, self.fPixmapWidth, self.height(), self.fPixmapR)
  129. QListWidget.paintEvent(self, event)
  130. # ------------------------------------------------------------------------------------------------------------
  131. # Rack widget
  132. class CarlaRackW(QFrame):
  133. def __init__(self, parent, doSetup = True):
  134. QFrame.__init__(self, parent)
  135. self.fLayout = QHBoxLayout(self)
  136. self.fLayout.setContentsMargins(0, 0, 0, 0)
  137. self.fLayout.setSpacing(0)
  138. self.setLayout(self.fLayout)
  139. self.fPadLeft = QLabel(self)
  140. self.fPadLeft.setFixedWidth(25)
  141. self.fPadLeft.setObjectName("PadLeft")
  142. self.fPadLeft.setText("")
  143. self.fPadRight = QLabel(self)
  144. self.fPadRight.setFixedWidth(25)
  145. self.fPadRight.setObjectName("PadRight")
  146. self.fPadRight.setText("")
  147. self.fRack = CarlaRackList(self)
  148. self.fRack.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
  149. self.fRack.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
  150. self.fRack.currentRowChanged.connect(self.slot_currentRowChanged)
  151. sb = self.fRack.verticalScrollBar()
  152. self.fScrollBar = QScrollBar(Qt.Vertical, self)
  153. self.fScrollBar.setMinimum(sb.minimum())
  154. self.fScrollBar.setMaximum(sb.maximum())
  155. self.fScrollBar.setValue(sb.value())
  156. #sb.actionTriggered.connect(self.fScrollBar.triggerAction)
  157. #sb.sliderMoved.connect(self.fScrollBar.)
  158. #sb.sliderPressed.connect(self.fScrollBar.)
  159. #sb.sliderReleased.connect(self.fScrollBar.)
  160. sb.rangeChanged.connect(self.fScrollBar.setRange)
  161. sb.valueChanged.connect(self.fScrollBar.setValue)
  162. self.fScrollBar.rangeChanged.connect(sb.setRange)
  163. self.fScrollBar.valueChanged.connect(sb.setValue)
  164. self.fLayout.addWidget(self.fPadLeft)
  165. self.fLayout.addWidget(self.fRack)
  166. self.fLayout.addWidget(self.fPadRight)
  167. self.fLayout.addWidget(self.fScrollBar)
  168. # -------------------------------------------------------------
  169. # Internal stuff
  170. self.fParent = parent
  171. self.fPluginCount = 0
  172. self.fPluginList = []
  173. self.fCurrentRow = -1
  174. self.fLastSelectedItem = None
  175. # -------------------------------------------------------------
  176. # Set-up GUI stuff
  177. #app = QApplication.instance()
  178. #pal1 = app.palette().base().color()
  179. #pal2 = app.palette().button().color()
  180. #col1 = "stop:0 rgb(%i, %i, %i)" % (pal1.red(), pal1.green(), pal1.blue())
  181. #col2 = "stop:1 rgb(%i, %i, %i)" % (pal2.red(), pal2.green(), pal2.blue())
  182. self.setStyleSheet("""
  183. QLabel#PadLeft {
  184. background-image: url(:/bitmaps/rack_padding_left.png);
  185. background-repeat: repeat-y;
  186. }
  187. QLabel#PadRight {
  188. background-image: url(:/bitmaps/rack_padding_right.png);
  189. background-repeat: repeat-y;
  190. }
  191. QListWidget {
  192. background-color: black;
  193. }
  194. """)
  195. # -------------------------------------------------------------
  196. # Connect actions to functions
  197. if not doSetup: return
  198. parent.ui.menu_Canvas.hide()
  199. parent.ui.act_plugins_enable.triggered.connect(self.slot_pluginsEnable)
  200. parent.ui.act_plugins_disable.triggered.connect(self.slot_pluginsDisable)
  201. parent.ui.act_plugins_volume100.triggered.connect(self.slot_pluginsVolume100)
  202. parent.ui.act_plugins_mute.triggered.connect(self.slot_pluginsMute)
  203. parent.ui.act_plugins_wet100.triggered.connect(self.slot_pluginsWet100)
  204. parent.ui.act_plugins_bypass.triggered.connect(self.slot_pluginsBypass)
  205. parent.ui.act_plugins_center.triggered.connect(self.slot_pluginsCenter)
  206. parent.ui.act_plugins_panic.triggered.connect(self.slot_pluginsDisable)
  207. parent.ui.act_settings_configure.triggered.connect(self.slot_configureCarla)
  208. parent.ParameterValueChangedCallback.connect(self.slot_handleParameterValueChangedCallback)
  209. parent.ParameterDefaultChangedCallback.connect(self.slot_handleParameterDefaultChangedCallback)
  210. parent.ParameterMidiChannelChangedCallback.connect(self.slot_handleParameterMidiChannelChangedCallback)
  211. parent.ParameterMidiCcChangedCallback.connect(self.slot_handleParameterMidiCcChangedCallback)
  212. parent.ProgramChangedCallback.connect(self.slot_handleProgramChangedCallback)
  213. parent.MidiProgramChangedCallback.connect(self.slot_handleMidiProgramChangedCallback)
  214. parent.UiStateChangedCallback.connect(self.slot_handleUiStateChangedCallback)
  215. parent.NoteOnCallback.connect(self.slot_handleNoteOnCallback)
  216. parent.NoteOffCallback.connect(self.slot_handleNoteOffCallback)
  217. parent.UpdateCallback.connect(self.slot_handleUpdateCallback)
  218. parent.ReloadInfoCallback.connect(self.slot_handleReloadInfoCallback)
  219. parent.ReloadParametersCallback.connect(self.slot_handleReloadParametersCallback)
  220. parent.ReloadProgramsCallback.connect(self.slot_handleReloadProgramsCallback)
  221. parent.ReloadAllCallback.connect(self.slot_handleReloadAllCallback)
  222. # -----------------------------------------------------------------
  223. def getPluginCount(self):
  224. return self.fPluginCount
  225. # -----------------------------------------------------------------
  226. def addPlugin(self, pluginId, isProjectLoading):
  227. pitem = CarlaRackItem(self.fRack, pluginId)
  228. self.fPluginList.append(pitem)
  229. self.fPluginCount += 1
  230. if not isProjectLoading:
  231. pitem.widget.setActive(True, True, True)
  232. def removePlugin(self, pluginId):
  233. if pluginId >= self.fPluginCount:
  234. return
  235. pitem = self.fPluginList[pluginId]
  236. if pitem is None:
  237. return
  238. self.fPluginCount -= 1
  239. self.fPluginList.pop(pluginId)
  240. self.fRack.takeItem(pluginId)
  241. pitem.close()
  242. del pitem
  243. # push all plugins 1 slot back
  244. for i in range(pluginId, self.fPluginCount):
  245. pitem = self.fPluginList[i]
  246. pitem.widget.setId(i)
  247. def renamePlugin(self, pluginId, newName):
  248. if pluginId >= self.fPluginCount:
  249. return
  250. pitem = self.fPluginList[pluginId]
  251. if pitem is None:
  252. return
  253. pitem.widget.setName(newName)
  254. def disablePlugin(self, pluginId, errorMsg):
  255. if pluginId >= self.fPluginCount:
  256. return
  257. pitem = self.fPluginList[pluginId]
  258. if pitem is None:
  259. return
  260. def removeAllPlugins(self):
  261. while self.fRack.takeItem(0):
  262. pass
  263. for i in range(self.fPluginCount):
  264. pitem = self.fPluginList[i]
  265. if pitem is None:
  266. break
  267. pitem.close()
  268. del pitem
  269. self.fPluginCount = 0
  270. self.fPluginList = []
  271. # -----------------------------------------------------------------
  272. def engineStarted(self):
  273. pass
  274. def engineStopped(self):
  275. pass
  276. def engineChanged(self):
  277. pass
  278. # -----------------------------------------------------------------
  279. def idleFast(self):
  280. for i in range(self.fPluginCount):
  281. pitem = self.fPluginList[i]
  282. if pitem is None:
  283. break
  284. pitem.widget.idleFast()
  285. def idleSlow(self):
  286. for i in range(self.fPluginCount):
  287. pitem = self.fPluginList[i]
  288. if pitem is None:
  289. break
  290. pitem.widget.idleSlow()
  291. # -----------------------------------------------------------------
  292. def projectLoaded(self):
  293. pass
  294. def saveSettings(self, settings):
  295. pass
  296. def showEditDialog(self, pluginId):
  297. if pluginId >= self.fPluginCount:
  298. return
  299. pitem = self.fPluginList[pluginId]
  300. if pitem is None:
  301. return
  302. pitem.widget.slot_showEditDialog(True)
  303. # -----------------------------------------------------------------
  304. @pyqtSlot()
  305. def slot_pluginsEnable(self):
  306. if not gCarla.host.is_engine_running():
  307. return
  308. for i in range(self.fPluginCount):
  309. pitem = self.fPluginList[i]
  310. if pitem is None:
  311. break
  312. pitem.widget.setActive(True, True, True)
  313. @pyqtSlot()
  314. def slot_pluginsDisable(self):
  315. if not gCarla.host.is_engine_running():
  316. return
  317. for i in range(self.fPluginCount):
  318. pitem = self.fPluginList[i]
  319. if pitem is None:
  320. break
  321. pitem.widget.setActive(False, True, True)
  322. @pyqtSlot()
  323. def slot_pluginsVolume100(self):
  324. if not gCarla.host.is_engine_running():
  325. return
  326. for i in range(self.fPluginCount):
  327. pitem = self.fPluginList[i]
  328. if pitem is None:
  329. break
  330. pitem.widget.setInternalParameter(PLUGIN_CAN_VOLUME, 1.0)
  331. @pyqtSlot()
  332. def slot_pluginsMute(self):
  333. if not gCarla.host.is_engine_running():
  334. return
  335. for i in range(self.fPluginCount):
  336. pitem = self.fPluginList[i]
  337. if pitem is None:
  338. break
  339. pitem.widget.setInternalParameter(PLUGIN_CAN_VOLUME, 0.0)
  340. @pyqtSlot()
  341. def slot_pluginsWet100(self):
  342. if not gCarla.host.is_engine_running():
  343. return
  344. for i in range(self.fPluginCount):
  345. pitem = self.fPluginList[i]
  346. if pitem is None:
  347. break
  348. pitem.widget.setInternalParameter(PLUGIN_CAN_DRYWET, 1.0)
  349. @pyqtSlot()
  350. def slot_pluginsBypass(self):
  351. if not gCarla.host.is_engine_running():
  352. return
  353. for i in range(self.fPluginCount):
  354. pitem = self.fPluginList[i]
  355. if pitem is None:
  356. break
  357. pitem.widget.setInternalParameter(PLUGIN_CAN_DRYWET, 0.0)
  358. @pyqtSlot()
  359. def slot_pluginsCenter(self):
  360. if not gCarla.host.is_engine_running():
  361. return
  362. for i in range(self.fPluginCount):
  363. pitem = self.fPluginList[i]
  364. if pitem is None:
  365. break
  366. pitem.widget.setInternalParameter(PARAMETER_BALANCE_LEFT, -1.0)
  367. pitem.widget.setInternalParameter(PARAMETER_BALANCE_RIGHT, 1.0)
  368. pitem.widget.setInternalParameter(PARAMETER_PANNING, 0.0)
  369. # -----------------------------------------------------------------
  370. @pyqtSlot()
  371. def slot_configureCarla(self):
  372. if self.fParent is None or not self.fParent.openSettingsWindow(False, False):
  373. return
  374. self.fParent.loadSettings(False)
  375. # -----------------------------------------------------------------
  376. @pyqtSlot(int, int, float)
  377. def slot_handleParameterValueChangedCallback(self, pluginId, index, value):
  378. if pluginId >= self.fPluginCount:
  379. return
  380. pitem = self.fPluginList[pluginId]
  381. if pitem is None:
  382. return
  383. pitem.widget.setParameterValue(index, value, True)
  384. @pyqtSlot(int, int, float)
  385. def slot_handleParameterDefaultChangedCallback(self, pluginId, index, value):
  386. if pluginId >= self.fPluginCount:
  387. return
  388. pitem = self.fPluginList[pluginId]
  389. if pitem is None:
  390. return
  391. pitem.widget.setParameterDefault(index, value)
  392. @pyqtSlot(int, int, int)
  393. def slot_handleParameterMidiCcChangedCallback(self, pluginId, index, cc):
  394. if pluginId >= self.fPluginCount:
  395. return
  396. pitem = self.fPluginList[pluginId]
  397. if pitem is None:
  398. return
  399. pitem.widget.setParameterMidiControl(index, cc)
  400. @pyqtSlot(int, int, int)
  401. def slot_handleParameterMidiChannelChangedCallback(self, pluginId, index, channel):
  402. if pluginId >= self.fPluginCount:
  403. return
  404. pitem = self.fPluginList[pluginId]
  405. if pitem is None:
  406. return
  407. pitem.widget.setParameterMidiChannel(index, channel)
  408. # -----------------------------------------------------------------
  409. @pyqtSlot(int, int)
  410. def slot_handleProgramChangedCallback(self, pluginId, index):
  411. if pluginId >= self.fPluginCount:
  412. return
  413. pitem = self.fPluginList[pluginId]
  414. if pitem is None:
  415. return
  416. pitem.widget.setProgram(index, True)
  417. @pyqtSlot(int, int)
  418. def slot_handleMidiProgramChangedCallback(self, pluginId, index):
  419. if pluginId >= self.fPluginCount:
  420. return
  421. pitem = self.fPluginList[pluginId]
  422. if pitem is None:
  423. return
  424. pitem.widget.setMidiProgram(index, True)
  425. # -----------------------------------------------------------------
  426. @pyqtSlot(int, int)
  427. def slot_handleUiStateChangedCallback(self, pluginId, state):
  428. if pluginId >= self.fPluginCount:
  429. return
  430. pitem = self.fPluginList[pluginId]
  431. if pitem is None:
  432. return
  433. pitem.widget.customUiStateChanged(state)
  434. # -----------------------------------------------------------------
  435. @pyqtSlot(int, int, int, int)
  436. def slot_handleNoteOnCallback(self, pluginId, channel, note, velo):
  437. if pluginId >= self.fPluginCount:
  438. return
  439. pitem = self.fPluginList[pluginId]
  440. if pitem is None:
  441. return
  442. pitem.widget.sendNoteOn(channel, note)
  443. @pyqtSlot(int, int, int)
  444. def slot_handleNoteOffCallback(self, pluginId, channel, note):
  445. if pluginId >= self.fPluginCount:
  446. return
  447. pitem = self.fPluginList[pluginId]
  448. if pitem is None:
  449. return
  450. pitem.widget.sendNoteOff(channel, note)
  451. # -----------------------------------------------------------------
  452. @pyqtSlot(int)
  453. def slot_handleUpdateCallback(self, pluginId):
  454. if pluginId >= self.fPluginCount:
  455. return
  456. pitem = self.fPluginList[pluginId]
  457. if pitem is None:
  458. return
  459. pitem.widget.fEditDialog.updateInfo()
  460. @pyqtSlot(int)
  461. def slot_handleReloadInfoCallback(self, pluginId):
  462. if pluginId >= self.fPluginCount:
  463. return
  464. pitem = self.fPluginList[pluginId]
  465. if pitem is None:
  466. return
  467. pitem.widget.fEditDialog.reloadInfo()
  468. @pyqtSlot(int)
  469. def slot_handleReloadParametersCallback(self, pluginId):
  470. if pluginId >= self.fPluginCount:
  471. return
  472. pitem = self.fPluginList[pluginId]
  473. if pitem is None:
  474. return
  475. pitem.widget.fEditDialog.reloadParameters()
  476. @pyqtSlot(int)
  477. def slot_handleReloadProgramsCallback(self, pluginId):
  478. if pluginId >= self.fPluginCount:
  479. return
  480. pitem = self.fPluginList[pluginId]
  481. if pitem is None:
  482. return
  483. pitem.widget.fEditDialog.reloadPrograms()
  484. @pyqtSlot(int)
  485. def slot_handleReloadAllCallback(self, pluginId):
  486. if pluginId >= self.fPluginCount:
  487. return
  488. pitem = self.fPluginList[pluginId]
  489. if pitem is None:
  490. return
  491. self.fRack.setCurrentRow(-1)
  492. self.fCurrentRow = -1
  493. self.fLastSelectedItem = None
  494. pitem.reloadAll(pluginId)
  495. # -----------------------------------------------------------------
  496. def slot_currentRowChanged(self, row):
  497. self.fCurrentRow = row
  498. if self.fLastSelectedItem is not None:
  499. self.fLastSelectedItem.setSelected(False)
  500. if row < 0 or row >= self.fPluginCount or self.fPluginList[row] is None:
  501. self.fLastSelectedItem = None
  502. return
  503. pitem = self.fPluginList[row]
  504. pitem.widget.setSelected(True)
  505. self.fLastSelectedItem = pitem.widget
  506. # -----------------------------------------------------------------