Как отключить событие колеса мыши в Qcombobox, помещенном в QTableWidget в Python? - PullRequest
1 голос
/ 07 апреля 2020

У меня есть QTablewidget от QtDesigner, в который я добавил 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_())

1 Ответ

1 голос
/ 07 апреля 2020

Вы делаете ошибку, аналогичную той, о которой я вас предупреждал в этом ответе :

не устанавливайте атрибуты экземпляра, если они не требуются: все эти .rowPosition, self.monthList, et c, изменяются каждый раз в течение l oop циклов

Каждый раз, когда вы делаете это:

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) 

        <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 функцией, которая ничего не делает , Это почти то же самое, что создание подкласса и перезапись метода wheelEvent простым pass; это может быть не очень элегантно, но отлично работает для очень простых ситуаций , например:

        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)
                # ...
...