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.

1938 lines
71KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla plugin/slot skin code
  4. # Copyright (C) 2013-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, QRectF
  24. from PyQt5.QtGui import QFont, QFontDatabase, QPen, QPixmap
  25. from PyQt5.QtWidgets import QFrame, QPushButton
  26. else:
  27. from PyQt4.QtCore import Qt, QRectF
  28. from PyQt4.QtGui import QFont, QFontDatabase, QFrame, QPen, QPixmap, QPushButton
  29. # ------------------------------------------------------------------------------------------------------------
  30. # Imports (Custom)
  31. import ui_carla_plugin_default
  32. import ui_carla_plugin_basic_fx
  33. import ui_carla_plugin_calf
  34. import ui_carla_plugin_sf2
  35. import ui_carla_plugin_zynfx
  36. from carla_widgets import *
  37. from digitalpeakmeter import DigitalPeakMeter
  38. from pixmapdial import PixmapDial
  39. # ------------------------------------------------------------------------------------------------------------
  40. # Try to "shortify" a parameter name
  41. def getParameterShortName(paramName):
  42. paramName = paramName.split("/",1)[0].split(" (",1)[0].split(" [",1)[0].strip()
  43. paramLow = paramName.lower()
  44. # Cut useless prefix
  45. if paramLow.startswith("compressor "):
  46. paramName = paramName.replace("ompressor ", ".", 1)
  47. paramLow = paramName.lower()
  48. elif paramLow.startswith("room "):
  49. paramName = paramName.split(" ",1)[1]
  50. paramLow = paramName.lower()
  51. # Cut useless suffix
  52. if paramLow.endswith(" level"):
  53. paramName = paramName.rsplit(" ",1)[0]
  54. paramLow = paramName.lower()
  55. elif paramLow.endswith(" time"):
  56. paramName = paramName.rsplit(" ",1)[0]
  57. paramLow = paramName.lower()
  58. # Cut generic names
  59. if "attack" in paramLow:
  60. paramName = paramName.replace("ttack", "tk")
  61. elif "bandwidth" in paramLow:
  62. paramName = paramName.replace("andwidth", "w")
  63. elif "damping" in paramLow:
  64. paramName = paramName.replace("amping", "amp")
  65. elif "distortion" in paramLow:
  66. paramName = paramName.replace("istortion", "ist")
  67. elif "feedback" in paramLow:
  68. paramName = paramName.replace("eedback", "b")
  69. elif "frequency" in paramLow:
  70. paramName = paramName.replace("requency", "req")
  71. elif "input" in paramLow:
  72. paramName = paramName.replace("nput", "n")
  73. elif "makeup" in paramLow:
  74. paramName = paramName.replace("akeup", "kUp" if "Make" in paramName else "kup")
  75. elif "output" in paramLow:
  76. paramName = paramName.replace("utput", "ut")
  77. elif "random" in paramLow:
  78. paramName = paramName.replace("andom", "nd")
  79. elif "threshold" in paramLow:
  80. paramName = paramName.replace("hreshold", "hres")
  81. # remove space if 1st last word is lowercase and the 2nd first is uppercase, or if 2nd is number
  82. if " " in paramName:
  83. name1, name2 = paramName.split(" ", 1)
  84. if (name1[-1].islower() and name2[0].isupper()) or name2.isdigit():
  85. paramName = paramName.replace(" ", "", 1)
  86. # cut stuff if too big
  87. if len(paramName) > 7:
  88. paramName = paramName.replace("a","").replace("e","").replace("i","").replace("o","").replace("u","")
  89. if len(paramName) > 7:
  90. paramName = paramName[:7]
  91. return paramName.strip()
  92. # ------------------------------------------------------------------------------------------------------------
  93. # Abstract plugin slot
  94. class AbstractPluginSlot(QFrame, PluginEditParentMeta):
  95. #class AbstractPluginSlot(QFrame, PluginEditParentMeta, metaclass=PyQtMetaClass):
  96. def __init__(self, parent, host, pluginId):
  97. QFrame.__init__(self, parent)
  98. self.host = host
  99. if False:
  100. # kdevelop likes this :)
  101. host = CarlaHostMeta()
  102. self.host = host
  103. # -------------------------------------------------------------
  104. # Get plugin info
  105. self.fPluginId = pluginId
  106. self.fPluginInfo = host.get_plugin_info(self.fPluginId)
  107. #if not gCarla.isLocal:
  108. #self.fPluginInfo['hints'] &= ~PLUGIN_HAS_CUSTOM_UI
  109. # -------------------------------------------------------------
  110. # Internal stuff
  111. self.fIsActive = bool(host.get_internal_parameter_value(self.fPluginId, PARAMETER_ACTIVE) >= 0.5)
  112. self.fIsCollapsed = False
  113. self.fIsSelected = False
  114. self.fLastGreenLedState = False
  115. self.fLastBlueLedState = False
  116. self.fParameterIconTimer = ICON_STATE_NULL
  117. self.fParameterList = [] # index, widget
  118. audioCountInfo = host.get_audio_port_count_info(self.fPluginId)
  119. self.fPeaksInputCount = int(audioCountInfo['ins'])
  120. self.fPeaksOutputCount = int(audioCountInfo['outs'])
  121. if self.fPeaksInputCount > 2:
  122. self.fPeaksInputCount = 2
  123. if self.fPeaksOutputCount > 2:
  124. self.fPeaksOutputCount = 2
  125. # used during testing
  126. self.fIdleTimerId = 0
  127. # -------------------------------------------------------------
  128. # Set-up GUI
  129. self.fEditDialog = PluginEdit(self, host, self.fPluginId)
  130. # -------------------------------------------------------------
  131. # Set-up common widgets (as none)
  132. self.b_enable = None
  133. self.b_gui = None
  134. self.b_edit = None
  135. self.b_remove = None
  136. self.cb_presets = None
  137. self.label_name = None
  138. self.label_type = None
  139. self.led_control = None
  140. self.led_midi = None
  141. self.led_audio_in = None
  142. self.led_audio_out = None
  143. self.line = None
  144. self.peak_in = None
  145. self.peak_out = None
  146. # -------------------------------------------------------------
  147. # Set-up connections
  148. host.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
  149. host.PluginUnavailableCallback.connect(self.slot_handlePluginUnavailableCallback)
  150. host.ParameterValueChangedCallback.connect(self.slot_handleParameterValueChangedCallback)
  151. host.ParameterDefaultChangedCallback.connect(self.slot_handleParameterDefaultChangedCallback)
  152. host.ParameterMidiChannelChangedCallback.connect(self.slot_handleParameterMidiChannelChangedCallback)
  153. host.ParameterMidiCcChangedCallback.connect(self.slot_handleParameterMidiCcChangedCallback)
  154. host.ProgramChangedCallback.connect(self.slot_handleProgramChangedCallback)
  155. host.MidiProgramChangedCallback.connect(self.slot_handleMidiProgramChangedCallback)
  156. host.OptionChangedCallback.connect(self.slot_handleOptionChangedCallback)
  157. host.UiStateChangedCallback.connect(self.slot_handleUiStateChangedCallback)
  158. # -----------------------------------------------------------------
  159. @pyqtSlot(int, str)
  160. def slot_handlePluginRenamedCallback(self, pluginId, newName):
  161. if self.fPluginId == pluginId:
  162. self.setName(newName)
  163. @pyqtSlot(int, str)
  164. def slot_handlePluginUnavailableCallback(self, pluginId, errorMsg):
  165. if self.fPluginId == pluginId:
  166. pass
  167. @pyqtSlot(int, int, float)
  168. def slot_handleParameterValueChangedCallback(self, pluginId, index, value):
  169. if self.fPluginId == pluginId:
  170. self.setParameterValue(index, value, True)
  171. @pyqtSlot(int, int, float)
  172. def slot_handleParameterDefaultChangedCallback(self, pluginId, index, value):
  173. if self.fPluginId == pluginId:
  174. self.setParameterDefault(index, value)
  175. @pyqtSlot(int, int, int)
  176. def slot_handleParameterMidiCcChangedCallback(self, pluginId, index, cc):
  177. if self.fPluginId == pluginId:
  178. self.setParameterMidiControl(index, cc)
  179. @pyqtSlot(int, int, int)
  180. def slot_handleParameterMidiChannelChangedCallback(self, pluginId, index, channel):
  181. if self.fPluginId == pluginId:
  182. self.setParameterMidiChannel(index, channel)
  183. @pyqtSlot(int, int)
  184. def slot_handleProgramChangedCallback(self, pluginId, index):
  185. if self.fPluginId == pluginId:
  186. self.setProgram(index, True)
  187. @pyqtSlot(int, int)
  188. def slot_handleMidiProgramChangedCallback(self, pluginId, index):
  189. if self.fPluginId == pluginId:
  190. self.setMidiProgram(index, True)
  191. @pyqtSlot(int, int, bool)
  192. def slot_handleOptionChangedCallback(self, pluginId, option, yesNo):
  193. if self.fPluginId == pluginId:
  194. self.setOption(option, yesNo)
  195. @pyqtSlot(int, int)
  196. def slot_handleUiStateChangedCallback(self, pluginId, state):
  197. if self.fPluginId == pluginId:
  198. self.customUiStateChanged(state)
  199. #------------------------------------------------------------------
  200. def ready(self):
  201. if self.b_enable is not None:
  202. self.b_enable.setChecked(self.fIsActive)
  203. self.b_enable.clicked.connect(self.slot_enableClicked)
  204. if self.b_gui is not None:
  205. self.b_gui.clicked.connect(self.slot_showCustomUi)
  206. self.b_gui.setEnabled(bool(self.fPluginInfo['hints'] & PLUGIN_HAS_CUSTOM_UI))
  207. if self.b_edit is None:
  208. # Edit dialog *must* be available
  209. self.b_edit = QPushButton(self)
  210. self.b_edit.setCheckable(True)
  211. self.b_edit.hide()
  212. self.b_edit.clicked.connect(self.slot_showEditDialog)
  213. if self.b_remove is not None:
  214. self.b_remove.clicked.connect(self.slot_removePlugin)
  215. if self.label_name is not None:
  216. self.label_name.setEnabled(self.fIsActive)
  217. self.label_name.setText(self.fPluginInfo['name'])
  218. if self.label_type is not None:
  219. self.label_type.setText(getPluginTypeAsString(self.fPluginInfo['type']))
  220. if self.led_control is not None:
  221. self.led_control.setColor(self.led_control.YELLOW)
  222. self.led_control.setEnabled(False)
  223. if self.led_midi is not None:
  224. self.led_midi.setColor(self.led_midi.RED)
  225. self.led_midi.setEnabled(False)
  226. if self.led_audio_in is not None:
  227. self.led_audio_in.setColor(self.led_audio_in.GREEN)
  228. self.led_audio_in.setEnabled(False)
  229. if self.led_audio_out is not None:
  230. self.led_audio_out.setColor(self.led_audio_out.BLUE)
  231. self.led_audio_out.setEnabled(False)
  232. if self.peak_in is not None:
  233. self.peak_in.setChannelCount(self.fPeaksInputCount)
  234. self.peak_in.setMeterColor(DigitalPeakMeter.COLOR_GREEN)
  235. self.peak_in.setMeterOrientation(DigitalPeakMeter.HORIZONTAL)
  236. if (self.fPeaksInputCount == 0 and not isinstance(self, PluginSlot_Default)) or self.fIsCollapsed:
  237. self.peak_in.hide()
  238. if self.peak_out is not None:
  239. self.peak_out.setChannelCount(self.fPeaksOutputCount)
  240. self.peak_out.setMeterColor(DigitalPeakMeter.COLOR_BLUE)
  241. self.peak_out.setMeterOrientation(DigitalPeakMeter.HORIZONTAL)
  242. if (self.fPeaksOutputCount == 0 and not isinstance(self, PluginSlot_Default)) or self.fIsCollapsed:
  243. self.peak_out.hide()
  244. if self.line is not None:
  245. if self.fIsCollapsed:
  246. self.line.hide()
  247. for paramIndex, paramWidget in self.fParameterList:
  248. paramWidget.setContextMenuPolicy(Qt.CustomContextMenu)
  249. paramWidget.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  250. paramWidget.realValueChanged.connect(self.slot_parameterValueChanged)
  251. paramWidget.setValue(self.host.get_internal_parameter_value(self.fPluginId, paramIndex))
  252. if self.fIsCollapsed:
  253. paramWidget.hide()
  254. self.setWindowTitle(self.fPluginInfo['name'])
  255. #------------------------------------------------------------------
  256. def getFixedHeight(self):
  257. return 32
  258. def getHints(self):
  259. return self.fPluginInfo['hints']
  260. def getPluginId(self):
  261. return self.fPluginId
  262. #------------------------------------------------------------------
  263. def setPluginId(self, idx):
  264. self.fPluginId = idx
  265. self.fEditDialog.setPluginId(idx)
  266. def setName(self, name):
  267. self.fEditDialog.setName(name)
  268. if self.label_name is not None:
  269. self.label_name.setText(name)
  270. def setSelected(self, yesNo):
  271. if self.fIsSelected == yesNo:
  272. return
  273. self.fIsSelected = yesNo
  274. self.update()
  275. #------------------------------------------------------------------
  276. def setActive(self, active, sendCallback=False, sendHost=True):
  277. self.fIsActive = active
  278. if sendCallback:
  279. self.fParameterIconTimer = ICON_STATE_ON
  280. self.activeChanged(active)
  281. if sendHost:
  282. self.host.set_active(self.fPluginId, active)
  283. if active:
  284. self.fEditDialog.clearNotes()
  285. self.midiActivityChanged(False)
  286. if self.label_name is not None:
  287. self.label_name.setEnabled(self.fIsActive)
  288. # called from rack, checks if param is possible first
  289. def setInternalParameter(self, parameterId, value):
  290. if parameterId <= PARAMETER_MAX or parameterId >= PARAMETER_NULL:
  291. return
  292. elif parameterId == PARAMETER_ACTIVE:
  293. return self.setActive(bool(value), True, True)
  294. elif parameterId == PARAMETER_DRYWET:
  295. if (self.fPluginInfo['hints'] & PLUGIN_CAN_DRYWET) == 0: return
  296. self.host.set_drywet(self.fPluginId, value)
  297. elif parameterId == PARAMETER_VOLUME:
  298. if (self.fPluginInfo['hints'] & PLUGIN_CAN_VOLUME) == 0: return
  299. self.host.set_volume(self.fPluginId, value)
  300. elif parameterId == PARAMETER_BALANCE_LEFT:
  301. if (self.fPluginInfo['hints'] & PLUGIN_CAN_BALANCE) == 0: return
  302. self.host.set_balance_left(self.fPluginId, value)
  303. elif parameterId == PARAMETER_BALANCE_RIGHT:
  304. if (self.fPluginInfo['hints'] & PLUGIN_CAN_BALANCE) == 0: return
  305. self.host.set_balance_right(self.fPluginId, value)
  306. elif parameterId == PARAMETER_PANNING:
  307. if (self.fPluginInfo['hints'] & PLUGIN_CAN_PANNING) == 0: return
  308. self.host.set_panning(self.fPluginId, value)
  309. elif parameterId == PARAMETER_CTRL_CHANNEL:
  310. self.host.set_ctrl_channel(self.fPluginId, value)
  311. self.fEditDialog.setParameterValue(parameterId, value)
  312. #------------------------------------------------------------------
  313. def setParameterValue(self, parameterId, value, sendCallback):
  314. if parameterId == PARAMETER_ACTIVE:
  315. return self.setActive(bool(value), True, False)
  316. self.fEditDialog.setParameterValue(parameterId, value)
  317. if sendCallback:
  318. self.fParameterIconTimer = ICON_STATE_ON
  319. self.editDialogParameterValueChanged(self.fPluginId, parameterId, value)
  320. def setParameterDefault(self, parameterId, value):
  321. self.fEditDialog.setParameterDefault(parameterId, value)
  322. def setParameterMidiControl(self, parameterId, control):
  323. self.fEditDialog.setParameterMidiControl(parameterId, control)
  324. def setParameterMidiChannel(self, parameterId, channel):
  325. self.fEditDialog.setParameterMidiChannel(parameterId, channel)
  326. #------------------------------------------------------------------
  327. def setProgram(self, index, sendCallback):
  328. self.fEditDialog.setProgram(index)
  329. if sendCallback:
  330. self.fParameterIconTimer = ICON_STATE_ON
  331. self.editDialogProgramChanged(self.fPluginId, index)
  332. self.updateParameterValues()
  333. def setMidiProgram(self, index, sendCallback):
  334. self.fEditDialog.setMidiProgram(index)
  335. if sendCallback:
  336. self.fParameterIconTimer = ICON_STATE_ON
  337. self.editDialogMidiProgramChanged(self.fPluginId, index)
  338. self.updateParameterValues()
  339. #------------------------------------------------------------------
  340. def setOption(self, option, yesNo):
  341. self.fEditDialog.setOption(option, yesNo)
  342. #------------------------------------------------------------------
  343. def activeChanged(self, onOff):
  344. self.fIsActive = onOff
  345. if self.b_enable is None:
  346. return
  347. self.b_enable.blockSignals(True)
  348. self.b_enable.setChecked(onOff)
  349. self.b_enable.blockSignals(False)
  350. def customUiStateChanged(self, state):
  351. if self.b_gui is None:
  352. return
  353. self.b_gui.blockSignals(True)
  354. if state == 0:
  355. self.b_gui.setChecked(False)
  356. self.b_gui.setEnabled(True)
  357. elif state == 1:
  358. self.b_gui.setChecked(True)
  359. self.b_gui.setEnabled(True)
  360. elif state == -1:
  361. self.b_gui.setChecked(False)
  362. self.b_gui.setEnabled(False)
  363. self.b_gui.blockSignals(False)
  364. def parameterActivityChanged(self, onOff):
  365. if self.led_control is None:
  366. return
  367. self.led_control.setChecked(onOff)
  368. def midiActivityChanged(self, onOff):
  369. if self.led_midi is None:
  370. return
  371. self.led_midi.setChecked(onOff)
  372. def optionChanged(self, option, yesNo):
  373. pass
  374. # -----------------------------------------------------------------
  375. # PluginEdit callbacks
  376. def editDialogVisibilityChanged(self, pluginId, visible):
  377. if self.b_edit is None:
  378. return
  379. self.b_edit.blockSignals(True)
  380. self.b_edit.setChecked(visible)
  381. self.b_edit.blockSignals(False)
  382. def editDialogPluginHintsChanged(self, pluginId, hints):
  383. self.fPluginInfo['hints'] = hints
  384. for paramIndex, paramWidget in self.fParameterList:
  385. if paramIndex == PARAMETER_DRYWET:
  386. paramWidget.setVisible(hints & PLUGIN_CAN_DRYWET)
  387. elif paramIndex == PARAMETER_VOLUME:
  388. paramWidget.setVisible(hints & PLUGIN_CAN_VOLUME)
  389. if self.b_gui is not None:
  390. self.b_gui.setEnabled(bool(hints & PLUGIN_HAS_CUSTOM_UI))
  391. def editDialogParameterValueChanged(self, pluginId, parameterId, value):
  392. for paramIndex, paramWidget in self.fParameterList:
  393. if paramIndex != parameterId:
  394. continue
  395. paramWidget.blockSignals(True)
  396. paramWidget.setValue(value)
  397. paramWidget.blockSignals(False)
  398. break
  399. def editDialogProgramChanged(self, pluginId, index):
  400. if self.cb_presets is None:
  401. return
  402. self.cb_presets.blockSignals(True)
  403. self.cb_presets.setCurrentIndex(index)
  404. self.cb_presets.blockSignals(False)
  405. # FIXME
  406. self.updateParameterValues()
  407. def editDialogMidiProgramChanged(self, pluginId, index):
  408. if self.cb_presets is None:
  409. return
  410. self.cb_presets.blockSignals(True)
  411. self.cb_presets.setCurrentIndex(index)
  412. self.cb_presets.blockSignals(False)
  413. # FIXME
  414. self.updateParameterValues()
  415. def editDialogNotePressed(self, pluginId, note):
  416. pass
  417. def editDialogNoteReleased(self, pluginId, note):
  418. pass
  419. def editDialogMidiActivityChanged(self, pluginId, onOff):
  420. self.midiActivityChanged(onOff)
  421. #------------------------------------------------------------------
  422. def idleFast(self):
  423. # Input peaks
  424. if self.fPeaksInputCount > 0:
  425. if self.fPeaksInputCount > 1:
  426. peak1 = self.host.get_input_peak_value(self.fPluginId, True)
  427. peak2 = self.host.get_input_peak_value(self.fPluginId, False)
  428. ledState = bool(peak1 != 0.0 or peak2 != 0.0)
  429. if self.peak_in is not None:
  430. self.peak_in.displayMeter(1, peak1)
  431. self.peak_in.displayMeter(2, peak2)
  432. else:
  433. peak = self.host.get_input_peak_value(self.fPluginId, True)
  434. ledState = bool(peak != 0.0)
  435. if self.peak_in is not None:
  436. self.peak_in.displayMeter(1, peak)
  437. if self.fLastGreenLedState != ledState and self.led_audio_in is not None:
  438. self.fLastGreenLedState = ledState
  439. self.led_audio_in.setChecked(ledState)
  440. # Output peaks
  441. if self.fPeaksOutputCount > 0:
  442. if self.fPeaksOutputCount > 1:
  443. peak1 = self.host.get_output_peak_value(self.fPluginId, True)
  444. peak2 = self.host.get_output_peak_value(self.fPluginId, False)
  445. ledState = bool(peak1 != 0.0 or peak2 != 0.0)
  446. if self.peak_out is not None:
  447. self.peak_out.displayMeter(1, peak1)
  448. self.peak_out.displayMeter(2, peak2)
  449. else:
  450. peak = self.host.get_output_peak_value(self.fPluginId, True)
  451. ledState = bool(peak != 0.0)
  452. if self.peak_out is not None:
  453. self.peak_out.displayMeter(1, peak)
  454. if self.fLastBlueLedState != ledState and self.led_audio_out is not None:
  455. self.fLastBlueLedState = ledState
  456. self.led_audio_out.setChecked(ledState)
  457. def idleSlow(self):
  458. if self.fParameterIconTimer == ICON_STATE_ON:
  459. self.parameterActivityChanged(True)
  460. self.fParameterIconTimer = ICON_STATE_WAIT
  461. elif self.fParameterIconTimer == ICON_STATE_WAIT:
  462. self.fParameterIconTimer = ICON_STATE_OFF
  463. elif self.fParameterIconTimer == ICON_STATE_OFF:
  464. self.parameterActivityChanged(False)
  465. self.fParameterIconTimer = ICON_STATE_NULL
  466. self.fEditDialog.idleSlow()
  467. #------------------------------------------------------------------
  468. def drawOutline(self):
  469. painter = QPainter(self)
  470. if self.fIsSelected:
  471. painter.setPen(QPen(Qt.cyan, 4))
  472. painter.setBrush(Qt.transparent)
  473. painter.drawRect(0, 0, self.width(), self.height())
  474. else:
  475. painter.setPen(QPen(Qt.black, 1))
  476. painter.setBrush(Qt.black)
  477. painter.drawLine(0, self.height()-1, self.width(), self.height()-1)
  478. def showDefaultCustomMenu(self, isEnabled, bEdit = None, bGui = None):
  479. menu = QMenu(self)
  480. actActive = menu.addAction(self.tr("Disable") if isEnabled else self.tr("Enable"))
  481. menu.addSeparator()
  482. actReset = menu.addAction(self.tr("Reset parameters"))
  483. actRandom = menu.addAction(self.tr("Randomize parameters"))
  484. menu.addSeparator()
  485. if bEdit is not None:
  486. actEdit = menu.addAction(self.tr("Edit"))
  487. actEdit.setCheckable(True)
  488. actEdit.setChecked(bEdit.isChecked())
  489. else:
  490. actEdit = None
  491. if bGui is not None:
  492. actGui = menu.addAction(self.tr("Show Custom UI"))
  493. actGui.setCheckable(True)
  494. actGui.setChecked(bGui.isChecked())
  495. actGui.setEnabled(bGui.isEnabled())
  496. else:
  497. actGui = None
  498. menu.addSeparator()
  499. actClone = menu.addAction(self.tr("Clone"))
  500. actReplace = menu.addAction(self.tr("Replace..."))
  501. actRename = menu.addAction(self.tr("Rename..."))
  502. actRemove = menu.addAction(self.tr("Remove"))
  503. if self.fIdleTimerId != 0:
  504. actRemove.setVisible(False)
  505. actSel = menu.exec_(QCursor.pos())
  506. if not actSel:
  507. return
  508. if actSel == actActive:
  509. self.setActive(not isEnabled, True, True)
  510. elif actSel == actReset:
  511. self.host.reset_parameters(self.fPluginId)
  512. elif actSel == actRandom:
  513. self.host.randomize_parameters(self.fPluginId)
  514. elif actSel == actGui:
  515. bGui.click()
  516. elif actSel == actEdit:
  517. bEdit.click()
  518. elif actSel == actClone:
  519. if not self.host.clone_plugin(self.fPluginId):
  520. CustomMessageBox(self, QMessageBox.Warning, self.tr("Error"), self.tr("Operation failed"),
  521. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  522. elif actSel == actRename:
  523. oldName = self.fPluginInfo['name']
  524. newNameTry = QInputDialog.getText(self, self.tr("Rename Plugin"), self.tr("New plugin name:"), QLineEdit.Normal, oldName)
  525. if not (newNameTry[1] and newNameTry[0] and oldName != newNameTry[0]):
  526. return
  527. newName = newNameTry[0]
  528. if self.host.rename_plugin(self.fPluginId, newName):
  529. self.setName(newName)
  530. else:
  531. CustomMessageBox(self, QMessageBox.Warning, self.tr("Error"), self.tr("Operation failed"),
  532. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  533. elif actSel == actReplace:
  534. gCarla.gui.slot_pluginAdd(self.fPluginId)
  535. elif actSel == actRemove:
  536. if not self.host.remove_plugin(self.fPluginId):
  537. CustomMessageBox(self, QMessageBox.Warning, self.tr("Error"), self.tr("Operation failed"),
  538. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  539. def updateParameterValues(self):
  540. for paramIndex, paramWidget in self.fParameterList:
  541. if paramIndex < 0:
  542. continue
  543. paramWidget.blockSignals(True)
  544. paramWidget.setValue(self.host.get_current_parameter_value(self.fPluginId, paramIndex))
  545. paramWidget.blockSignals(False)
  546. #------------------------------------------------------------------
  547. @pyqtSlot(bool)
  548. def slot_enableClicked(self, yesNo):
  549. self.setActive(yesNo, False, True)
  550. @pyqtSlot()
  551. def slot_showDefaultCustomMenu(self):
  552. self.showDefaultCustomMenu(self.fIsActive, self.b_edit, self.b_gui)
  553. @pyqtSlot()
  554. def slot_knobCustomMenu(self):
  555. sender = self.sender()
  556. index = sender.fIndex
  557. minimum = sender.fMinimum
  558. maximum = sender.fMaximum
  559. current = sender.fRealValue
  560. label = sender.fLabel
  561. if index in (PARAMETER_DRYWET, PARAMETER_VOLUME):
  562. default = 1.0
  563. resetText = self.tr("Reset (%i%%)" % int(default*100.0))
  564. minimText = self.tr("Set to Minimum (%i%%)" % int(minimum*100.0))
  565. maximText = self.tr("Set to Maximum (%i%%)" % int(maximum*100.0))
  566. else:
  567. default = self.host.get_default_parameter_value(self.fPluginId, index)
  568. resetText = self.tr("Reset (%f)" % default)
  569. minimText = self.tr("Set to Minimum (%f)" % minimum)
  570. maximText = self.tr("Set to Maximum (%f)" % maximum)
  571. menu = QMenu(self)
  572. actReset = menu.addAction(resetText)
  573. menu.addSeparator()
  574. actMinimum = menu.addAction(minimText)
  575. actMaximum = menu.addAction(maximText)
  576. menu.addSeparator()
  577. actSet = menu.addAction(self.tr("Set value..."))
  578. actSelected = menu.exec_(QCursor.pos())
  579. if actSelected == actSet:
  580. valueTry = QInputDialog.getDouble(self, self.tr("Set value"), label, current, minimum, maximum, 3) # FIXME - 3 decimals
  581. if valueTry[1]:
  582. value = valueTry[0] * 10
  583. else:
  584. return
  585. elif actSelected == actMinimum:
  586. value = minimum
  587. elif actSelected == actMaximum:
  588. value = maximum
  589. elif actSelected == actReset:
  590. value = default
  591. else:
  592. return
  593. self.sender().setValue(value)
  594. #------------------------------------------------------------------
  595. @pyqtSlot(bool)
  596. def slot_showCustomUi(self, show):
  597. self.host.show_custom_ui(self.fPluginId, show)
  598. @pyqtSlot(bool)
  599. def slot_showEditDialog(self, show):
  600. self.fEditDialog.setVisible(show)
  601. @pyqtSlot()
  602. def slot_removePlugin(self):
  603. if not self.host.remove_plugin(self.fPluginId):
  604. CustomMessageBox(self, QMessageBox.Warning, self.tr("Error"), self.tr("Operation failed"),
  605. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  606. #------------------------------------------------------------------
  607. @pyqtSlot(int)
  608. def slot_parameterValueChanged(self, value):
  609. index = self.sender().getIndex()
  610. if index < 0:
  611. self.setInternalParameter(index, value)
  612. else:
  613. self.host.set_parameter_value(self.fPluginId, index, value)
  614. self.setParameterValue(index, value, False)
  615. @pyqtSlot(int)
  616. def slot_programChanged(self, index):
  617. self.host.set_program(self.fPluginId, index)
  618. self.setProgram(index, False)
  619. @pyqtSlot(int)
  620. def slot_midiProgramChanged(self, index):
  621. self.host.set_midi_program(self.fPluginId, index)
  622. self.setMidiProgram(index, False)
  623. #------------------------------------------------------------------
  624. def testTimer(self):
  625. self.fIdleTimerId = self.startTimer(25)
  626. #------------------------------------------------------------------
  627. def closeEvent(self, event):
  628. if self.fIdleTimerId != 0:
  629. self.killTimer(self.fIdleTimerId)
  630. self.fIdleTimerId = 0
  631. self.host.engine_close()
  632. QFrame.closeEvent(self, event)
  633. def timerEvent(self, event):
  634. if event.timerId() == self.fIdleTimerId:
  635. self.host.engine_idle()
  636. self.idleFast()
  637. self.idleSlow()
  638. QFrame.timerEvent(self, event)
  639. def paintEvent(self, event):
  640. self.drawOutline()
  641. QFrame.paintEvent(self, event)
  642. # ------------------------------------------------------------------------------------------------------------
  643. class PluginSlot_Default(AbstractPluginSlot):
  644. def __init__(self, parent, host, pluginId):
  645. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  646. self.ui = ui_carla_plugin_default.Ui_PluginWidget()
  647. self.ui.setupUi(self)
  648. # -------------------------------------------------------------
  649. # Internal stuff
  650. self.fColorTop = QColor(60, 60, 60)
  651. self.fColorBottom = QColor(47, 47, 47)
  652. self.fColorSeprtr = QColor(70, 70, 70)
  653. # -------------------------------------------------------------
  654. # Set-up GUI
  655. self.setStyleSheet("""
  656. QLabel#label_name {
  657. color: #BBB;
  658. }
  659. """)
  660. self.ui.b_enable.setPixmaps(":/bitmaps/button_off.png", ":/bitmaps/button_on.png", ":/bitmaps/button_off.png")
  661. self.ui.b_edit.setPixmaps(":/bitmaps/button_edit.png", ":/bitmaps/button_edit_down.png", ":/bitmaps/button_edit_hover.png")
  662. if self.fPluginInfo['iconName'] == "distrho":
  663. self.ui.b_gui.setPixmaps(":/bitmaps/button_distrho.png", ":/bitmaps/button_distrho_down.png", ":/bitmaps/button_distrho_hover.png")
  664. elif self.fPluginInfo['iconName'] == "file":
  665. self.ui.b_gui.setPixmaps(":/bitmaps/button_file.png", ":/bitmaps/button_file_down.png", ":/bitmaps/button_file_hover.png")
  666. else:
  667. self.ui.b_gui.setPixmaps(":/bitmaps/button_gui.png", ":/bitmaps/button_gui_down.png", ":/bitmaps/button_gui_hover.png")
  668. # -------------------------------------------------------------
  669. self.b_enable = self.ui.b_enable
  670. self.b_gui = self.ui.b_gui
  671. self.b_edit = self.ui.b_edit
  672. self.label_name = self.ui.label_name
  673. self.led_control = self.ui.led_control
  674. self.led_midi = self.ui.led_midi
  675. self.led_audio_in = self.ui.led_audio_in
  676. self.led_audio_out = self.ui.led_audio_out
  677. self.peak_in = self.ui.peak_in
  678. self.peak_out = self.ui.peak_out
  679. self.ready()
  680. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  681. #------------------------------------------------------------------
  682. def getFixedHeight(self):
  683. return 36
  684. #------------------------------------------------------------------
  685. def paintEvent(self, event):
  686. painter = QPainter(self)
  687. painter.save()
  688. areaX = self.ui.area_right.x()+7
  689. width = self.width()
  690. height = self.height()
  691. painter.setPen(QPen(QColor(17, 17, 17), 1))
  692. painter.setBrush(QColor(17, 17, 17))
  693. painter.drawRect(0, 0, width, height)
  694. painter.setPen(self.fColorSeprtr.lighter(110))
  695. painter.setBrush(self.fColorBottom)
  696. painter.setRenderHint(QPainter.Antialiasing, True)
  697. # name -> leds arc
  698. path = QPainterPath()
  699. path.moveTo(areaX-20, height-4)
  700. path.cubicTo(areaX, height-5, areaX-20, 4.75, areaX, 4.75)
  701. path.lineTo(areaX, height-5)
  702. painter.drawPath(path)
  703. painter.setPen(self.fColorSeprtr)
  704. painter.setRenderHint(QPainter.Antialiasing, False)
  705. # separator lines
  706. painter.drawLine(0, height-5, areaX-20, height-5)
  707. painter.drawLine(areaX, 4, width, 4)
  708. painter.setPen(self.fColorBottom)
  709. painter.setBrush(self.fColorBottom)
  710. # top, bottom and left lines
  711. painter.drawLine(0, 0, width, 0)
  712. painter.drawRect(0, height-4, areaX, 4)
  713. painter.drawRoundedRect(areaX-20, height-5, areaX, 5, 22, 22)
  714. painter.drawLine(0, 0, 0, height)
  715. # fill the rest
  716. painter.drawRect(areaX-1, 5, width, height)
  717. # bottom 1px line
  718. painter.setPen(self.fColorSeprtr)
  719. painter.drawLine(0, height-1, width, height-1)
  720. painter.restore()
  721. AbstractPluginSlot.paintEvent(self, event)
  722. # ------------------------------------------------------------------------------------------------------------
  723. class PluginSlot_BasicFX(AbstractPluginSlot):
  724. def __init__(self, parent, host, pluginId):
  725. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  726. self.ui = ui_carla_plugin_basic_fx.Ui_PluginWidget()
  727. self.ui.setupUi(self)
  728. if self.fPluginInfo['type'] == PLUGIN_INTERNAL and self.fPluginInfo['label'] == "midifile":
  729. self.fIsCollapsed = True
  730. # -------------------------------------------------------------
  731. # Set-up GUI
  732. labelFont = self.ui.label_name.font()
  733. labelFont.setBold(True)
  734. labelFont.setPointSize(9)
  735. self.ui.label_name.setFont(labelFont)
  736. r = 40
  737. g = 40
  738. b = 40
  739. if self.fPluginInfo['category'] == PLUGIN_CATEGORY_MODULATOR:
  740. r += 10
  741. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_EQ:
  742. g += 10
  743. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_FILTER:
  744. b += 10
  745. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_DELAY:
  746. r += 15
  747. b -= 15
  748. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_DISTORTION:
  749. g += 10
  750. b += 10
  751. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_DYNAMICS:
  752. r += 10
  753. b += 10
  754. elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_UTILITY:
  755. r += 10
  756. g += 10
  757. bg = "noise1"
  758. if self.fPluginInfo['maker'] in ("falkTX, Michael Gruhn", "DISTRHO") and "3bandeq" in self.fPluginInfo['label'].lower():
  759. bg = "3bandeq"
  760. self.setStyleSheet("""
  761. PluginSlot_BasicFX#PluginWidget {
  762. background-color: rgb(%i, %i, %i);
  763. background-image: url(:/bitmaps/background_%s.png);
  764. background-repeat: repeat-xy;
  765. }
  766. QLabel#label_name {
  767. color: #BBB;
  768. }
  769. QLabel#label_name:disabled {
  770. color: #555;
  771. }
  772. """ % (r, g, b, bg))
  773. self.ui.b_enable.setPixmaps(":/bitmaps/button_off.png", ":/bitmaps/button_on.png", ":/bitmaps/button_off.png")
  774. self.ui.b_edit.setPixmaps(":/bitmaps/button_edit.png", ":/bitmaps/button_edit_down.png", ":/bitmaps/button_edit_hover.png")
  775. if self.fPluginInfo['iconName'] == "distrho":
  776. self.ui.b_gui.setPixmaps(":/bitmaps/button_distrho.png", ":/bitmaps/button_distrho_down.png", ":/bitmaps/button_distrho_hover.png")
  777. elif self.fPluginInfo['iconName'] == "file":
  778. self.ui.b_gui.setPixmaps(":/bitmaps/button_file.png", ":/bitmaps/button_file_down.png", ":/bitmaps/button_file_hover.png")
  779. else:
  780. self.ui.b_gui.setPixmaps(":/bitmaps/button_gui.png", ":/bitmaps/button_gui_down.png", ":/bitmaps/button_gui_hover.png")
  781. if self.fIsCollapsed:
  782. self.layout().setAlignment(Qt.AlignCenter)
  783. self.layout().setContentsMargins(2,2,2,0)
  784. self.ui.layout_leds.setContentsMargins(0,4,0,0)
  785. self.ui.w_knobs.hide()
  786. self.ui.w_screws_left.hide()
  787. self.ui.w_screws_right.hide()
  788. self.ui.label_name.setFixedHeight(self.ui.b_enable.height())
  789. print("TO BOTTOM")
  790. # -------------------------------------------------------------
  791. # Set-up parameters
  792. parameterCount = self.host.get_parameter_count(self.fPluginId) if not self.fIsCollapsed else 0
  793. index = 0
  794. for i in range(parameterCount):
  795. if index >= 8:
  796. break
  797. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  798. paramData = self.host.get_parameter_data(self.fPluginId, i)
  799. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  800. if paramData['type'] != PARAMETER_INPUT:
  801. continue
  802. if paramData['hints'] & PARAMETER_IS_BOOLEAN:
  803. continue
  804. if paramData['hints'] & PARAMETER_IS_INTEGER:
  805. continue
  806. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  807. continue
  808. paramName = getParameterShortName(paramInfo['name'])
  809. #if self.fPluginInfo['category'] == PLUGIN_CATEGORY_FILTER:
  810. #_r = 55 + float(i)/8*200
  811. #_g = 255 - float(i)/8*200
  812. #_b = 127 - r*2
  813. #elif self.fPluginInfo['category'] == PLUGIN_CATEGORY_DELAY:
  814. #_r = 127
  815. #_g = 55 + float(i)/8*200
  816. #_b = 255 - float(i)/8*200
  817. #elif r < b < g:
  818. #_r = 55 + float(i)/8*200
  819. #_g = 127
  820. #_b = 255 - float(i)/8*200
  821. #else:
  822. _r = 255 - float(index)/8*200
  823. _g = 55 + float(index)/8*200
  824. _b = (r-40)*4
  825. #if _r < 140: _r = 140
  826. #if _g < 140: _g = 140
  827. #if _b < 140: _b = 140
  828. widget = PixmapDial(self, i)
  829. widget.setPixmap(3)
  830. widget.setLabel(paramName)
  831. widget.setCustomPaintColor(QColor(_r, _g, _b))
  832. widget.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_COLOR)
  833. widget.forceWhiteLabelGradientText()
  834. widget.setMinimum(paramRanges['min'])
  835. widget.setMaximum(paramRanges['max'])
  836. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  837. widget.setEnabled(False)
  838. self.ui.w_knobs.layout().insertWidget(index, widget)
  839. index += 1
  840. self.fParameterList.append([i, widget])
  841. self.ui.dial_drywet.setIndex(PARAMETER_DRYWET)
  842. self.ui.dial_drywet.setPixmap(3)
  843. self.ui.dial_drywet.setLabel("Dry/Wet")
  844. self.ui.dial_drywet.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_CARLA_WET)
  845. self.ui.dial_drywet.setMinimum(0.0)
  846. self.ui.dial_drywet.setMaximum(1.0)
  847. self.ui.dial_drywet.forceWhiteLabelGradientText()
  848. self.ui.dial_drywet.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_DRYWET)
  849. self.ui.dial_vol.setIndex(PARAMETER_VOLUME)
  850. self.ui.dial_vol.setPixmap(3)
  851. self.ui.dial_vol.setLabel("Volume")
  852. self.ui.dial_vol.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_CARLA_VOL)
  853. self.ui.dial_vol.setMinimum(0.0)
  854. self.ui.dial_vol.setMaximum(1.27)
  855. self.ui.dial_vol.forceWhiteLabelGradientText()
  856. self.ui.dial_vol.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_VOLUME)
  857. self.fParameterList.append([PARAMETER_DRYWET, self.ui.dial_drywet])
  858. self.fParameterList.append([PARAMETER_VOLUME, self.ui.dial_vol])
  859. # -------------------------------------------------------------
  860. self.b_enable = self.ui.b_enable
  861. self.b_gui = self.ui.b_gui
  862. self.b_edit = self.ui.b_edit
  863. self.label_name = self.ui.label_name
  864. self.led_control = self.ui.led_control
  865. self.led_midi = self.ui.led_midi
  866. self.led_audio_in = self.ui.led_audio_in
  867. self.led_audio_out = self.ui.led_audio_out
  868. self.line = self.ui.line
  869. self.peak_in = self.ui.peak_in
  870. self.peak_out = self.ui.peak_out
  871. self.ready()
  872. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  873. #------------------------------------------------------------------
  874. def getFixedHeight(self):
  875. return 28 if self.fIsCollapsed else 79
  876. #------------------------------------------------------------------
  877. def paintEvent(self, event):
  878. painter = QPainter(self)
  879. painter.setBrush(Qt.transparent)
  880. painter.setPen(QPen(QColor(42, 42, 42), 1))
  881. painter.drawRect(0, 1, self.width()-1, self.getFixedHeight()-3)
  882. painter.setPen(QPen(QColor(60, 60, 60), 1))
  883. painter.drawLine(0, 0, self.width(), 0)
  884. AbstractPluginSlot.paintEvent(self, event)
  885. # ------------------------------------------------------------------------------------------------------------
  886. class PluginSlot_Calf(AbstractPluginSlot):
  887. def __init__(self, parent, host, pluginId):
  888. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  889. self.ui = ui_carla_plugin_calf.Ui_PluginWidget()
  890. self.ui.setupUi(self)
  891. # FIXME
  892. self.fIsCollapsed = False
  893. audioCount = self.host.get_audio_port_count_info(self.fPluginId)
  894. midiCount = self.host.get_midi_port_count_info(self.fPluginId)
  895. # -------------------------------------------------------------
  896. # Internal stuff
  897. self.fButtonFont = self.ui.b_gui.font()
  898. self.fButtonFont.setBold(False)
  899. self.fButtonFont.setPointSize(8)
  900. # Use black for mono plugins
  901. self.fBackgroundBlack = bool("mono" in self.fPluginInfo["label"].lower())
  902. self.fButtonColorOn = QColor( 18, 41, 87)
  903. self.fButtonColorOff = QColor(150, 150, 150)
  904. # -------------------------------------------------------------
  905. # Set-up GUI
  906. self.setStyleSheet("""
  907. QLabel#label_name, QLabel#label_audio_in, QLabel#label_audio_out, QLabel#label_midi {
  908. color: #BBB;
  909. }
  910. QLabel#label_name:disabled {
  911. color: #667;
  912. }
  913. PluginSlot_Calf#PluginWidget {
  914. background-image: url(:/bitmaps/background_calf_%s.png);
  915. background-repeat: repeat-xy;
  916. border: 2px;
  917. }
  918. """ % ("black" if self.fBackgroundBlack else "blue"))
  919. self.ui.line.setStyleSheet("* { color: #FF5100; }")
  920. self.ui.b_enable.setPixmaps(":/bitmaps/button_calf3.png", ":/bitmaps/button_calf3_down.png", ":/bitmaps/button_calf3.png")
  921. self.ui.b_gui.setPixmaps(":/bitmaps/button_calf2.png", ":/bitmaps/button_calf2_down.png", ":/bitmaps/button_calf2_hover.png")
  922. self.ui.b_edit.setPixmaps(":/bitmaps/button_calf2.png", ":/bitmaps/button_calf2_down.png", ":/bitmaps/button_calf2_hover.png")
  923. self.ui.b_remove.setPixmaps(":/bitmaps/button_calf1.png", ":/bitmaps/button_calf1_down.png", ":/bitmaps/button_calf1_hover.png")
  924. self.ui.b_edit.setTopText(self.tr("Edit"), self.fButtonColorOn, self.fButtonFont)
  925. self.ui.b_remove.setTopText(self.tr("Remove"), self.fButtonColorOn, self.fButtonFont)
  926. if self.fPluginInfo['hints'] & PLUGIN_HAS_CUSTOM_UI:
  927. self.ui.b_gui.setTopText(self.tr("GUI"), self.fButtonColorOn, self.fButtonFont)
  928. else:
  929. self.ui.b_gui.setTopText(self.tr("GUI"), self.fButtonColorOff, self.fButtonFont)
  930. labelFont = self.ui.label_name.font()
  931. labelFont.setBold(True)
  932. labelFont.setPointSize(10)
  933. self.ui.label_name.setFont(labelFont)
  934. self.ui.peak_in.setMeterStyle(DigitalPeakMeter.STYLE_CALF)
  935. self.ui.peak_out.setMeterStyle(DigitalPeakMeter.STYLE_CALF)
  936. if audioCount['ins'] == 0:
  937. self.ui.label_audio_in.hide()
  938. self.ui.peak_in.hide()
  939. if audioCount['outs'] > 0:
  940. self.ui.peak_out.setMinimumWidth(200)
  941. if audioCount['outs'] == 0:
  942. self.ui.label_audio_out.hide()
  943. self.ui.peak_out.hide()
  944. if midiCount['ins'] == 0:
  945. self.ui.label_midi.hide()
  946. self.ui.led_midi.hide()
  947. if self.fIdleTimerId != 0:
  948. self.ui.b_remove.setEnabled(False)
  949. self.ui.b_remove.setVisible(False)
  950. # -------------------------------------------------------------
  951. # Set-up parameters
  952. parameterCount = self.host.get_parameter_count(self.fPluginId) if not self.fIsCollapsed else 0
  953. index = 0
  954. limit = 7 if midiCount['ins'] == 0 else 6
  955. for i in range(parameterCount):
  956. if index >= limit:
  957. break
  958. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  959. paramData = self.host.get_parameter_data(self.fPluginId, i)
  960. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  961. if paramData['type'] != PARAMETER_INPUT:
  962. continue
  963. if paramData['hints'] & PARAMETER_IS_BOOLEAN:
  964. continue
  965. if paramData['hints'] & PARAMETER_IS_INTEGER:
  966. continue
  967. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  968. continue
  969. paramName = getParameterShortName(paramInfo['name'])
  970. widget = PixmapDial(self, i)
  971. widget.setPixmap(7)
  972. widget.setLabel(paramName)
  973. widget.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  974. widget.setMinimum(paramRanges['min'])
  975. widget.setMaximum(paramRanges['max'])
  976. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  977. widget.setEnabled(False)
  978. self.ui.w_knobs.layout().insertWidget(index, widget)
  979. index += 1
  980. self.fParameterList.append([i, widget])
  981. # -------------------------------------------------------------
  982. self.b_enable = self.ui.b_enable
  983. self.b_gui = self.ui.b_gui
  984. self.b_edit = self.ui.b_edit
  985. self.b_remove = self.ui.b_remove
  986. self.label_name = self.ui.label_name
  987. self.led_midi = self.ui.led_midi
  988. self.line = self.ui.line
  989. self.peak_in = self.ui.peak_in
  990. self.peak_out = self.ui.peak_out
  991. self.ready()
  992. self.ui.led_midi.setColor(self.ui.led_midi.CALF)
  993. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  994. #------------------------------------------------------------------
  995. def getFixedHeight(self):
  996. return 94 if max(self.peak_in.channelCount(), self.peak_out.channelCount()) < 2 else 106
  997. #------------------------------------------------------------------
  998. def editDialogPluginHintsChanged(self, pluginId, hints):
  999. if hints & PLUGIN_HAS_CUSTOM_UI:
  1000. self.ui.b_gui.setTopText(self.tr("GUI"), self.fButtonColorOn, self.fButtonFont)
  1001. else:
  1002. self.ui.b_gui.setTopText(self.tr("GUI"), self.fButtonColorOff, self.fButtonFont)
  1003. AbstractPluginSlot.editDialogPluginHintsChanged(self, pluginId, hints)
  1004. #------------------------------------------------------------------
  1005. def paintEvent(self, event):
  1006. painter = QPainter(self)
  1007. painter.setBrush(Qt.transparent)
  1008. painter.setPen(QPen(QColor(20, 20, 20) if self.fBackgroundBlack else QColor(75, 86, 99), 1))
  1009. painter.drawRect(0, 1, self.width()-1, 88-3)
  1010. painter.setPen(QPen(QColor(45, 45, 45) if self.fBackgroundBlack else QColor(86, 99, 114), 1))
  1011. painter.drawLine(0, 0, self.width(), 0)
  1012. AbstractPluginSlot.paintEvent(self, event)
  1013. # ------------------------------------------------------------------------------------------------------------
  1014. class PluginSlot_OpenAV(AbstractPluginSlot):
  1015. def __init__(self, parent, host, pluginId):
  1016. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  1017. self.ui = ui_carla_plugin_basic_fx.Ui_PluginWidget()
  1018. self.ui.setupUi(self)
  1019. # -------------------------------------------------------------
  1020. # Set-up GUI
  1021. QFontDatabase.addApplicationFont(":/fonts/uranium.ttf")
  1022. labelFont = QFont()
  1023. labelFont.setFamily("Uranium")
  1024. labelFont.setPointSize(13)
  1025. labelFont.setCapitalization(QFont.AllUppercase)
  1026. self.ui.label_name.setFont(labelFont)
  1027. self.setStyleSheet("""
  1028. PluginSlot_OpenAV#PluginWidget {
  1029. background-color: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,
  1030. stop: 0 #303030, stop: 0.35 #111111, stop: 1.0 #111111);
  1031. }
  1032. """)
  1033. self.ui.label_name.setStyleSheet("* { color: #FF5100; }")
  1034. #self.ui.line.setStyleSheet("* { background-color: #FF5100; color: #FF5100; }")
  1035. self.ui.peak_in.setMeterStyle(DigitalPeakMeter.STYLE_OPENAV)
  1036. self.ui.peak_out.setMeterStyle(DigitalPeakMeter.STYLE_OPENAV)
  1037. self.ui.b_enable.setPixmaps(":/bitmaps/button_off.png", ":/bitmaps/button_on.png", ":/bitmaps/button_off.png")
  1038. self.ui.b_edit.setPixmaps(":/bitmaps/button_edit.png", ":/bitmaps/button_edit_down.png", ":/bitmaps/button_edit_hover.png")
  1039. self.ui.b_gui.setPixmaps(":/bitmaps/button_gui.png", ":/bitmaps/button_gui_down.png", ":/bitmaps/button_gui_hover.png")
  1040. # -------------------------------------------------------------
  1041. # Set-up parameters
  1042. parameterCount = self.host.get_parameter_count(self.fPluginId) if not self.fIsCollapsed else 0
  1043. index = 0
  1044. for i in range(parameterCount):
  1045. if index >= 8:
  1046. break
  1047. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  1048. paramData = self.host.get_parameter_data(self.fPluginId, i)
  1049. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  1050. if paramData['type'] != PARAMETER_INPUT:
  1051. continue
  1052. if paramData['hints'] & PARAMETER_IS_BOOLEAN:
  1053. continue
  1054. if paramData['hints'] & PARAMETER_IS_INTEGER:
  1055. continue
  1056. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1057. continue
  1058. paramName = getParameterShortName(paramInfo['name'])
  1059. widget = PixmapDial(self, i)
  1060. widget.setPixmap(11)
  1061. widget.setLabel(paramName)
  1062. widget.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1063. widget.setMinimum(paramRanges['min'])
  1064. widget.setMaximum(paramRanges['max'])
  1065. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1066. widget.setEnabled(False)
  1067. self.ui.w_knobs.layout().insertWidget(index, widget)
  1068. index += 1
  1069. self.fParameterList.append([i, widget])
  1070. self.ui.dial_drywet.setIndex(PARAMETER_DRYWET)
  1071. self.ui.dial_drywet.setPixmap(13)
  1072. self.ui.dial_drywet.setLabel("Dry/Wet")
  1073. self.ui.dial_drywet.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1074. self.ui.dial_drywet.setMinimum(0.0)
  1075. self.ui.dial_drywet.setMaximum(1.0)
  1076. self.ui.dial_drywet.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_DRYWET)
  1077. self.ui.dial_vol.setIndex(PARAMETER_VOLUME)
  1078. self.ui.dial_vol.setPixmap(12)
  1079. self.ui.dial_vol.setLabel("Volume")
  1080. self.ui.dial_vol.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1081. self.ui.dial_vol.setMinimum(0.0)
  1082. self.ui.dial_vol.setMaximum(1.27)
  1083. self.ui.dial_vol.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_VOLUME)
  1084. self.fParameterList.append([PARAMETER_DRYWET, self.ui.dial_drywet])
  1085. self.fParameterList.append([PARAMETER_VOLUME, self.ui.dial_vol])
  1086. # -------------------------------------------------------------
  1087. self.b_enable = self.ui.b_enable
  1088. self.b_gui = self.ui.b_gui
  1089. self.b_edit = self.ui.b_edit
  1090. self.label_name = self.ui.label_name
  1091. self.led_control = self.ui.led_control
  1092. self.led_midi = self.ui.led_midi
  1093. self.led_audio_in = self.ui.led_audio_in
  1094. self.led_audio_out = self.ui.led_audio_out
  1095. self.line = self.ui.line
  1096. self.peak_in = self.ui.peak_in
  1097. self.peak_out = self.ui.peak_out
  1098. self.ready()
  1099. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  1100. #------------------------------------------------------------------
  1101. def getFixedHeight(self):
  1102. return 28 if self.fIsCollapsed else 79
  1103. #------------------------------------------------------------------
  1104. def paintEvent(self, event):
  1105. painter = QPainter(self)
  1106. painter.setBrush(Qt.transparent)
  1107. painter.setPen(QPen(QColor(42, 42, 42), 1))
  1108. painter.drawRect(0, 1, self.width()-1, 79-3)
  1109. painter.setPen(QPen(QColor(60, 60, 60), 1))
  1110. painter.drawLine(0, 0, self.width(), 0)
  1111. AbstractPluginSlot.paintEvent(self, event)
  1112. # ------------------------------------------------------------------------------------------------------------
  1113. class PluginSlot_Nekobi(AbstractPluginSlot):
  1114. def __init__(self, parent, host, pluginId):
  1115. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  1116. #self.ui = ui_carla_plugin_basic_fx.Ui_PluginWidget()
  1117. #self.ui.setupUi(self)
  1118. # -------------------------------------------------------------
  1119. # Set-up GUI
  1120. self.fPixmapCenter = QPixmap(":/bitmaps/background_nekobi.png")
  1121. self.fPixmapLeft = QPixmap(":/bitmaps/background_nekobi_left.png")
  1122. self.fPixmapLeftRect = QRectF(0, 0, self.fPixmapLeft.width(), self.fPixmapLeft.height())
  1123. self.fPixmapRight = QPixmap(":/bitmaps/background_nekobi_right.png")
  1124. self.fPixmapRightRect = QRectF(0, 0, self.fPixmapRight.width(), self.fPixmapRight.height())
  1125. #self.setStyleSheet("""
  1126. #PluginSlot_Nekobi#PluginWidget {
  1127. #background-image: url(:/bitmaps/background_nekobi.png);
  1128. #background-repeat: repeat-xy;
  1129. #}
  1130. #QLabel#label_name {
  1131. #color: #BBB;
  1132. #}
  1133. #""")
  1134. #------------------------------------------------------------------
  1135. def getFixedHeight(self):
  1136. return 108
  1137. #------------------------------------------------------------------
  1138. def paintEvent(self, event):
  1139. painter = QPainter(self)
  1140. # main bg (center)
  1141. painter.drawTiledPixmap(0, 0, self.width(), self.height(), self.fPixmapCenter)
  1142. # left side
  1143. painter.drawPixmap(self.fPixmapLeftRect, self.fPixmapLeft, self.fPixmapLeftRect)
  1144. # right side
  1145. rightTarget = QRectF(self.fPixmapRightRect)
  1146. rightTarget.moveLeft(self.width()-rightTarget.width())
  1147. painter.drawPixmap(rightTarget, self.fPixmapRight, self.fPixmapRightRect)
  1148. AbstractPluginSlot.paintEvent(self, event)
  1149. # ------------------------------------------------------------------------------------------------------------
  1150. class PluginSlot_SF2(AbstractPluginSlot):
  1151. def __init__(self, parent, host, pluginId):
  1152. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  1153. self.ui = ui_carla_plugin_sf2.Ui_PluginWidget()
  1154. self.ui.setupUi(self)
  1155. # -------------------------------------------------------------
  1156. # Set-up GUI
  1157. #labelFont = self.ui.label_name.font()
  1158. #labelFont.setBold(True)
  1159. #labelFont.setPointSize(9)
  1160. #self.ui.label_name.setFont(labelFont)
  1161. self.setStyleSheet("""
  1162. PluginSlot_SF2#PluginWidget {
  1163. background-image: url(:/bitmaps/background_3bandeq.png);
  1164. background-repeat: repeat-xy;
  1165. }
  1166. QLabel#label_name {
  1167. color: #BBB;
  1168. }
  1169. """)
  1170. self.ui.b_enable.setPixmaps(":/bitmaps/button_off.png", ":/bitmaps/button_on.png", ":/bitmaps/button_off.png")
  1171. self.ui.b_edit.setPixmaps(":/bitmaps/button_edit.png", ":/bitmaps/button_edit_down.png", ":/bitmaps/button_edit_hover.png")
  1172. # -------------------------------------------------------------
  1173. # Set-up parameters
  1174. parameterCount = self.host.get_parameter_count(self.fPluginId) if not self.fIsCollapsed else 0
  1175. index = 0
  1176. for i in range(parameterCount):
  1177. if index >= 8:
  1178. break
  1179. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  1180. paramData = self.host.get_parameter_data(self.fPluginId, i)
  1181. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  1182. if paramData['type'] != PARAMETER_INPUT:
  1183. continue
  1184. if paramData['hints'] & PARAMETER_IS_BOOLEAN:
  1185. continue
  1186. if paramData['hints'] & PARAMETER_IS_INTEGER:
  1187. continue
  1188. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1189. continue
  1190. paramName = getParameterShortName(paramInfo['name'])
  1191. widget = PixmapDial(self, i)
  1192. widget.setPixmap(3)
  1193. widget.setLabel(paramName)
  1194. widget.setMinimum(paramRanges['min'])
  1195. widget.setMaximum(paramRanges['max'])
  1196. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1197. widget.setEnabled(False)
  1198. self.ui.w_knobs.layout().insertWidget(index, widget)
  1199. index += 1
  1200. self.fParameterList.append([i, widget])
  1201. self.ui.dial_vol.setIndex(PARAMETER_VOLUME)
  1202. self.ui.dial_vol.setPixmap(3)
  1203. self.ui.dial_vol.setLabel("Volume")
  1204. self.ui.dial_vol.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_CARLA_VOL)
  1205. self.ui.dial_vol.setMinimum(0.0)
  1206. self.ui.dial_vol.setMaximum(1.27)
  1207. self.ui.dial_vol.forceWhiteLabelGradientText()
  1208. self.ui.dial_vol.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_VOLUME)
  1209. self.fParameterList.append([PARAMETER_VOLUME, self.ui.dial_vol])
  1210. # -------------------------------------------------------------
  1211. self.b_enable = self.ui.b_enable
  1212. self.b_edit = self.ui.b_edit
  1213. self.label_name = self.ui.label_name
  1214. self.led_control = self.ui.led_control
  1215. self.led_midi = self.ui.led_midi
  1216. self.led_audio_out = self.ui.led_audio_out
  1217. self.line = self.ui.line
  1218. self.peak_out = self.ui.peak_out
  1219. self.ready()
  1220. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  1221. #------------------------------------------------------------------
  1222. def getFixedHeight(self):
  1223. return 79
  1224. #------------------------------------------------------------------
  1225. def paintEvent(self, event):
  1226. painter = QPainter(self)
  1227. painter.setBrush(Qt.transparent)
  1228. painter.setPen(QPen(QColor(42, 42, 42), 1))
  1229. painter.drawRect(0, 1, self.width()-1, 79-3)
  1230. painter.setPen(QPen(QColor(60, 60, 60), 1))
  1231. painter.drawLine(0, 0, self.width(), 0)
  1232. AbstractPluginSlot.paintEvent(self, event)
  1233. # ------------------------------------------------------------------------------------------------------------
  1234. class PluginSlot_ZynFX(AbstractPluginSlot):
  1235. def __init__(self, parent, host, pluginId):
  1236. AbstractPluginSlot.__init__(self, parent, host, pluginId)
  1237. self.ui = ui_carla_plugin_zynfx.Ui_PluginWidget()
  1238. self.ui.setupUi(self)
  1239. # -------------------------------------------------------------
  1240. # Set-up GUI
  1241. self.setStyleSheet("""
  1242. PluginSlot_ZynFX#PluginWidget {
  1243. background-image: url(:/bitmaps/background_zynfx.png);
  1244. background-repeat: repeat-xy;
  1245. border: 2px;
  1246. }
  1247. QLabel#label_name, QLabel#label_presets {
  1248. color: #BBB;
  1249. }
  1250. """)
  1251. self.ui.b_enable.setPixmaps(":/bitmaps/button_off.png", ":/bitmaps/button_on.png", ":/bitmaps/button_off.png")
  1252. self.ui.b_edit.setPixmaps(":/bitmaps/button_edit.png", ":/bitmaps/button_edit_down.png", ":/bitmaps/button_edit_hover.png")
  1253. labelFont = self.ui.label_name.font()
  1254. labelFont.setBold(True)
  1255. labelFont.setPointSize(9)
  1256. self.ui.label_name.setFont(labelFont)
  1257. presetFont = self.ui.label_presets.font()
  1258. presetFont.setBold(True)
  1259. presetFont.setPointSize(8)
  1260. self.ui.label_presets.setFont(presetFont)
  1261. self.ui.peak_in.setMeterStyle(DigitalPeakMeter.STYLE_OPENAV)
  1262. self.ui.peak_out.setMeterStyle(DigitalPeakMeter.STYLE_OPENAV)
  1263. # -------------------------------------------------------------
  1264. # Set-up parameters
  1265. parameterCount = self.host.get_parameter_count(self.fPluginId) if not self.fIsCollapsed else 0
  1266. index = 0
  1267. for i in range(parameterCount):
  1268. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  1269. paramData = self.host.get_parameter_data(self.fPluginId, i)
  1270. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  1271. if paramData['type'] != PARAMETER_INPUT:
  1272. continue
  1273. if paramData['hints'] & PARAMETER_IS_BOOLEAN:
  1274. continue
  1275. if paramData['hints'] & PARAMETER_IS_INTEGER:
  1276. continue
  1277. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1278. continue
  1279. paramName = paramInfo['name']
  1280. # real zyn fx plugins
  1281. if self.fPluginInfo['label'] == "zynalienwah":
  1282. if i == 0: paramName = "Freq"
  1283. elif i == 1: paramName = "Rnd"
  1284. elif i == 2: paramName = "L type" # combobox
  1285. elif i == 3: paramName = "St.df"
  1286. elif i == 5: paramName = "Fb"
  1287. elif i == 7: paramName = "L/R"
  1288. elif self.fPluginInfo['label'] == "zynchorus":
  1289. if i == 0: paramName = "Freq"
  1290. elif i == 1: paramName = "Rnd"
  1291. elif i == 2: paramName = "L type" # combobox
  1292. elif i == 3: paramName = "St.df"
  1293. elif i == 6: paramName = "Fb"
  1294. elif i == 7: paramName = "L/R"
  1295. elif i == 8: paramName = "Flngr" # button
  1296. elif i == 9: paramName = "Subst" # button
  1297. elif self.fPluginInfo['label'] == "zyndistortion":
  1298. if i == 0: paramName = "LRc."
  1299. elif i == 4: paramName = "Neg." # button
  1300. elif i == 5: paramName = "LPF"
  1301. elif i == 6: paramName = "HPF"
  1302. elif i == 7: paramName = "St." # button
  1303. elif i == 8: paramName = "PF" # button
  1304. elif self.fPluginInfo['label'] == "zyndynamicfilter":
  1305. if i == 0: paramName = "Freq"
  1306. elif i == 1: paramName = "Rnd"
  1307. elif i == 2: paramName = "L type" # combobox
  1308. elif i == 3: paramName = "St.df"
  1309. elif i == 4: paramName = "LfoD"
  1310. elif i == 5: paramName = "A.S."
  1311. elif i == 6: paramName = "A.Inv." # button
  1312. elif i == 7: paramName = "A.M."
  1313. elif self.fPluginInfo['label'] == "zynecho":
  1314. if i == 1: paramName = "LRdl."
  1315. elif i == 2: paramName = "LRc."
  1316. elif i == 3: paramName = "Fb."
  1317. elif i == 4: paramName = "Damp"
  1318. elif self.fPluginInfo['label'] == "zynphaser":
  1319. if i == 0: paramName = "Freq"
  1320. elif i == 1: paramName = "Rnd"
  1321. elif i == 2: paramName = "L type" # combobox
  1322. elif i == 3: paramName = "St.df"
  1323. elif i == 5: paramName = "Fb"
  1324. elif i == 7: paramName = "L/R"
  1325. elif i == 8: paramName = "Subst" # button
  1326. elif i == 9: paramName = "Phase"
  1327. elif i == 11: paramName = "Dist"
  1328. elif self.fPluginInfo['label'] == "zynreverb":
  1329. if i == 2: paramName = "I.delfb"
  1330. elif i == 5: paramName = "LPF"
  1331. elif i == 6: paramName = "HPF"
  1332. elif i == 9: paramName = "R.S."
  1333. elif i == 10: paramName = "I.del"
  1334. else:
  1335. paramName = getParameterShortName(paramInfo['name'])
  1336. widget = PixmapDial(self, i)
  1337. widget.setPixmap(5)
  1338. widget.setLabel(paramName)
  1339. widget.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1340. widget.setMinimum(paramRanges['min'])
  1341. widget.setMaximum(paramRanges['max'])
  1342. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  1343. widget.setEnabled(False)
  1344. self.ui.w_knobs.layout().insertWidget(index, widget)
  1345. index += 1
  1346. self.fParameterList.append([i, widget])
  1347. self.ui.dial_drywet.setIndex(PARAMETER_DRYWET)
  1348. self.ui.dial_drywet.setPixmap(5)
  1349. self.ui.dial_drywet.setLabel("Wet")
  1350. self.ui.dial_drywet.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1351. self.ui.dial_drywet.setMinimum(0.0)
  1352. self.ui.dial_drywet.setMaximum(1.0)
  1353. self.ui.dial_drywet.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_DRYWET)
  1354. self.ui.dial_vol.setIndex(PARAMETER_VOLUME)
  1355. self.ui.dial_vol.setPixmap(5)
  1356. self.ui.dial_vol.setLabel("Vol")
  1357. self.ui.dial_vol.setCustomPaintMode(PixmapDial.CUSTOM_PAINT_MODE_NO_GRADIENT)
  1358. self.ui.dial_vol.setMinimum(0.0)
  1359. self.ui.dial_vol.setMaximum(1.27)
  1360. self.ui.dial_vol.setVisible(self.fPluginInfo['hints'] & PLUGIN_CAN_VOLUME)
  1361. self.fParameterList.append([PARAMETER_DRYWET, self.ui.dial_drywet])
  1362. self.fParameterList.append([PARAMETER_VOLUME, self.ui.dial_vol])
  1363. # -------------------------------------------------------------
  1364. # Set-up MIDI programs
  1365. midiProgramCount = self.host.get_midi_program_count(self.fPluginId)
  1366. if midiProgramCount > 0:
  1367. self.ui.cb_presets.setEnabled(True)
  1368. self.ui.label_presets.setEnabled(True)
  1369. for i in range(midiProgramCount):
  1370. mpData = self.host.get_midi_program_data(self.fPluginId, i)
  1371. mpName = mpData['name']
  1372. self.ui.cb_presets.addItem(mpName)
  1373. self.fCurrentMidiProgram = self.host.get_current_midi_program_index(self.fPluginId)
  1374. self.ui.cb_presets.setCurrentIndex(self.fCurrentMidiProgram)
  1375. else:
  1376. self.fCurrentMidiProgram = -1
  1377. self.ui.cb_presets.setEnabled(False)
  1378. self.ui.cb_presets.setVisible(False)
  1379. self.ui.label_presets.setEnabled(False)
  1380. self.ui.label_presets.setVisible(False)
  1381. # -------------------------------------------------------------
  1382. self.b_enable = self.ui.b_enable
  1383. self.b_edit = self.ui.b_edit
  1384. self.cb_presets = self.ui.cb_presets
  1385. self.label_name = self.ui.label_name
  1386. self.led_control = self.ui.led_control
  1387. self.led_midi = self.ui.led_midi
  1388. self.led_audio_in = self.ui.led_audio_in
  1389. self.led_audio_out = self.ui.led_audio_out
  1390. self.line = self.ui.line
  1391. self.peak_in = self.ui.peak_in
  1392. self.peak_out = self.ui.peak_out
  1393. self.ready()
  1394. self.customContextMenuRequested.connect(self.slot_showDefaultCustomMenu)
  1395. self.ui.cb_presets.currentIndexChanged.connect(self.slot_midiProgramChanged)
  1396. #------------------------------------------------------------------
  1397. def getFixedHeight(self):
  1398. return 30 if self.fIsCollapsed else 79
  1399. #------------------------------------------------------------------
  1400. def paintEvent(self, event):
  1401. painter = QPainter(self)
  1402. painter.setBrush(Qt.transparent)
  1403. painter.setPen(QPen(QColor(60, 60, 60), 1))
  1404. painter.drawRect(0, 1, self.width()-1, self.height()-3)
  1405. painter.setPen(QPen(QColor(94, 94, 94), 1))
  1406. painter.drawLine(0, 0, self.width(), 0)
  1407. AbstractPluginSlot.paintEvent(self, event)
  1408. # ------------------------------------------------------------------------------------------------------------
  1409. def createPluginSlot(parent, host, pluginId, useSkins):
  1410. if not useSkins:
  1411. return PluginSlot_Default(parent, host, pluginId)
  1412. pluginInfo = host.get_plugin_info(pluginId)
  1413. pluginName = host.get_real_plugin_name(pluginId)
  1414. pluginLabel = pluginInfo['label']
  1415. pluginMaker = pluginInfo['maker']
  1416. uniqueId = pluginInfo['uniqueId']
  1417. #pluginIcon = pluginInfo['iconName']
  1418. if pluginInfo['type'] == PLUGIN_SF2:
  1419. return PluginSlot_SF2(parent, host, pluginId)
  1420. if pluginMaker == "OpenAV Productions":
  1421. return PluginSlot_OpenAV(parent, host, pluginId)
  1422. if pluginInfo['type'] == PLUGIN_INTERNAL:
  1423. if pluginLabel.startswith("zyn") and pluginInfo['category'] != PLUGIN_CATEGORY_SYNTH:
  1424. return PluginSlot_ZynFX(parent, host, pluginId)
  1425. elif pluginInfo['type'] == PLUGIN_LADSPA:
  1426. if pluginLabel.startswith("Zyn") and pluginMaker.startswith("Josep Andreu"):
  1427. return PluginSlot_ZynFX(parent, host, pluginId)
  1428. if pluginName.split(" ", 1)[0].lower() == "calf":
  1429. return PluginSlot_Calf(parent, host, pluginId)
  1430. #if pluginName.lower() == "nekobi":
  1431. #return PluginSlot_Nekobi(parent, pluginId)
  1432. return PluginSlot_BasicFX(parent, host, pluginId)
  1433. # ------------------------------------------------------------------------------------------------------------
  1434. # Main Testing
  1435. if __name__ == '__main__':
  1436. from carla_app import CarlaApplication
  1437. from carla_host import initHost, loadHostSettings
  1438. import resources_rc
  1439. app = CarlaApplication("Carla-Skins")
  1440. host = initHost("Skins", None, False, False, False)
  1441. loadHostSettings(host)
  1442. host.engine_init("JACK", "Carla-Widgets")
  1443. host.add_plugin(BINARY_NATIVE, PLUGIN_INTERNAL, "", "", "zynreverb", 0, None, 0x0)
  1444. #host.add_plugin(BINARY_NATIVE, PLUGIN_DSSI, "/usr/lib/dssi/karplong.so", "karplong", "karplong", 0, None, 0x0)
  1445. #host.add_plugin(BINARY_NATIVE, PLUGIN_LV2, "", "", "http://www.openavproductions.com/sorcer", 0, None, 0x0)
  1446. #host.add_plugin(BINARY_NATIVE, PLUGIN_LV2, "", "", "http://calf.sourceforge.net/plugins/Compressor", 0, None, 0x0)
  1447. host.set_active(0, True)
  1448. gui = createPluginSlot(None, host, 0, True)
  1449. gui.testTimer()
  1450. gui.show()
  1451. app.exec_()