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.

814 lines
28KB

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