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.

1797 lines
66KB

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