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.

713 lines
25KB

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