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.

1831 lines
73KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla widgets code
  4. # Copyright (C) 2011-2020 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. from abc import abstractmethod
  20. # ------------------------------------------------------------------------------------------------------------
  21. # Imports (PyQt5)
  22. from PyQt5.QtCore import pyqtSignal, pyqtSlot, Qt, QByteArray
  23. from PyQt5.QtGui import QCursor, QIcon, QPalette, QPixmap
  24. from PyQt5.QtWidgets import QDialog, QFileDialog, QInputDialog, QMenu, QMessageBox, QScrollArea, QVBoxLayout, QWidget
  25. # ------------------------------------------------------------------------------------------------------------
  26. # Imports (Custom)
  27. import ui_carla_about
  28. import ui_carla_about_juce
  29. import ui_carla_edit
  30. import ui_carla_parameter
  31. from carla_backend import (
  32. MACOS, WINDOWS,
  33. BINARY_NATIVE,
  34. PLUGIN_INTERNAL,
  35. PLUGIN_DSSI,
  36. PLUGIN_LV2,
  37. PLUGIN_VST2,
  38. PLUGIN_SF2,
  39. PLUGIN_SFZ,
  40. PLUGIN_CAN_DRYWET,
  41. PLUGIN_CAN_VOLUME,
  42. PLUGIN_CAN_BALANCE,
  43. PLUGIN_CAN_PANNING,
  44. PLUGIN_CATEGORY_SYNTH,
  45. PLUGIN_OPTION_FIXED_BUFFERS,
  46. PLUGIN_OPTION_FORCE_STEREO,
  47. PLUGIN_OPTION_MAP_PROGRAM_CHANGES,
  48. PLUGIN_OPTION_USE_CHUNKS,
  49. PLUGIN_OPTION_SEND_CONTROL_CHANGES,
  50. PLUGIN_OPTION_SEND_CHANNEL_PRESSURE,
  51. PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH,
  52. PLUGIN_OPTION_SEND_PITCHBEND,
  53. PLUGIN_OPTION_SEND_ALL_SOUND_OFF,
  54. PLUGIN_OPTION_SEND_PROGRAM_CHANGES,
  55. PARAMETER_DRYWET,
  56. PARAMETER_VOLUME,
  57. PARAMETER_BALANCE_LEFT,
  58. PARAMETER_BALANCE_RIGHT,
  59. PARAMETER_PANNING,
  60. PARAMETER_CTRL_CHANNEL,
  61. PARAMETER_IS_ENABLED,
  62. PARAMETER_IS_AUTOMABLE,
  63. PARAMETER_IS_READ_ONLY,
  64. PARAMETER_USES_SCALEPOINTS,
  65. PARAMETER_USES_CUSTOM_TEXT,
  66. PARAMETER_CAN_BE_CV_CONTROLLED,
  67. PARAMETER_INPUT, PARAMETER_OUTPUT,
  68. CONTROL_VALUE_NONE,
  69. CONTROL_VALUE_MIDI_PITCHBEND,
  70. CONTROL_VALUE_CV
  71. )
  72. from carla_shared import (
  73. MIDI_CC_LIST, MAX_MIDI_CC_LIST_ITEM,
  74. VERSION,
  75. countDecimalPoints,
  76. fontMetricsHorizontalAdvance,
  77. setUpSignals,
  78. gCarla
  79. )
  80. from carla_utils import getPluginTypeAsString
  81. #)
  82. from widgets.collapsablewidget import CollapsibleBox
  83. from widgets.pixmapkeyboard import PixmapKeyboardHArea
  84. # ------------------------------------------------------------------------------------------------------------
  85. # Carla GUI defines
  86. ICON_STATE_ON = 3 # turns on, sets as wait
  87. ICON_STATE_WAIT = 2 # nothing, sets as off
  88. ICON_STATE_OFF = 1 # turns off, sets as null
  89. ICON_STATE_NULL = 0 # nothing
  90. # ------------------------------------------------------------------------------------------------------------
  91. # Carla About dialog
  92. class CarlaAboutW(QDialog):
  93. def __init__(self, parent, host):
  94. QDialog.__init__(self, parent)
  95. self.ui = ui_carla_about.Ui_CarlaAboutW()
  96. self.ui.setupUi(self)
  97. if host.isControl:
  98. extraInfo = " - <b>%s</b>" % self.tr("OSC Bridge Version")
  99. elif host.isPlugin:
  100. extraInfo = " - <b>%s</b>" % self.tr("Plugin Version")
  101. else:
  102. extraInfo = ""
  103. self.ui.l_about.setText(self.tr(""
  104. "<br>Version %s"
  105. "<br>Carla is a fully-featured audio plugin host%s.<br>"
  106. "<br>Copyright (C) 2011-2020 falkTX<br>"
  107. "" % (VERSION, extraInfo)))
  108. if self.ui.about.palette().color(QPalette.Background).blackF() < 0.5:
  109. self.ui.l_icons.setPixmap(QPixmap(":/bitmaps/carla_about_black.png"))
  110. self.ui.ico_example_edit.setPixmap(QPixmap(":/bitmaps/button_file-black.png"))
  111. self.ui.ico_example_file.setPixmap(QPixmap(":/bitmaps/button_edit-black.png"))
  112. self.ui.ico_example_gui.setPixmap(QPixmap(":/bitmaps/button_gui-black.png"))
  113. if host.isControl:
  114. self.ui.l_extended.hide()
  115. self.ui.tabWidget.removeTab(3)
  116. self.ui.tabWidget.removeTab(2)
  117. self.ui.l_extended.setText(gCarla.utils.get_complete_license_text())
  118. if host.is_engine_running() and not host.isControl:
  119. self.ui.le_osc_url_tcp.setText(host.get_host_osc_url_tcp())
  120. self.ui.le_osc_url_udp.setText(host.get_host_osc_url_udp())
  121. else:
  122. self.ui.le_osc_url_tcp.setText(self.tr("(Engine not running)"))
  123. self.ui.le_osc_url_udp.setText(self.tr("(Engine not running)"))
  124. # pylint: disable=line-too-long
  125. self.ui.l_osc_cmds.setText("<table>"
  126. "<tr><td>" "/set_active" "&nbsp;</td><td>&lt;i-value&gt;</td></tr>"
  127. "<tr><td>" "/set_drywet" "&nbsp;</td><td>&lt;f-value&gt;</td></tr>"
  128. "<tr><td>" "/set_volume" "&nbsp;</td><td>&lt;f-value&gt;</td></tr>"
  129. "<tr><td>" "/set_balance_left" "&nbsp;</td><td>&lt;f-value&gt;</td></tr>"
  130. "<tr><td>" "/set_balance_right" "&nbsp;</td><td>&lt;f-value&gt;</td></tr>"
  131. "<tr><td>" "/set_panning" "&nbsp;</td><td>&lt;f-value&gt;</td></tr>"
  132. "<tr><td>" "/set_parameter_value" "&nbsp;</td><td>&lt;i-index&gt; &lt;f-value&gt;</td></tr>"
  133. "<tr><td>" "/set_parameter_midi_cc" "&nbsp;</td><td>&lt;i-index&gt; &lt;i-cc&gt;</td></tr>"
  134. "<tr><td>" "/set_parameter_midi_channel" "&nbsp;</td><td>&lt;i-index&gt; &lt;i-channel&gt;</td></tr>"
  135. "<tr><td>" "/set_program" "&nbsp;</td><td>&lt;i-index&gt;</td></tr>"
  136. "<tr><td>" "/set_midi_program" "&nbsp;</td><td>&lt;i-index&gt;</td></tr>"
  137. "<tr><td>" "/note_on" "&nbsp;</td><td>&lt;i-channel&gt; &lt;i-note&gt; &lt;i-velo&gt;</td></tr>"
  138. "<tr><td>" "/note_off" "&nbsp;</td><td>&lt;i-channel&gt; &lt;i-note</td></tr>"
  139. "</table>"
  140. )
  141. self.ui.l_example.setText("/Carla/2/set_parameter_value 5 1.0")
  142. self.ui.l_example_help.setText("<i>(as in this example, \"2\" is the plugin number and \"5\" the parameter)</i>")
  143. # pylint: enable=line-too-long
  144. self.ui.l_ladspa.setText(self.tr("Everything! (Including LRDF)"))
  145. self.ui.l_dssi.setText(self.tr("Everything! (Including CustomData/Chunks)"))
  146. self.ui.l_lv2.setText(self.tr("About 110&#37; complete (using custom extensions)<br/>"
  147. "Implemented Feature/Extensions:"
  148. "<ul>"
  149. "<li>http://lv2plug.in/ns/ext/atom</li>"
  150. "<li>http://lv2plug.in/ns/ext/buf-size</li>"
  151. "<li>http://lv2plug.in/ns/ext/data-access</li>"
  152. #"<li>http://lv2plug.in/ns/ext/dynmanifest</li>"
  153. "<li>http://lv2plug.in/ns/ext/event</li>"
  154. "<li>http://lv2plug.in/ns/ext/instance-access</li>"
  155. "<li>http://lv2plug.in/ns/ext/log</li>"
  156. "<li>http://lv2plug.in/ns/ext/midi</li>"
  157. #"<li>http://lv2plug.in/ns/ext/morph</li>"
  158. "<li>http://lv2plug.in/ns/ext/options</li>"
  159. "<li>http://lv2plug.in/ns/ext/parameters</li>"
  160. #"<li>http://lv2plug.in/ns/ext/patch</li>"
  161. "<li>http://lv2plug.in/ns/ext/port-props</li>"
  162. "<li>http://lv2plug.in/ns/ext/presets</li>"
  163. "<li>http://lv2plug.in/ns/ext/resize-port</li>"
  164. "<li>http://lv2plug.in/ns/ext/state</li>"
  165. "<li>http://lv2plug.in/ns/ext/time</li>"
  166. "<li>http://lv2plug.in/ns/ext/uri-map</li>"
  167. "<li>http://lv2plug.in/ns/ext/urid</li>"
  168. "<li>http://lv2plug.in/ns/ext/worker</li>"
  169. "<li>http://lv2plug.in/ns/extensions/ui</li>"
  170. "<li>http://lv2plug.in/ns/extensions/units</li>"
  171. "<li>http://home.gna.org/lv2dynparam/rtmempool/v1</li>"
  172. "<li>http://kxstudio.sf.net/ns/lv2ext/external-ui</li>"
  173. "<li>http://kxstudio.sf.net/ns/lv2ext/programs</li>"
  174. "<li>http://kxstudio.sf.net/ns/lv2ext/props</li>"
  175. "<li>http://kxstudio.sf.net/ns/lv2ext/rtmempool</li>"
  176. "<li>http://ll-plugins.nongnu.org/lv2/ext/midimap</li>"
  177. "<li>http://ll-plugins.nongnu.org/lv2/ext/miditype</li>"
  178. "</ul>"))
  179. usingJuce = "juce" in gCarla.utils.get_supported_features()
  180. if usingJuce and (MACOS or WINDOWS):
  181. self.ui.l_vst2.setText(self.tr("Using JUCE host"))
  182. self.ui.l_vst3.setText(self.tr("Using JUCE host"))
  183. else:
  184. self.ui.l_vst2.setText(self.tr("About 85&#37; complete (missing vst bank/presets and some minor stuff)"))
  185. self.ui.line_vst2.hide()
  186. self.ui.l_vst3.hide()
  187. self.ui.lid_vst3.hide()
  188. if MACOS:
  189. self.ui.l_au.setText(self.tr("Using JUCE host"))
  190. else:
  191. self.ui.line_vst3.hide()
  192. self.ui.l_au.hide()
  193. self.ui.lid_au.hide()
  194. # 3rd tab is usually longer than the 1st
  195. # adjust appropriately
  196. self.ui.tabWidget.setCurrentIndex(2)
  197. self.adjustSize()
  198. self.ui.tabWidget.setCurrentIndex(0)
  199. self.setFixedSize(self.size())
  200. flags = self.windowFlags()
  201. flags &= ~Qt.WindowContextHelpButtonHint
  202. if WINDOWS:
  203. flags |= Qt.MSWindowsFixedSizeDialogHint
  204. self.setWindowFlags(flags)
  205. # ------------------------------------------------------------------------------------------------------------
  206. # JUCE About dialog
  207. class JuceAboutW(QDialog):
  208. def __init__(self, parent):
  209. QDialog.__init__(self, parent)
  210. self.ui = ui_carla_about_juce.Ui_JuceAboutW()
  211. self.ui.setupUi(self)
  212. self.ui.l_text2.setText(self.tr("This program uses JUCE version %s." % gCarla.utils.get_juce_version()))
  213. self.adjustSize()
  214. self.setFixedSize(self.size())
  215. flags = self.windowFlags()
  216. flags &= ~Qt.WindowContextHelpButtonHint
  217. if WINDOWS:
  218. flags |= Qt.MSWindowsFixedSizeDialogHint
  219. self.setWindowFlags(flags)
  220. # ------------------------------------------------------------------------------------------------------------
  221. # Plugin Parameter
  222. class PluginParameter(QWidget):
  223. mappedControlChanged = pyqtSignal(int, int)
  224. mappedRangeChanged = pyqtSignal(int, float, float)
  225. midiChannelChanged = pyqtSignal(int, int)
  226. valueChanged = pyqtSignal(int, float)
  227. def __init__(self, parent, host, pInfo, pluginId, tabIndex):
  228. QWidget.__init__(self, parent)
  229. self.host = host
  230. self.ui = ui_carla_parameter.Ui_PluginParameter()
  231. self.ui.setupUi(self)
  232. # -------------------------------------------------------------
  233. # Internal stuff
  234. self.fDecimalPoints = max(2, countDecimalPoints(pInfo['step'], pInfo['stepSmall']))
  235. self.fCanBeInCV = pInfo['hints'] & PARAMETER_CAN_BE_CV_CONTROLLED
  236. self.fMappedCtrl = pInfo['mappedControlIndex']
  237. self.fMappedMinimum = pInfo['mappedMinimum']
  238. self.fMappedMaximum = pInfo['mappedMaximum']
  239. self.fMinimum = pInfo['minimum']
  240. self.fMaximum = pInfo['maximum']
  241. self.fMidiChannel = pInfo['midiChannel']
  242. self.fParameterId = pInfo['index']
  243. self.fPluginId = pluginId
  244. self.fTabIndex = tabIndex
  245. # -------------------------------------------------------------
  246. # Set-up GUI
  247. pType = pInfo['type']
  248. pHints = pInfo['hints']
  249. self.ui.l_name.setText(pInfo['name'])
  250. self.ui.widget.setName(pInfo['name'])
  251. self.ui.widget.setMinimum(pInfo['minimum'])
  252. self.ui.widget.setMaximum(pInfo['maximum'])
  253. self.ui.widget.setDefault(pInfo['default'])
  254. self.ui.widget.setLabel(pInfo['unit'])
  255. self.ui.widget.setStep(pInfo['step'])
  256. self.ui.widget.setStepSmall(pInfo['stepSmall'])
  257. self.ui.widget.setStepLarge(pInfo['stepLarge'])
  258. self.ui.widget.setScalePoints(pInfo['scalePoints'], bool(pHints & PARAMETER_USES_SCALEPOINTS))
  259. if pInfo['comment']:
  260. self.ui.l_name.setToolTip(pInfo['comment'])
  261. self.ui.widget.setToolTip(pInfo['comment'])
  262. if pType == PARAMETER_INPUT:
  263. if not pHints & PARAMETER_IS_ENABLED:
  264. self.ui.l_name.setEnabled(False)
  265. self.ui.l_status.setEnabled(False)
  266. self.ui.widget.setEnabled(False)
  267. self.ui.widget.setReadOnly(True)
  268. self.ui.tb_options.setEnabled(False)
  269. elif not pHints & PARAMETER_IS_AUTOMABLE:
  270. self.ui.l_status.setEnabled(False)
  271. self.ui.tb_options.setEnabled(False)
  272. if pHints & PARAMETER_IS_READ_ONLY:
  273. self.ui.l_status.setEnabled(False)
  274. self.ui.widget.setReadOnly(True)
  275. self.ui.tb_options.setEnabled(False)
  276. elif pType == PARAMETER_OUTPUT:
  277. self.ui.widget.setReadOnly(True)
  278. else:
  279. self.ui.l_status.setVisible(False)
  280. self.ui.widget.setVisible(False)
  281. self.ui.tb_options.setVisible(False)
  282. # Only set value after all hints are handled
  283. self.ui.widget.setValue(pInfo['current'])
  284. if pHints & PARAMETER_USES_CUSTOM_TEXT and not host.isPlugin:
  285. self.ui.widget.setTextCallback(self._textCallBack)
  286. self.ui.l_status.setFixedWidth(fontMetricsHorizontalAdvance(self.ui.l_status.fontMetrics(),
  287. self.tr("CC%i Ch%i" % (119,16))))
  288. self.ui.widget.setValueCallback(self._valueCallBack)
  289. self.ui.widget.updateAll()
  290. self.setMappedControlIndex(pInfo['mappedControlIndex'])
  291. self.setMidiChannel(pInfo['midiChannel'])
  292. self.updateStatusLabel()
  293. # -------------------------------------------------------------
  294. # Set-up connections
  295. self.ui.tb_options.clicked.connect(self.slot_optionsCustomMenu)
  296. self.ui.widget.dragStateChanged.connect(self.slot_parameterDragStateChanged)
  297. # -------------------------------------------------------------
  298. def getPluginId(self):
  299. return self.fPluginId
  300. def getTabIndex(self):
  301. return self.fTabIndex
  302. def setPluginId(self, pluginId):
  303. self.fPluginId = pluginId
  304. def setDefault(self, value):
  305. self.ui.widget.setDefault(value)
  306. def setValue(self, value):
  307. self.ui.widget.blockSignals(True)
  308. self.ui.widget.setValue(value)
  309. self.ui.widget.blockSignals(False)
  310. def setMappedControlIndex(self, control):
  311. self.fMappedCtrl = control
  312. self.updateStatusLabel()
  313. def setMappedRange(self, minimum, maximum):
  314. self.fMappedMinimum = minimum
  315. self.fMappedMaximum = maximum
  316. def setMidiChannel(self, channel):
  317. self.fMidiChannel = channel
  318. self.updateStatusLabel()
  319. def setLabelWidth(self, width):
  320. self.ui.l_name.setFixedWidth(width)
  321. def updateStatusLabel(self):
  322. if self.fMappedCtrl == CONTROL_VALUE_NONE:
  323. text = self.tr("Unmapped")
  324. elif self.fMappedCtrl == CONTROL_VALUE_CV:
  325. text = self.tr("CV export")
  326. else:
  327. text = self.tr("CC%i Ch%i" % (self.fMappedCtrl, self.fMidiChannel))
  328. self.ui.l_status.setText(text)
  329. @pyqtSlot()
  330. def slot_optionsCustomMenu(self):
  331. menu = QMenu(self)
  332. if self.fMappedCtrl == CONTROL_VALUE_NONE:
  333. title = self.tr("Unmapped")
  334. elif self.fMappedCtrl == CONTROL_VALUE_CV:
  335. title = self.tr("Exposed as CV port")
  336. else:
  337. title = self.tr("Mapped to MIDI control %i, channel %i" % (self.fMappedCtrl, self.fMidiChannel))
  338. if self.fMappedCtrl != CONTROL_VALUE_NONE:
  339. title += " (range: %g-%g)" % (self.fMappedMinimum, self.fMappedMaximum)
  340. actTitle = menu.addAction(title)
  341. actTitle.setEnabled(False)
  342. menu.addSeparator()
  343. actUnmap = menu.addAction(self.tr("Unmap"))
  344. if self.fMappedCtrl == CONTROL_VALUE_NONE:
  345. actUnmap.setCheckable(True)
  346. actUnmap.setChecked(True)
  347. if self.fCanBeInCV:
  348. menu.addSection("CV")
  349. actCV = menu.addAction(self.tr("Expose as CV port"))
  350. if self.fMappedCtrl == CONTROL_VALUE_CV:
  351. actCV.setCheckable(True)
  352. actCV.setChecked(True)
  353. else:
  354. actCV = None
  355. menu.addSection("MIDI")
  356. menuMIDI = menu.addMenu(self.tr("MIDI Control"))
  357. if self.fMappedCtrl not in (CONTROL_VALUE_NONE, CONTROL_VALUE_CV, CONTROL_VALUE_MIDI_PITCHBEND):
  358. action = menuMIDI.menuAction()
  359. action.setCheckable(True)
  360. action.setChecked(True)
  361. inlist = False
  362. actCCs = []
  363. for cc in MIDI_CC_LIST:
  364. action = menuMIDI.addAction(cc)
  365. actCCs.append(action)
  366. if self.fMappedCtrl >= 0 and self.fMappedCtrl <= MAX_MIDI_CC_LIST_ITEM:
  367. ccx = int(cc.split(" [", 1)[0], 10)
  368. if ccx > self.fMappedCtrl and not inlist:
  369. inlist = True
  370. action = menuMIDI.addAction(self.tr("%02i [0x%02X] (Custom)" % (self.fMappedCtrl,
  371. self.fMappedCtrl)))
  372. action.setCheckable(True)
  373. action.setChecked(True)
  374. actCCs.append(action)
  375. elif ccx == self.fMappedCtrl:
  376. inlist = True
  377. action.setCheckable(True)
  378. action.setChecked(True)
  379. if self.fMappedCtrl > MAX_MIDI_CC_LIST_ITEM and self.fMappedCtrl <= 0x77:
  380. action = menuMIDI.addAction(self.tr("%02i [0x%02X] (Custom)" % (self.fMappedCtrl, self.fMappedCtrl)))
  381. action.setCheckable(True)
  382. action.setChecked(True)
  383. actCCs.append(action)
  384. actCustomCC = menuMIDI.addAction(self.tr("Custom..."))
  385. # TODO
  386. #actPitchbend = menu.addAction(self.tr("MIDI Pitchbend"))
  387. #if self.fMappedCtrl == CONTROL_VALUE_MIDI_PITCHBEND:
  388. #actPitchbend.setCheckable(True)
  389. #actPitchbend.setChecked(True)
  390. menuChannel = menu.addMenu(self.tr("MIDI Channel"))
  391. actChannels = []
  392. for i in range(1, 16+1):
  393. action = menuChannel.addAction("%i" % i)
  394. actChannels.append(action)
  395. if self.fMidiChannel == i:
  396. action.setCheckable(True)
  397. action.setChecked(True)
  398. if self.fMappedCtrl != CONTROL_VALUE_NONE:
  399. if self.fMappedCtrl == CONTROL_VALUE_CV:
  400. menu.addSection("Range (Scaled CV input)")
  401. else:
  402. menu.addSection("Range (MIDI bounds)")
  403. actRangeMinimum = menu.addAction(self.tr("Set minimum... (%g)" % self.fMappedMinimum))
  404. actRangeMaximum = menu.addAction(self.tr("Set maximum... (%g)" % self.fMappedMaximum))
  405. else:
  406. actRangeMinimum = actRangeMaximum = None
  407. actSel = menu.exec_(QCursor.pos())
  408. if not actSel:
  409. return
  410. if actSel in actChannels:
  411. channel = int(actSel.text())
  412. self.fMidiChannel = channel
  413. self.updateStatusLabel()
  414. self.midiChannelChanged.emit(self.fParameterId, channel)
  415. return
  416. if actSel == actRangeMinimum:
  417. value, ok = QInputDialog.getDouble(self,
  418. self.tr("Custom Minimum"),
  419. "Custom minimum value to use:",
  420. self.fMappedMinimum,
  421. self.fMinimum if self.fMappedCtrl != CONTROL_VALUE_CV else -9e6,
  422. self.fMaximum if self.fMappedCtrl != CONTROL_VALUE_CV else 9e6,
  423. self.fDecimalPoints)
  424. if not ok:
  425. return
  426. self.fMappedMinimum = value
  427. self.mappedRangeChanged.emit(self.fParameterId, self.fMappedMinimum, self.fMappedMaximum)
  428. return
  429. if actSel == actRangeMaximum:
  430. value, ok = QInputDialog.getDouble(self,
  431. self.tr("Custom Maximum"),
  432. "Custom maximum value to use:",
  433. self.fMappedMaximum,
  434. self.fMinimum if self.fMappedCtrl != CONTROL_VALUE_CV else -9e6,
  435. self.fMaximum if self.fMappedCtrl != CONTROL_VALUE_CV else 9e6,
  436. self.fDecimalPoints)
  437. if not ok:
  438. return
  439. self.fMappedMaximum = value
  440. self.mappedRangeChanged.emit(self.fParameterId, self.fMappedMinimum, self.fMappedMaximum)
  441. return
  442. if actSel == actUnmap:
  443. ctrl = CONTROL_VALUE_NONE
  444. elif actSel == actCV:
  445. ctrl = CONTROL_VALUE_CV
  446. elif actSel in actCCs:
  447. ctrl = int(actSel.text().split(" ", 1)[0].replace("&",""), 10)
  448. elif actSel == actCustomCC:
  449. value = self.fMappedCtrl if self.fMappedCtrl >= 0x01 and self.fMappedCtrl <= 0x77 else 1
  450. ctrl, ok = QInputDialog.getInt(self,
  451. self.tr("Custom CC"),
  452. "Custom MIDI CC to use:",
  453. value,
  454. 0x01, 0x77, 1)
  455. if not ok:
  456. return
  457. #elif actSel in actPitchbend:
  458. #ctrl = CONTROL_VALUE_MIDI_PITCHBEND
  459. else:
  460. return
  461. self.fMappedCtrl = ctrl
  462. self.updateStatusLabel()
  463. self.mappedControlChanged.emit(self.fParameterId, ctrl)
  464. @pyqtSlot(bool)
  465. def slot_parameterDragStateChanged(self, touch):
  466. self.host.set_parameter_touch(self.fPluginId, self.fParameterId, touch)
  467. def _textCallBack(self):
  468. return self.host.get_parameter_text(self.fPluginId, self.fParameterId)
  469. def _valueCallBack(self, value):
  470. self.valueChanged.emit(self.fParameterId, value)
  471. # ------------------------------------------------------------------------------------------------------------
  472. # Plugin Editor Parent (Meta class)
  473. class PluginEditParentMeta():
  474. #class PluginEditParentMeta(metaclass=ABCMeta):
  475. @abstractmethod
  476. def editDialogVisibilityChanged(self, pluginId, visible):
  477. raise NotImplementedError
  478. @abstractmethod
  479. def editDialogPluginHintsChanged(self, pluginId, hints):
  480. raise NotImplementedError
  481. @abstractmethod
  482. def editDialogParameterValueChanged(self, pluginId, parameterId, value):
  483. raise NotImplementedError
  484. @abstractmethod
  485. def editDialogProgramChanged(self, pluginId, index):
  486. raise NotImplementedError
  487. @abstractmethod
  488. def editDialogMidiProgramChanged(self, pluginId, index):
  489. raise NotImplementedError
  490. @abstractmethod
  491. def editDialogNotePressed(self, pluginId, note):
  492. raise NotImplementedError
  493. @abstractmethod
  494. def editDialogNoteReleased(self, pluginId, note):
  495. raise NotImplementedError
  496. @abstractmethod
  497. def editDialogMidiActivityChanged(self, pluginId, onOff):
  498. raise NotImplementedError
  499. # ------------------------------------------------------------------------------------------------------------
  500. # Plugin Editor (Built-in)
  501. class PluginEdit(QDialog):
  502. # signals
  503. SIGTERM = pyqtSignal()
  504. SIGUSR1 = pyqtSignal()
  505. def __init__(self, parent, host, pluginId):
  506. QDialog.__init__(self, parent.window() if parent is not None else None)
  507. self.host = host
  508. self.ui = ui_carla_edit.Ui_PluginEdit()
  509. self.ui.setupUi(self)
  510. # -------------------------------------------------------------
  511. # Internal stuff
  512. self.fGeometry = QByteArray()
  513. self.fParent = parent
  514. self.fPluginId = pluginId
  515. self.fPluginInfo = None
  516. self.fCurrentStateFilename = None
  517. self.fControlChannel = round(host.get_internal_parameter_value(pluginId, PARAMETER_CTRL_CHANNEL))
  518. self.fFirstInit = True
  519. self.fParameterList = [] # (type, id, widget)
  520. self.fParametersToUpdate = [] # (id, value)
  521. self.fPlayingNotes = [] # (channel, note)
  522. self.fTabIconOff = QIcon(":/scalable/led_off.svg")
  523. self.fTabIconOn = QIcon(":/scalable/led_yellow.svg")
  524. self.fTabIconTimers = []
  525. # used during testing
  526. self.fIdleTimerId = 0
  527. # -------------------------------------------------------------
  528. # Set-up GUI
  529. labelPluginFont = self.ui.label_plugin.font()
  530. labelPluginFont.setPixelSize(15)
  531. labelPluginFont.setWeight(75)
  532. self.ui.label_plugin.setFont(labelPluginFont)
  533. self.ui.dial_drywet.setCustomPaintMode(self.ui.dial_drywet.CUSTOM_PAINT_MODE_CARLA_WET)
  534. self.ui.dial_drywet.setImage(3)
  535. self.ui.dial_drywet.setLabel("Dry/Wet")
  536. self.ui.dial_drywet.setMinimum(0.0)
  537. self.ui.dial_drywet.setMaximum(1.0)
  538. self.ui.dial_drywet.setValue(host.get_internal_parameter_value(pluginId, PARAMETER_DRYWET))
  539. self.ui.dial_vol.setCustomPaintMode(self.ui.dial_vol.CUSTOM_PAINT_MODE_CARLA_VOL)
  540. self.ui.dial_vol.setImage(3)
  541. self.ui.dial_vol.setLabel("Volume")
  542. self.ui.dial_vol.setMinimum(0.0)
  543. self.ui.dial_vol.setMaximum(1.27)
  544. self.ui.dial_vol.setValue(host.get_internal_parameter_value(pluginId, PARAMETER_VOLUME))
  545. self.ui.dial_b_left.setCustomPaintMode(self.ui.dial_b_left.CUSTOM_PAINT_MODE_CARLA_L)
  546. self.ui.dial_b_left.setImage(4)
  547. self.ui.dial_b_left.setLabel("L")
  548. self.ui.dial_b_left.setMinimum(-1.0)
  549. self.ui.dial_b_left.setMaximum(1.0)
  550. self.ui.dial_b_left.setValue(host.get_internal_parameter_value(pluginId, PARAMETER_BALANCE_LEFT))
  551. self.ui.dial_b_right.setCustomPaintMode(self.ui.dial_b_right.CUSTOM_PAINT_MODE_CARLA_R)
  552. self.ui.dial_b_right.setImage(4)
  553. self.ui.dial_b_right.setLabel("R")
  554. self.ui.dial_b_right.setMinimum(-1.0)
  555. self.ui.dial_b_right.setMaximum(1.0)
  556. self.ui.dial_b_right.setValue(host.get_internal_parameter_value(pluginId, PARAMETER_BALANCE_RIGHT))
  557. self.ui.dial_pan.setCustomPaintMode(self.ui.dial_b_right.CUSTOM_PAINT_MODE_CARLA_PAN)
  558. self.ui.dial_pan.setImage(4)
  559. self.ui.dial_pan.setLabel("Pan")
  560. self.ui.dial_pan.setMinimum(-1.0)
  561. self.ui.dial_pan.setMaximum(1.0)
  562. self.ui.dial_pan.setValue(host.get_internal_parameter_value(pluginId, PARAMETER_PANNING))
  563. self.ui.sb_ctrl_channel.setValue(self.fControlChannel+1)
  564. self.ui.scrollArea = PixmapKeyboardHArea(self)
  565. self.ui.keyboard = self.ui.scrollArea.keyboard
  566. self.ui.keyboard.setEnabled(self.fControlChannel >= 0)
  567. self.layout().addWidget(self.ui.scrollArea)
  568. self.ui.scrollArea.setEnabled(False)
  569. self.ui.scrollArea.setVisible(False)
  570. # todo
  571. self.ui.rb_balance.setEnabled(False)
  572. self.ui.rb_balance.setVisible(False)
  573. self.ui.rb_pan.setEnabled(False)
  574. self.ui.rb_pan.setVisible(False)
  575. self.setWindowFlags(self.windowFlags() & ~Qt.WindowContextHelpButtonHint)
  576. self.reloadAll()
  577. self.fFirstInit = False
  578. # -------------------------------------------------------------
  579. # Set-up connections
  580. self.finished.connect(self.slot_finished)
  581. self.ui.ch_fixed_buffer.clicked.connect(self.slot_optionChanged)
  582. self.ui.ch_force_stereo.clicked.connect(self.slot_optionChanged)
  583. self.ui.ch_map_program_changes.clicked.connect(self.slot_optionChanged)
  584. self.ui.ch_use_chunks.clicked.connect(self.slot_optionChanged)
  585. self.ui.ch_send_program_changes.clicked.connect(self.slot_optionChanged)
  586. self.ui.ch_send_control_changes.clicked.connect(self.slot_optionChanged)
  587. self.ui.ch_send_channel_pressure.clicked.connect(self.slot_optionChanged)
  588. self.ui.ch_send_note_aftertouch.clicked.connect(self.slot_optionChanged)
  589. self.ui.ch_send_pitchbend.clicked.connect(self.slot_optionChanged)
  590. self.ui.ch_send_all_sound_off.clicked.connect(self.slot_optionChanged)
  591. self.ui.dial_drywet.realValueChanged.connect(self.slot_dryWetChanged)
  592. self.ui.dial_vol.realValueChanged.connect(self.slot_volumeChanged)
  593. self.ui.dial_b_left.realValueChanged.connect(self.slot_balanceLeftChanged)
  594. self.ui.dial_b_right.realValueChanged.connect(self.slot_balanceRightChanged)
  595. self.ui.dial_pan.realValueChanged.connect(self.slot_panChanged)
  596. self.ui.sb_ctrl_channel.valueChanged.connect(self.slot_ctrlChannelChanged)
  597. self.ui.dial_drywet.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  598. self.ui.dial_vol.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  599. self.ui.dial_b_left.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  600. self.ui.dial_b_right.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  601. self.ui.dial_pan.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  602. self.ui.sb_ctrl_channel.customContextMenuRequested.connect(self.slot_channelCustomMenu)
  603. self.ui.keyboard.noteOn.connect(self.slot_noteOn)
  604. self.ui.keyboard.noteOff.connect(self.slot_noteOff)
  605. self.ui.cb_programs.currentIndexChanged.connect(self.slot_programIndexChanged)
  606. self.ui.cb_midi_programs.currentIndexChanged.connect(self.slot_midiProgramIndexChanged)
  607. self.ui.b_save_state.clicked.connect(self.slot_stateSave)
  608. self.ui.b_load_state.clicked.connect(self.slot_stateLoad)
  609. host.NoteOnCallback.connect(self.slot_handleNoteOnCallback)
  610. host.NoteOffCallback.connect(self.slot_handleNoteOffCallback)
  611. host.UpdateCallback.connect(self.slot_handleUpdateCallback)
  612. host.ReloadInfoCallback.connect(self.slot_handleReloadInfoCallback)
  613. host.ReloadParametersCallback.connect(self.slot_handleReloadParametersCallback)
  614. host.ReloadProgramsCallback.connect(self.slot_handleReloadProgramsCallback)
  615. host.ReloadAllCallback.connect(self.slot_handleReloadAllCallback)
  616. #------------------------------------------------------------------
  617. @pyqtSlot(int, int, int, int)
  618. def slot_handleNoteOnCallback(self, pluginId, channel, note, velocity):
  619. if self.fPluginId != pluginId:
  620. return
  621. if self.fControlChannel == channel:
  622. self.ui.keyboard.sendNoteOn(note, False)
  623. playItem = (channel, note)
  624. if playItem not in self.fPlayingNotes:
  625. self.fPlayingNotes.append(playItem)
  626. if len(self.fPlayingNotes) == 1 and self.fParent is not None:
  627. self.fParent.editDialogMidiActivityChanged(self.fPluginId, True)
  628. @pyqtSlot(int, int, int)
  629. def slot_handleNoteOffCallback(self, pluginId, channel, note):
  630. if self.fPluginId != pluginId:
  631. return
  632. if self.fControlChannel == channel:
  633. self.ui.keyboard.sendNoteOff(note, False)
  634. playItem = (channel, note)
  635. if playItem in self.fPlayingNotes:
  636. self.fPlayingNotes.remove(playItem)
  637. if self.fPlayingNotes and self.fParent is not None:
  638. self.fParent.editDialogMidiActivityChanged(self.fPluginId, False)
  639. @pyqtSlot(int)
  640. def slot_handleUpdateCallback(self, pluginId):
  641. if self.fPluginId == pluginId:
  642. self.updateInfo()
  643. @pyqtSlot(int)
  644. def slot_handleReloadInfoCallback(self, pluginId):
  645. if self.fPluginId == pluginId:
  646. self.reloadInfo()
  647. @pyqtSlot(int)
  648. def slot_handleReloadParametersCallback(self, pluginId):
  649. if self.fPluginId == pluginId:
  650. self.reloadParameters()
  651. @pyqtSlot(int)
  652. def slot_handleReloadProgramsCallback(self, pluginId):
  653. if self.fPluginId == pluginId:
  654. self.reloadPrograms()
  655. @pyqtSlot(int)
  656. def slot_handleReloadAllCallback(self, pluginId):
  657. if self.fPluginId == pluginId:
  658. self.reloadAll()
  659. #------------------------------------------------------------------
  660. def updateInfo(self):
  661. # Update current program text
  662. if self.ui.cb_programs.count() > 0:
  663. pIndex = self.ui.cb_programs.currentIndex()
  664. if pIndex >= 0:
  665. pName = self.host.get_program_name(self.fPluginId, pIndex)
  666. #pName = pName[:40] + (pName[40:] and "...")
  667. self.ui.cb_programs.setItemText(pIndex, pName)
  668. # Update current midi program text
  669. if self.ui.cb_midi_programs.count() > 0:
  670. mpIndex = self.ui.cb_midi_programs.currentIndex()
  671. if mpIndex >= 0:
  672. mpData = self.host.get_midi_program_data(self.fPluginId, mpIndex)
  673. mpBank = mpData['bank']
  674. mpProg = mpData['program']
  675. mpName = mpData['name']
  676. #mpName = mpName[:40] + (mpName[40:] and "...")
  677. self.ui.cb_midi_programs.setItemText(mpIndex, "%03i:%03i - %s" % (mpBank+1, mpProg+1, mpName))
  678. # Update all parameter values
  679. for _, paramId, paramWidget in self.fParameterList:
  680. paramWidget.blockSignals(True)
  681. paramWidget.setValue(self.host.get_current_parameter_value(self.fPluginId, paramId))
  682. paramWidget.blockSignals(False)
  683. # and the internal ones too
  684. self.ui.dial_drywet.blockSignals(True)
  685. self.ui.dial_drywet.setValue(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_DRYWET))
  686. self.ui.dial_drywet.blockSignals(False)
  687. self.ui.dial_vol.blockSignals(True)
  688. self.ui.dial_vol.setValue(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_VOLUME))
  689. self.ui.dial_vol.blockSignals(False)
  690. self.ui.dial_b_left.blockSignals(True)
  691. self.ui.dial_b_left.setValue(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_BALANCE_LEFT))
  692. self.ui.dial_b_left.blockSignals(False)
  693. self.ui.dial_b_right.blockSignals(True)
  694. self.ui.dial_b_right.setValue(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_BALANCE_RIGHT))
  695. self.ui.dial_b_right.blockSignals(False)
  696. self.ui.dial_pan.blockSignals(True)
  697. self.ui.dial_pan.setValue(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_PANNING))
  698. self.ui.dial_pan.blockSignals(False)
  699. self.fControlChannel = round(self.host.get_internal_parameter_value(self.fPluginId, PARAMETER_CTRL_CHANNEL))
  700. self.ui.sb_ctrl_channel.blockSignals(True)
  701. self.ui.sb_ctrl_channel.setValue(self.fControlChannel+1)
  702. self.ui.sb_ctrl_channel.blockSignals(False)
  703. self.ui.keyboard.allNotesOff()
  704. self._updateCtrlPrograms()
  705. self.fParametersToUpdate = []
  706. #------------------------------------------------------------------
  707. def reloadAll(self):
  708. self.fPluginInfo = self.host.get_plugin_info(self.fPluginId)
  709. self.reloadInfo()
  710. self.reloadParameters()
  711. self.reloadPrograms()
  712. if self.fPluginInfo['type'] == PLUGIN_LV2:
  713. self.ui.b_save_state.setEnabled(False)
  714. if not self.ui.scrollArea.isEnabled():
  715. self.resize(self.width(), self.height()-self.ui.scrollArea.height())
  716. #------------------------------------------------------------------
  717. def reloadInfo(self):
  718. realPluginName = self.host.get_real_plugin_name(self.fPluginId)
  719. #audioCountInfo = self.host.get_audio_port_count_info(self.fPluginId)
  720. midiCountInfo = self.host.get_midi_port_count_info(self.fPluginId)
  721. #paramCountInfo = self.host.get_parameter_count_info(self.fPluginId)
  722. pluginHints = self.fPluginInfo['hints']
  723. self.ui.le_type.setText(getPluginTypeAsString(self.fPluginInfo['type']))
  724. self.ui.label_name.setEnabled(bool(realPluginName))
  725. self.ui.le_name.setEnabled(bool(realPluginName))
  726. self.ui.le_name.setText(realPluginName)
  727. self.ui.le_name.setToolTip(realPluginName)
  728. self.ui.label_label.setEnabled(bool(self.fPluginInfo['label']))
  729. self.ui.le_label.setEnabled(bool(self.fPluginInfo['label']))
  730. self.ui.le_label.setText(self.fPluginInfo['label'])
  731. self.ui.le_label.setToolTip(self.fPluginInfo['label'])
  732. self.ui.label_maker.setEnabled(bool(self.fPluginInfo['maker']))
  733. self.ui.le_maker.setEnabled(bool(self.fPluginInfo['maker']))
  734. self.ui.le_maker.setText(self.fPluginInfo['maker'])
  735. self.ui.le_maker.setToolTip(self.fPluginInfo['maker'])
  736. self.ui.label_copyright.setEnabled(bool(self.fPluginInfo['copyright']))
  737. self.ui.le_copyright.setEnabled(bool(self.fPluginInfo['copyright']))
  738. self.ui.le_copyright.setText(self.fPluginInfo['copyright'])
  739. self.ui.le_copyright.setToolTip(self.fPluginInfo['copyright'])
  740. self.ui.label_unique_id.setEnabled(bool(self.fPluginInfo['uniqueId']))
  741. self.ui.le_unique_id.setEnabled(bool(self.fPluginInfo['uniqueId']))
  742. self.ui.le_unique_id.setText(str(self.fPluginInfo['uniqueId']))
  743. self.ui.le_unique_id.setToolTip(str(self.fPluginInfo['uniqueId']))
  744. self.ui.label_plugin.setText("\n%s\n" % (self.fPluginInfo['name'] or "(none)"))
  745. self.setWindowTitle(self.fPluginInfo['name'] or "(none)")
  746. self.ui.dial_drywet.setEnabled(pluginHints & PLUGIN_CAN_DRYWET)
  747. self.ui.dial_vol.setEnabled(pluginHints & PLUGIN_CAN_VOLUME)
  748. self.ui.dial_b_left.setEnabled(pluginHints & PLUGIN_CAN_BALANCE)
  749. self.ui.dial_b_right.setEnabled(pluginHints & PLUGIN_CAN_BALANCE)
  750. self.ui.dial_pan.setEnabled(pluginHints & PLUGIN_CAN_PANNING)
  751. optsAvailable = self.fPluginInfo['optionsAvailable']
  752. optsEnabled = self.fPluginInfo['optionsEnabled']
  753. self.ui.ch_use_chunks.setEnabled(optsAvailable & PLUGIN_OPTION_USE_CHUNKS)
  754. self.ui.ch_use_chunks.setChecked(optsEnabled & PLUGIN_OPTION_USE_CHUNKS)
  755. self.ui.ch_fixed_buffer.setEnabled(optsAvailable & PLUGIN_OPTION_FIXED_BUFFERS)
  756. self.ui.ch_fixed_buffer.setChecked(optsEnabled & PLUGIN_OPTION_FIXED_BUFFERS)
  757. self.ui.ch_force_stereo.setEnabled(optsAvailable & PLUGIN_OPTION_FORCE_STEREO)
  758. self.ui.ch_force_stereo.setChecked(optsEnabled & PLUGIN_OPTION_FORCE_STEREO)
  759. self.ui.ch_map_program_changes.setEnabled(optsAvailable & PLUGIN_OPTION_MAP_PROGRAM_CHANGES)
  760. self.ui.ch_map_program_changes.setChecked(optsEnabled & PLUGIN_OPTION_MAP_PROGRAM_CHANGES)
  761. self.ui.ch_send_control_changes.setEnabled(optsAvailable & PLUGIN_OPTION_SEND_CONTROL_CHANGES)
  762. self.ui.ch_send_control_changes.setChecked(optsEnabled & PLUGIN_OPTION_SEND_CONTROL_CHANGES)
  763. self.ui.ch_send_channel_pressure.setEnabled(optsAvailable & PLUGIN_OPTION_SEND_CHANNEL_PRESSURE)
  764. self.ui.ch_send_channel_pressure.setChecked(optsEnabled & PLUGIN_OPTION_SEND_CHANNEL_PRESSURE)
  765. self.ui.ch_send_note_aftertouch.setEnabled(optsAvailable & PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH)
  766. self.ui.ch_send_note_aftertouch.setChecked(optsEnabled & PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH)
  767. self.ui.ch_send_pitchbend.setEnabled(optsAvailable & PLUGIN_OPTION_SEND_PITCHBEND)
  768. self.ui.ch_send_pitchbend.setChecked(optsEnabled & PLUGIN_OPTION_SEND_PITCHBEND)
  769. self.ui.ch_send_all_sound_off.setEnabled(optsAvailable & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  770. self.ui.ch_send_all_sound_off.setChecked(optsEnabled & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  771. canSendPrograms = bool((optsAvailable & PLUGIN_OPTION_SEND_PROGRAM_CHANGES) != 0 and
  772. (optsEnabled & PLUGIN_OPTION_MAP_PROGRAM_CHANGES) == 0)
  773. self.ui.ch_send_program_changes.setEnabled(canSendPrograms)
  774. self.ui.ch_send_program_changes.setChecked(optsEnabled & PLUGIN_OPTION_SEND_PROGRAM_CHANGES)
  775. self.ui.sw_programs.setCurrentIndex(0 if self.fPluginInfo['type'] in (PLUGIN_VST2, PLUGIN_SFZ) else 1)
  776. # Show/hide keyboard
  777. showKeyboard = (self.fPluginInfo['category'] == PLUGIN_CATEGORY_SYNTH or midiCountInfo['ins'] > 0)
  778. self.ui.scrollArea.setEnabled(showKeyboard)
  779. self.ui.scrollArea.setVisible(showKeyboard)
  780. # Force-update parent for new hints
  781. if self.fParent is not None and not self.fFirstInit:
  782. self.fParent.editDialogPluginHintsChanged(self.fPluginId, pluginHints)
  783. def reloadParameters(self):
  784. # Reset
  785. self.fParameterList = []
  786. self.fParametersToUpdate = []
  787. self.fTabIconTimers = []
  788. # Save current tab state
  789. tabIndex = self.ui.tabWidget.currentIndex()
  790. tabWidget = self.ui.tabWidget.currentWidget()
  791. scrollVal = tabWidget.verticalScrollBar().value() if isinstance(tabWidget, QScrollArea) else None
  792. del tabWidget
  793. # Remove all previous parameters
  794. for _ in range(self.ui.tabWidget.count()-1):
  795. self.ui.tabWidget.widget(1).deleteLater()
  796. self.ui.tabWidget.removeTab(1)
  797. parameterCount = self.host.get_parameter_count(self.fPluginId)
  798. # -----------------------------------------------------------------
  799. if parameterCount <= 0:
  800. return
  801. # -----------------------------------------------------------------
  802. paramInputList = []
  803. paramOutputList = []
  804. paramInputWidth = 0
  805. paramOutputWidth = 0
  806. paramInputListFull = [] # ([params], width)
  807. paramOutputListFull = [] # ([params], width)
  808. for i in range(min(parameterCount, self.host.maxParameters)):
  809. paramInfo = self.host.get_parameter_info(self.fPluginId, i)
  810. paramData = self.host.get_parameter_data(self.fPluginId, i)
  811. paramRanges = self.host.get_parameter_ranges(self.fPluginId, i)
  812. paramValue = self.host.get_current_parameter_value(self.fPluginId, i)
  813. if paramData['type'] not in (PARAMETER_INPUT, PARAMETER_OUTPUT):
  814. continue
  815. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  816. continue
  817. parameter = {
  818. 'type': paramData['type'],
  819. 'hints': paramData['hints'],
  820. 'name': paramInfo['name'],
  821. 'unit': paramInfo['unit'],
  822. 'scalePoints': [],
  823. 'index': paramData['index'],
  824. 'default': paramRanges['def'],
  825. 'minimum': paramRanges['min'],
  826. 'maximum': paramRanges['max'],
  827. 'step': paramRanges['step'],
  828. 'stepSmall': paramRanges['stepSmall'],
  829. 'stepLarge': paramRanges['stepLarge'],
  830. 'mappedControlIndex': paramData['mappedControlIndex'],
  831. 'mappedMinimum': paramData['mappedMinimum'],
  832. 'mappedMaximum': paramData['mappedMaximum'],
  833. 'midiChannel': paramData['midiChannel']+1,
  834. 'comment': paramInfo['comment'],
  835. 'groupName': paramInfo['groupName'],
  836. 'current': paramValue
  837. }
  838. for j in range(paramInfo['scalePointCount']):
  839. scalePointInfo = self.host.get_parameter_scalepoint_info(self.fPluginId, i, j)
  840. parameter['scalePoints'].append({
  841. 'value': scalePointInfo['value'],
  842. 'label': scalePointInfo['label']
  843. })
  844. #parameter['name'] = parameter['name'][:30] + (parameter['name'][30:] and "...")
  845. # -----------------------------------------------------------------
  846. # Get width values, in packs of 20
  847. if parameter['type'] == PARAMETER_INPUT:
  848. paramInputWidthTMP = fontMetricsHorizontalAdvance(self.fontMetrics(), parameter['name'])
  849. if paramInputWidthTMP > paramInputWidth:
  850. paramInputWidth = paramInputWidthTMP
  851. paramInputList.append(parameter)
  852. else:
  853. paramOutputWidthTMP = fontMetricsHorizontalAdvance(self.fontMetrics(), parameter['name'])
  854. if paramOutputWidthTMP > paramOutputWidth:
  855. paramOutputWidth = paramOutputWidthTMP
  856. paramOutputList.append(parameter)
  857. paramInputListFull.append((paramInputList, paramInputWidth))
  858. paramOutputListFull.append((paramOutputList, paramOutputWidth))
  859. # Create parameter tabs + widgets
  860. self._createParameterWidgets(PARAMETER_INPUT, paramInputListFull, self.tr("Parameters"))
  861. self._createParameterWidgets(PARAMETER_OUTPUT, paramOutputListFull, self.tr("Outputs"))
  862. # Restore tab state
  863. if tabIndex < self.ui.tabWidget.count():
  864. self.ui.tabWidget.setCurrentIndex(tabIndex)
  865. if scrollVal is not None:
  866. self.ui.tabWidget.currentWidget().verticalScrollBar().setValue(scrollVal)
  867. def reloadPrograms(self):
  868. # Programs
  869. self.ui.cb_programs.blockSignals(True)
  870. self.ui.cb_programs.clear()
  871. programCount = self.host.get_program_count(self.fPluginId)
  872. if programCount > 0:
  873. self.ui.cb_programs.setEnabled(True)
  874. self.ui.label_programs.setEnabled(True)
  875. for i in range(programCount):
  876. pName = self.host.get_program_name(self.fPluginId, i)
  877. #pName = pName[:40] + (pName[40:] and "...")
  878. self.ui.cb_programs.addItem(pName)
  879. self.ui.cb_programs.setCurrentIndex(self.host.get_current_program_index(self.fPluginId))
  880. else:
  881. self.ui.cb_programs.setEnabled(False)
  882. self.ui.label_programs.setEnabled(False)
  883. self.ui.cb_programs.blockSignals(False)
  884. # MIDI Programs
  885. self.ui.cb_midi_programs.blockSignals(True)
  886. self.ui.cb_midi_programs.clear()
  887. midiProgramCount = self.host.get_midi_program_count(self.fPluginId)
  888. if midiProgramCount > 0:
  889. self.ui.cb_midi_programs.setEnabled(True)
  890. self.ui.label_midi_programs.setEnabled(True)
  891. for i in range(midiProgramCount):
  892. mpData = self.host.get_midi_program_data(self.fPluginId, i)
  893. mpBank = mpData['bank']
  894. mpProg = mpData['program']
  895. mpName = mpData['name']
  896. #mpName = mpName[:40] + (mpName[40:] and "...")
  897. self.ui.cb_midi_programs.addItem("%03i:%03i - %s" % (mpBank+1, mpProg+1, mpName))
  898. self.ui.cb_midi_programs.setCurrentIndex(self.host.get_current_midi_program_index(self.fPluginId))
  899. else:
  900. self.ui.cb_midi_programs.setEnabled(False)
  901. self.ui.label_midi_programs.setEnabled(False)
  902. self.ui.cb_midi_programs.blockSignals(False)
  903. self.ui.sw_programs.setEnabled(programCount > 0 or midiProgramCount > 0)
  904. if self.fPluginInfo['type'] == PLUGIN_LV2:
  905. self.ui.b_load_state.setEnabled(programCount > 0)
  906. #------------------------------------------------------------------
  907. def clearNotes(self):
  908. self.fPlayingNotes = []
  909. self.ui.keyboard.allNotesOff()
  910. def noteOn(self, channel, note, velocity):
  911. if self.fControlChannel == channel:
  912. self.ui.keyboard.sendNoteOn(note, False)
  913. def noteOff(self, channel, note):
  914. if self.fControlChannel == channel:
  915. self.ui.keyboard.sendNoteOff(note, False)
  916. #------------------------------------------------------------------
  917. def getHints(self):
  918. return self.fPluginInfo['hints']
  919. def setPluginId(self, idx):
  920. self.fPluginId = idx
  921. def setName(self, name):
  922. self.fPluginInfo['name'] = name
  923. self.ui.label_plugin.setText("\n%s\n" % name)
  924. self.setWindowTitle(name)
  925. #------------------------------------------------------------------
  926. def setParameterValue(self, parameterId, value):
  927. for paramItem in self.fParametersToUpdate:
  928. if paramItem[0] == parameterId:
  929. paramItem[1] = value
  930. break
  931. else:
  932. self.fParametersToUpdate.append([parameterId, value])
  933. def setParameterDefault(self, parameterId, value):
  934. for _, paramId, paramWidget in self.fParameterList:
  935. if paramId == parameterId:
  936. paramWidget.setDefault(value)
  937. break
  938. def setParameterMappedControlIndex(self, parameterId, control):
  939. for _, paramId, paramWidget in self.fParameterList:
  940. if paramId == parameterId:
  941. paramWidget.setMappedControlIndex(control)
  942. break
  943. def setParameterMappedRange(self, parameterId, minimum, maximum):
  944. for _, paramId, paramWidget in self.fParameterList:
  945. if paramId == parameterId:
  946. paramWidget.setMappedRange(minimum, maximum)
  947. break
  948. def setParameterMidiChannel(self, parameterId, channel):
  949. for _, paramId, paramWidget in self.fParameterList:
  950. if paramId == parameterId:
  951. paramWidget.setMidiChannel(channel+1)
  952. break
  953. def setProgram(self, index):
  954. self.ui.cb_programs.blockSignals(True)
  955. self.ui.cb_programs.setCurrentIndex(index)
  956. self.ui.cb_programs.blockSignals(False)
  957. self._updateParameterValues()
  958. def setMidiProgram(self, index):
  959. self.ui.cb_midi_programs.blockSignals(True)
  960. self.ui.cb_midi_programs.setCurrentIndex(index)
  961. self.ui.cb_midi_programs.blockSignals(False)
  962. self._updateParameterValues()
  963. def setOption(self, option, yesNo):
  964. if option == PLUGIN_OPTION_USE_CHUNKS:
  965. widget = self.ui.ch_use_chunks
  966. elif option == PLUGIN_OPTION_FIXED_BUFFERS:
  967. widget = self.ui.ch_fixed_buffer
  968. elif option == PLUGIN_OPTION_FORCE_STEREO:
  969. widget = self.ui.ch_force_stereo
  970. elif option == PLUGIN_OPTION_MAP_PROGRAM_CHANGES:
  971. widget = self.ui.ch_map_program_changes
  972. elif option == PLUGIN_OPTION_SEND_PROGRAM_CHANGES:
  973. widget = self.ui.ch_send_program_changes
  974. elif option == PLUGIN_OPTION_SEND_CONTROL_CHANGES:
  975. widget = self.ui.ch_send_control_changes
  976. elif option == PLUGIN_OPTION_SEND_CHANNEL_PRESSURE:
  977. widget = self.ui.ch_send_channel_pressure
  978. elif option == PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH:
  979. widget = self.ui.ch_send_note_aftertouch
  980. elif option == PLUGIN_OPTION_SEND_PITCHBEND:
  981. widget = self.ui.ch_send_pitchbend
  982. elif option == PLUGIN_OPTION_SEND_ALL_SOUND_OFF:
  983. widget = self.ui.ch_send_all_sound_off
  984. else:
  985. return
  986. widget.blockSignals(True)
  987. widget.setChecked(yesNo)
  988. widget.blockSignals(False)
  989. #------------------------------------------------------------------
  990. def setVisible(self, yesNo):
  991. if yesNo:
  992. if not self.fGeometry.isNull():
  993. self.restoreGeometry(self.fGeometry)
  994. else:
  995. self.fGeometry = self.saveGeometry()
  996. QDialog.setVisible(self, yesNo)
  997. #------------------------------------------------------------------
  998. def idleSlow(self):
  999. # Check Tab icons
  1000. for i in range(len(self.fTabIconTimers)):
  1001. if self.fTabIconTimers[i] == ICON_STATE_ON:
  1002. self.fTabIconTimers[i] = ICON_STATE_WAIT
  1003. elif self.fTabIconTimers[i] == ICON_STATE_WAIT:
  1004. self.fTabIconTimers[i] = ICON_STATE_OFF
  1005. elif self.fTabIconTimers[i] == ICON_STATE_OFF:
  1006. self.fTabIconTimers[i] = ICON_STATE_NULL
  1007. self.ui.tabWidget.setTabIcon(i+1, self.fTabIconOff)
  1008. # Check parameters needing update
  1009. for index, value in self.fParametersToUpdate:
  1010. if index == PARAMETER_DRYWET:
  1011. self.ui.dial_drywet.blockSignals(True)
  1012. self.ui.dial_drywet.setValue(value)
  1013. self.ui.dial_drywet.blockSignals(False)
  1014. elif index == PARAMETER_VOLUME:
  1015. self.ui.dial_vol.blockSignals(True)
  1016. self.ui.dial_vol.setValue(value)
  1017. self.ui.dial_vol.blockSignals(False)
  1018. elif index == PARAMETER_BALANCE_LEFT:
  1019. self.ui.dial_b_left.blockSignals(True)
  1020. self.ui.dial_b_left.setValue(value)
  1021. self.ui.dial_b_left.blockSignals(False)
  1022. elif index == PARAMETER_BALANCE_RIGHT:
  1023. self.ui.dial_b_right.blockSignals(True)
  1024. self.ui.dial_b_right.setValue(value)
  1025. self.ui.dial_b_right.blockSignals(False)
  1026. elif index == PARAMETER_PANNING:
  1027. self.ui.dial_pan.blockSignals(True)
  1028. self.ui.dial_pan.setValue(value)
  1029. self.ui.dial_pan.blockSignals(False)
  1030. elif index == PARAMETER_CTRL_CHANNEL:
  1031. self.fControlChannel = round(value)
  1032. self.ui.sb_ctrl_channel.blockSignals(True)
  1033. self.ui.sb_ctrl_channel.setValue(self.fControlChannel+1)
  1034. self.ui.sb_ctrl_channel.blockSignals(False)
  1035. self.ui.keyboard.allNotesOff()
  1036. self._updateCtrlPrograms()
  1037. elif index >= 0:
  1038. for paramType, paramId, paramWidget in self.fParameterList:
  1039. if paramId != index:
  1040. continue
  1041. # FIXME see below
  1042. if paramType != PARAMETER_INPUT:
  1043. continue
  1044. paramWidget.blockSignals(True)
  1045. paramWidget.setValue(value)
  1046. paramWidget.blockSignals(False)
  1047. #if paramType == PARAMETER_INPUT:
  1048. tabIndex = paramWidget.getTabIndex()
  1049. if self.fTabIconTimers[tabIndex-1] == ICON_STATE_NULL:
  1050. self.ui.tabWidget.setTabIcon(tabIndex, self.fTabIconOn)
  1051. self.fTabIconTimers[tabIndex-1] = ICON_STATE_ON
  1052. break
  1053. # Clear all parameters
  1054. self.fParametersToUpdate = []
  1055. # Update parameter outputs | FIXME needed?
  1056. for paramType, paramId, paramWidget in self.fParameterList:
  1057. if paramType != PARAMETER_OUTPUT:
  1058. continue
  1059. paramWidget.blockSignals(True)
  1060. paramWidget.setValue(self.host.get_current_parameter_value(self.fPluginId, paramId))
  1061. paramWidget.blockSignals(False)
  1062. #------------------------------------------------------------------
  1063. @pyqtSlot()
  1064. def slot_stateSave(self):
  1065. if self.fPluginInfo['type'] == PLUGIN_LV2:
  1066. # TODO
  1067. return
  1068. if self.fCurrentStateFilename:
  1069. askTry = QMessageBox.question(self,
  1070. self.tr("Overwrite?"),
  1071. self.tr("Overwrite previously created file?"),
  1072. QMessageBox.Ok|QMessageBox.Cancel)
  1073. if askTry == QMessageBox.Ok:
  1074. self.host.save_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  1075. return
  1076. self.fCurrentStateFilename = None
  1077. fileFilter = self.tr("Carla State File (*.carxs)")
  1078. filename, _ = QFileDialog.getSaveFileName(self, self.tr("Save Plugin State File"), filter=fileFilter)
  1079. # FIXME use ok value, test if it works as expected
  1080. if not filename:
  1081. return
  1082. if not filename.lower().endswith(".carxs"):
  1083. filename += ".carxs"
  1084. self.fCurrentStateFilename = filename
  1085. self.host.save_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  1086. @pyqtSlot()
  1087. def slot_stateLoad(self):
  1088. if self.fPluginInfo['type'] == PLUGIN_LV2:
  1089. presetList = []
  1090. for i in range(self.host.get_program_count(self.fPluginId)):
  1091. presetList.append("%03i - %s" % (i+1, self.host.get_program_name(self.fPluginId, i)))
  1092. ret = QInputDialog.getItem(self,
  1093. self.tr("Open LV2 Preset"),
  1094. self.tr("Select an LV2 Preset:"),
  1095. presetList, 0, False)
  1096. if ret[1]:
  1097. index = int(ret[0].split(" - ", 1)[0])-1
  1098. self.host.set_program(self.fPluginId, index)
  1099. self.setMidiProgram(-1)
  1100. return
  1101. fileFilter = self.tr("Carla State File (*.carxs)")
  1102. filename, _ = QFileDialog.getOpenFileName(self, self.tr("Open Plugin State File"), filter=fileFilter)
  1103. # FIXME use ok value, test if it works as expected
  1104. if not filename:
  1105. return
  1106. self.fCurrentStateFilename = filename
  1107. self.host.load_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  1108. #------------------------------------------------------------------
  1109. @pyqtSlot(bool)
  1110. def slot_optionChanged(self, clicked):
  1111. sender = self.sender()
  1112. if sender == self.ui.ch_use_chunks:
  1113. option = PLUGIN_OPTION_USE_CHUNKS
  1114. elif sender == self.ui.ch_fixed_buffer:
  1115. option = PLUGIN_OPTION_FIXED_BUFFERS
  1116. elif sender == self.ui.ch_force_stereo:
  1117. option = PLUGIN_OPTION_FORCE_STEREO
  1118. elif sender == self.ui.ch_map_program_changes:
  1119. option = PLUGIN_OPTION_MAP_PROGRAM_CHANGES
  1120. elif sender == self.ui.ch_send_program_changes:
  1121. option = PLUGIN_OPTION_SEND_PROGRAM_CHANGES
  1122. elif sender == self.ui.ch_send_control_changes:
  1123. option = PLUGIN_OPTION_SEND_CONTROL_CHANGES
  1124. elif sender == self.ui.ch_send_channel_pressure:
  1125. option = PLUGIN_OPTION_SEND_CHANNEL_PRESSURE
  1126. elif sender == self.ui.ch_send_note_aftertouch:
  1127. option = PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH
  1128. elif sender == self.ui.ch_send_pitchbend:
  1129. option = PLUGIN_OPTION_SEND_PITCHBEND
  1130. elif sender == self.ui.ch_send_all_sound_off:
  1131. option = PLUGIN_OPTION_SEND_ALL_SOUND_OFF
  1132. else:
  1133. return
  1134. #--------------------------------------------------------------
  1135. # handle map-program-changes and send-program-changes conflict
  1136. if option == PLUGIN_OPTION_MAP_PROGRAM_CHANGES and clicked:
  1137. self.ui.ch_send_program_changes.setEnabled(False)
  1138. # disable send-program-changes if needed
  1139. if self.ui.ch_send_program_changes.isChecked():
  1140. self.host.set_option(self.fPluginId, PLUGIN_OPTION_SEND_PROGRAM_CHANGES, False)
  1141. #--------------------------------------------------------------
  1142. # set option
  1143. self.host.set_option(self.fPluginId, option, clicked)
  1144. #--------------------------------------------------------------
  1145. # handle map-program-changes and send-program-changes conflict
  1146. if option == PLUGIN_OPTION_MAP_PROGRAM_CHANGES and not clicked:
  1147. self.ui.ch_send_program_changes.setEnabled(
  1148. self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_PROGRAM_CHANGES)
  1149. # restore send-program-changes if needed
  1150. if self.ui.ch_send_program_changes.isChecked():
  1151. self.host.set_option(self.fPluginId, PLUGIN_OPTION_SEND_PROGRAM_CHANGES, True)
  1152. #------------------------------------------------------------------
  1153. @pyqtSlot(float)
  1154. def slot_dryWetChanged(self, value):
  1155. self.host.set_drywet(self.fPluginId, value)
  1156. if self.fParent is not None:
  1157. self.fParent.editDialogParameterValueChanged(self.fPluginId, PARAMETER_DRYWET, value)
  1158. @pyqtSlot(float)
  1159. def slot_volumeChanged(self, value):
  1160. self.host.set_volume(self.fPluginId, value)
  1161. if self.fParent is not None:
  1162. self.fParent.editDialogParameterValueChanged(self.fPluginId, PARAMETER_VOLUME, value)
  1163. @pyqtSlot(float)
  1164. def slot_balanceLeftChanged(self, value):
  1165. self.host.set_balance_left(self.fPluginId, value)
  1166. if self.fParent is not None:
  1167. self.fParent.editDialogParameterValueChanged(self.fPluginId, PARAMETER_BALANCE_LEFT, value)
  1168. @pyqtSlot(float)
  1169. def slot_balanceRightChanged(self, value):
  1170. self.host.set_balance_right(self.fPluginId, value)
  1171. if self.fParent is not None:
  1172. self.fParent.editDialogParameterValueChanged(self.fPluginId, PARAMETER_BALANCE_RIGHT, value)
  1173. @pyqtSlot(float)
  1174. def slot_panChanged(self, value):
  1175. self.host.set_panning(self.fPluginId, value)
  1176. if self.fParent is not None:
  1177. self.fParent.editDialogParameterValueChanged(self.fPluginId, PARAMETER_PANNING, value)
  1178. @pyqtSlot(int)
  1179. def slot_ctrlChannelChanged(self, value):
  1180. self.fControlChannel = value-1
  1181. self.host.set_ctrl_channel(self.fPluginId, self.fControlChannel)
  1182. self.ui.keyboard.allNotesOff()
  1183. self._updateCtrlPrograms()
  1184. #------------------------------------------------------------------
  1185. @pyqtSlot(int, float)
  1186. def slot_parameterValueChanged(self, parameterId, value):
  1187. self.host.set_parameter_value(self.fPluginId, parameterId, value)
  1188. if self.fParent is not None:
  1189. self.fParent.editDialogParameterValueChanged(self.fPluginId, parameterId, value)
  1190. @pyqtSlot(int, int)
  1191. def slot_parameterMappedControlChanged(self, parameterId, control):
  1192. self.host.set_parameter_mapped_control_index(self.fPluginId, parameterId, control)
  1193. @pyqtSlot(int, float, float)
  1194. def slot_parameterMappedRangeChanged(self, parameterId, minimum, maximum):
  1195. self.host.set_parameter_mapped_range(self.fPluginId, parameterId, minimum, maximum)
  1196. @pyqtSlot(int, int)
  1197. def slot_parameterMidiChannelChanged(self, parameterId, channel):
  1198. self.host.set_parameter_midi_channel(self.fPluginId, parameterId, channel-1)
  1199. #------------------------------------------------------------------
  1200. @pyqtSlot(int)
  1201. def slot_programIndexChanged(self, index):
  1202. self.host.set_program(self.fPluginId, index)
  1203. if self.fParent is not None:
  1204. self.fParent.editDialogProgramChanged(self.fPluginId, index)
  1205. self._updateParameterValues()
  1206. @pyqtSlot(int)
  1207. def slot_midiProgramIndexChanged(self, index):
  1208. self.host.set_midi_program(self.fPluginId, index)
  1209. if self.fParent is not None:
  1210. self.fParent.editDialogMidiProgramChanged(self.fPluginId, index)
  1211. self._updateParameterValues()
  1212. #------------------------------------------------------------------
  1213. @pyqtSlot(int)
  1214. def slot_noteOn(self, note):
  1215. if self.fControlChannel >= 0:
  1216. self.host.send_midi_note(self.fPluginId, self.fControlChannel, note, 100)
  1217. if self.fParent is not None:
  1218. self.fParent.editDialogNotePressed(self.fPluginId, note)
  1219. @pyqtSlot(int)
  1220. def slot_noteOff(self, note):
  1221. if self.fControlChannel >= 0:
  1222. self.host.send_midi_note(self.fPluginId, self.fControlChannel, note, 0)
  1223. if self.fParent is not None:
  1224. self.fParent.editDialogNoteReleased(self.fPluginId, note)
  1225. #------------------------------------------------------------------
  1226. @pyqtSlot()
  1227. def slot_finished(self):
  1228. if self.fParent is not None:
  1229. self.fParent.editDialogVisibilityChanged(self.fPluginId, False)
  1230. #------------------------------------------------------------------
  1231. @pyqtSlot()
  1232. def slot_knobCustomMenu(self):
  1233. sender = self.sender()
  1234. knobName = sender.objectName()
  1235. if knobName == "dial_drywet":
  1236. minimum = 0.0
  1237. maximum = 1.0
  1238. default = 1.0
  1239. label = "Dry/Wet"
  1240. elif knobName == "dial_vol":
  1241. minimum = 0.0
  1242. maximum = 1.27
  1243. default = 1.0
  1244. label = "Volume"
  1245. elif knobName == "dial_b_left":
  1246. minimum = -1.0
  1247. maximum = 1.0
  1248. default = -1.0
  1249. label = "Balance-Left"
  1250. elif knobName == "dial_b_right":
  1251. minimum = -1.0
  1252. maximum = 1.0
  1253. default = 1.0
  1254. label = "Balance-Right"
  1255. elif knobName == "dial_pan":
  1256. minimum = -1.0
  1257. maximum = 1.0
  1258. default = 0.0
  1259. label = "Panning"
  1260. else:
  1261. minimum = 0.0
  1262. maximum = 1.0
  1263. default = 0.5
  1264. label = "Unknown"
  1265. menu = QMenu(self)
  1266. actReset = menu.addAction(self.tr("Reset (%i%%)" % (default*100)))
  1267. menu.addSeparator()
  1268. actMinimum = menu.addAction(self.tr("Set to Minimum (%i%%)" % (minimum*100)))
  1269. actCenter = menu.addAction(self.tr("Set to Center"))
  1270. actMaximum = menu.addAction(self.tr("Set to Maximum (%i%%)" % (maximum*100)))
  1271. menu.addSeparator()
  1272. actSet = menu.addAction(self.tr("Set value..."))
  1273. if label not in ("Balance-Left", "Balance-Right", "Panning"):
  1274. menu.removeAction(actCenter)
  1275. actSelected = menu.exec_(QCursor.pos())
  1276. if actSelected == actSet:
  1277. current = minimum + (maximum-minimum)*(float(sender.value())/10000)
  1278. value, ok = QInputDialog.getInt(self,
  1279. self.tr("Set value"),
  1280. label,
  1281. round(current*100.0),
  1282. round(minimum*100.0),
  1283. round(maximum*100.0),
  1284. 1)
  1285. if ok:
  1286. value = float(value)/100.0
  1287. if not ok:
  1288. return
  1289. elif actSelected == actMinimum:
  1290. value = minimum
  1291. elif actSelected == actMaximum:
  1292. value = maximum
  1293. elif actSelected == actReset:
  1294. value = default
  1295. elif actSelected == actCenter:
  1296. value = 0.0
  1297. else:
  1298. return
  1299. sender.setValue(value, True)
  1300. #------------------------------------------------------------------
  1301. @pyqtSlot()
  1302. def slot_channelCustomMenu(self):
  1303. menu = QMenu(self)
  1304. actNone = menu.addAction(self.tr("None"))
  1305. if self.fControlChannel+1 == 0:
  1306. actNone.setCheckable(True)
  1307. actNone.setChecked(True)
  1308. for i in range(1, 16+1):
  1309. action = menu.addAction("%i" % i)
  1310. if self.fControlChannel+1 == i:
  1311. action.setCheckable(True)
  1312. action.setChecked(True)
  1313. actSel = menu.exec_(QCursor.pos())
  1314. if not actSel:
  1315. pass
  1316. elif actSel == actNone:
  1317. self.ui.sb_ctrl_channel.setValue(0)
  1318. elif actSel:
  1319. selChannel = int(actSel.text())
  1320. self.ui.sb_ctrl_channel.setValue(selChannel)
  1321. #------------------------------------------------------------------
  1322. def _createParameterWidgets(self, paramType, paramListFull, tabPageName):
  1323. groupWidgets = {}
  1324. for paramList, width in paramListFull:
  1325. if not paramList:
  1326. break
  1327. tabIndex = self.ui.tabWidget.count()
  1328. scrollArea = QScrollArea(self.ui.tabWidget)
  1329. scrollArea.setWidgetResizable(True)
  1330. scrollArea.setFrameStyle(0)
  1331. palette1 = scrollArea.palette()
  1332. palette1.setColor(QPalette.Background, Qt.transparent)
  1333. scrollArea.setPalette(palette1)
  1334. palette2 = scrollArea.palette()
  1335. palette2.setColor(QPalette.Background, palette2.color(QPalette.Button))
  1336. scrollAreaWidget = QWidget(scrollArea)
  1337. scrollAreaLayout = QVBoxLayout(scrollAreaWidget)
  1338. scrollAreaLayout.setSpacing(3)
  1339. for paramInfo in paramList:
  1340. groupName = paramInfo['groupName']
  1341. if groupName:
  1342. groupSymbol, groupName = groupName.split(":",1)
  1343. groupLayout, groupWidget = groupWidgets.get(groupSymbol, (None, None))
  1344. if groupLayout is None:
  1345. groupWidget = CollapsibleBox(groupName, scrollAreaWidget)
  1346. groupLayout = groupWidget.getContentLayout()
  1347. groupWidget.setPalette(palette2)
  1348. scrollAreaLayout.addWidget(groupWidget)
  1349. groupWidgets[groupSymbol] = (groupLayout, groupWidget)
  1350. else:
  1351. groupLayout = scrollAreaLayout
  1352. groupWidget = scrollAreaWidget
  1353. paramWidget = PluginParameter(groupWidget, self.host, paramInfo, self.fPluginId, tabIndex)
  1354. paramWidget.setLabelWidth(width)
  1355. groupLayout.addWidget(paramWidget)
  1356. self.fParameterList.append((paramType, paramInfo['index'], paramWidget))
  1357. if paramType == PARAMETER_INPUT:
  1358. paramWidget.valueChanged.connect(self.slot_parameterValueChanged)
  1359. paramWidget.mappedControlChanged.connect(self.slot_parameterMappedControlChanged)
  1360. paramWidget.mappedRangeChanged.connect(self.slot_parameterMappedRangeChanged)
  1361. paramWidget.midiChannelChanged.connect(self.slot_parameterMidiChannelChanged)
  1362. scrollAreaLayout.addStretch()
  1363. scrollArea.setWidget(scrollAreaWidget)
  1364. self.ui.tabWidget.addTab(scrollArea, tabPageName)
  1365. if paramType == PARAMETER_INPUT:
  1366. self.ui.tabWidget.setTabIcon(tabIndex, self.fTabIconOff)
  1367. self.fTabIconTimers.append(ICON_STATE_NULL)
  1368. def _updateCtrlPrograms(self):
  1369. self.ui.keyboard.setEnabled(self.fControlChannel >= 0)
  1370. if self.fPluginInfo['category'] != PLUGIN_CATEGORY_SYNTH:
  1371. return
  1372. if self.fPluginInfo['type'] not in (PLUGIN_INTERNAL, PLUGIN_SF2):
  1373. return
  1374. if self.fControlChannel < 0:
  1375. self.ui.cb_programs.setEnabled(False)
  1376. self.ui.cb_midi_programs.setEnabled(False)
  1377. return
  1378. self.ui.cb_programs.setEnabled(True)
  1379. self.ui.cb_midi_programs.setEnabled(True)
  1380. pIndex = self.host.get_current_program_index(self.fPluginId)
  1381. if self.ui.cb_programs.currentIndex() != pIndex:
  1382. self.setProgram(pIndex)
  1383. mpIndex = self.host.get_current_midi_program_index(self.fPluginId)
  1384. if self.ui.cb_midi_programs.currentIndex() != mpIndex:
  1385. self.setMidiProgram(mpIndex)
  1386. def _updateParameterValues(self):
  1387. for _, paramId, paramWidget in self.fParameterList:
  1388. paramWidget.blockSignals(True)
  1389. paramWidget.setValue(self.host.get_current_parameter_value(self.fPluginId, paramId))
  1390. paramWidget.blockSignals(False)
  1391. #------------------------------------------------------------------
  1392. def testTimer(self):
  1393. self.fIdleTimerId = self.startTimer(50)
  1394. self.SIGTERM.connect(self.testTimerClose)
  1395. gCarla.gui = self
  1396. setUpSignals()
  1397. def testTimerClose(self):
  1398. self.close()
  1399. _app.quit()
  1400. #------------------------------------------------------------------
  1401. def closeEvent(self, event):
  1402. if self.fIdleTimerId != 0:
  1403. self.killTimer(self.fIdleTimerId)
  1404. self.fIdleTimerId = 0
  1405. self.host.engine_close()
  1406. QDialog.closeEvent(self, event)
  1407. def timerEvent(self, event):
  1408. if event.timerId() == self.fIdleTimerId:
  1409. self.host.engine_idle()
  1410. self.idleSlow()
  1411. QDialog.timerEvent(self, event)
  1412. # ------------------------------------------------------------------------------------------------------------
  1413. # Main
  1414. if __name__ == '__main__':
  1415. from carla_app import CarlaApplication
  1416. from carla_host import initHost as _initHost, loadHostSettings as _loadHostSettings
  1417. _app = CarlaApplication()
  1418. _host = _initHost("Widgets", None, False, False, False)
  1419. _loadHostSettings(_host)
  1420. _host.engine_init("JACK", "Carla-Widgets")
  1421. _host.add_plugin(BINARY_NATIVE, PLUGIN_DSSI, "/usr/lib/dssi/karplong.so", "karplong", "karplong", 0, None, 0x0)
  1422. _host.set_active(0, True)
  1423. gui1 = CarlaAboutW(None, _host)
  1424. gui1.show()
  1425. gui2 = PluginEdit(None, _host, 0)
  1426. gui2.testTimer()
  1427. gui2.show()
  1428. _app.exit_exec()