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.

carla_widgets.py 56KB

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