Как вызвать событие, когда в QListWidget добавлено или удалено содержимое - PullRequest
1 голос
/ 08 апреля 2019

При добавлении элементов в QListWidget я хочу включить или отключить действие в зависимости от того, есть ли элементы в QListWidget. Есть ли что-то в списке listWidget.itemChanged.connect(self.checkListLength), кроме изменения количества элементов в QlistWidget, которое я мог бы использовать? Обычно после каждого изменения в виджете я просто выполняю проверку, но многие различные источники будут ее менять, и было бы больше смысла просто слушать изменение.

Я перепробовал каждое соединение, которое смог найти, чтобы увидеть, вернет ли оно желаемый результат, а это не так.

Вот пример, который компилируется, но не включает действие Rename Selection. Мы импортируем имена файлов в QListWidget. В этом примере мы могли бы просто включить его после использования открытой папки, но для моего кода это будет не так просто.

import sys
from os import listdir
from os.path import isfile, join
from PyQt5.QtWidgets import (
    QMainWindow, QAction, QHBoxLayout, QWidget, QListWidget, 
    QListWidgetItem, QAbstractItemView, QApplication, QDialog, qApp, QToolBar, QFileDialog
)
from PyQt5.QtGui import QIcon
from PyQt5 import QtCore

class MainGui(QMainWindow):

    def __init__(self):
        super().__init__()
        self.mainWidget = QWidget()
        self.setCentralWidget(self.mainWidget)
        self.initUI()

    def initUI(self):  
        #TopIcon
        self.exitAct = QAction(QIcon('img/x-square.svg'), 'Exit', self)
        self.exitAct.setShortcut('Ctrl+Q')
        self.exitAct.triggered.connect(qApp.quit)

        #Import Folder
        self.importAct = QAction(QIcon('img/folder.svg'), 'Open Folder', self)
        self.importAct.setShortcut('Ctrl+N')
        self.importAct.triggered.connect(self.getFolder)

        #Rename Button
        self.renameAct = QAction(QIcon('img/edit.svg'), 'Rename Selection', self)
        self.renameAct.setShortcut('Ctrl+R')
        self.renameAct.setEnabled(False)

        self.toolbar = QToolBar('ToolBar')
        self.addToolBar(QtCore.Qt.LeftToolBarArea, self.toolbar)
        self.toolbar.addAction(self.exitAct)
        self.toolbar.addSeparator()
        self.toolbar.addAction(self.importAct)
        self.toolbar.addAction(self.renameAct)
        self.hbox = QHBoxLayout()
        self.listWidget = QListWidget()
        self.listWidget.setSelectionMode(QAbstractItemView.ExtendedSelection)
        self.listWidget.setGeometry(QtCore.QRect(10, 10, 211, 291))
        self.hbox.addWidget(self.listWidget)
        self.mainWidget.setLayout(self.hbox)
        self.show()

    def getFolder(self):
        self.ImportFolder = QFileDialog.getExistingDirectory(None, "Select Directory") 
        self.getFilesInDir(self.ImportFolder)

    def getFilesInDir(self, mypath):
        f = []
        files = [f for f in listdir(mypath) if isfile(join(mypath, f))]
        if files:
            self.listWidget.addItems(files)

if __name__ == '__main__':
    app = QApplication(sys.argv)
    gui = MainGui()
    sys.exit(app.exec_())

Ответы [ 2 ]

1 голос
/ 08 апреля 2019

Если вы хотите узнать, был ли элемент добавлен или удален в QListWidget, вы должны сделать это через внутреннюю модель, которая имеет сигналы rowsInserted и rowsRemoved соответственно.

import sys
from os import listdir
from os.path import isfile, join
from PyQt5 import QtCore, QtGui, QtWidgets


class MainGui(QtWidgets.QMainWindow):
    def __init__(self):
        super().__init__()
        self.mainWidget = QtWidgets.QWidget()
        self.setCentralWidget(self.mainWidget)
        self.initUI()

    def initUI(self):
        # TopIcon
        self.exitAct = QtWidgets.QAction(
            QtGui.QIcon("img/x-square.svg"), "Exit", self
        )
        self.exitAct.setShortcut("Ctrl+Q")
        self.exitAct.triggered.connect(QtWidgets.qApp.quit)

        # Import Folder
        self.importAct = QtWidgets.QAction(
            QtGui.QIcon("img/folder.svg"), "Open Folder", self
        )
        self.importAct.setShortcut("Ctrl+N")
        self.importAct.triggered.connect(self.getFolder)

        # Rename Button
        self.renameAct = QtWidgets.QAction(
            QtGui.QIcon("img/edit.svg"), "Rename Selection", self
        )
        self.renameAct.setShortcut("Ctrl+R")
        self.renameAct.setEnabled(False)

        self.toolbar = QtWidgets.QToolBar("ToolBar")
        self.addToolBar(QtCore.Qt.LeftToolBarArea, self.toolbar)
        self.toolbar.addAction(self.exitAct)
        self.toolbar.addSeparator()
        self.toolbar.addAction(self.importAct)
        self.toolbar.addAction(self.renameAct)
        hbox = QtWidgets.QHBoxLayout(self.mainWidget)
        self.listWidget = QtWidgets.QListWidget()
        self.listWidget.setSelectionMode(
            QtWidgets.QAbstractItemView.ExtendedSelection
        )
        self.listWidget.model().rowsInserted.connect(self.on_rowsInserted)
        self.listWidget.model().rowsRemoved.connect(self.on_rowsRemoved)
        hbox.addWidget(self.listWidget)
        self.show()

    @QtCore.pyqtSlot(QtCore.QModelIndex, int, int)
    def on_rowsInserted(self, parent, first, last):
        print("Insert:", parent, first, last)

    @QtCore.pyqtSlot(QtCore.QModelIndex, int, int)
    def on_rowsRemoved(self, parent, first, last):
        print("Remove:", parent, first, last)

    @QtCore.pyqtSlot()
    def getFolder(self):
        importFolder = QtWidgets.QFileDialog.getExistingDirectory(
            None, "Select Directory"
        )
        self.getFilesInDir(importFolder)

    def getFilesInDir(self, mypath):
        files = [f for f in listdir(mypath) if isfile(join(mypath, f))]
        if files:
            self.listWidget.addItems(files)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    gui = MainGui()
    sys.exit(app.exec_())
0 голосов
/ 08 апреля 2019

Я прошу прощения за ответ на свой вопрос, но я понял это и не хотел оставлять кого-либо в покое.

Вам нужно сослаться на модель QListWidget, чтобы сделать это:

self.lwModel = self.listWidget.model()
self.lwModel.rowsInserted.connect(self.checkListLength) #Any time an element is added run function
self.lwModel.rowsRemoved.connect(self.checkListLength) #Any time an element is removed run function

Надеюсь, это поможет любому с той же проблемой.

...