使用liswidget的itemdelegate和信号时,如何保护tablewidget中的单元格发生变化?

时间:2019-06-21 15:32:08

标签: python pyqt pyqt4 qtablewidget

我有代码,可以在tablewidget中开始编辑单元格,然后双击列表小部件中的名称将其添加到单元格中。但是当我在第二个单元格上执行此操作时,两个单元格都在变化。如何保护单元格不被更改,代码如下

from functools import partial
from PyQt4 import QtCore, QtGui


class StyledItemDelegate(QtGui.QStyledItemDelegate):
    textChanged = QtCore.pyqtSignal(int, int, str)
    #editingFinished = QtCore.pyqtSignal()

    def createEditor(self, parent, option, index):
        editor = super(StyledItemDelegate, self).createEditor(
            parent, option, index
        )
        if isinstance(editor, QtGui.QLineEdit):
            editor.textChanged.connect(
               partial(self.textChanged.emit, index.row(), index.column())
            )
            #editor.editingFinished.connect(self.editingFinished)
        return editor


class Widget(QtGui.QWidget):
    def __init__(self, parent=None):
        super(Widget, self).__init__(parent)

        self.table_widget = QtGui.QTableWidget(4, 4)
        self.table_widget.setHorizontalHeaderLabels(("Name", "1", "2", 
                      "3"))
        delegate = StyledItemDelegate(self.table_widget)
        delegate.textChanged.connect(self.filter)
        #delegate.editingFinished.connect(self.clear_filter)
        self.table_widget.setItemDelegate(delegate)

        self.list_widget = QtGui.QListWidget()



        hlay = QtGui.QHBoxLayout(self)
        hlay.addWidget(self.table_widget)
        hlay.addWidget(self.list_widget)

        for letter1 in "ABCDEFGHIJKLMNOPQRSTUVWXYZ":
            for letter2 in "ABCDEFGHIJKLMNOPQRSTUVWXYZ":
                text = letter1 + letter2
                it = QtGui.QListWidgetItem(text)
                self.list_widget.addItem(it)

    @QtCore.pyqtSlot(int, int, str)
    def filter(self, row, column, text):
        print(row, column)
        self.clear_filter()
        for r in range(self.list_widget.count()):
            it = self.list_widget.item(r)
            # filter algorithm
            is_showing = text in it.text()
            # Hide the row if necessary
            it.setHidden(not is_showing)      
        self.list_widget.itemDoubleClicked.connect(
                           lambda:self.nameselected(row,column))

    @QtCore.pyqtSlot()
    def clear_filter(self):
        for r in range(self.list_widget.count()):
            it = self.list_widget.item(r)
            it.setHidden(False)

    def nameselected(self,row,column):
        name=self.list_widget.currentItem().text()
        print(name,row,column)
        #self.table_Widget.blockSignals(True)
        self.table_widget.setItem(row,column, 
                      QtGui.QTableWidgetItem(name))
        #self.table_Widget.blockSignals(False)

if __name__ == "__main__":
    import sys

    app = QtGui.QApplication(sys.argv)
    w = Widget()
    w.show()
    sys.exit(app.exec_())

假设我在列表小部件中双击“ AA”时,在第一个单元格中以“ A”开头的所有名称都以“ A”开头,第一个单元格中的文字为“ AA”。当我在第二个单元格中键入“ B”时,当我双击“ listwidget”中的“ BA”时,listwidget的所有名称均以“ B”开头,第二个单元格应具有文本“ BA”。但是这里两个单元都设置为“ BA”。如何保护其他单元不被更改。

1 个答案:

答案 0 :(得分:0)

问题来自于每次应用新过滤器时您都连接到self.nameselected插槽的事实。如果您多次更改过滤器,则可以轻松看到它:每当在列表视图中双击一个项目时,它都会多次打印nameselected的广告位输出。

每次应用过滤器时,您都可以使用try并断开itemDoubleClicked信号,但这并不是一个好的解决方案(您可能需要从该信号中获得其他连接)。

只需将itemDoubleClicked信号连接到nameselected中的__init__,然后按如下所示更改该功能:

def nameselected(self):
    name=self.list_widget.currentItem().text()
    tableIndex = self.table_widget.currentIndex()
    self.table_widget.setItem(tableIndex(), tableIndex(), 
                  QtGui.QTableWidgetItem(name))

如果您不想这样做(例如,如果用户错误地单击了表中的另一个项目,则为避免错误的索引),只需保留对最后一个“已编辑”项目索引的内部引用并使用在名称选择功能中。