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.

1659 lines
66KB

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