#!/usr/bin/env python3 # -*- coding: utf-8 -*- # Carla rack widget code # Copyright (C) 2011-2014 Filipe Coelho # # This program is free software; you can redistribute it and/or # modify it under the terms of the GNU General Public License as # published by the Free Software Foundation; either version 2 of # the License, or any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # For a full copy of the GNU General Public License see the doc/GPL.txt file. # ------------------------------------------------------------------------------------------------------------ # Imports (Config) from carla_config import * # ------------------------------------------------------------------------------------------------------------ # Imports (Global) if config_UseQt5: from PyQt5.QtCore import Qt, QSize, QTimer from PyQt5.QtGui import QPixmap from PyQt5.QtWidgets import QAbstractItemView, QApplication, QHBoxLayout, QLabel, QListWidget, QListWidgetItem, QScrollBar else: from PyQt4.QtCore import Qt, QSize, QTimer from PyQt4.QtGui import QAbstractItemView, QApplication, QHBoxLayout, QLabel, QListWidget, QListWidgetItem, QPixmap, QScrollBar # ------------------------------------------------------------------------------------------------------------ # Imports (Custom Stuff) from carla_host import * from carla_skin import * # ------------------------------------------------------------------------------------------------------------ # Rack widget item class CarlaRackItem(QListWidgetItem): kRackItemType = QListWidgetItem.UserType + 1 def __init__(self, parent, pluginId, useSkins): QListWidgetItem.__init__(self, parent, self.kRackItemType) # ---------------------------------------------------------------------------------------------------- # Internal stuff self.fParent = parent self.fPluginId = pluginId self.fUseSkins = useSkins self.fWidget = None self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled) #self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled|Qt.ItemIsDragEnabled|Qt.ItemIsDropEnabled) # ---------------------------------------------------------------------------------------------------- # Set-up GUI self.recreateWidget() if False: self.fWidget = AbstractPluginSlot(parent, parent.host, pluginId) # -------------------------------------------------------------------------------------------------------- def setPluginId(self, pluginId): self.fPluginId = pluginId self.fWidget.setPluginId(pluginId) # -------------------------------------------------------------------------------------------------------- def getEditDialog(self): return self.fWidget.fEditDialog def closeEditDialog(self): self.fWidget.fEditDialog.close() # -------------------------------------------------------------------------------------------------------- def getWidget(self): return self.fWidget def recreateWidget(self): if self.fWidget is not None: #self.fWidget.fEditDialog.close() del self.fWidget self.fWidget = createPluginSlot(self.fParent, self.fParent.host, self.fPluginId, self.fUseSkins) self.fWidget.setFixedHeight(self.fWidget.getFixedHeight()) self.setSizeHint(QSize(640, self.fWidget.getFixedHeight())) self.fParent.setItemWidget(self, self.fWidget) # ------------------------------------------------------------------------------------------------------------ # Rack widget list class CarlaRackList(QListWidget): def __init__(self, parent, host): QListWidget.__init__(self, parent) self.host = host if False: # kdevelop likes this :) host = CarlaHostMeta() self.host = host # ------------------------------------------------------------- exts = host.get_supported_file_extensions().split(";") # plugin files exts.append("dll") if MACOS: exts.append("dylib") if not WINDOWS: exts.append("so") self.fSupportedExtensions = tuple(i.replace("*.","") for i in exts) self.fWasLastDragValid = False self.setMinimumWidth(640+20) # required by zita, 591 was old value self.setSelectionMode(QAbstractItemView.SingleSelection) self.setSortingEnabled(False) #self.setSortingEnabled(True) self.setDragEnabled(True) self.setDragDropMode(QAbstractItemView.DropOnly) self.setDropIndicatorShown(True) self.viewport().setAcceptDrops(True) self.setFrameShape(QFrame.NoFrame) self.setFrameShadow(QFrame.Plain) self.fPixmapL = QPixmap(":/bitmaps/rack_interior_left.png") self.fPixmapR = QPixmap(":/bitmaps/rack_interior_right.png") self.fPixmapWidth = self.fPixmapL.width() def isDragEventValid(self, urls): for url in urls: filename = url.toLocalFile() if os.path.isdir(filename): if os.path.exists(os.path.join(filename, "manifest.ttl")): return True elif os.path.isfile(filename): if filename.lower().endswith(self.fSupportedExtensions): return True return False def dragEnterEvent(self, event): if self.isDragEventValid(event.mimeData().urls()): self.fWasLastDragValid = True event.acceptProposedAction() return self.fWasLastDragValid = False QListWidget.dragEnterEvent(self, event) def dragMoveEvent(self, event): if self.fWasLastDragValid: event.acceptProposedAction() tryItem = self.itemAt(event.pos()) if tryItem is not None: self.setCurrentRow(tryItem.widget.getPluginId()) else: self.setCurrentRow(-1) return QListWidget.dragMoveEvent(self, event) #def dragLeaveEvent(self, event): #self.fWasLastDragValid = False #QListWidget.dragLeaveEvent(self, event) def dropEvent(self, event): event.acceptProposedAction() urls = event.mimeData().urls() if len(urls) == 0: return tryItem = self.itemAt(event.pos()) if tryItem is not None: pluginId = tryItem.widget.getPluginId() self.host.replace_plugin(pluginId) for url in urls: filename = url.toLocalFile() if not self.host.load_file(filename): CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"), self.tr("Failed to load file"), self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok) if tryItem is not None: self.host.replace_plugin(self.parent().fPluginCount) #tryItem.widget.setActive(True, True, True) def mousePressEvent(self, event): if self.itemAt(event.pos()) is None: event.accept() self.setCurrentRow(-1) return QListWidget.mousePressEvent(self, event) def paintEvent(self, event): painter = QPainter(self.viewport()) painter.drawTiledPixmap(0, 0, self.fPixmapWidth, self.height(), self.fPixmapL) painter.drawTiledPixmap(self.width()-self.fPixmapWidth-2, 0, self.fPixmapWidth, self.height(), self.fPixmapR) QListWidget.paintEvent(self, event) # ------------------------------------------------------------------------------------------------------------ # Rack widget class CarlaRackW(QFrame, HostWidgetMeta): #class CarlaRackW(QFrame, HostWidgetMeta, metaclass=PyQtMetaClass): def __init__(self, parent, host, doSetup = True): QFrame.__init__(self, parent) self.host = host if False: # kdevelop likes this :) host = CarlaHostMeta() self.host = host # ------------------------------------------------------------- self.fLayout = QHBoxLayout(self) self.fLayout.setContentsMargins(0, 0, 0, 0) self.fLayout.setSpacing(0) self.setLayout(self.fLayout) self.fPadLeft = QLabel(self) self.fPadLeft.setFixedWidth(25) self.fPadLeft.setObjectName("PadLeft") self.fPadLeft.setText("") self.fPadRight = QLabel(self) self.fPadRight.setFixedWidth(25) self.fPadRight.setObjectName("PadRight") self.fPadRight.setText("") self.fRack = CarlaRackList(self, host) self.fRack.setObjectName("CarlaRackList") self.fRack.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff) self.fRack.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff) self.fRack.currentRowChanged.connect(self.slot_currentRowChanged) sb = self.fRack.verticalScrollBar() self.fScrollBar = QScrollBar(Qt.Vertical, self) self.fScrollBar.setMinimum(sb.minimum()) self.fScrollBar.setMaximum(sb.maximum()) self.fScrollBar.setValue(sb.value()) #sb.actionTriggered.connect(self.fScrollBar.triggerAction) #sb.sliderMoved.connect(self.fScrollBar.) #sb.sliderPressed.connect(self.fScrollBar.) #sb.sliderReleased.connect(self.fScrollBar.) sb.rangeChanged.connect(self.fScrollBar.setRange) sb.valueChanged.connect(self.fScrollBar.setValue) self.fScrollBar.rangeChanged.connect(sb.setRange) self.fScrollBar.valueChanged.connect(sb.setValue) self.fLayout.addWidget(self.fPadLeft) self.fLayout.addWidget(self.fRack) self.fLayout.addWidget(self.fPadRight) self.fLayout.addWidget(self.fScrollBar) # ------------------------------------------------------------- # Internal stuff self.fParent = parent self.fPluginCount = 0 self.fPluginList = [] self.fCurrentRow = -1 self.fLastSelectedItem = None # ------------------------------------------------------------- # Set-up GUI stuff #app = QApplication.instance() #pal1 = app.palette().base().color() #pal2 = app.palette().button().color() #col1 = "stop:0 rgb(%i, %i, %i)" % (pal1.red(), pal1.green(), pal1.blue()) #col2 = "stop:1 rgb(%i, %i, %i)" % (pal2.red(), pal2.green(), pal2.blue()) self.setStyleSheet(""" QLabel#PadLeft { background-image: url(:/bitmaps/rack_padding_left.png); background-repeat: repeat-y; } QLabel#PadRight { background-image: url(:/bitmaps/rack_padding_right.png); background-repeat: repeat-y; } CarlaRackList#CarlaRackList { background-color: black; } """) # ------------------------------------------------------------- # Connect actions to functions host.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback) host.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback) host.ReloadAllCallback.connect(self.slot_handleReloadAllCallback) if not doSetup: return parent.ui.menu_Canvas.hide() parent.ui.act_plugins_enable.triggered.connect(self.slot_pluginsEnable) parent.ui.act_plugins_disable.triggered.connect(self.slot_pluginsDisable) parent.ui.act_plugins_volume100.triggered.connect(self.slot_pluginsVolume100) parent.ui.act_plugins_mute.triggered.connect(self.slot_pluginsMute) parent.ui.act_plugins_wet100.triggered.connect(self.slot_pluginsWet100) parent.ui.act_plugins_bypass.triggered.connect(self.slot_pluginsBypass) parent.ui.act_plugins_center.triggered.connect(self.slot_pluginsCenter) parent.ui.act_plugins_panic.triggered.connect(self.slot_pluginsDisable) parent.ui.act_settings_configure.triggered.connect(self.slot_configureCarla) # ----------------------------------------------------------------- @pyqtSlot(int, str) def slot_handlePluginAddedCallback(self, pluginId, pluginName): pitem = CarlaRackItem(self.fRack, pluginId, self.fParent.getSavedSettings()[CARLA_KEY_MAIN_USE_CUSTOM_SKINS]) self.fPluginList.append(pitem) self.fPluginCount += 1 if not self.fParent.isProjectLoading(): pitem.getWidget().setActive(True, True, True) @pyqtSlot(int) def slot_handlePluginRemovedCallback(self, pluginId): pitem = self.getPluginItem(pluginId) self.fPluginCount -= 1 self.fPluginList.pop(pluginId) self.fRack.takeItem(pluginId) if pitem is not None: pitem.closeEditDialog() del pitem # push all plugins 1 slot back for i in range(pluginId, self.fPluginCount): pitem = self.fPluginList[i] pitem.setPluginId(i) # ----------------------------------------------------------------- # HostWidgetMeta methods def removeAllPlugins(self): while self.fRack.takeItem(0): pass for pitem in self.fPluginList: if pitem is None: break pitem.closeEditDialog() del pitem self.fPluginCount = 0 self.fPluginList = [] def engineStarted(self): pass def engineStopped(self): pass def idleFast(self): for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().idleFast() def idleSlow(self): for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().idleSlow() def projectLoadingStarted(self): self.fRack.setEnabled(False) def projectLoadingFinished(self): self.fRack.setEnabled(True) def saveSettings(self, settings): pass def showEditDialog(self, pluginId): dialog = self.getPluginEditDialog(pluginId) if dialog is None: return dialog.show() # ----------------------------------------------------------------- @pyqtSlot() def slot_pluginsEnable(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setActive(True, True, True) @pyqtSlot() def slot_pluginsDisable(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setActive(False, True, True) @pyqtSlot() def slot_pluginsVolume100(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setInternalParameter(PLUGIN_CAN_VOLUME, 1.0) @pyqtSlot() def slot_pluginsMute(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setInternalParameter(PLUGIN_CAN_VOLUME, 0.0) @pyqtSlot() def slot_pluginsWet100(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setInternalParameter(PLUGIN_CAN_DRYWET, 1.0) @pyqtSlot() def slot_pluginsBypass(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setInternalParameter(PLUGIN_CAN_DRYWET, 0.0) @pyqtSlot() def slot_pluginsCenter(self): if not self.host.is_engine_running(): return for pitem in self.fPluginList: if pitem is None: break pitem.getWidget().setInternalParameter(PARAMETER_BALANCE_LEFT, -1.0) pitem.getWidget().setInternalParameter(PARAMETER_BALANCE_RIGHT, 1.0) pitem.getWidget().setInternalParameter(PARAMETER_PANNING, 0.0) # ----------------------------------------------------------------- @pyqtSlot() def slot_configureCarla(self): dialog = CarlaSettingsW(self, self.host, False, False) if not dialog.exec_(): return self.fParent.loadSettings(False) # ----------------------------------------------------------------- @pyqtSlot(int) def slot_handleReloadAllCallback(self, pluginId): if pluginId >= self.fPluginCount: return pitem = self.fPluginList[pluginId] if pitem is None: return self.fRack.setCurrentRow(-1) self.fCurrentRow = -1 self.fLastSelectedItem = None pitem.recreateWidget() # ----------------------------------------------------------------- @pyqtSlot(int) def slot_currentRowChanged(self, row): self.fCurrentRow = row if self.fLastSelectedItem is not None: self.fLastSelectedItem.setSelected(False) if row < 0 or row >= self.fPluginCount or self.fPluginList[row] is None: self.fLastSelectedItem = None return pitem = self.fPluginList[row] pitem.getWidget().setSelected(True) self.fLastSelectedItem = pitem.getWidget() # ----------------------------------------------------------------- def getPluginItem(self, pluginId): if pluginId >= self.fPluginCount: return None pitem = self.fPluginList[pluginId] if pitem is None: return None if False: pitem = CarlaRackItem(self, 0, False) return pitem def getPluginEditDialog(self, pluginId): if pluginId >= self.fPluginCount: return None pitem = self.fPluginList[pluginId] if pitem is None: return None if False: pitem = CarlaRackItem(self, 0, False) return pitem.getEditDialog() def getPluginSlotWidget(self, pluginId): if pluginId >= self.fPluginCount: return None pitem = self.fPluginList[pluginId] if pitem is None: return None if False: pitem = CarlaRackItem(self, 0, False) return pitem.getWidget() # -----------------------------------------------------------------