如何在发送单元格变化的信号中区分颜色和值

时间:2019-09-09 12:28:11

标签: python-3.x pyqt5 qtablewidget

其中有一个包含一些值的表。我想要做的是捕获表内容的更改并将其保存在SQLite数据库中。因此我将表格连接到cellChangeditemSelectionChanged信号。当我尝试更改某些单元格的内容时,它将对其进行检查以确认它不包含字母,并且是数字。因此,如果在更改单元格后在其中包含字母,它将尝试将内容更改为之前的值。 我还有一个搜索工具栏,可以在表中搜索某些关键字。搜索工具栏的作用是为与该特定关键字匹配的单元格着色。但是这样做会发出cellChanged信号,而不是cellSelectionChanged信号。导致错误(myapp对象没有属性previous_value),因为我尚未选择任何单元格。但是,如果我选择一个单元格然后开始搜索,则在搜索框中写入任何内容都会更改每个单元格的内容,因为它会触发cellChanged信号。我的问题是如何在cellChanged函数中区分颜色和文本更改。先感谢您。 这是示例代码:(python 3.7.4,pyqt5 5.13.0)

from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
import sys
class myapp(QWidget):
    def __init__(self):
        QWidget.__init__(self)
        self.tableWidget = QTableWidget()
        self.tableWidget.setSortingEnabled(True)
        self.tableWidget.setRowCount(5)
        self.tableWidget.setColumnCount(5)
        for i in range(5):
            for j in range(5):
                item=QTableWidgetItem("cell["+str(i) + ","+ str(j) + "]")
                self.tableWidget.setItem(i,j,item)
        self.tableWidget.cellChanged.connect(self.on_tablewidget_cell_changed)
        self.tableWidget.itemSelectionChanged.connect(self.on_tablewidget_item_selection_changed)

        self.search_entry = QLineEdit()
        self.search_entry.textChanged.connect(self.seek)
        layout = QVBoxLayout()
        layout.addWidget(self.tableWidget)
        layout.addWidget(self.search_entry)
        self.setLayout(layout)
    def on_tablewidget_cell_changed(self,row,col):
        value = self.tableWidget.item(row,col).text()
        if not self.check_numeric(value):
            self.tableWidget.item(row,col).setText(self.previous_value)
    def on_tablewidget_item_selection_changed(self):
        row = self.tableWidget.currentRow()
        col = self.tableWidget.currentColumn()
        self.previous_value = self.tableWidget.item(row,col).text()
    def seek(self):
        keyword = self.search_entry.text()
        items = self.tableWidget.findItems(keyword, Qt.MatchContains)
        for item in items:
            item.setBackground(Qt.red)

    def check_numeric(self,value):
        value = value.strip()
        tmp = "".join([i for i in value if i in "0123456789"])
        return(len(value) == len(tmp))


app = QApplication(sys.argv)
win = myapp()
win.show()
sys.exit(app.exec_())

1 个答案:

答案 0 :(得分:0)

除了捕获cellChanged信号以检查用户输入是否有效之外,您还可以为表格小部件设置项目委托,并在设置模型数据之前在其中检查用户输入。例如:

class IntItemDelegate(QItemDelegate):
    def setModelData(self, editor, item_model, model_index):
        # only update the model data if the user input consists of digits only.
        text = editor.text()
        if self.check_numeric(text):
            item_model.setData(model_index, text)

    def check_numeric(self, value):
        return value.isdecimal()


class myapp(QWidget):
    def __init__(self):
        QWidget.__init__(self)
        self.tableWidget = QTableWidget()
        self.tableWidget.setSortingEnabled(True)
        self.tableWidget.setRowCount(5)
        self.tableWidget.setColumnCount(5)
        for i in range(5):
            for j in range(5):
                item=QTableWidgetItem("cell["+str(i) + ","+ str(j) + "]")
                self.tableWidget.setItem(i,j,item)

        # set a custom item delegate which will take care of checking the user input
        self.tableWidget.setItemDelegate( IntItemDelegate())

        self.search_entry = QLineEdit()
        self.search_entry.textChanged.connect(self.seek)
        layout = QVBoxLayout()
        layout.addWidget(self.tableWidget)
        layout.addWidget(self.search_entry)
        self.setLayout(layout)

    def seek(self):
        keyword = self.search_entry.text()
        items = self.tableWidget.findItems(keyword, Qt.MatchContains)
        for item in items:
            item.setBackground(Qt.red)