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.

319 lines
10KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Rack List Widget, a custom Qt4 widget
  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 Qt, QSize
  24. from PyQt5.QtGui import QPainter, QPixmap
  25. from PyQt5.QtWidgets import QAbstractItemView, QFrame, QListWidget, QListWidgetItem
  26. else:
  27. from PyQt4.QtCore import Qt, QSize
  28. from PyQt4.QtGui import QAbstractItemView, QFrame, QListWidget, QListWidgetItem, QPainter, QPixmap
  29. # ------------------------------------------------------------------------------------------------------------
  30. # Imports (Custom Stuff)
  31. from carla_skin import *
  32. # ------------------------------------------------------------------------------------------------------------
  33. # Rack Widget item
  34. class RackListItem(QListWidgetItem):
  35. kRackItemType = QListWidgetItem.UserType + 1
  36. kMinimumWidth = 620
  37. def __init__(self, parent, pluginId, useSkins):
  38. QListWidgetItem.__init__(self, parent, self.kRackItemType)
  39. self.host = parent.host
  40. if False:
  41. # kdevelop likes this :)
  42. parent = RackListWidget()
  43. host = CarlaHostMeta()
  44. self.host = host
  45. self.fWidget = AbstractPluginSlot()
  46. # ----------------------------------------------------------------------------------------------------
  47. # Internal stuff
  48. self.fParent = parent
  49. self.fPluginId = pluginId
  50. self.fWidget = None
  51. self.fOptions = {
  52. 'compact': False,
  53. 'useSkins': useSkins
  54. }
  55. self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled)
  56. #self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled|Qt.ItemIsDragEnabled)
  57. # ----------------------------------------------------------------------------------------------------
  58. # Set-up GUI
  59. self.recreateWidget()
  60. # --------------------------------------------------------------------------------------------------------
  61. def close(self):
  62. if self.fWidget is None:
  63. return
  64. widget = self.fWidget
  65. self.fWidget = None
  66. self.fParent.customClearSelection()
  67. self.fParent.setItemWidget(self, None)
  68. widget.fEditDialog.close()
  69. widget.fEditDialog.setParent(None)
  70. widget.fEditDialog.deleteLater()
  71. del widget.fEditDialog
  72. widget.close()
  73. widget.setParent(None)
  74. widget.deleteLater()
  75. del widget
  76. def getEditDialog(self):
  77. if self.fWidget is None:
  78. return None
  79. return self.fWidget.fEditDialog
  80. def getPluginId(self):
  81. return self.fPluginId
  82. def getWidget(self):
  83. return self.fWidget
  84. # --------------------------------------------------------------------------------------------------------
  85. def setPluginId(self, pluginId):
  86. self.fPluginId = pluginId
  87. if self.fWidget is not None:
  88. self.fWidget.setPluginId(pluginId)
  89. def setSelected(self, select):
  90. if self.fWidget is not None:
  91. self.fWidget.setSelected(select)
  92. QListWidgetItem.setSelected(self, select)
  93. # --------------------------------------------------------------------------------------------------------
  94. def recreateWidget(self, invertCompactOption = False):
  95. if invertCompactOption:
  96. self.fOptions['compact'] = not self.fOptions['compact']
  97. self.close()
  98. self.fWidget = createPluginSlot(self.fParent, self.host, self.fPluginId, self.fOptions)
  99. self.fWidget.setFixedHeight(self.fWidget.getFixedHeight())
  100. self.setSizeHint(QSize(self.kMinimumWidth, self.fWidget.getFixedHeight()))
  101. self.fParent.setItemWidget(self, self.fWidget)
  102. # ------------------------------------------------------------------------------------------------------------
  103. # Rack Widget
  104. class RackListWidget(QListWidget):
  105. def __init__(self, parent):
  106. QListWidget.__init__(self, parent)
  107. self.host = None
  108. if False:
  109. # kdevelop likes this :)
  110. from carla_backend import CarlaHostMeta
  111. host = CarlaHostMeta()
  112. self.host = host
  113. exts = gCarla.utils.get_supported_file_extensions().split(";")
  114. exts.append(".dll")
  115. if MACOS:
  116. exts.append(".dylib")
  117. if not WINDOWS:
  118. exts.append(".so")
  119. self.fSupportedExtensions = tuple(i.replace("*","").lower() for i in exts)
  120. self.fLastSelectedItem = None
  121. self.fWasLastDragValid = False
  122. self.fPixmapL = QPixmap(":/bitmaps/rack_interior_left.png")
  123. self.fPixmapR = QPixmap(":/bitmaps/rack_interior_right.png")
  124. self.fPixmapWidth = self.fPixmapL.width()
  125. self.setMinimumWidth(RackListItem.kMinimumWidth)
  126. self.setSelectionMode(QAbstractItemView.SingleSelection)
  127. self.setSortingEnabled(False)
  128. self.setDragEnabled(True)
  129. self.setDragDropMode(QAbstractItemView.DropOnly)
  130. self.setDropIndicatorShown(True)
  131. self.viewport().setAcceptDrops(True)
  132. self.setFrameShape(QFrame.NoFrame)
  133. self.setFrameShadow(QFrame.Plain)
  134. # --------------------------------------------------------------------------------------------------------
  135. def createItem(self, pluginId, useSkins):
  136. return RackListItem(self, pluginId, useSkins)
  137. def setHost(self, host):
  138. self.host = host
  139. # --------------------------------------------------------------------------------------------------------
  140. def customClearSelection(self):
  141. self.setCurrentRow(-1)
  142. self.clearSelection()
  143. self.clearFocus()
  144. def isDragUrlValid(self, url):
  145. filename = url.toLocalFile()
  146. if os.path.isdir(filename):
  147. if os.path.exists(os.path.join(filename, "manifest.ttl")):
  148. return True
  149. if MACOS and filename.lower().endswith((".vst", ".vst3")):
  150. return True
  151. elif os.path.isfile(filename):
  152. if filename.lower().endswith(self.fSupportedExtensions):
  153. return True
  154. return False
  155. # --------------------------------------------------------------------------------------------------------
  156. def dragEnterEvent(self, event):
  157. urls = event.mimeData().urls()
  158. for url in urls:
  159. if self.isDragUrlValid(url):
  160. self.fWasLastDragValid = True
  161. event.acceptProposedAction()
  162. return
  163. self.fWasLastDragValid = False
  164. QListWidget.dragEnterEvent(self, event)
  165. def dragMoveEvent(self, event):
  166. if not self.fWasLastDragValid:
  167. QListWidget.dragMoveEvent(self, event)
  168. return
  169. event.acceptProposedAction()
  170. tryItem = self.itemAt(event.pos())
  171. if tryItem is not None:
  172. self.setCurrentRow(tryItem.getPluginId())
  173. else:
  174. self.setCurrentRow(-1)
  175. def dragLeaveEvent(self, event):
  176. self.fWasLastDragValid = False
  177. QListWidget.dragLeaveEvent(self, event)
  178. # --------------------------------------------------------------------------------------------------------
  179. # FIXME: this needs some attention
  180. # if dropping project file over 1 plugin, load it in rack or patchbay
  181. # if dropping regular files over 1 plugin, keep replacing plugins
  182. def dropEvent(self, event):
  183. event.acceptProposedAction()
  184. urls = event.mimeData().urls()
  185. if len(urls) == 0:
  186. return
  187. tryItem = self.itemAt(event.pos())
  188. if tryItem is not None:
  189. pluginId = tryItem.getPluginId()
  190. else:
  191. pluginId = -1
  192. for url in urls:
  193. if pluginId >= 0:
  194. self.host.replace_plugin(pluginId)
  195. pluginId += 1
  196. if pluginId > self.host.get_current_plugin_count():
  197. pluginId = -1
  198. filename = url.toLocalFile()
  199. if not self.host.load_file(filename):
  200. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
  201. self.tr("Failed to load file"),
  202. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  203. if tryItem is not None:
  204. self.host.replace_plugin(self.host.get_max_plugin_number())
  205. #tryItem.widget.setActive(True, True, True)
  206. # --------------------------------------------------------------------------------------------------------
  207. def mousePressEvent(self, event):
  208. if self.itemAt(event.pos()) is None and self.currentRow() != -1:
  209. event.accept()
  210. self.customClearSelection()
  211. return
  212. QListWidget.mousePressEvent(self, event)
  213. def paintEvent(self, event):
  214. painter = QPainter(self.viewport())
  215. painter.drawTiledPixmap(0, 0, self.fPixmapWidth, self.height(), self.fPixmapL)
  216. painter.drawTiledPixmap(self.width()-self.fPixmapWidth-2, 0, self.fPixmapWidth, self.height(), self.fPixmapR)
  217. QListWidget.paintEvent(self, event)
  218. # --------------------------------------------------------------------------------------------------------
  219. def selectionChanged(self, selected, deselected):
  220. for index in deselected.indexes():
  221. item = self.itemFromIndex(index)
  222. if item is not None:
  223. item.setSelected(False)
  224. for index in selected.indexes():
  225. item = self.itemFromIndex(index)
  226. if item is not None:
  227. item.setSelected(True)
  228. QListWidget.selectionChanged(self, selected, deselected)
  229. # ------------------------------------------------------------------------------------------------------------