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.

1425 lines
56KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla widgets code
  4. # Copyright (C) 2011-2014 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Config)
  19. from carla_config import *
  20. # ------------------------------------------------------------------------------------------------------------
  21. # Imports (Global)
  22. if config_UseQt5:
  23. from PyQt5.QtCore import pyqtSignal, pyqtSlot, QByteArray, QSettings
  24. from PyQt5.QtGui import QColor, QCursor, QFontMetrics, QPainter, QPainterPath
  25. from PyQt5.QtWidgets import QDialog, QInputDialog, QLineEdit, QMenu, QVBoxLayout, QWidget
  26. else:
  27. from PyQt4.QtCore import pyqtSignal, pyqtSlot, QByteArray, QSettings
  28. from PyQt4.QtGui import QColor, QCursor, QFontMetrics, QPainter, QPainterPath
  29. from PyQt4.QtGui import QDialog, QInputDialog, QLineEdit, QMenu, QVBoxLayout, QWidget
  30. # ------------------------------------------------------------------------------------------------------------
  31. # Imports (Custom)
  32. import ui_carla_about
  33. import ui_carla_edit
  34. import ui_carla_parameter
  35. from carla_shared import *
  36. from pixmapkeyboard import PixmapKeyboardHArea
  37. # ------------------------------------------------------------------------------------------------------------
  38. # Carla GUI defines
  39. ICON_STATE_NULL = 0
  40. ICON_STATE_OFF = 1
  41. ICON_STATE_WAIT = 2
  42. ICON_STATE_ON = 3
  43. # ------------------------------------------------------------------------------------------------------------
  44. # Fake plugin info for easy testing
  45. gFakePluginInfo = {
  46. "type": PLUGIN_NONE,
  47. "category": PLUGIN_CATEGORY_SYNTH,
  48. "hints": PLUGIN_IS_SYNTH|PLUGIN_CAN_DRYWET|PLUGIN_CAN_VOLUME|PLUGIN_CAN_PANNING,
  49. "optionsAvailable": 0x1FF, # all
  50. "optionsEnabled": 0x1FF, # all
  51. "filename": "AwesoomeFilename.what",
  52. "name": "Awesoome Name",
  53. "label": "awesoomeLabel",
  54. "maker": "Awesoome Maker",
  55. "copyright": "Awesoome Copyright",
  56. "iconName": "plugin",
  57. "uniqueId": 0
  58. }
  59. gFakeParamInfo = {
  60. "type": PARAMETER_INPUT,
  61. "hints": PARAMETER_IS_ENABLED|PARAMETER_IS_AUTOMABLE,
  62. "name": "Parameter Name",
  63. "unit": "",
  64. "scalePoints": [],
  65. "index": 0,
  66. "default": 0.0,
  67. "minimum": 0.0,
  68. "maximum": 1.0,
  69. "step": 0.001,
  70. "stepSmall": 0.0001,
  71. "stepLarge": 0.01,
  72. "midiCC": -1,
  73. "midiChannel": 1,
  74. "current": 0.0
  75. }
  76. gFakePortCountInfo = {
  77. "ins": 0,
  78. "outs": 0
  79. }
  80. # ------------------------------------------------------------------------------------------------------------
  81. # Carla About dialog
  82. class CarlaAboutW(QDialog):
  83. def __init__(self, parent):
  84. QDialog.__init__(self, parent)
  85. self.ui = ui_carla_about.Ui_CarlaAboutW()
  86. self.ui.setupUi(self)
  87. if gCarla.isControl:
  88. extraInfo = " - <b>%s</b>" % self.tr("OSC Bridge Version")
  89. elif gCarla.isPlugin:
  90. extraInfo = " - <b>%s</b>" % self.tr("Plugin Version")
  91. else:
  92. extraInfo = ""
  93. self.ui.l_about.setText(self.tr(""
  94. "<br>Version %s"
  95. "<br>Carla is a fully-featured audio plugin host%s.<br>"
  96. "<br>Copyright (C) 2011-2014 falkTX<br>"
  97. "" % (VERSION, extraInfo)))
  98. if gCarla.isControl or gCarla.isPlugin or gCarla.host is None:
  99. self.ui.l_extended.hide()
  100. self.ui.tabWidget.removeTab(1)
  101. self.ui.tabWidget.removeTab(1)
  102. self.adjustSize()
  103. else:
  104. self.ui.l_extended.setText(gCarla.host.get_complete_license_text())
  105. if gCarla.host.is_engine_running():
  106. self.ui.le_osc_url_tcp.setText(gCarla.host.get_host_osc_url_tcp())
  107. self.ui.le_osc_url_udp.setText(gCarla.host.get_host_osc_url_udp())
  108. else:
  109. self.ui.le_osc_url_tcp.setText(self.tr("(Engine not running)"))
  110. self.ui.le_osc_url_udp.setText(self.tr("(Engine not running)"))
  111. self.ui.l_osc_cmds.setText(""
  112. " /set_active <i-value>\n"
  113. " /set_drywet <f-value>\n"
  114. " /set_volume <f-value>\n"
  115. " /set_balance_left <f-value>\n"
  116. " /set_balance_right <f-value>\n"
  117. " /set_panning <f-value>\n"
  118. " /set_parameter_value <i-index> <f-value>\n"
  119. " /set_parameter_midi_cc <i-index> <i-cc>\n"
  120. " /set_parameter_midi_channel <i-index> <i-channel>\n"
  121. " /set_program <i-index>\n"
  122. " /set_midi_program <i-index>\n"
  123. " /note_on <i-note> <i-velo>\n"
  124. " /note_off <i-note>\n"
  125. )
  126. self.ui.l_example.setText("/Carla/2/set_parameter_value 5 1.0")
  127. self.ui.l_example_help.setText("<i>(as in this example, \"2\" is the plugin number and \"5\" the parameter)</i>")
  128. self.ui.l_ladspa.setText(self.tr("Everything! (Including LRDF)"))
  129. self.ui.l_dssi.setText(self.tr("Everything! (Including CustomData/Chunks)"))
  130. self.ui.l_lv2.setText(self.tr("About 90&#37; complete (using custom extensions)<br/>"
  131. "Implemented Feature/Extensions:"
  132. "<ul>"
  133. "<li>http://lv2plug.in/ns/ext/atom</li>"
  134. "<li>http://lv2plug.in/ns/ext/buf-size</li>"
  135. "<li>http://lv2plug.in/ns/ext/data-access</li>"
  136. #"<li>http://lv2plug.in/ns/ext/dynmanifest</li>"
  137. "<li>http://lv2plug.in/ns/ext/event</li>"
  138. "<li>http://lv2plug.in/ns/ext/instance-access</li>"
  139. "<li>http://lv2plug.in/ns/ext/log</li>"
  140. "<li>http://lv2plug.in/ns/ext/midi</li>"
  141. #"<li>http://lv2plug.in/ns/ext/morph</li>"
  142. "<li>http://lv2plug.in/ns/ext/options</li>"
  143. "<li>http://lv2plug.in/ns/ext/parameters</li>"
  144. #"<li>http://lv2plug.in/ns/ext/patch</li>"
  145. #"<li>http://lv2plug.in/ns/ext/port-groups</li>"
  146. "<li>http://lv2plug.in/ns/ext/port-props</li>"
  147. "<li>http://lv2plug.in/ns/ext/presets</li>"
  148. "<li>http://lv2plug.in/ns/ext/resize-port</li>"
  149. "<li>http://lv2plug.in/ns/ext/state</li>"
  150. "<li>http://lv2plug.in/ns/ext/time</li>"
  151. "<li>http://lv2plug.in/ns/ext/uri-map</li>"
  152. "<li>http://lv2plug.in/ns/ext/urid</li>"
  153. "<li>http://lv2plug.in/ns/ext/worker</li>"
  154. "<li>http://lv2plug.in/ns/extensions/ui</li>"
  155. "<li>http://lv2plug.in/ns/extensions/units</li>"
  156. "<li>http://home.gna.org/lv2dynparam/rtmempool/v1</li>"
  157. "<li>http://kxstudio.sf.net/ns/lv2ext/external-ui</li>"
  158. "<li>http://kxstudio.sf.net/ns/lv2ext/programs</li>"
  159. "<li>http://kxstudio.sf.net/ns/lv2ext/rtmempool</li>"
  160. "<li>http://ll-plugins.nongnu.org/lv2/ext/midimap</li>"
  161. "<li>http://ll-plugins.nongnu.org/lv2/ext/miditype</li>"
  162. "</ul>"))
  163. self.ui.l_vst.setText(self.tr("<p>About 85&#37; complete (missing vst bank/presets and some minor stuff)</p>"))
  164. def done(self, r):
  165. QDialog.done(self, r)
  166. self.close()
  167. # ------------------------------------------------------------------------------------------------------------
  168. # Plugin Parameter
  169. class PluginParameter(QWidget):
  170. midiControlChanged = pyqtSignal(int, int)
  171. midiChannelChanged = pyqtSignal(int, int)
  172. valueChanged = pyqtSignal(int, float)
  173. def __init__(self, parent, pInfo, pluginId, tabIndex):
  174. QWidget.__init__(self, parent)
  175. self.ui = ui_carla_parameter.Ui_PluginParameter()
  176. self.ui.setupUi(self)
  177. # -------------------------------------------------------------
  178. # Internal stuff
  179. self.fMidiControl = -1
  180. self.fMidiChannel = 1
  181. self.fParameterId = pInfo['index']
  182. self.fPluginId = pluginId
  183. self.fTabIndex = tabIndex
  184. # -------------------------------------------------------------
  185. # Set-up GUI
  186. pType = pInfo['type']
  187. pHints = pInfo['hints']
  188. self.ui.label.setText(pInfo['name'])
  189. self.ui.widget.setName(pInfo['name'])
  190. self.ui.widget.setMinimum(pInfo['minimum'])
  191. self.ui.widget.setMaximum(pInfo['maximum'])
  192. self.ui.widget.setDefault(pInfo['default'])
  193. self.ui.widget.setValue(pInfo['current'], False)
  194. self.ui.widget.setLabel(pInfo['unit'])
  195. self.ui.widget.setStep(pInfo['step'])
  196. self.ui.widget.setStepSmall(pInfo['stepSmall'])
  197. self.ui.widget.setStepLarge(pInfo['stepLarge'])
  198. self.ui.widget.setScalePoints(pInfo['scalePoints'], bool(pHints & PARAMETER_USES_SCALEPOINTS))
  199. if not pHints & PARAMETER_IS_AUTOMABLE:
  200. self.ui.sb_control.setEnabled(False)
  201. self.ui.sb_channel.setEnabled(False)
  202. if pType == PARAMETER_INPUT:
  203. if not pHints & PARAMETER_IS_ENABLED:
  204. self.ui.label.setEnabled(False)
  205. self.ui.widget.setEnabled(False)
  206. self.ui.widget.setReadOnly(True)
  207. self.ui.sb_control.setEnabled(False)
  208. self.ui.sb_channel.setEnabled(False)
  209. if pHints & PARAMETER_IS_READ_ONLY:
  210. self.ui.widget.setReadOnly(True)
  211. elif pType == PARAMETER_OUTPUT:
  212. self.ui.widget.setReadOnly(True)
  213. else:
  214. self.ui.widget.setVisible(False)
  215. self.ui.sb_control.setVisible(False)
  216. self.ui.sb_channel.setVisible(False)
  217. if pHints & PARAMETER_USES_CUSTOM_TEXT and not gCarla.isPlugin:
  218. self.ui.widget.setTextCallback(self._textCallBack)
  219. self.ui.widget.updateAll()
  220. self.setMidiControl(pInfo['midiCC'])
  221. self.setMidiChannel(pInfo['midiChannel'])
  222. # -------------------------------------------------------------
  223. # Set-up connections
  224. self.ui.sb_control.customContextMenuRequested.connect(self.slot_controlSpinboxCustomMenu)
  225. self.ui.sb_channel.customContextMenuRequested.connect(self.slot_channelSpinboxCustomMenu)
  226. self.ui.sb_control.valueChanged.connect(self.slot_controlSpinboxChanged)
  227. self.ui.sb_channel.valueChanged.connect(self.slot_channelSpinboxChanged)
  228. self.ui.widget.valueChanged.connect(self.slot_widgetValueChanged)
  229. # -------------------------------------------------------------
  230. def getPluginId(self):
  231. return self.fPluginId
  232. def getTabIndex(self):
  233. return self.fTabIndex
  234. def setPluginId(self, pluginId):
  235. self.fPluginId = pluginId
  236. def setDefault(self, value):
  237. self.ui.widget.setDefault(value)
  238. def setValue(self, value, send=True):
  239. self.ui.widget.setValue(value, send)
  240. def setMidiControl(self, control):
  241. self.fMidiControl = control
  242. self.ui.sb_control.blockSignals(True)
  243. self.ui.sb_control.setValue(control)
  244. self.ui.sb_control.blockSignals(False)
  245. def setMidiChannel(self, channel):
  246. self.fMidiChannel = channel
  247. self.ui.sb_channel.blockSignals(True)
  248. self.ui.sb_channel.setValue(channel)
  249. self.ui.sb_channel.blockSignals(False)
  250. def setLabelWidth(self, width):
  251. self.ui.label.setFixedWidth(width)
  252. @pyqtSlot()
  253. def slot_controlSpinboxCustomMenu(self):
  254. menu = QMenu(self)
  255. actNone = menu.addAction(self.tr("None"))
  256. if self.fMidiControl == -1:
  257. actNone.setCheckable(True)
  258. actNone.setChecked(True)
  259. for cc in MIDI_CC_LIST:
  260. action = menu.addAction(cc)
  261. if self.fMidiControl != -1 and int(cc.split(" ", 1)[0], 16) == self.fMidiControl:
  262. action.setCheckable(True)
  263. action.setChecked(True)
  264. actSel = menu.exec_(QCursor.pos())
  265. if not actSel:
  266. pass
  267. elif actSel == actNone:
  268. self.ui.sb_control.setValue(-1)
  269. else:
  270. selControlStr = actSel.text()
  271. selControl = int(selControlStr.split(" ", 1)[0], 16)
  272. self.ui.sb_control.setValue(selControl)
  273. @pyqtSlot()
  274. def slot_channelSpinboxCustomMenu(self):
  275. menu = QMenu(self)
  276. for i in range(1, 16+1):
  277. action = menu.addAction("%i" % i)
  278. if self.fMidiChannel == i:
  279. action.setCheckable(True)
  280. action.setChecked(True)
  281. actSel = menu.exec_(QCursor.pos())
  282. if actSel:
  283. selChannel = int(actSel.text())
  284. self.ui.sb_channel.setValue(selChannel)
  285. @pyqtSlot(int)
  286. def slot_controlSpinboxChanged(self, control):
  287. self.fMidiControl = control
  288. self.midiControlChanged.emit(self.fParameterId, control)
  289. @pyqtSlot(int)
  290. def slot_channelSpinboxChanged(self, channel):
  291. self.fMidiChannel = channel
  292. self.midiChannelChanged.emit(self.fParameterId, channel)
  293. @pyqtSlot(float)
  294. def slot_widgetValueChanged(self, value):
  295. self.valueChanged.emit(self.fParameterId, value)
  296. def _textCallBack(self):
  297. return gCarla.host.get_parameter_text(self.fPluginId, self.fParameterId)
  298. # ------------------------------------------------------------------------------------------------------------
  299. # Plugin Editor (Built-in)
  300. class PluginEdit(QDialog):
  301. kParamsPerPage = 8
  302. def __init__(self, parent, pluginId):
  303. QDialog.__init__(self, gCarla.gui)
  304. self.ui = ui_carla_edit.Ui_PluginEdit()
  305. self.ui.setupUi(self)
  306. # -------------------------------------------------------------
  307. # Internal stuff
  308. self.fGeometry = QByteArray()
  309. self.fPluginId = pluginId
  310. self.fPuginInfo = None
  311. self.fRealParent = parent
  312. self.fCurrentProgram = -1
  313. self.fCurrentMidiProgram = -1
  314. self.fCurrentStateFilename = None
  315. self.fControlChannel = int(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_CTRL_CHANNEL)) if gCarla.host is not None else 0
  316. self.fFirstInit = True
  317. self.fParameterCount = 0
  318. self.fParameterList = [] # (type, id, widget)
  319. self.fParametersToUpdate = [] # (id, value)
  320. self.fPlayingNotes = [] # (channel, note)
  321. self.fTabIconOff = QIcon(":/bitmaps/led_off.png")
  322. self.fTabIconOn = QIcon(":/bitmaps/led_yellow.png")
  323. self.fTabIconCount = 0
  324. self.fTabIconTimers = []
  325. # -------------------------------------------------------------
  326. # Set-up GUI
  327. self.ui.dial_drywet.setCustomPaintMode(self.ui.dial_drywet.CUSTOM_PAINT_MODE_CARLA_WET)
  328. self.ui.dial_drywet.setPixmap(3)
  329. self.ui.dial_drywet.setLabel("Dry/Wet")
  330. self.ui.dial_drywet.setMinimum(0.0)
  331. self.ui.dial_drywet.setMaximum(1.0)
  332. self.ui.dial_vol.setCustomPaintMode(self.ui.dial_vol.CUSTOM_PAINT_MODE_CARLA_VOL)
  333. self.ui.dial_vol.setPixmap(3)
  334. self.ui.dial_vol.setLabel("Volume")
  335. self.ui.dial_vol.setMinimum(0.0)
  336. self.ui.dial_vol.setMaximum(1.27)
  337. self.ui.dial_b_left.setCustomPaintMode(self.ui.dial_b_left.CUSTOM_PAINT_MODE_CARLA_L)
  338. self.ui.dial_b_left.setPixmap(4)
  339. self.ui.dial_b_left.setLabel("L")
  340. self.ui.dial_b_left.setMinimum(-1.0)
  341. self.ui.dial_b_left.setMaximum(1.0)
  342. self.ui.dial_b_right.setCustomPaintMode(self.ui.dial_b_right.CUSTOM_PAINT_MODE_CARLA_R)
  343. self.ui.dial_b_right.setPixmap(4)
  344. self.ui.dial_b_right.setLabel("R")
  345. self.ui.dial_b_right.setMinimum(-1.0)
  346. self.ui.dial_b_right.setMaximum(1.0)
  347. self.ui.dial_pan.setCustomPaintMode(self.ui.dial_b_right.CUSTOM_PAINT_MODE_CARLA_PAN)
  348. self.ui.dial_pan.setPixmap(4)
  349. self.ui.dial_pan.setLabel("Pan")
  350. self.ui.dial_pan.setMinimum(-1.0)
  351. self.ui.dial_pan.setMaximum(1.0)
  352. if gCarla.host is not None:
  353. self.ui.dial_drywet.setValue(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_DRYWET))
  354. self.ui.dial_vol.setValue(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_VOLUME))
  355. self.ui.dial_b_left.setValue(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_BALANCE_LEFT))
  356. self.ui.dial_b_right.setValue(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_BALANCE_RIGHT))
  357. self.ui.dial_pan.setValue(gCarla.host.get_internal_parameter_value(pluginId, PARAMETER_PANNING))
  358. else:
  359. self.ui.dial_drywet.setValue(1.0)
  360. self.ui.dial_vol.setValue(1.0)
  361. self.ui.dial_b_left.setValue(-1.0)
  362. self.ui.dial_b_right.setValue(1.0)
  363. self.ui.dial_pan.setValue(0.0)
  364. self.ui.sb_ctrl_channel.setValue(self.fControlChannel+1)
  365. self.ui.scrollArea = PixmapKeyboardHArea(self)
  366. self.ui.keyboard = self.ui.scrollArea.keyboard
  367. self.layout().addWidget(self.ui.scrollArea)
  368. self.ui.scrollArea.setEnabled(False)
  369. self.ui.scrollArea.setVisible(False)
  370. # todo
  371. self.ui.rb_balance.setEnabled(False)
  372. self.ui.rb_pan.setEnabled(False)
  373. self.reloadAll()
  374. # -------------------------------------------------------------
  375. # Set-up connections
  376. self.finished.connect(self.slot_finished)
  377. self.ui.ch_fixed_buffer.clicked.connect(self.slot_optionChanged)
  378. self.ui.ch_force_stereo.clicked.connect(self.slot_optionChanged)
  379. self.ui.ch_map_program_changes.clicked.connect(self.slot_optionChanged)
  380. self.ui.ch_use_chunks.clicked.connect(self.slot_optionChanged)
  381. self.ui.ch_send_control_changes.clicked.connect(self.slot_optionChanged)
  382. self.ui.ch_send_channel_pressure.clicked.connect(self.slot_optionChanged)
  383. self.ui.ch_send_note_aftertouch.clicked.connect(self.slot_optionChanged)
  384. self.ui.ch_send_pitchbend.clicked.connect(self.slot_optionChanged)
  385. self.ui.ch_send_all_sound_off.clicked.connect(self.slot_optionChanged)
  386. self.ui.dial_drywet.realValueChanged.connect(self.slot_dryWetChanged)
  387. self.ui.dial_vol.realValueChanged.connect(self.slot_volumeChanged)
  388. self.ui.dial_b_left.realValueChanged.connect(self.slot_balanceLeftChanged)
  389. self.ui.dial_b_right.realValueChanged.connect(self.slot_balanceRightChanged)
  390. self.ui.dial_pan.realValueChanged.connect(self.slot_panChanged)
  391. self.ui.sb_ctrl_channel.valueChanged.connect(self.slot_ctrlChannelChanged)
  392. self.ui.dial_drywet.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  393. self.ui.dial_vol.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  394. self.ui.dial_b_left.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  395. self.ui.dial_b_right.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  396. self.ui.dial_pan.customContextMenuRequested.connect(self.slot_knobCustomMenu)
  397. self.ui.sb_ctrl_channel.customContextMenuRequested.connect(self.slot_channelCustomMenu)
  398. self.ui.keyboard.noteOn.connect(self.slot_noteOn)
  399. self.ui.keyboard.noteOff.connect(self.slot_noteOff)
  400. self.ui.cb_programs.currentIndexChanged.connect(self.slot_programIndexChanged)
  401. self.ui.cb_midi_programs.currentIndexChanged.connect(self.slot_midiProgramIndexChanged)
  402. if gCarla.isLocal:
  403. self.ui.b_save_state.clicked.connect(self.slot_stateSave)
  404. self.ui.b_load_state.clicked.connect(self.slot_stateLoad)
  405. else:
  406. self.ui.b_load_state.setEnabled(False)
  407. self.ui.b_save_state.setEnabled(False)
  408. #------------------------------------------------------------------
  409. def updateInfo(self):
  410. # Update current program text
  411. if self.ui.cb_programs.count() > 0:
  412. pIndex = self.ui.cb_programs.currentIndex()
  413. pName = gCarla.host.get_program_name(self.fPluginId, pIndex)
  414. #pName = pName[:40] + (pName[40:] and "...")
  415. self.ui.cb_programs.setItemText(pIndex, pName)
  416. # Update current midi program text
  417. if self.ui.cb_midi_programs.count() > 0:
  418. mpIndex = self.ui.cb_midi_programs.currentIndex()
  419. mpData = gCarla.host.get_midi_program_data(self.fPluginId, mpIndex)
  420. mpBank = mpData['bank']
  421. mpProg = mpData['program']
  422. mpName = mpData['name']
  423. #mpName = mpName[:40] + (mpName[40:] and "...")
  424. self.ui.cb_midi_programs.setItemText(mpIndex, "%03i:%03i - %s" % (mpBank+1, mpProg+1, mpName))
  425. # Update all parameter values
  426. for paramType, paramId, paramWidget in self.fParameterList:
  427. paramWidget.setValue(gCarla.host.get_current_parameter_value(self.fPluginId, paramId), False)
  428. paramWidget.update()
  429. self.fParametersToUpdate = []
  430. #------------------------------------------------------------------
  431. def reloadAll(self):
  432. if gCarla.host is not None:
  433. self.fPluginInfo = gCarla.host.get_plugin_info(self.fPluginId)
  434. if not gCarla.isLocal:
  435. self.fPluginInfo['hints'] &= ~PLUGIN_HAS_CUSTOM_UI
  436. else:
  437. self.fPluginInfo = gFakePluginInfo
  438. self.reloadInfo()
  439. self.reloadParameters()
  440. self.reloadPrograms()
  441. if self.fPluginInfo['type'] == PLUGIN_LV2:
  442. self.ui.b_save_state.setEnabled(False)
  443. if not self.ui.scrollArea.isEnabled():
  444. self.resize(self.width(), self.height()-self.ui.scrollArea.height())
  445. self.fFirstInit = False
  446. #------------------------------------------------------------------
  447. def reloadInfo(self):
  448. if gCarla.host is not None:
  449. pluginName = gCarla.host.get_real_plugin_name(self.fPluginId)
  450. #audioCountInfo = gCarla.host.get_audio_port_count_info(self.fPluginId)
  451. midiCountInfo = gCarla.host.get_midi_port_count_info(self.fPluginId)
  452. #paramCountInfo = gCarla.host.get_parameter_count_info(self.fPluginId)
  453. else:
  454. pluginName = ""
  455. #audioCountInfo = gFakePortCountInfo
  456. midiCountInfo = gFakePortCountInfo
  457. #paramCountInfo = gFakePortCountInfo
  458. pluginHints = self.fPluginInfo['hints']
  459. self.ui.le_type.setText(getPluginTypeAsString(self.fPluginInfo['type']))
  460. if pluginName:
  461. self.ui.label_name.setEnabled(True)
  462. self.ui.le_name.setEnabled(True)
  463. self.ui.le_name.setText(pluginName)
  464. self.ui.le_name.setToolTip(pluginName)
  465. else:
  466. self.ui.label_name.setEnabled(False)
  467. self.ui.le_name.setEnabled(False)
  468. self.ui.le_name.setText("")
  469. self.ui.le_name.setToolTip("")
  470. if self.fPluginInfo['label']:
  471. self.ui.label_label.setEnabled(True)
  472. self.ui.le_label.setEnabled(True)
  473. self.ui.le_label.setText(self.fPluginInfo['label'])
  474. self.ui.le_label.setToolTip(self.fPluginInfo['label'])
  475. else:
  476. self.ui.label_label.setEnabled(False)
  477. self.ui.le_label.setEnabled(False)
  478. self.ui.le_label.setText("")
  479. self.ui.le_label.setToolTip("")
  480. if self.fPluginInfo['maker']:
  481. self.ui.label_maker.setEnabled(True)
  482. self.ui.le_maker.setEnabled(True)
  483. self.ui.le_maker.setText(self.fPluginInfo['maker'])
  484. self.ui.le_maker.setToolTip(self.fPluginInfo['maker'])
  485. else:
  486. self.ui.label_maker.setEnabled(False)
  487. self.ui.le_maker.setEnabled(False)
  488. self.ui.le_maker.setText("")
  489. self.ui.le_maker.setToolTip("")
  490. if self.fPluginInfo['copyright']:
  491. self.ui.label_copyright.setEnabled(True)
  492. self.ui.le_copyright.setEnabled(True)
  493. self.ui.le_copyright.setText(self.fPluginInfo['copyright'])
  494. self.ui.le_copyright.setToolTip(self.fPluginInfo['copyright'])
  495. else:
  496. self.ui.label_copyright.setEnabled(False)
  497. self.ui.le_copyright.setEnabled(False)
  498. self.ui.le_copyright.setText("")
  499. self.ui.le_copyright.setToolTip("")
  500. if self.fPluginInfo['uniqueId'] != 0:
  501. self.ui.label_unique_id.setEnabled(True)
  502. self.ui.le_unique_id.setEnabled(True)
  503. self.ui.le_unique_id.setText(str(self.fPluginInfo['uniqueId']))
  504. self.ui.le_unique_id.setToolTip(str(self.fPluginInfo['uniqueId']))
  505. else:
  506. self.ui.label_unique_id.setEnabled(False)
  507. self.ui.le_unique_id.setEnabled(False)
  508. self.ui.le_unique_id.setText("")
  509. self.ui.le_unique_id.setToolTip("")
  510. self.ui.label_plugin.setText("\n%s\n" % self.fPluginInfo['name'])
  511. self.setWindowTitle(self.fPluginInfo['name'])
  512. self.ui.dial_drywet.setEnabled(pluginHints & PLUGIN_CAN_DRYWET)
  513. self.ui.dial_vol.setEnabled(pluginHints & PLUGIN_CAN_VOLUME)
  514. self.ui.dial_b_left.setEnabled(pluginHints & PLUGIN_CAN_BALANCE)
  515. self.ui.dial_b_right.setEnabled(pluginHints & PLUGIN_CAN_BALANCE)
  516. self.ui.dial_pan.setEnabled(pluginHints & PLUGIN_CAN_PANNING)
  517. self.ui.ch_fixed_buffer.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_FIXED_BUFFERS)
  518. self.ui.ch_fixed_buffer.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_FIXED_BUFFERS)
  519. self.ui.ch_force_stereo.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_FORCE_STEREO)
  520. self.ui.ch_force_stereo.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_FORCE_STEREO)
  521. self.ui.ch_map_program_changes.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_MAP_PROGRAM_CHANGES)
  522. self.ui.ch_map_program_changes.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_MAP_PROGRAM_CHANGES)
  523. self.ui.ch_use_chunks.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_USE_CHUNKS)
  524. self.ui.ch_use_chunks.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_USE_CHUNKS)
  525. self.ui.ch_send_control_changes.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_CONTROL_CHANGES)
  526. self.ui.ch_send_control_changes.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_SEND_CONTROL_CHANGES)
  527. self.ui.ch_send_channel_pressure.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_CHANNEL_PRESSURE)
  528. self.ui.ch_send_channel_pressure.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_SEND_CHANNEL_PRESSURE)
  529. self.ui.ch_send_note_aftertouch.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH)
  530. self.ui.ch_send_note_aftertouch.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH)
  531. self.ui.ch_send_pitchbend.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_PITCHBEND)
  532. self.ui.ch_send_pitchbend.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_SEND_PITCHBEND)
  533. self.ui.ch_send_all_sound_off.setEnabled(self.fPluginInfo['optionsAvailable'] & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  534. self.ui.ch_send_all_sound_off.setChecked(self.fPluginInfo['optionsEnabled'] & PLUGIN_OPTION_SEND_ALL_SOUND_OFF)
  535. self.ui.sw_programs.setCurrentIndex(0 if self.fPluginInfo['type'] == PLUGIN_VST else 1)
  536. # Show/hide keyboard
  537. showKeyboard = (self.fPluginInfo['category'] == PLUGIN_CATEGORY_SYNTH or midiCountInfo['ins'] > 0 < midiCountInfo['outs'])
  538. self.ui.scrollArea.setEnabled(showKeyboard)
  539. self.ui.scrollArea.setVisible(showKeyboard)
  540. # Force-Update parent for new hints
  541. if self.fRealParent and not self.fFirstInit:
  542. self.fRealParent.recheckPluginHints(pluginHints)
  543. def reloadParameters(self):
  544. # Reset
  545. self.fParameterCount = 0
  546. self.fParameterList = []
  547. self.fParametersToUpdate = []
  548. self.fTabIconCount = 0
  549. self.fTabIconTimers = []
  550. # Remove all previous parameters
  551. for x in range(self.ui.tabWidget.count()-1):
  552. self.ui.tabWidget.widget(1).deleteLater()
  553. self.ui.tabWidget.removeTab(1)
  554. if gCarla.host is None:
  555. paramFakeListFull = []
  556. paramFakeList = []
  557. paramFakeWidth = QFontMetrics(self.font()).width(gFakeParamInfo['name'])
  558. paramFakeList.append(gFakeParamInfo)
  559. paramFakeListFull.append((paramFakeList, paramFakeWidth))
  560. self._createParameterWidgets(PARAMETER_INPUT, paramFakeListFull, self.tr("Parameters"))
  561. return
  562. parameterCount = gCarla.host.get_parameter_count(self.fPluginId)
  563. if parameterCount <= 0:
  564. pass
  565. elif parameterCount <= gCarla.maxParameters:
  566. paramInputListFull = []
  567. paramOutputListFull = []
  568. paramInputList = [] # ([params], width)
  569. paramInputWidth = 0
  570. paramOutputList = [] # ([params], width)
  571. paramOutputWidth = 0
  572. for i in range(parameterCount):
  573. paramInfo = gCarla.host.get_parameter_info(self.fPluginId, i)
  574. paramData = gCarla.host.get_parameter_data(self.fPluginId, i)
  575. paramRanges = gCarla.host.get_parameter_ranges(self.fPluginId, i)
  576. paramValue = gCarla.host.get_current_parameter_value(self.fPluginId, i)
  577. if paramData['type'] not in (PARAMETER_INPUT, PARAMETER_OUTPUT):
  578. continue
  579. if (paramData['hints'] & PARAMETER_IS_ENABLED) == 0:
  580. continue
  581. parameter = {
  582. 'type': paramData['type'],
  583. 'hints': paramData['hints'],
  584. 'name': paramInfo['name'],
  585. 'unit': paramInfo['unit'],
  586. 'scalePoints': [],
  587. 'index': paramData['index'],
  588. 'default': paramRanges['def'],
  589. 'minimum': paramRanges['min'],
  590. 'maximum': paramRanges['max'],
  591. 'step': paramRanges['step'],
  592. 'stepSmall': paramRanges['stepSmall'],
  593. 'stepLarge': paramRanges['stepLarge'],
  594. 'midiCC': paramData['midiCC'],
  595. 'midiChannel': paramData['midiChannel']+1,
  596. 'current': paramValue
  597. }
  598. for j in range(paramInfo['scalePointCount']):
  599. scalePointInfo = gCarla.host.get_parameter_scalepoint_info(self.fPluginId, i, j)
  600. parameter['scalePoints'].append({
  601. 'value': scalePointInfo['value'],
  602. 'label': scalePointInfo['label']
  603. })
  604. #parameter['name'] = parameter['name'][:30] + (parameter['name'][30:] and "...")
  605. # -----------------------------------------------------------------
  606. # Get width values, in packs of 10
  607. if parameter['type'] == PARAMETER_INPUT:
  608. paramInputWidthTMP = QFontMetrics(self.font()).width(parameter['name'])
  609. if paramInputWidthTMP > paramInputWidth:
  610. paramInputWidth = paramInputWidthTMP
  611. paramInputList.append(parameter)
  612. if len(paramInputList) == self.kParamsPerPage:
  613. paramInputListFull.append((paramInputList, paramInputWidth))
  614. paramInputList = []
  615. paramInputWidth = 0
  616. else:
  617. paramOutputWidthTMP = QFontMetrics(self.font()).width(parameter['name'])
  618. if paramOutputWidthTMP > paramOutputWidth:
  619. paramOutputWidth = paramOutputWidthTMP
  620. paramOutputList.append(parameter)
  621. if len(paramOutputList) == self.kParamsPerPage:
  622. paramOutputListFull.append((paramOutputList, paramOutputWidth))
  623. paramOutputList = []
  624. paramOutputWidth = 0
  625. # for i in range(parameterCount)
  626. else:
  627. # Final page width values
  628. if 0 < len(paramInputList) < 10:
  629. paramInputListFull.append((paramInputList, paramInputWidth))
  630. if 0 < len(paramOutputList) < 10:
  631. paramOutputListFull.append((paramOutputList, paramOutputWidth))
  632. # -----------------------------------------------------------------
  633. # Create parameter tabs + widgets
  634. self._createParameterWidgets(PARAMETER_INPUT, paramInputListFull, self.tr("Parameters"))
  635. self._createParameterWidgets(PARAMETER_OUTPUT, paramOutputListFull, self.tr("Outputs"))
  636. else: # > gCarla.maxParameters
  637. fakeName = self.tr("This plugin has too many parameters to display here!")
  638. paramFakeListFull = []
  639. paramFakeList = []
  640. paramFakeWidth = QFontMetrics(self.font()).width(fakeName)
  641. parameter = {
  642. 'type': PARAMETER_UNKNOWN,
  643. 'hints': 0x0,
  644. 'name': fakeName,
  645. 'unit': "",
  646. 'scalePoints': [],
  647. 'index': 0,
  648. 'default': 0.0,
  649. 'minimum': 0.0,
  650. 'maximum': 1.0,
  651. 'step': 0.0,
  652. 'stepSmall': 0.0,
  653. 'stepLarge': 0.0,
  654. 'midiCC': -1,
  655. 'midiChannel': 1,
  656. 'current': 0.0
  657. }
  658. paramFakeList.append(parameter)
  659. paramFakeListFull.append((paramFakeList, paramFakeWidth))
  660. self._createParameterWidgets(PARAMETER_UNKNOWN, paramFakeListFull, self.tr("Information"))
  661. def reloadPrograms(self):
  662. # Programs
  663. self.ui.cb_programs.blockSignals(True)
  664. self.ui.cb_programs.clear()
  665. programCount = gCarla.host.get_program_count(self.fPluginId) if gCarla.host is not None else 0
  666. if programCount > 0:
  667. self.ui.cb_programs.setEnabled(True)
  668. self.ui.label_programs.setEnabled(True)
  669. for i in range(programCount):
  670. pName = gCarla.host.get_program_name(self.fPluginId, i)
  671. #pName = pName[:40] + (pName[40:] and "...")
  672. self.ui.cb_programs.addItem(pName)
  673. self.fCurrentProgram = gCarla.host.get_current_program_index(self.fPluginId)
  674. self.ui.cb_programs.setCurrentIndex(self.fCurrentProgram)
  675. else:
  676. self.fCurrentProgram = -1
  677. self.ui.cb_programs.setEnabled(False)
  678. self.ui.label_programs.setEnabled(False)
  679. self.ui.cb_programs.blockSignals(False)
  680. # MIDI Programs
  681. self.ui.cb_midi_programs.blockSignals(True)
  682. self.ui.cb_midi_programs.clear()
  683. midiProgramCount = gCarla.host.get_midi_program_count(self.fPluginId) if gCarla.host is not None else 0
  684. if midiProgramCount > 0:
  685. self.ui.cb_midi_programs.setEnabled(True)
  686. self.ui.label_midi_programs.setEnabled(True)
  687. for i in range(midiProgramCount):
  688. mpData = gCarla.host.get_midi_program_data(self.fPluginId, i)
  689. mpBank = mpData['bank']
  690. mpProg = mpData['program']
  691. mpName = mpData['name']
  692. #mpName = mpName[:40] + (mpName[40:] and "...")
  693. self.ui.cb_midi_programs.addItem("%03i:%03i - %s" % (mpBank+1, mpProg+1, mpName))
  694. self.fCurrentMidiProgram = gCarla.host.get_current_midi_program_index(self.fPluginId)
  695. self.ui.cb_midi_programs.setCurrentIndex(self.fCurrentMidiProgram)
  696. else:
  697. self.fCurrentMidiProgram = -1
  698. self.ui.cb_midi_programs.setEnabled(False)
  699. self.ui.label_midi_programs.setEnabled(False)
  700. self.ui.cb_midi_programs.blockSignals(False)
  701. self.ui.sw_programs.setEnabled(programCount > 0 or midiProgramCount > 0)
  702. if self.fPluginInfo['type'] == PLUGIN_LV2:
  703. self.ui.b_load_state.setEnabled(programCount > 0)
  704. #------------------------------------------------------------------
  705. def clearNotes(self):
  706. self.fPlayingNotes = []
  707. self.ui.keyboard.allNotesOff()
  708. #------------------------------------------------------------------
  709. def getHints(self):
  710. return self.fPluginInfo['hints']
  711. def setId(self, idx):
  712. self.fPluginId = idx
  713. def setName(self, name):
  714. self.fPluginInfo['name'] = name
  715. self.ui.label_plugin.setText("\n%s\n" % name)
  716. self.setWindowTitle(name)
  717. #------------------------------------------------------------------
  718. def setParameterValue(self, parameterId, value):
  719. for paramItem in self.fParametersToUpdate:
  720. if paramItem[0] == parameterId:
  721. paramItem[1] = value
  722. break
  723. else:
  724. self.fParametersToUpdate.append([parameterId, value])
  725. def setParameterDefault(self, parameterId, value):
  726. for paramType, paramId, paramWidget in self.fParameterList:
  727. if paramId == parameterId:
  728. paramWidget.setDefault(value)
  729. break
  730. def setParameterMidiControl(self, parameterId, control):
  731. for paramType, paramId, paramWidget in self.fParameterList:
  732. if paramId == parameterId:
  733. paramWidget.setMidiControl(control)
  734. break
  735. def setParameterMidiChannel(self, parameterId, channel):
  736. for paramType, paramId, paramWidget in self.fParameterList:
  737. if paramId == parameterId:
  738. paramWidget.setMidiChannel(channel+1)
  739. break
  740. def setProgram(self, index):
  741. self.ui.cb_programs.blockSignals(True)
  742. self.ui.cb_programs.setCurrentIndex(index)
  743. self.ui.cb_programs.blockSignals(False)
  744. def setMidiProgram(self, index):
  745. self.ui.cb_midi_programs.blockSignals(True)
  746. self.ui.cb_midi_programs.setCurrentIndex(index)
  747. self.ui.cb_midi_programs.blockSignals(False)
  748. #------------------------------------------------------------------
  749. def sendNoteOn(self, channel, note):
  750. if self.fControlChannel == channel:
  751. self.ui.keyboard.sendNoteOn(note, False)
  752. playItem = (channel, note)
  753. if playItem not in self.fPlayingNotes:
  754. self.fPlayingNotes.append(playItem)
  755. return bool(len(self.fPlayingNotes) == 1)
  756. def sendNoteOff(self, channel, note):
  757. if self.fControlChannel == channel:
  758. self.ui.keyboard.sendNoteOff(note, False)
  759. playItem = (channel, note)
  760. if playItem in self.fPlayingNotes:
  761. self.fPlayingNotes.remove(playItem)
  762. return bool(len(self.fPlayingNotes) == 0)
  763. #------------------------------------------------------------------
  764. def setVisible(self, yesNo):
  765. if yesNo:
  766. if not self.fGeometry.isNull():
  767. self.restoreGeometry(self.fGeometry)
  768. else:
  769. self.fGeometry = self.saveGeometry()
  770. QDialog.setVisible(self, yesNo)
  771. #------------------------------------------------------------------
  772. def idleSlow(self):
  773. # Check Tab icons
  774. for i in range(len(self.fTabIconTimers)):
  775. if self.fTabIconTimers[i] == ICON_STATE_ON:
  776. self.fTabIconTimers[i] = ICON_STATE_WAIT
  777. elif self.fTabIconTimers[i] == ICON_STATE_WAIT:
  778. self.fTabIconTimers[i] = ICON_STATE_OFF
  779. elif self.fTabIconTimers[i] == ICON_STATE_OFF:
  780. self.fTabIconTimers[i] = ICON_STATE_NULL
  781. self.ui.tabWidget.setTabIcon(i+1, self.fTabIconOff)
  782. # Check parameters needing update
  783. for index, value in self.fParametersToUpdate:
  784. if index == PARAMETER_DRYWET:
  785. self.ui.dial_drywet.blockSignals(True)
  786. self.ui.dial_drywet.setValue(value)
  787. self.ui.dial_drywet.blockSignals(False)
  788. elif index == PARAMETER_VOLUME:
  789. self.ui.dial_vol.blockSignals(True)
  790. self.ui.dial_vol.setValue(value)
  791. self.ui.dial_vol.blockSignals(False)
  792. elif index == PARAMETER_BALANCE_LEFT:
  793. self.ui.dial_b_left.blockSignals(True)
  794. self.ui.dial_b_left.setValue(value)
  795. self.ui.dial_b_left.blockSignals(False)
  796. elif index == PARAMETER_BALANCE_RIGHT:
  797. self.ui.dial_b_right.blockSignals(True)
  798. self.ui.dial_b_right.setValue(value)
  799. self.ui.dial_b_right.blockSignals(False)
  800. elif index == PARAMETER_PANNING:
  801. self.ui.dial_pan.blockSignals(True)
  802. self.ui.dial_pan.setValue(value)
  803. self.ui.dial_pan.blockSignals(False)
  804. elif index == PARAMETER_CTRL_CHANNEL:
  805. self.fControlChannel = int(value)
  806. self.ui.sb_ctrl_channel.blockSignals(True)
  807. self.ui.sb_ctrl_channel.setValue(self.fControlChannel+1)
  808. self.ui.sb_ctrl_channel.blockSignals(False)
  809. self.ui.keyboard.allNotesOff()
  810. self._updateCtrlMidiProgram()
  811. elif index >= 0:
  812. for paramType, paramId, paramWidget in self.fParameterList:
  813. if paramId != index:
  814. continue
  815. paramWidget.setValue(value, False)
  816. if paramType == PARAMETER_INPUT:
  817. tabIndex = paramWidget.getTabIndex()
  818. if self.fTabIconTimers[tabIndex-1] == ICON_STATE_NULL:
  819. self.ui.tabWidget.setTabIcon(tabIndex, self.fTabIconOn)
  820. self.fTabIconTimers[tabIndex-1] = ICON_STATE_ON
  821. break
  822. # Clear all parameters
  823. self.fParametersToUpdate = []
  824. # Update parameter outputs
  825. for paramType, paramId, paramWidget in self.fParameterList:
  826. if paramType == PARAMETER_OUTPUT:
  827. value = gCarla.host.get_current_parameter_value(self.fPluginId, paramId)
  828. paramWidget.setValue(value, False)
  829. #------------------------------------------------------------------
  830. @pyqtSlot()
  831. def slot_stateSave(self):
  832. if self.fPluginInfo['type'] == PLUGIN_LV2:
  833. # TODO
  834. return
  835. if self.fCurrentStateFilename:
  836. askTry = QMessageBox.question(self, self.tr("Overwrite?"), self.tr("Overwrite previously created file?"), QMessageBox.Ok|QMessageBox.Cancel)
  837. if askTry == QMessageBox.Ok:
  838. gCarla.host.save_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  839. return
  840. self.fCurrentStateFilename = None
  841. fileFilter = self.tr("Carla State File (*.carxs)")
  842. filenameTry = QFileDialog.getSaveFileName(self, self.tr("Save Plugin State File"), filter=fileFilter)
  843. if filenameTry:
  844. if not filenameTry.lower().endswith(".carxs"):
  845. filenameTry += ".carxs"
  846. self.fCurrentStateFilename = filenameTry
  847. gCarla.host.save_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  848. @pyqtSlot()
  849. def slot_stateLoad(self):
  850. if self.fPluginInfo['type'] == PLUGIN_LV2:
  851. presetList = []
  852. for i in range(gCarla.host.get_program_count(self.fPluginId)):
  853. presetList.append("%03i - %s" % (i+1, gCarla.host.get_program_name(self.fPluginId, i)))
  854. ret = QInputDialog.getItem(self, self.tr("Open LV2 Preset"), self.tr("Select an LV2 Preset:"), presetList, 0, False)
  855. if ret[1]:
  856. index = int(ret[0].split(" - ", 1)[0])-1
  857. gCarla.host.set_midi_program(self.fPluginId, -1)
  858. gCarla.host.set_program(self.fPluginId, index)
  859. self.setMidiProgram(-1)
  860. return
  861. fileFilter = self.tr("Carla State File (*.carxs)")
  862. filenameTry = QFileDialog.getOpenFileName(self, self.tr("Open Plugin State File"), filter=fileFilter)
  863. if filenameTry:
  864. self.fCurrentStateFilename = filenameTry
  865. gCarla.host.load_plugin_state(self.fPluginId, self.fCurrentStateFilename)
  866. #------------------------------------------------------------------
  867. @pyqtSlot(bool)
  868. def slot_optionChanged(self, clicked):
  869. if gCarla.host is None:
  870. return
  871. sender = self.sender()
  872. if sender == self.ui.ch_fixed_buffer:
  873. option = PLUGIN_OPTION_FIXED_BUFFERS
  874. elif sender == self.ui.ch_force_stereo:
  875. option = PLUGIN_OPTION_FORCE_STEREO
  876. elif sender == self.ui.ch_map_program_changes:
  877. option = PLUGIN_OPTION_MAP_PROGRAM_CHANGES
  878. elif sender == self.ui.ch_use_chunks:
  879. option = PLUGIN_OPTION_USE_CHUNKS
  880. elif sender == self.ui.ch_send_control_changes:
  881. option = PLUGIN_OPTION_SEND_CONTROL_CHANGES
  882. elif sender == self.ui.ch_send_channel_pressure:
  883. option = PLUGIN_OPTION_SEND_CHANNEL_PRESSURE
  884. elif sender == self.ui.ch_send_note_aftertouch:
  885. option = PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH
  886. elif sender == self.ui.ch_send_pitchbend:
  887. option = PLUGIN_OPTION_SEND_PITCHBEND
  888. elif sender == self.ui.ch_send_all_sound_off:
  889. option = PLUGIN_OPTION_SEND_ALL_SOUND_OFF
  890. else:
  891. return
  892. gCarla.host.set_option(self.fPluginId, option, clicked)
  893. #------------------------------------------------------------------
  894. @pyqtSlot(int)
  895. def slot_dryWetChanged(self, value):
  896. if gCarla.host is not None:
  897. gCarla.host.set_drywet(self.fPluginId, value)
  898. if self.fRealParent is not None:
  899. self.fRealParent.parameterValueChanged(PARAMETER_DRYWET, value)
  900. @pyqtSlot(int)
  901. def slot_volumeChanged(self, value):
  902. if gCarla.host is not None:
  903. gCarla.host.set_volume(self.fPluginId, value)
  904. if self.fRealParent is not None:
  905. self.fRealParent.parameterValueChanged(PARAMETER_VOLUME, value)
  906. @pyqtSlot(int)
  907. def slot_balanceLeftChanged(self, value):
  908. if gCarla.host is not None:
  909. gCarla.host.set_balance_left(self.fPluginId, value)
  910. if self.fRealParent is not None:
  911. self.fRealParent.parameterValueChanged(PARAMETER_BALANCE_LEFT, value)
  912. @pyqtSlot(int)
  913. def slot_balanceRightChanged(self, value):
  914. if gCarla.host is not None:
  915. gCarla.host.set_balance_right(self.fPluginId, value)
  916. if self.fRealParent is not None:
  917. self.fRealParent.parameterValueChanged(PARAMETER_BALANCE_RIGHT, value)
  918. @pyqtSlot(int)
  919. def slot_panChanged(self, value):
  920. if gCarla.host is not None:
  921. gCarla.host.set_panning(self.fPluginId, value)
  922. if self.fRealParent is not None:
  923. self.fRealParent.parameterValueChanged(PARAMETER_PANNING, value)
  924. @pyqtSlot(int)
  925. def slot_ctrlChannelChanged(self, value):
  926. self.fControlChannel = value-1
  927. if gCarla.host is not None:
  928. gCarla.host.set_ctrl_channel(self.fPluginId, self.fControlChannel)
  929. self.ui.keyboard.allNotesOff()
  930. self._updateCtrlMidiProgram()
  931. #------------------------------------------------------------------
  932. @pyqtSlot(int, float)
  933. def slot_parameterValueChanged(self, parameterId, value):
  934. if gCarla.host is not None:
  935. gCarla.host.set_parameter_value(self.fPluginId, parameterId, value)
  936. if self.fRealParent is not None:
  937. self.fRealParent.parameterValueChanged(parameterId, value)
  938. @pyqtSlot(int, int)
  939. def slot_parameterMidiControlChanged(self, parameterId, control):
  940. if gCarla.host is not None:
  941. gCarla.host.set_parameter_midi_cc(self.fPluginId, parameterId, control)
  942. @pyqtSlot(int, int)
  943. def slot_parameterMidiChannelChanged(self, parameterId, channel):
  944. if gCarla.host is not None:
  945. gCarla.host.set_parameter_midi_channel(self.fPluginId, parameterId, channel-1)
  946. #------------------------------------------------------------------
  947. @pyqtSlot(int)
  948. def slot_programIndexChanged(self, index):
  949. self.fCurrentProgram = index
  950. if gCarla.host is not None:
  951. gCarla.host.set_program(self.fPluginId, index)
  952. if self.fRealParent is not None:
  953. self.fRealParent.programChanged(index)
  954. @pyqtSlot(int)
  955. def slot_midiProgramIndexChanged(self, index):
  956. self.fCurrentMidiProgram = index
  957. if gCarla.host is not None:
  958. gCarla.host.set_midi_program(self.fPluginId, index)
  959. if self.fRealParent is not None:
  960. self.fRealParent.midiProgramChanged(index)
  961. #------------------------------------------------------------------
  962. @pyqtSlot(int)
  963. def slot_noteOn(self, note):
  964. if self.fControlChannel >= 0 and gCarla.host is not None:
  965. gCarla.host.send_midi_note(self.fPluginId, self.fControlChannel, note, 100)
  966. if self.fRealParent is not None:
  967. self.fRealParent.notePressed(note)
  968. @pyqtSlot(int)
  969. def slot_noteOff(self, note):
  970. if self.fControlChannel >= 0 and gCarla.host is not None:
  971. gCarla.host.send_midi_note(self.fPluginId, self.fControlChannel, note, 0)
  972. if self.fRealParent is not None:
  973. self.fRealParent.noteReleased(note)
  974. #------------------------------------------------------------------
  975. @pyqtSlot()
  976. def slot_finished(self):
  977. if self.fRealParent is not None:
  978. self.fRealParent.editDialogChanged(False)
  979. #------------------------------------------------------------------
  980. @pyqtSlot()
  981. def slot_knobCustomMenu(self):
  982. knobName = self.sender().objectName()
  983. if knobName == "dial_drywet":
  984. minimum = 0.0
  985. maximum = 1.0
  986. default = 1.0
  987. label = "Dry/Wet"
  988. elif knobName == "dial_vol":
  989. minimum = 0.0
  990. maximum = 1.27
  991. default = 1.0
  992. label = "Volume"
  993. elif knobName == "dial_b_left":
  994. minimum = -1.0
  995. maximum = 1.0
  996. default = -1.0
  997. label = "Balance-Left"
  998. elif knobName == "dial_b_right":
  999. minimum = -1.0
  1000. maximum = 1.0
  1001. default = 1.0
  1002. label = "Balance-Right"
  1003. elif knobName == "dial_pan":
  1004. minimum = -1.0
  1005. maximum = 1.0
  1006. default = 0.0
  1007. label = "Panning"
  1008. else:
  1009. minimum = 0.0
  1010. maximum = 1.0
  1011. default = 0.5
  1012. label = "Unknown"
  1013. current = self.sender().value() / 10
  1014. menu = QMenu(self)
  1015. actReset = menu.addAction(self.tr("Reset (%i%%)" % (default*100)))
  1016. menu.addSeparator()
  1017. actMinimum = menu.addAction(self.tr("Set to Minimum (%i%%)" % (minimum*100)))
  1018. actCenter = menu.addAction(self.tr("Set to Center"))
  1019. actMaximum = menu.addAction(self.tr("Set to Maximum (%i%%)" % (maximum*100)))
  1020. menu.addSeparator()
  1021. actSet = menu.addAction(self.tr("Set value..."))
  1022. if label not in ("Balance-Left", "Balance-Right"):
  1023. menu.removeAction(actCenter)
  1024. actSelected = menu.exec_(QCursor.pos())
  1025. if actSelected == actSet:
  1026. valueTry = QInputDialog.getDouble(self, self.tr("Set value"), label, current, minimum, maximum, 3)
  1027. if valueTry[1]:
  1028. value = valueTry[0] * 10
  1029. else:
  1030. return
  1031. elif actSelected == actMinimum:
  1032. value = minimum
  1033. elif actSelected == actMaximum:
  1034. value = maximum
  1035. elif actSelected == actReset:
  1036. value = default
  1037. elif actSelected == actCenter:
  1038. value = 0.0
  1039. else:
  1040. return
  1041. self.sender().setValue(value)
  1042. #------------------------------------------------------------------
  1043. @pyqtSlot()
  1044. def slot_channelCustomMenu(self):
  1045. menu = QMenu(self)
  1046. actNone = menu.addAction(self.tr("None"))
  1047. if self.fControlChannel+1 == 0:
  1048. actNone.setCheckable(True)
  1049. actNone.setChecked(True)
  1050. for i in range(1, 16+1):
  1051. action = menu.addAction("%i" % i)
  1052. if self.fControlChannel+1 == i:
  1053. action.setCheckable(True)
  1054. action.setChecked(True)
  1055. actSel = menu.exec_(QCursor.pos())
  1056. if not actSel:
  1057. pass
  1058. elif actSel == actNone:
  1059. self.ui.sb_ctrl_channel.setValue(0)
  1060. elif actSel:
  1061. selChannel = int(actSel.text())
  1062. self.ui.sb_ctrl_channel.setValue(selChannel)
  1063. #------------------------------------------------------------------
  1064. def _createParameterWidgets(self, paramType, paramListFull, tabPageName):
  1065. i = 1
  1066. for paramList, width in paramListFull:
  1067. if len(paramList) == 0:
  1068. break
  1069. tabIndex = self.ui.tabWidget.count()
  1070. tabPageContainer = QWidget(self.ui.tabWidget)
  1071. tabPageLayout = QVBoxLayout(tabPageContainer)
  1072. tabPageContainer.setLayout(tabPageLayout)
  1073. for paramInfo in paramList:
  1074. paramWidget = PluginParameter(tabPageContainer, paramInfo, self.fPluginId, tabIndex)
  1075. paramWidget.setLabelWidth(width)
  1076. tabPageLayout.addWidget(paramWidget)
  1077. self.fParameterList.append((paramType, paramInfo['index'], paramWidget))
  1078. if paramType == PARAMETER_INPUT:
  1079. paramWidget.valueChanged.connect(self.slot_parameterValueChanged)
  1080. paramWidget.midiControlChanged.connect(self.slot_parameterMidiControlChanged)
  1081. paramWidget.midiChannelChanged.connect(self.slot_parameterMidiChannelChanged)
  1082. tabPageLayout.addStretch()
  1083. self.ui.tabWidget.addTab(tabPageContainer, "%s (%i)" % (tabPageName, i))
  1084. i += 1
  1085. if paramType == PARAMETER_INPUT:
  1086. self.ui.tabWidget.setTabIcon(tabIndex, self.fTabIconOff)
  1087. self.fTabIconTimers.append(ICON_STATE_NULL)
  1088. def _updateCtrlMidiProgram(self):
  1089. if self.fPluginInfo['type'] not in (PLUGIN_INTERNAL, PLUGIN_SF2):
  1090. return
  1091. elif self.fPluginInfo['category'] != PLUGIN_CATEGORY_SYNTH:
  1092. return
  1093. if self.fControlChannel < 0:
  1094. self.ui.cb_midi_programs.setEnabled(False)
  1095. return
  1096. self.ui.cb_midi_programs.setEnabled(True)
  1097. mpIndex = gCarla.host.get_current_midi_program_index(self.fPluginId)
  1098. if self.ui.cb_midi_programs.currentIndex() != mpIndex:
  1099. self.setMidiProgram(mpIndex)
  1100. #------------------------------------------------------------------
  1101. def done(self, r):
  1102. QDialog.done(self, r)
  1103. self.close()
  1104. # ------------------------------------------------------------------------------------------------------------
  1105. # Main
  1106. if __name__ == '__main__':
  1107. from carla_style import CarlaApplication
  1108. app = CarlaApplication()
  1109. gui1 = CarlaAboutW(None)
  1110. gui1.show()
  1111. gui2 = PluginParameter(None, gFakeParamInfo, 0, 0)
  1112. gui2.show()
  1113. gui3 = PluginEdit(None, 0)
  1114. gui3.show()
  1115. app.exit_exec()