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.

745 lines
23KB

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