【问题标题】:How to disable Mouse Wheel event of Qcombobox placed in QTableWidget in Python?如何在Python中禁用放置在QTableWidget中的Qcombobox的鼠标滚轮事件?
【发布时间】:2020-07-19 22:27:38
【问题描述】:

我有 QtDesigner 的 QTablewidget,我在其中一列中添加了 Qcombobox。我还添加了事件过滤器来禁用鼠标滚轮事件,但它只适用于最近添加的组合框。如何使它适用于表格列中的所有组合框?

我在这里错过了什么?

谢谢。

QtDesigner 编写的代码:

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(504, 346)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.tableWidget = QtWidgets.QTableWidget(self.centralwidget)
        self.tableWidget.setGeometry(QtCore.QRect(10, 10, 481, 301))
        self.tableWidget.setRowCount(4)
        self.tableWidget.setColumnCount(3)
        self.tableWidget.setObjectName("tableWidget")
        item = QtWidgets.QTableWidgetItem()
        self.tableWidget.setHorizontalHeaderItem(0, item)
        item = QtWidgets.QTableWidgetItem()
        self.tableWidget.setHorizontalHeaderItem(1, item)
        item = QtWidgets.QTableWidgetItem()
        self.tableWidget.setHorizontalHeaderItem(2, item)
        self.tableWidget.verticalHeader().setVisible(False)
        MainWindow.setCentralWidget(self.centralwidget)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
        item = self.tableWidget.horizontalHeaderItem(0)
        item.setText(_translate("MainWindow", "Name"))
        item = self.tableWidget.horizontalHeaderItem(1)
        item.setText(_translate("MainWindow", "Sex"))
        item = self.tableWidget.horizontalHeaderItem(2)
        item.setText(_translate("MainWindow", "Age"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

我的脚本:

from PyQt5 import QtCore, QtWidgets
from PyQt5.QtWidgets import QTableWidgetItem

from demoui import Ui_MainWindow

class democode(QtWidgets.QMainWindow, Ui_MainWindow):                 
    def __init__(self):
        super(democode, self).__init__()
        self.setupUi(self) 

        #Adding Data
        for row in range(0,4):
            for column in range(0,3):
                if column == 1:
                    self.combo = QtWidgets.QComboBox()
                    self.combo.addItem("Male")
                    self.combo.addItem("Female")
                    self.combo.installEventFilter(self)
                    self.tableWidget.setCellWidget(row, column, self.combo)
                else:
                    item = QTableWidgetItem('')
                    self.tableWidget.setItem(row, column, item)

    #Event Filter
    def eventFilter(self, source, event):
        if (event.type() == QtCore.QEvent.Wheel and
            source is self.combo):
            return True
        return super(democode, self).eventFilter(source, event)

if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    window = democode()   
    window.show()
    sys.exit(app.exec_())

【问题讨论】:

    标签: python-3.x pyqt5 qtablewidget qcombobox


    【解决方案1】:

    您犯的错误类似于我在this answer 中警告过的错误:

    如果不需要,请不要设置实例属性:所有这些 self.rowPosition、self.monthList 等,每次 for 循环循环时都会更改

    每次你这样做:

    self.combo = QtWidgets.QComboBox()
    

    您正在覆盖实例的combo 属性 (self);这意味着分配给该属性的最后一个组合将是 self.combo 所指的内容,这就是它仅适用于最新组合的原因。

    你正在做的实际上是这样的:

    self.combo = QtWidgets.QComboBox()
    self.combo = QtWidgets.QComboBox()
    self.combo = QtWidgets.QComboBox()
    self.combo = QtWidgets.QComboBox()
    # ...
    

    想一想:self.combo 指向什么组合?显然是最后一个。

    在像您这样的简单情况下,您只需将一组有限的小部件添加到事件过滤器中,使用 isinstance() 可能就足够了:

        def eventFilter(self, source, event):
            if (event.type() == QtCore.QEvent.Wheel and
                isinstance(source, QtWidgets.QComboBox)):
                return True
            return super(democode, self).eventFilter(source, event)
    

    但这也意味着过滤器将匹配您为其安装了事件过滤器的任何 QCombobox。如果出于任何其他原因,您需要在该表中没有的另一个组合上安装过滤器,则该组合也不会接收到轮子事件。

    最简单且可能正确的方法是将这些组合添加到列表中,并检查源是否是该列表的成员:

    class democode(QtWidgets.QMainWindow, Ui_MainWindow):                 
        def __init__(self):
            super(democode, self).__init__()
            self.setupUi(self) 
    
            self.noWheelCombos = []
            for row in range(0,4):
                for column in range(0,3):
                    if column == 1:
                        # note that I'm *NOT* using "self"
                        combo = QtWidgets.QComboBox()
                        self.noWheelCombos.append(combo)
                        combo.addItem("Male")
                        combo.addItem("Female")
                        combo.installEventFilter(self)
                        self.tableWidget.setCellWidget(row, column, combo)
                    # ...
    
        def eventFilter(self, source, event):
            if (event.type() == QtCore.QEvent.Wheel and
                source in self.noWheelCombos):
                return True
            return super(democode, self).eventFilter(source, event)
    

    另一种可能性是用一个什么都不做的函数覆盖wheelEvent。这与使用简单的pass 子类化和覆盖wheelEvent 方法几乎相同;它可能不是很优雅,但适用于非常简单的情况,如下所示:

            combo = QtWidgets.QComboBox()
            combo.wheelEvent = lambda event: None
    

    最后,一个小建议:如果您总是有多个组合的相同元素,您可以在 for 循环之外创建一个列表并使用 addItems() 而不是每次都添加单个项目:

        items = "Male", "Female"
        for row in range(0,4):
            for column in range(0,3):
                if column == 1:
                    combo = QtWidgets.QComboBox()
                    combo.addItems(items)
                    # ...
    

    【讨论】:

    • 谢谢!我会记下它。 :)
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-03-15
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多