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.

881 lines
30KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla patchbay widget 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 QPointF, QTimer
  20. from PyQt4.QtGui import QFrame, QGraphicsView, QImage, QPrinter, QPrintDialog, QVBoxLayout
  21. # ------------------------------------------------------------------------------------------------------------
  22. # Imports (Custom Stuff)
  23. import patchcanvas
  24. from carla_widgets import *
  25. from pixmapkeyboard import PixmapKeyboardHArea
  26. # ------------------------------------------------------------------------------------------------------------
  27. # Try Import OpenGL
  28. try:
  29. from PyQt4.QtOpenGL import QGLWidget
  30. hasGL = True
  31. except:
  32. hasGL = False
  33. # ------------------------------------------------------------------------------------------------------------
  34. # Carla Canvas defaults
  35. CARLA_DEFAULT_CANVAS_SIZE_WIDTH = 3100
  36. CARLA_DEFAULT_CANVAS_SIZE_HEIGHT = 2400
  37. # ------------------------------------------------------------------------------------------------
  38. # Patchbay widget
  39. class CarlaPatchbayW(QFrame):
  40. def __init__(self, parent, doSetup = True):
  41. QFrame.__init__(self, parent)
  42. self.fLayout = QVBoxLayout(self)
  43. self.fLayout.setContentsMargins(0, 0, 0, 0)
  44. self.fLayout.setSpacing(1)
  45. self.setLayout(self.fLayout)
  46. self.fView = QGraphicsView(self)
  47. self.fKeys = PixmapKeyboardHArea(self)
  48. self.fLayout.addWidget(self.fView)
  49. self.fLayout.addWidget(self.fKeys)
  50. # -------------------------------------------------------------
  51. # Internal stuff
  52. self.fParent = parent
  53. self.fPluginCount = 0
  54. self.fPluginList = []
  55. self.fCanvasWidth = 0
  56. self.fCanvasHeight = 0
  57. # -------------------------------------------------------------
  58. # Set-up Canvas Preview
  59. self.fMiniCanvasPreview = self.fParent.ui.miniCanvasPreview
  60. self.fMiniCanvasPreview.setRealParent(self)
  61. self.fMovingViaMiniCanvas = False
  62. # -------------------------------------------------------------
  63. # Set-up Canvas
  64. self.scene = patchcanvas.PatchScene(self, self.fView)
  65. self.fView.setScene(self.scene)
  66. self.fView.setRenderHint(QPainter.Antialiasing, bool(parent.fSavedSettings[CARLA_KEY_CANVAS_ANTIALIASING] == patchcanvas.ANTIALIASING_FULL))
  67. if parent.fSavedSettings[CARLA_KEY_CANVAS_USE_OPENGL] and hasGL:
  68. self.fView.setViewport(QGLWidget(self))
  69. self.fView.setRenderHint(QPainter.HighQualityAntialiasing, parent.fSavedSettings[CARLA_KEY_CANVAS_HQ_ANTIALIASING])
  70. self.setupCanvas()
  71. QTimer.singleShot(100, self.slot_restoreScrollbarValues)
  72. # -------------------------------------------------------------
  73. # Connect actions to functions
  74. self.fView.horizontalScrollBar().valueChanged.connect(self.slot_horizontalScrollBarChanged)
  75. self.fView.verticalScrollBar().valueChanged.connect(self.slot_verticalScrollBarChanged)
  76. self.scene.scaleChanged.connect(self.slot_canvasScaleChanged)
  77. self.scene.sceneGroupMoved.connect(self.slot_canvasItemMoved)
  78. self.scene.pluginSelected.connect(self.slot_canvasPluginSelected)
  79. self.fMiniCanvasPreview.miniCanvasMoved.connect(self.slot_miniCanvasMoved)
  80. if not doSetup: return
  81. parent.ui.act_plugins_enable.triggered.connect(self.slot_pluginsEnable)
  82. parent.ui.act_plugins_disable.triggered.connect(self.slot_pluginsDisable)
  83. parent.ui.act_plugins_volume100.triggered.connect(self.slot_pluginsVolume100)
  84. parent.ui.act_plugins_mute.triggered.connect(self.slot_pluginsMute)
  85. parent.ui.act_plugins_wet100.triggered.connect(self.slot_pluginsWet100)
  86. parent.ui.act_plugins_bypass.triggered.connect(self.slot_pluginsBypass)
  87. parent.ui.act_plugins_center.triggered.connect(self.slot_pluginsCenter)
  88. parent.ui.act_plugins_panic.triggered.connect(self.slot_pluginsDisable)
  89. parent.ui.act_canvas_arrange.setEnabled(False) # TODO, later
  90. parent.ui.act_canvas_arrange.triggered.connect(self.slot_canvasArrange)
  91. parent.ui.act_canvas_refresh.triggered.connect(self.slot_canvasRefresh)
  92. parent.ui.act_canvas_zoom_fit.triggered.connect(self.slot_canvasZoomFit)
  93. parent.ui.act_canvas_zoom_in.triggered.connect(self.slot_canvasZoomIn)
  94. parent.ui.act_canvas_zoom_out.triggered.connect(self.slot_canvasZoomOut)
  95. parent.ui.act_canvas_zoom_100.triggered.connect(self.slot_canvasZoomReset)
  96. parent.ui.act_canvas_print.triggered.connect(self.slot_canvasPrint)
  97. parent.ui.act_canvas_save_image.triggered.connect(self.slot_canvasSaveImage)
  98. parent.ui.act_settings_configure.triggered.connect(self.slot_configureCarla)
  99. parent.ParameterValueChangedCallback.connect(self.slot_handleParameterValueChangedCallback)
  100. parent.ParameterDefaultChangedCallback.connect(self.slot_handleParameterDefaultChangedCallback)
  101. parent.ParameterMidiChannelChangedCallback.connect(self.slot_handleParameterMidiChannelChangedCallback)
  102. parent.ParameterMidiCcChangedCallback.connect(self.slot_handleParameterMidiCcChangedCallback)
  103. parent.ProgramChangedCallback.connect(self.slot_handleProgramChangedCallback)
  104. parent.MidiProgramChangedCallback.connect(self.slot_handleMidiProgramChangedCallback)
  105. parent.NoteOnCallback.connect(self.slot_handleNoteOnCallback)
  106. parent.NoteOffCallback.connect(self.slot_handleNoteOffCallback)
  107. parent.UpdateCallback.connect(self.slot_handleUpdateCallback)
  108. parent.ReloadInfoCallback.connect(self.slot_handleReloadInfoCallback)
  109. parent.ReloadParametersCallback.connect(self.slot_handleReloadParametersCallback)
  110. parent.ReloadProgramsCallback.connect(self.slot_handleReloadProgramsCallback)
  111. parent.ReloadAllCallback.connect(self.slot_handleReloadAllCallback)
  112. parent.PatchbayClientAddedCallback.connect(self.slot_handlePatchbayClientAddedCallback)
  113. parent.PatchbayClientRemovedCallback.connect(self.slot_handlePatchbayClientRemovedCallback)
  114. parent.PatchbayClientRenamedCallback.connect(self.slot_handlePatchbayClientRenamedCallback)
  115. parent.PatchbayClientDataChangedCallback.connect(self.slot_handlePatchbayClientDataChangedCallback)
  116. parent.PatchbayPortAddedCallback.connect(self.slot_handlePatchbayPortAddedCallback)
  117. parent.PatchbayPortRemovedCallback.connect(self.slot_handlePatchbayPortRemovedCallback)
  118. parent.PatchbayPortRenamedCallback.connect(self.slot_handlePatchbayPortRenamedCallback)
  119. parent.PatchbayConnectionAddedCallback.connect(self.slot_handlePatchbayConnectionAddedCallback)
  120. parent.PatchbayConnectionRemovedCallback.connect(self.slot_handlePatchbayConnectionRemovedCallback)
  121. # -----------------------------------------------------------------
  122. def getPluginCount(self):
  123. return self.fPluginCount
  124. # -----------------------------------------------------------------
  125. def addPlugin(self, pluginId, isProjectLoading):
  126. pitem = PluginEdit(self, pluginId)
  127. self.fPluginList.append(pitem)
  128. self.fPluginCount += 1
  129. if not isProjectLoading:
  130. Carla.host.set_active(pluginId, True)
  131. def removePlugin(self, pluginId):
  132. if pluginId >= self.fPluginCount:
  133. return
  134. pitem = self.fPluginList[pluginId]
  135. if pitem is None:
  136. return
  137. self.fPluginCount -= 1
  138. self.fPluginList.pop(pluginId)
  139. pitem.close()
  140. del pitem
  141. # push all plugins 1 slot back
  142. for i in range(pluginId, self.fPluginCount):
  143. pitem = self.fPluginList[i]
  144. pitem.setId(i)
  145. def renamePlugin(self, pluginId, newName):
  146. if pluginId >= self.fPluginCount:
  147. return
  148. pitem = self.fPluginList[pluginId]
  149. if pitem is None:
  150. return
  151. pitem.setName(newName)
  152. def disablePlugin(self, pluginId, errorMsg):
  153. if pluginId >= self.fPluginCount:
  154. return
  155. pitem = self.fPluginList[pluginId]
  156. if pitem is None:
  157. return
  158. def removeAllPlugins(self):
  159. for i in range(self.fPluginCount):
  160. pitem = self.fPluginList[i]
  161. if pitem is None:
  162. break
  163. pitem.close()
  164. del pitem
  165. self.fPluginCount = 0
  166. self.fPluginList = []
  167. # -----------------------------------------------------------------
  168. def engineStarted(self):
  169. pass
  170. def engineStopped(self):
  171. patchcanvas.clear()
  172. def engineChanged(self):
  173. pass
  174. # -----------------------------------------------------------------
  175. def idleFast(self):
  176. pass
  177. def idleSlow(self):
  178. for i in range(self.fPluginCount):
  179. pitem = self.fPluginList[i]
  180. if pitem is None:
  181. break
  182. pitem.idleSlow()
  183. # -----------------------------------------------------------------
  184. def saveSettings(self, settings):
  185. settings.setValue("HorizontalScrollBarValue", self.fView.horizontalScrollBar().value())
  186. settings.setValue("VerticalScrollBarValue", self.fView.verticalScrollBar().value())
  187. # -----------------------------------------------------------------
  188. # called by PluginEdit, ignored here
  189. def recheckPluginHints(self, hints):
  190. pass
  191. # -----------------------------------------------------------------
  192. def setupCanvas(self):
  193. pOptions = patchcanvas.options_t()
  194. pOptions.theme_name = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_THEME]
  195. pOptions.auto_hide_groups = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS]
  196. pOptions.use_bezier_lines = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_USE_BEZIER_LINES]
  197. pOptions.antialiasing = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_ANTIALIASING]
  198. pOptions.eyecandy = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_EYE_CANDY]
  199. pFeatures = patchcanvas.features_t()
  200. pFeatures.group_info = False
  201. pFeatures.group_rename = False
  202. pFeatures.port_info = False
  203. pFeatures.port_rename = False
  204. pFeatures.handle_group_pos = True
  205. patchcanvas.setOptions(pOptions)
  206. patchcanvas.setFeatures(pFeatures)
  207. patchcanvas.init("Carla2", self.scene, canvasCallback, False)
  208. tryCanvasSize = self.fParent.fSavedSettings[CARLA_KEY_CANVAS_SIZE].split("x")
  209. if len(tryCanvasSize) == 2 and tryCanvasSize[0].isdigit() and tryCanvasSize[1].isdigit():
  210. self.fCanvasWidth = int(tryCanvasSize[0])
  211. self.fCanvasHeight = int(tryCanvasSize[1])
  212. else:
  213. self.fCanvasWidth = CARLA_DEFAULT_CANVAS_SIZE_WIDTH
  214. self.fCanvasHeight = CARLA_DEFAULT_CANVAS_SIZE_HEIGHT
  215. patchcanvas.setCanvasSize(0, 0, self.fCanvasWidth, self.fCanvasHeight)
  216. patchcanvas.setInitialPos(self.fCanvasWidth / 2, self.fCanvasHeight / 2)
  217. self.fView.setSceneRect(0, 0, self.fCanvasWidth, self.fCanvasHeight)
  218. self.themeData = [self.fCanvasWidth, self.fCanvasHeight, patchcanvas.canvas.theme.canvas_bg, patchcanvas.canvas.theme.rubberband_brush, patchcanvas.canvas.theme.rubberband_pen.color()]
  219. def updateCanvasInitialPos(self):
  220. x = self.fView.horizontalScrollBar().value() + self.width()/4
  221. y = self.fView.verticalScrollBar().value() + self.height()/4
  222. patchcanvas.setInitialPos(x, y)
  223. # -----------------------------------------------------------------
  224. @pyqtSlot()
  225. def slot_miniCanvasCheckAll(self):
  226. self.slot_miniCanvasCheckSize()
  227. self.slot_horizontalScrollBarChanged(self.fView.horizontalScrollBar().value())
  228. self.slot_verticalScrollBarChanged(self.fView.verticalScrollBar().value())
  229. @pyqtSlot()
  230. def slot_miniCanvasCheckSize(self):
  231. self.fMiniCanvasPreview.setViewSize(float(self.width()) / self.fCanvasWidth, float(self.height()) / self.fCanvasHeight)
  232. @pyqtSlot(int)
  233. def slot_horizontalScrollBarChanged(self, value):
  234. if self.fMovingViaMiniCanvas: return
  235. maximum = self.fView.horizontalScrollBar().maximum()
  236. if maximum == 0:
  237. xp = 0
  238. else:
  239. xp = float(value) / maximum
  240. self.fMiniCanvasPreview.setViewPosX(xp)
  241. self.updateCanvasInitialPos()
  242. @pyqtSlot(int)
  243. def slot_verticalScrollBarChanged(self, value):
  244. if self.fMovingViaMiniCanvas: return
  245. maximum = self.fView.verticalScrollBar().maximum()
  246. if maximum == 0:
  247. yp = 0
  248. else:
  249. yp = float(value) / maximum
  250. self.fMiniCanvasPreview.setViewPosY(yp)
  251. self.updateCanvasInitialPos()
  252. @pyqtSlot()
  253. def slot_restoreScrollbarValues(self):
  254. settings = QSettings()
  255. self.fView.horizontalScrollBar().setValue(settings.value("HorizontalScrollBarValue", self.fView.horizontalScrollBar().maximum()/2, type=int))
  256. self.fView.verticalScrollBar().setValue(settings.value("VerticalScrollBarValue", self.fView.verticalScrollBar().maximum()/2, type=int))
  257. # -----------------------------------------------------------------
  258. @pyqtSlot(float)
  259. def slot_canvasScaleChanged(self, scale):
  260. self.fMiniCanvasPreview.setViewScale(scale)
  261. @pyqtSlot(int, int, QPointF)
  262. def slot_canvasItemMoved(self, group_id, split_mode, pos):
  263. self.fMiniCanvasPreview.update()
  264. @pyqtSlot(list)
  265. def slot_canvasPluginSelected(self, pluginList):
  266. self.fKeys.setEnabled(len(pluginList) != 0)
  267. @pyqtSlot(float, float)
  268. def slot_miniCanvasMoved(self, xp, yp):
  269. self.fMovingViaMiniCanvas = True
  270. self.fView.horizontalScrollBar().setValue(xp * self.fView.horizontalScrollBar().maximum())
  271. self.fView.verticalScrollBar().setValue(yp * self.fView.verticalScrollBar().maximum())
  272. self.fMovingViaMiniCanvas = False
  273. self.updateCanvasInitialPos()
  274. # -----------------------------------------------------------------
  275. @pyqtSlot()
  276. def slot_pluginsEnable(self):
  277. if not Carla.host.is_engine_running():
  278. return
  279. for i in range(self.fPluginCount):
  280. Carla.host.set_active(i, True)
  281. @pyqtSlot()
  282. def slot_pluginsDisable(self):
  283. if not Carla.host.is_engine_running():
  284. return
  285. for i in range(self.fPluginCount):
  286. Carla.host.set_active(i, False)
  287. @pyqtSlot()
  288. def slot_pluginsVolume100(self):
  289. if not Carla.host.is_engine_running():
  290. return
  291. for i in range(self.fPluginCount):
  292. pitem = self.fPluginList[i]
  293. if pitem is None:
  294. break
  295. if pitem.getHints() & PLUGIN_CAN_VOLUME:
  296. pitem.setParameterValue(PARAMETER_VOLUME, 1.0)
  297. Carla.host.set_volume(i, 1.0)
  298. @pyqtSlot()
  299. def slot_pluginsMute(self):
  300. if not Carla.host.is_engine_running():
  301. return
  302. for i in range(self.fPluginCount):
  303. pitem = self.fPluginList[i]
  304. if pitem is None:
  305. break
  306. if pitem.getHints() & PLUGIN_CAN_VOLUME:
  307. pitem.setParameterValue(PARAMETER_VOLUME, 0.0)
  308. Carla.host.set_volume(i, 0.0)
  309. @pyqtSlot()
  310. def slot_pluginsWet100(self):
  311. if not Carla.host.is_engine_running():
  312. return
  313. for i in range(self.fPluginCount):
  314. pitem = self.fPluginList[i]
  315. if pitem is None:
  316. break
  317. if pitem.getHints() & PLUGIN_CAN_DRYWET:
  318. pitem.setParameterValue(PARAMETER_DRYWET, 1.0)
  319. Carla.host.set_drywet(i, 1.0)
  320. @pyqtSlot()
  321. def slot_pluginsBypass(self):
  322. if not Carla.host.is_engine_running():
  323. return
  324. for i in range(self.fPluginCount):
  325. pitem = self.fPluginList[i]
  326. if pitem is None:
  327. break
  328. if pitem.getHints() & PLUGIN_CAN_DRYWET:
  329. pitem.setParameterValue(PARAMETER_DRYWET, 0.0)
  330. Carla.host.set_drywet(i, 0.0)
  331. @pyqtSlot()
  332. def slot_pluginsCenter(self):
  333. if not Carla.host.is_engine_running():
  334. return
  335. for i in range(self.fPluginCount):
  336. pitem = self.fPluginList[i]
  337. if pitem is None:
  338. break
  339. if pitem.getHints() & PLUGIN_CAN_BALANCE:
  340. pitem.setParameterValue(PARAMETER_BALANCE_LEFT, -1.0)
  341. pitem.setParameterValue(PARAMETER_BALANCE_RIGHT, 1.0)
  342. Carla.host.set_balance_left(i, -1.0)
  343. Carla.host.set_balance_right(i, 1.0)
  344. if pitem.getHints() & PLUGIN_CAN_PANNING:
  345. pitem.setParameterValue(PARAMETER_PANNING, 0.0)
  346. Carla.host.set_panning(i, 0.0)
  347. # -----------------------------------------------------------------
  348. @pyqtSlot()
  349. def slot_configureCarla(self):
  350. if self.fParent is None or not self.fParent.openSettingsWindow(True, hasGL):
  351. return
  352. self.fParent.loadSettings(False)
  353. patchcanvas.clear()
  354. self.setupCanvas()
  355. self.fParent.updateContainer(self.themeData)
  356. self.slot_miniCanvasCheckAll()
  357. if Carla.host.is_engine_running():
  358. Carla.host.patchbay_refresh()
  359. # -----------------------------------------------------------------
  360. @pyqtSlot(int, int, float)
  361. def slot_handleParameterValueChangedCallback(self, pluginId, index, value):
  362. if pluginId >= self.fPluginCount:
  363. return
  364. pitem = self.fPluginList[pluginId]
  365. if pitem is None:
  366. return
  367. pitem.setParameterValue(index, value)
  368. @pyqtSlot(int, int, float)
  369. def slot_handleParameterDefaultChangedCallback(self, pluginId, index, value):
  370. if pluginId >= self.fPluginCount:
  371. return
  372. pitem = self.fPluginList[pluginId]
  373. if pitem is None:
  374. return
  375. pitem.setParameterDefault(index, value)
  376. @pyqtSlot(int, int, int)
  377. def slot_handleParameterMidiCcChangedCallback(self, pluginId, index, cc):
  378. if pluginId >= self.fPluginCount:
  379. return
  380. pitem = self.fPluginList[pluginId]
  381. if pitem is None:
  382. return
  383. pitem.setParameterMidiControl(index, cc)
  384. @pyqtSlot(int, int, int)
  385. def slot_handleParameterMidiChannelChangedCallback(self, pluginId, index, channel):
  386. if pluginId >= self.fPluginCount:
  387. return
  388. pitem = self.fPluginList[pluginId]
  389. if pitem is None:
  390. return
  391. pitem.setParameterMidiChannel(index, channel)
  392. # -----------------------------------------------------------------
  393. @pyqtSlot(int, int)
  394. def slot_handleProgramChangedCallback(self, pluginId, index):
  395. if pluginId >= self.fPluginCount:
  396. return
  397. pitem = self.fPluginList[pluginId]
  398. if pitem is None:
  399. return
  400. pitem.setProgram(index)
  401. @pyqtSlot(int, int)
  402. def slot_handleMidiProgramChangedCallback(self, pluginId, index):
  403. if pluginId >= self.fPluginCount:
  404. return
  405. pitem = self.fPluginList[pluginId]
  406. if pitem is None:
  407. return
  408. pitem.setMidiProgram(index)
  409. # -----------------------------------------------------------------
  410. @pyqtSlot(int, int, int, int)
  411. def slot_handleNoteOnCallback(self, pluginId, channel, note, velo):
  412. if pluginId >= self.fPluginCount:
  413. return
  414. pitem = self.fPluginList[pluginId]
  415. if pitem is None:
  416. return
  417. pitem.sendNoteOn(channel, note)
  418. @pyqtSlot(int, int, int)
  419. def slot_handleNoteOffCallback(self, pluginId, channel, note):
  420. if pluginId >= self.fPluginCount:
  421. return
  422. pitem = self.fPluginList[pluginId]
  423. if pitem is None:
  424. return
  425. pitem.sendNoteOff(channel, note)
  426. # -----------------------------------------------------------------
  427. @pyqtSlot(int)
  428. def slot_handleUpdateCallback(self, pluginId):
  429. if pluginId >= self.fPluginCount:
  430. return
  431. pitem = self.fPluginList[pluginId]
  432. if pitem is None:
  433. return
  434. pitem.updateInfo()
  435. @pyqtSlot(int)
  436. def slot_handleReloadInfoCallback(self, pluginId):
  437. if pluginId >= self.fPluginCount:
  438. return
  439. pitem = self.fPluginList[pluginId]
  440. if pitem is None:
  441. return
  442. pitem.reloadInfo()
  443. @pyqtSlot(int)
  444. def slot_handleReloadParametersCallback(self, pluginId):
  445. if pluginId >= self.fPluginCount:
  446. return
  447. pitem = self.fPluginList[pluginId]
  448. if pitem is None:
  449. return
  450. pitem.reloadParameters()
  451. @pyqtSlot(int)
  452. def slot_handleReloadProgramsCallback(self, pluginId):
  453. if pluginId >= self.fPluginCount:
  454. return
  455. pitem = self.fPluginList[pluginId]
  456. if pitem is None:
  457. return
  458. pitem.reloadPrograms()
  459. @pyqtSlot(int)
  460. def slot_handleReloadAllCallback(self, pluginId):
  461. if pluginId >= self.fPluginCount:
  462. return
  463. pitem = self.fPluginList[pluginId]
  464. if pitem is None:
  465. return
  466. pitem.reloadAll()
  467. # -----------------------------------------------------------------
  468. @pyqtSlot(int, int, int, str)
  469. def slot_handlePatchbayClientAddedCallback(self, clientId, clientIcon, pluginId, clientName):
  470. pcSplit = patchcanvas.SPLIT_UNDEF
  471. pcIcon = patchcanvas.ICON_APPLICATION
  472. if clientIcon == PATCHBAY_ICON_PLUGIN:
  473. pcIcon = patchcanvas.ICON_PLUGIN
  474. if clientIcon == PATCHBAY_ICON_HARDWARE:
  475. pcIcon = patchcanvas.ICON_HARDWARE
  476. elif clientIcon == PATCHBAY_ICON_CARLA:
  477. pass
  478. elif clientIcon == PATCHBAY_ICON_DISTRHO:
  479. pcIcon = patchcanvas.ICON_DISTRHO
  480. elif clientIcon == PATCHBAY_ICON_FILE:
  481. pcIcon = patchcanvas.ICON_FILE
  482. patchcanvas.addGroup(clientId, clientName, pcSplit, pcIcon)
  483. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  484. if pluginId < 0:
  485. return
  486. if pluginId >= self.getPluginCount():
  487. print("sorry, can't map this plugin to canvas client", pluginId, self.getPluginCount())
  488. return
  489. pitem = self.fPluginList[pluginId]
  490. if pitem is None:
  491. return
  492. patchcanvas.setGroupAsPlugin(clientId, pluginId, bool(pitem.fPluginInfo['hints'] & PLUGIN_HAS_CUSTOM_UI))
  493. @pyqtSlot(int)
  494. def slot_handlePatchbayClientRemovedCallback(self, clientId):
  495. #if not self.fEngineStarted: return
  496. patchcanvas.removeGroup(clientId)
  497. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  498. @pyqtSlot(int, str)
  499. def slot_handlePatchbayClientRenamedCallback(self, clientId, newClientName):
  500. patchcanvas.renameGroup(clientId, newClientName)
  501. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  502. @pyqtSlot(int, int, int)
  503. def slot_handlePatchbayClientDataChangedCallback(self, clientId, clientIcon, pluginId):
  504. pcIcon = patchcanvas.ICON_APPLICATION
  505. if clientIcon == PATCHBAY_ICON_PLUGIN:
  506. pcIcon = patchcanvas.ICON_PLUGIN
  507. if clientIcon == PATCHBAY_ICON_HARDWARE:
  508. pcIcon = patchcanvas.ICON_HARDWARE
  509. elif clientIcon == PATCHBAY_ICON_CARLA:
  510. pass
  511. elif clientIcon == PATCHBAY_ICON_DISTRHO:
  512. pcIcon = patchcanvas.ICON_DISTRHO
  513. elif clientIcon == PATCHBAY_ICON_FILE:
  514. pcIcon = patchcanvas.ICON_FILE
  515. patchcanvas.setGroupIcon(clientId, pcIcon)
  516. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  517. if pluginId < 0:
  518. return
  519. if pluginId >= self.getPluginCount():
  520. print("sorry, can't map this plugin to canvas client", pluginId, self.getPluginCount())
  521. return
  522. pitem = self.fPluginList[pluginId]
  523. if pitem is None:
  524. return
  525. patchcanvas.setGroupAsPlugin(clientId, pluginId, bool(pitem.fPluginInfo['hints'] & PLUGIN_HAS_CUSTOM_UI))
  526. @pyqtSlot(int, int, int, str)
  527. def slot_handlePatchbayPortAddedCallback(self, clientId, portId, portFlags, portName):
  528. if (portFlags & PATCHBAY_PORT_IS_INPUT):
  529. portMode = patchcanvas.PORT_MODE_INPUT
  530. else:
  531. portMode = patchcanvas.PORT_MODE_OUTPUT
  532. if (portFlags & PATCHBAY_PORT_TYPE_AUDIO):
  533. portType = patchcanvas.PORT_TYPE_AUDIO_JACK
  534. elif (portFlags & PATCHBAY_PORT_TYPE_CV):
  535. portType = patchcanvas.PORT_TYPE_AUDIO_JACK # TODO
  536. elif (portFlags & PATCHBAY_PORT_TYPE_MIDI):
  537. portType = patchcanvas.PORT_TYPE_MIDI_JACK
  538. else:
  539. portType = patchcanvas.PORT_TYPE_NULL
  540. patchcanvas.addPort(clientId, portId, portName, portMode, portType)
  541. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  542. @pyqtSlot(int, int)
  543. def slot_handlePatchbayPortRemovedCallback(self, groupId, portId):
  544. #if not self.fEngineStarted: return
  545. patchcanvas.removePort(portId)
  546. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  547. @pyqtSlot(int, int, str)
  548. def slot_handlePatchbayPortRenamedCallback(self, groupId, portId, newPortName):
  549. patchcanvas.renamePort(portId, newPortName)
  550. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  551. @pyqtSlot(int, int, int)
  552. def slot_handlePatchbayConnectionAddedCallback(self, connectionId, portOutId, portInId):
  553. patchcanvas.connectPorts(connectionId, portOutId, portInId)
  554. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  555. @pyqtSlot(int, int, int)
  556. def slot_handlePatchbayConnectionRemovedCallback(self, connectionId, portOutId, portInId):
  557. #if not self.fEngineStarted: return
  558. patchcanvas.disconnectPorts(connectionId)
  559. QTimer.singleShot(0, self.fMiniCanvasPreview.update)
  560. # -----------------------------------------------------------------
  561. @pyqtSlot()
  562. def slot_canvasArrange(self):
  563. patchcanvas.arrange()
  564. @pyqtSlot()
  565. def slot_canvasRefresh(self):
  566. patchcanvas.clear()
  567. if Carla.host.is_engine_running():
  568. Carla.host.patchbay_refresh()
  569. QTimer.singleShot(1000 if self.fParent.fSavedSettings[CARLA_KEY_CANVAS_EYE_CANDY] else 0, self.fMiniCanvasPreview.update)
  570. @pyqtSlot()
  571. def slot_canvasZoomFit(self):
  572. self.scene.zoom_fit()
  573. @pyqtSlot()
  574. def slot_canvasZoomIn(self):
  575. self.scene.zoom_in()
  576. @pyqtSlot()
  577. def slot_canvasZoomOut(self):
  578. self.scene.zoom_out()
  579. @pyqtSlot()
  580. def slot_canvasZoomReset(self):
  581. self.scene.zoom_reset()
  582. @pyqtSlot()
  583. def slot_canvasPrint(self):
  584. self.scene.clearSelection()
  585. self.fExportPrinter = QPrinter()
  586. dialog = QPrintDialog(self.fExportPrinter, self)
  587. if dialog.exec_():
  588. painter = QPainter(self.fExportPrinter)
  589. painter.save()
  590. painter.setRenderHint(QPainter.Antialiasing)
  591. painter.setRenderHint(QPainter.TextAntialiasing)
  592. self.scene.render(painter)
  593. painter.restore()
  594. @pyqtSlot()
  595. def slot_canvasSaveImage(self):
  596. newPath = QFileDialog.getSaveFileName(self, self.tr("Save Image"), filter=self.tr("PNG Image (*.png);;JPEG Image (*.jpg)"))
  597. if newPath:
  598. self.scene.clearSelection()
  599. if newPath.lower().endswith((".jpg",)):
  600. imgFormat = "JPG"
  601. elif newPath.lower().endswith((".png",)):
  602. imgFormat = "PNG"
  603. else:
  604. # File-dialog may not auto-add the extension
  605. imgFormat = "PNG"
  606. newPath += ".png"
  607. self.fExportImage = QImage(self.scene.sceneRect().width(), self.scene.sceneRect().height(), QImage.Format_RGB32)
  608. painter = QPainter(self.fExportImage)
  609. painter.save()
  610. painter.setRenderHint(QPainter.Antialiasing) # TODO - set true, cleanup this
  611. painter.setRenderHint(QPainter.TextAntialiasing)
  612. self.scene.render(painter)
  613. self.fExportImage.save(newPath, imgFormat, 100)
  614. painter.restore()
  615. # -----------------------------------------------------------------
  616. def resizeEvent(self, event):
  617. QFrame.resizeEvent(self, event)
  618. self.slot_miniCanvasCheckSize()
  619. # ------------------------------------------------------------------------------------------------
  620. # Canvas callback
  621. def canvasCallback(action, value1, value2, valueStr):
  622. if action == patchcanvas.ACTION_GROUP_INFO:
  623. pass
  624. elif action == patchcanvas.ACTION_GROUP_RENAME:
  625. pass
  626. elif action == patchcanvas.ACTION_GROUP_SPLIT:
  627. groupId = value1
  628. patchcanvas.splitGroup(groupId)
  629. Carla.gui.ui.miniCanvasPreview.update()
  630. elif action == patchcanvas.ACTION_GROUP_JOIN:
  631. groupId = value1
  632. patchcanvas.joinGroup(groupId)
  633. Carla.gui.ui.miniCanvasPreview.update()
  634. elif action == patchcanvas.ACTION_PORT_INFO:
  635. pass
  636. elif action == patchcanvas.ACTION_PORT_RENAME:
  637. pass
  638. elif action == patchcanvas.ACTION_PORTS_CONNECT:
  639. portIdA = value1
  640. portIdB = value2
  641. if not Carla.host.patchbay_connect(portIdA, portIdB):
  642. print("Connection failed:", Carla.host.get_last_error())
  643. elif action == patchcanvas.ACTION_PORTS_DISCONNECT:
  644. connectionId = value1
  645. if not Carla.host.patchbay_disconnect(connectionId):
  646. print("Disconnect failed:", Carla.host.get_last_error())
  647. elif action == patchcanvas.ACTION_PLUGIN_CLONE:
  648. pluginId = value1
  649. Carla.host.clone_plugin(pluginId)
  650. elif action == patchcanvas.ACTION_PLUGIN_EDIT:
  651. pluginId = value1
  652. # TODO
  653. elif action == patchcanvas.ACTION_PLUGIN_RENAME:
  654. pluginId = value1
  655. newName = valueStr
  656. Carla.host.rename_plugin(pluginId, newName)
  657. elif action == patchcanvas.ACTION_PLUGIN_REMOVE:
  658. pluginId = value1
  659. Carla.host.remove_plugin(pluginId)
  660. elif action == patchcanvas.ACTION_PLUGIN_SHOW_UI:
  661. pluginId = value1
  662. Carla.host.show_custom_ui(pluginId, True)