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.

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