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.

761 lines
24KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla rack widget code
  4. # Copyright (C) 2011-2014 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 (Config)
  19. from carla_config import *
  20. # ------------------------------------------------------------------------------------------------------------
  21. # Imports (Global)
  22. if config_UseQt5:
  23. from PyQt5.QtCore import Qt, QSize, QTimer
  24. from PyQt5.QtGui import QPixmap
  25. from PyQt5.QtWidgets import QAbstractItemView, QApplication, QHBoxLayout, QLabel, QListWidget, QListWidgetItem, QScrollBar
  26. else:
  27. from PyQt4.QtCore import Qt, QSize, QTimer
  28. from PyQt4.QtGui import QAbstractItemView, QApplication, QHBoxLayout, QLabel, QListWidget, QListWidgetItem, QPixmap, QScrollBar
  29. # ------------------------------------------------------------------------------------------------------------
  30. # Imports (Custom Stuff)
  31. from carla_skin import *
  32. # ------------------------------------------------------------------------------------------------------------
  33. # Rack widget item
  34. class CarlaRackItem(QListWidgetItem):
  35. kRackItemType = QListWidgetItem.UserType + 1
  36. def __init__(self, parent, pluginId):
  37. QListWidgetItem.__init__(self, parent, self.kRackItemType)
  38. self.fParent = parent
  39. self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled)
  40. #self.setFlags(Qt.ItemIsSelectable|Qt.ItemIsEnabled|Qt.ItemIsDragEnabled|Qt.ItemIsDropEnabled)
  41. self.createWidget(pluginId)
  42. # -----------------------------------------------------------------
  43. def createWidget(self, pluginId):
  44. self.widget = createPluginSlot(self.fParent, self.fParent.host, pluginId, True) # FIXME useSkins opt
  45. self.widget.setFixedHeight(self.widget.getFixedHeight())
  46. self.setSizeHint(QSize(640, self.widget.getFixedHeight()))
  47. self.fParent.setItemWidget(self, self.widget)
  48. # -----------------------------------------------------------------
  49. def close(self):
  50. self.widget.fEditDialog.close()
  51. def reloadAll(self, pluginId):
  52. self.widget.fEditDialog.close()
  53. del self.widget
  54. self.createWidget(pluginId)
  55. # ------------------------------------------------------------------------------------------------------------
  56. # Rack widget list
  57. class CarlaRackList(QListWidget):
  58. def __init__(self, parent, host):
  59. QListWidget.__init__(self, parent)
  60. self.host = host
  61. if False:
  62. # kdevelop likes this :)
  63. host = CarlaHostMeta()
  64. self.host = host
  65. # -------------------------------------------------------------
  66. exts = host.get_supported_file_extensions().split(";")
  67. # plugin files
  68. exts.append("dll")
  69. if MACOS:
  70. exts.append("dylib")
  71. if not WINDOWS:
  72. exts.append("so")
  73. self.fSupportedExtensions = tuple(i.replace("*.","") for i in exts)
  74. self.fWasLastDragValid = False
  75. self.setMinimumWidth(640+20) # required by zita, 591 was old value
  76. self.setSelectionMode(QAbstractItemView.SingleSelection)
  77. self.setSortingEnabled(False)
  78. #self.setSortingEnabled(True)
  79. self.setDragEnabled(True)
  80. self.setDragDropMode(QAbstractItemView.DropOnly)
  81. self.setDropIndicatorShown(True)
  82. self.viewport().setAcceptDrops(True)
  83. self.setFrameShape(QFrame.NoFrame)
  84. self.setFrameShadow(QFrame.Plain)
  85. self.fPixmapL = QPixmap(":/bitmaps/rack_interior_left.png")
  86. self.fPixmapR = QPixmap(":/bitmaps/rack_interior_right.png")
  87. self.fPixmapWidth = self.fPixmapL.width()
  88. def isDragEventValid(self, urls):
  89. for url in urls:
  90. filename = url.toLocalFile()
  91. if os.path.isdir(filename):
  92. if os.path.exists(os.path.join(filename, "manifest.ttl")):
  93. return True
  94. elif os.path.isfile(filename):
  95. if filename.lower().endswith(self.fSupportedExtensions):
  96. return True
  97. return False
  98. def dragEnterEvent(self, event):
  99. if self.isDragEventValid(event.mimeData().urls()):
  100. self.fWasLastDragValid = True
  101. event.acceptProposedAction()
  102. return
  103. self.fWasLastDragValid = False
  104. QListWidget.dragEnterEvent(self, event)
  105. def dragMoveEvent(self, event):
  106. if self.fWasLastDragValid:
  107. event.acceptProposedAction()
  108. tryItem = self.itemAt(event.pos())
  109. if tryItem is not None:
  110. self.setCurrentRow(tryItem.widget.getPluginId())
  111. else:
  112. self.setCurrentRow(-1)
  113. return
  114. QListWidget.dragMoveEvent(self, event)
  115. #def dragLeaveEvent(self, event):
  116. #self.fWasLastDragValid = False
  117. #QListWidget.dragLeaveEvent(self, event)
  118. def dropEvent(self, event):
  119. event.acceptProposedAction()
  120. urls = event.mimeData().urls()
  121. if len(urls) == 0:
  122. return
  123. tryItem = self.itemAt(event.pos())
  124. if tryItem is not None:
  125. pluginId = tryItem.widget.getPluginId()
  126. self.host.replace_plugin(pluginId)
  127. for url in urls:
  128. filename = url.toLocalFile()
  129. if not self.host.load_file(filename):
  130. CustomMessageBox(self, QMessageBox.Critical, self.tr("Error"),
  131. self.tr("Failed to load file"),
  132. self.host.get_last_error(), QMessageBox.Ok, QMessageBox.Ok)
  133. if tryItem is not None:
  134. self.host.replace_plugin(self.parent().fPluginCount)
  135. #tryItem.widget.setActive(True, True, True)
  136. def mousePressEvent(self, event):
  137. if self.itemAt(event.pos()) is None:
  138. event.accept()
  139. self.setCurrentRow(-1)
  140. return
  141. QListWidget.mousePressEvent(self, event)
  142. def paintEvent(self, event):
  143. painter = QPainter(self.viewport())
  144. painter.drawTiledPixmap(0, 0, self.fPixmapWidth, self.height(), self.fPixmapL)
  145. painter.drawTiledPixmap(self.width()-self.fPixmapWidth-2, 0, self.fPixmapWidth, self.height(), self.fPixmapR)
  146. QListWidget.paintEvent(self, event)
  147. # ------------------------------------------------------------------------------------------------------------
  148. # Rack widget
  149. class CarlaRackW(QFrame):
  150. def __init__(self, parent, host, doSetup = True):
  151. QFrame.__init__(self, parent)
  152. self.host = host
  153. if False:
  154. # kdevelop likes this :)
  155. host = CarlaHostMeta()
  156. self.host = host
  157. # -------------------------------------------------------------
  158. self.fLayout = QHBoxLayout(self)
  159. self.fLayout.setContentsMargins(0, 0, 0, 0)
  160. self.fLayout.setSpacing(0)
  161. self.setLayout(self.fLayout)
  162. self.fPadLeft = QLabel(self)
  163. self.fPadLeft.setFixedWidth(25)
  164. self.fPadLeft.setObjectName("PadLeft")
  165. self.fPadLeft.setText("")
  166. self.fPadRight = QLabel(self)
  167. self.fPadRight.setFixedWidth(25)
  168. self.fPadRight.setObjectName("PadRight")
  169. self.fPadRight.setText("")
  170. self.fRack = CarlaRackList(self, host)
  171. self.fRack.setObjectName("CarlaRackList")
  172. self.fRack.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
  173. self.fRack.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
  174. self.fRack.currentRowChanged.connect(self.slot_currentRowChanged)
  175. sb = self.fRack.verticalScrollBar()
  176. self.fScrollBar = QScrollBar(Qt.Vertical, self)
  177. self.fScrollBar.setMinimum(sb.minimum())
  178. self.fScrollBar.setMaximum(sb.maximum())
  179. self.fScrollBar.setValue(sb.value())
  180. #sb.actionTriggered.connect(self.fScrollBar.triggerAction)
  181. #sb.sliderMoved.connect(self.fScrollBar.)
  182. #sb.sliderPressed.connect(self.fScrollBar.)
  183. #sb.sliderReleased.connect(self.fScrollBar.)
  184. sb.rangeChanged.connect(self.fScrollBar.setRange)
  185. sb.valueChanged.connect(self.fScrollBar.setValue)
  186. self.fScrollBar.rangeChanged.connect(sb.setRange)
  187. self.fScrollBar.valueChanged.connect(sb.setValue)
  188. self.fLayout.addWidget(self.fPadLeft)
  189. self.fLayout.addWidget(self.fRack)
  190. self.fLayout.addWidget(self.fPadRight)
  191. self.fLayout.addWidget(self.fScrollBar)
  192. # -------------------------------------------------------------
  193. # Internal stuff
  194. self.fParent = parent
  195. self.fPluginCount = 0
  196. self.fPluginList = []
  197. self.fCurrentRow = -1
  198. self.fLastSelectedItem = None
  199. # -------------------------------------------------------------
  200. # Set-up GUI stuff
  201. #app = QApplication.instance()
  202. #pal1 = app.palette().base().color()
  203. #pal2 = app.palette().button().color()
  204. #col1 = "stop:0 rgb(%i, %i, %i)" % (pal1.red(), pal1.green(), pal1.blue())
  205. #col2 = "stop:1 rgb(%i, %i, %i)" % (pal2.red(), pal2.green(), pal2.blue())
  206. self.setStyleSheet("""
  207. QLabel#PadLeft {
  208. background-image: url(:/bitmaps/rack_padding_left.png);
  209. background-repeat: repeat-y;
  210. }
  211. QLabel#PadRight {
  212. background-image: url(:/bitmaps/rack_padding_right.png);
  213. background-repeat: repeat-y;
  214. }
  215. CarlaRackList#CarlaRackList {
  216. background-color: black;
  217. }
  218. """)
  219. # -------------------------------------------------------------
  220. # Connect actions to functions
  221. if not doSetup: return
  222. parent.ui.menu_Canvas.hide()
  223. parent.ui.act_plugins_enable.triggered.connect(self.slot_pluginsEnable)
  224. parent.ui.act_plugins_disable.triggered.connect(self.slot_pluginsDisable)
  225. parent.ui.act_plugins_volume100.triggered.connect(self.slot_pluginsVolume100)
  226. parent.ui.act_plugins_mute.triggered.connect(self.slot_pluginsMute)
  227. parent.ui.act_plugins_wet100.triggered.connect(self.slot_pluginsWet100)
  228. parent.ui.act_plugins_bypass.triggered.connect(self.slot_pluginsBypass)
  229. parent.ui.act_plugins_center.triggered.connect(self.slot_pluginsCenter)
  230. parent.ui.act_plugins_panic.triggered.connect(self.slot_pluginsDisable)
  231. parent.ui.act_settings_configure.triggered.connect(self.slot_configureCarla)
  232. host.PluginAddedCallback.connect(self.slot_handlePluginAddedCallback)
  233. host.PluginRemovedCallback.connect(self.slot_handlePluginRemovedCallback)
  234. host.PluginRenamedCallback.connect(self.slot_handlePluginRenamedCallback)
  235. host.PluginUnavailableCallback.connect(self.slot_handlePluginUnavailableCallback)
  236. host.ParameterValueChangedCallback.connect(self.slot_handleParameterValueChangedCallback)
  237. host.ParameterDefaultChangedCallback.connect(self.slot_handleParameterDefaultChangedCallback)
  238. host.ParameterMidiChannelChangedCallback.connect(self.slot_handleParameterMidiChannelChangedCallback)
  239. host.ParameterMidiCcChangedCallback.connect(self.slot_handleParameterMidiCcChangedCallback)
  240. host.ProgramChangedCallback.connect(self.slot_handleProgramChangedCallback)
  241. host.MidiProgramChangedCallback.connect(self.slot_handleMidiProgramChangedCallback)
  242. host.OptionChangedCallback.connect(self.slot_handleOptionChangedCallback)
  243. host.UiStateChangedCallback.connect(self.slot_handleUiStateChangedCallback)
  244. host.NoteOnCallback.connect(self.slot_handleNoteOnCallback)
  245. host.NoteOffCallback.connect(self.slot_handleNoteOffCallback)
  246. host.UpdateCallback.connect(self.slot_handleUpdateCallback)
  247. host.ReloadInfoCallback.connect(self.slot_handleReloadInfoCallback)
  248. host.ReloadParametersCallback.connect(self.slot_handleReloadParametersCallback)
  249. host.ReloadProgramsCallback.connect(self.slot_handleReloadProgramsCallback)
  250. host.ReloadAllCallback.connect(self.slot_handleReloadAllCallback)
  251. # -----------------------------------------------------------------
  252. def addPlugin(self, pluginId, isProjectLoading):
  253. pitem = CarlaRackItem(self.fRack, pluginId)
  254. self.fPluginList.append(pitem)
  255. self.fPluginCount += 1
  256. if not isProjectLoading:
  257. pitem.widget.setActive(True, True, True)
  258. def removePlugin(self, pluginId):
  259. if pluginId >= self.fPluginCount:
  260. return
  261. pitem = self.fPluginList[pluginId]
  262. if pitem is None:
  263. return
  264. self.fPluginCount -= 1
  265. self.fPluginList.pop(pluginId)
  266. self.fRack.takeItem(pluginId)
  267. pitem.close()
  268. del pitem
  269. # push all plugins 1 slot back
  270. for i in range(pluginId, self.fPluginCount):
  271. pitem = self.fPluginList[i]
  272. pitem.widget.setId(i)
  273. def renamePlugin(self, pluginId, newName):
  274. if pluginId >= self.fPluginCount:
  275. return
  276. pitem = self.fPluginList[pluginId]
  277. if pitem is None:
  278. return
  279. pitem.widget.setName(newName)
  280. def disablePlugin(self, pluginId, errorMsg):
  281. if pluginId >= self.fPluginCount:
  282. return
  283. pitem = self.fPluginList[pluginId]
  284. if pitem is None:
  285. return
  286. def removeAllPlugins(self):
  287. while self.fRack.takeItem(0):
  288. pass
  289. for i in range(self.fPluginCount):
  290. pitem = self.fPluginList[i]
  291. if pitem is None:
  292. break
  293. pitem.close()
  294. del pitem
  295. self.fPluginCount = 0
  296. self.fPluginList = []
  297. # -----------------------------------------------------------------
  298. def engineStarted(self):
  299. pass
  300. def engineStopped(self):
  301. pass
  302. # -----------------------------------------------------------------
  303. def idleFast(self):
  304. for i in range(self.fPluginCount):
  305. pitem = self.fPluginList[i]
  306. if pitem is None:
  307. break
  308. pitem.widget.idleFast()
  309. def idleSlow(self):
  310. for i in range(self.fPluginCount):
  311. pitem = self.fPluginList[i]
  312. if pitem is None:
  313. break
  314. pitem.widget.idleSlow()
  315. # -----------------------------------------------------------------
  316. def projectLoadingStarted(self):
  317. self.fRack.setEnabled(False)
  318. def projectLoadingFinished(self):
  319. self.fRack.setEnabled(True)
  320. # -----------------------------------------------------------------
  321. def saveSettings(self, settings):
  322. pass
  323. def showEditDialog(self, pluginId):
  324. if pluginId >= self.fPluginCount:
  325. return
  326. pitem = self.fPluginList[pluginId]
  327. if pitem is None:
  328. return
  329. pitem.widget.slot_showEditDialog(True)
  330. # -----------------------------------------------------------------
  331. @pyqtSlot()
  332. def slot_pluginsEnable(self):
  333. if not self.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. pitem.widget.setActive(True, True, True)
  340. @pyqtSlot()
  341. def slot_pluginsDisable(self):
  342. if not self.host.is_engine_running():
  343. return
  344. for i in range(self.fPluginCount):
  345. pitem = self.fPluginList[i]
  346. if pitem is None:
  347. break
  348. pitem.widget.setActive(False, True, True)
  349. @pyqtSlot()
  350. def slot_pluginsVolume100(self):
  351. if not self.host.is_engine_running():
  352. return
  353. for i in range(self.fPluginCount):
  354. pitem = self.fPluginList[i]
  355. if pitem is None:
  356. break
  357. pitem.widget.setInternalParameter(PLUGIN_CAN_VOLUME, 1.0)
  358. @pyqtSlot()
  359. def slot_pluginsMute(self):
  360. if not self.host.is_engine_running():
  361. return
  362. for i in range(self.fPluginCount):
  363. pitem = self.fPluginList[i]
  364. if pitem is None:
  365. break
  366. pitem.widget.setInternalParameter(PLUGIN_CAN_VOLUME, 0.0)
  367. @pyqtSlot()
  368. def slot_pluginsWet100(self):
  369. if not self.host.is_engine_running():
  370. return
  371. for i in range(self.fPluginCount):
  372. pitem = self.fPluginList[i]
  373. if pitem is None:
  374. break
  375. pitem.widget.setInternalParameter(PLUGIN_CAN_DRYWET, 1.0)
  376. @pyqtSlot()
  377. def slot_pluginsBypass(self):
  378. if not self.host.is_engine_running():
  379. return
  380. for i in range(self.fPluginCount):
  381. pitem = self.fPluginList[i]
  382. if pitem is None:
  383. break
  384. pitem.widget.setInternalParameter(PLUGIN_CAN_DRYWET, 0.0)
  385. @pyqtSlot()
  386. def slot_pluginsCenter(self):
  387. if not self.host.is_engine_running():
  388. return
  389. for i in range(self.fPluginCount):
  390. pitem = self.fPluginList[i]
  391. if pitem is None:
  392. break
  393. pitem.widget.setInternalParameter(PARAMETER_BALANCE_LEFT, -1.0)
  394. pitem.widget.setInternalParameter(PARAMETER_BALANCE_RIGHT, 1.0)
  395. pitem.widget.setInternalParameter(PARAMETER_PANNING, 0.0)
  396. # -----------------------------------------------------------------
  397. @pyqtSlot()
  398. def slot_configureCarla(self):
  399. if self.fParent is None or not self.fParent.openSettingsWindow(False, False):
  400. return
  401. self.fParent.loadSettings(False)
  402. # -----------------------------------------------------------------
  403. @pyqtSlot(int, str)
  404. def slot_handlePluginAddedCallback(self, pluginId, pluginName):
  405. self.addPlugin(pluginId, self.fParent.isProjectLoading())
  406. @pyqtSlot(int)
  407. def slot_handlePluginRemovedCallback(self, pluginId):
  408. self.removePlugin(pluginId)
  409. @pyqtSlot(int, str)
  410. def slot_handlePluginRenamedCallback(self, pluginId, newName):
  411. self.renamePlugin(pluginId, newName)
  412. @pyqtSlot(int, str)
  413. def slot_handlePluginUnavailableCallback(self, pluginId, errorMsg):
  414. self.disablePlugin(pluginId, errorMsg)
  415. # -----------------------------------------------------------------
  416. @pyqtSlot(int, int, float)
  417. def slot_handleParameterValueChangedCallback(self, pluginId, index, value):
  418. if pluginId >= self.fPluginCount:
  419. return
  420. pitem = self.fPluginList[pluginId]
  421. if pitem is None:
  422. return
  423. pitem.widget.setParameterValue(index, value, True)
  424. @pyqtSlot(int, int, float)
  425. def slot_handleParameterDefaultChangedCallback(self, pluginId, index, value):
  426. if pluginId >= self.fPluginCount:
  427. return
  428. pitem = self.fPluginList[pluginId]
  429. if pitem is None:
  430. return
  431. pitem.widget.setParameterDefault(index, value)
  432. @pyqtSlot(int, int, int)
  433. def slot_handleParameterMidiCcChangedCallback(self, pluginId, index, cc):
  434. if pluginId >= self.fPluginCount:
  435. return
  436. pitem = self.fPluginList[pluginId]
  437. if pitem is None:
  438. return
  439. pitem.widget.setParameterMidiControl(index, cc)
  440. @pyqtSlot(int, int, int)
  441. def slot_handleParameterMidiChannelChangedCallback(self, pluginId, index, channel):
  442. if pluginId >= self.fPluginCount:
  443. return
  444. pitem = self.fPluginList[pluginId]
  445. if pitem is None:
  446. return
  447. pitem.widget.setParameterMidiChannel(index, channel)
  448. # -----------------------------------------------------------------
  449. @pyqtSlot(int, int)
  450. def slot_handleProgramChangedCallback(self, pluginId, index):
  451. if pluginId >= self.fPluginCount:
  452. return
  453. pitem = self.fPluginList[pluginId]
  454. if pitem is None:
  455. return
  456. pitem.widget.setProgram(index, True)
  457. @pyqtSlot(int, int)
  458. def slot_handleMidiProgramChangedCallback(self, pluginId, index):
  459. if pluginId >= self.fPluginCount:
  460. return
  461. pitem = self.fPluginList[pluginId]
  462. if pitem is None:
  463. return
  464. pitem.widget.setMidiProgram(index, True)
  465. # -----------------------------------------------------------------
  466. @pyqtSlot(int, int, bool)
  467. def slot_handleOptionChangedCallback(self, pluginId, option, yesNo):
  468. if pluginId >= self.fPluginCount:
  469. return
  470. pitem = self.fPluginList[pluginId]
  471. if pitem is None:
  472. return
  473. pitem.widget.setOption(option, yesNo)
  474. # -----------------------------------------------------------------
  475. @pyqtSlot(int, int)
  476. def slot_handleUiStateChangedCallback(self, pluginId, state):
  477. if pluginId >= self.fPluginCount:
  478. return
  479. pitem = self.fPluginList[pluginId]
  480. if pitem is None:
  481. return
  482. pitem.widget.customUiStateChanged(state)
  483. # -----------------------------------------------------------------
  484. @pyqtSlot(int, int, int, int)
  485. def slot_handleNoteOnCallback(self, pluginId, channel, note, velo):
  486. if pluginId >= self.fPluginCount:
  487. return
  488. pitem = self.fPluginList[pluginId]
  489. if pitem is None:
  490. return
  491. pitem.widget.sendNoteOn(channel, note)
  492. @pyqtSlot(int, int, int)
  493. def slot_handleNoteOffCallback(self, pluginId, channel, note):
  494. if pluginId >= self.fPluginCount:
  495. return
  496. pitem = self.fPluginList[pluginId]
  497. if pitem is None:
  498. return
  499. pitem.widget.sendNoteOff(channel, note)
  500. # -----------------------------------------------------------------
  501. @pyqtSlot(int)
  502. def slot_handleUpdateCallback(self, pluginId):
  503. if pluginId >= self.fPluginCount:
  504. return
  505. pitem = self.fPluginList[pluginId]
  506. if pitem is None:
  507. return
  508. pitem.widget.fEditDialog.updateInfo()
  509. @pyqtSlot(int)
  510. def slot_handleReloadInfoCallback(self, pluginId):
  511. if pluginId >= self.fPluginCount:
  512. return
  513. pitem = self.fPluginList[pluginId]
  514. if pitem is None:
  515. return
  516. pitem.widget.fEditDialog.reloadInfo()
  517. @pyqtSlot(int)
  518. def slot_handleReloadParametersCallback(self, pluginId):
  519. if pluginId >= self.fPluginCount:
  520. return
  521. pitem = self.fPluginList[pluginId]
  522. if pitem is None:
  523. return
  524. pitem.widget.fEditDialog.reloadParameters()
  525. @pyqtSlot(int)
  526. def slot_handleReloadProgramsCallback(self, pluginId):
  527. if pluginId >= self.fPluginCount:
  528. return
  529. pitem = self.fPluginList[pluginId]
  530. if pitem is None:
  531. return
  532. pitem.widget.fEditDialog.reloadPrograms()
  533. @pyqtSlot(int)
  534. def slot_handleReloadAllCallback(self, pluginId):
  535. if pluginId >= self.fPluginCount:
  536. return
  537. pitem = self.fPluginList[pluginId]
  538. if pitem is None:
  539. return
  540. self.fRack.setCurrentRow(-1)
  541. self.fCurrentRow = -1
  542. self.fLastSelectedItem = None
  543. pitem.reloadAll(pluginId)
  544. # -----------------------------------------------------------------
  545. @pyqtSlot(int)
  546. def slot_currentRowChanged(self, row):
  547. self.fCurrentRow = row
  548. if self.fLastSelectedItem is not None:
  549. self.fLastSelectedItem.setSelected(False)
  550. if row < 0 or row >= self.fPluginCount or self.fPluginList[row] is None:
  551. self.fLastSelectedItem = None
  552. return
  553. pitem = self.fPluginList[row]
  554. pitem.widget.setSelected(True)
  555. self.fLastSelectedItem = pitem.widget
  556. # -----------------------------------------------------------------