Python如何禁用Qcombobox的鼠标滚轮事件放置在QTableWidget中?
How to disable Mouse Wheel event of Qcombobox placed in QTableWidget in Python?
我有 QtDesigner 的 QTablewidget,我在其中的一列中添加了 Qcombobox。我还添加了事件过滤器来禁用鼠标滚轮事件,但它只适用于最近添加的组合框。如何让它适用于 table 列中的所有组合框?
我在这里错过了什么?
谢谢。
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_())
您犯了一个类似于我在 中警告您的错误:
don't set instance attributes if they are not required: all those self.rowPosition, self.monthList, etc, change everytime the for loop cycles
每次你这样做:
self.combo = QtWidgets.QComboBox()
您正在覆盖实例 (self
) 的 combo
属性;这意味着分配给该属性的最后一个组合将是 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,您已经为其安装了事件过滤器。如果出于任何其他原因,您需要在另一个不是该 table 的组合上安装过滤器,该组合也不会接收到 wheel 事件。
最简单且可能是正确的方法是将这些组合添加到列表中,如果源是该列表的成员,则检查过滤器:
class democode(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self):
super(democode, self).__init__()
self.setupUi(self)
<b>self.noWheelCombos = []</b>
for row in range(0,4):
for column in range(0,3):
if column == 1:
# note that I'm *NOT* using "self"
<b>combo = QtWidgets.QComboBox()
self.noWheelCombos.append(combo)</b>
combo.addItem("Male")
combo.addItem("Female")
combo.installEventFilter(self)
self.tableWidget.setCellWidget(row, column, <b>combo</b>)
# ...
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)
# ...
我有 QtDesigner 的 QTablewidget,我在其中的一列中添加了 Qcombobox。我还添加了事件过滤器来禁用鼠标滚轮事件,但它只适用于最近添加的组合框。如何让它适用于 table 列中的所有组合框?
我在这里错过了什么?
谢谢。
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_())
您犯了一个类似于我在
don't set instance attributes if they are not required: all those self.rowPosition, self.monthList, etc, change everytime the for loop cycles
每次你这样做:
self.combo = QtWidgets.QComboBox()
您正在覆盖实例 (self
) 的 combo
属性;这意味着分配给该属性的最后一个组合将是 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,您已经为其安装了事件过滤器。如果出于任何其他原因,您需要在另一个不是该 table 的组合上安装过滤器,该组合也不会接收到 wheel 事件。
最简单且可能是正确的方法是将这些组合添加到列表中,如果源是该列表的成员,则检查过滤器:
class democode(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self):
super(democode, self).__init__()
self.setupUi(self)
<b>self.noWheelCombos = []</b>
for row in range(0,4):
for column in range(0,3):
if column == 1:
# note that I'm *NOT* using "self"
<b>combo = QtWidgets.QComboBox()
self.noWheelCombos.append(combo)</b>
combo.addItem("Male")
combo.addItem("Female")
combo.installEventFilter(self)
self.tableWidget.setCellWidget(row, column, <b>combo</b>)
# ...
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)
# ...