PyQt5中实现widget的功能,无需修改pyuic5生成的文件

Posted

技术标签:

【中文标题】PyQt5中实现widget的功能,无需修改pyuic5生成的文件【英文标题】:Implement function of a widget in PyQt5 without modify the file generated by pyuic5 【发布时间】:2020-12-28 19:42:40 【问题描述】:

我想向小部件的方法添加更多功能,但不修改 pyuic5 生成的文件,因为我正在 QtDesigner 中进行频繁修改,因此文件在不断发展。

我正在考虑将 Main 类中的自定义函数分配给我想要扩展的方法,例如:

pyuic5生成的spin.py

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(257, 181)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.spinBox = QtWidgets.QSpinBox(self.centralwidget)
        self.spinBox.setGeometry(QtCore.QRect(102, 60, 42, 22))
        self.spinBox.setObjectName("spinBox")
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 257, 21))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        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"))

自定义 .py 文件 spin_main.py

from PyQt5 import QtWidgets, QtCore, QtGui
from spin import Ui_MainWindow
class MainWindow(Ui_MainWindow, QtWidgets.QMainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.spinBox.mousePressEvent = self.clickEvent

    def clickEvent(self, event):
        print("click event")
    
if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    ui = MainWindow()
    ui.show()
    sys.exit(app.exec_())

在这种情况下,我只想添加打印并且该方法继续其功能,但这样我无法保留以前的功能。你有什么推荐给我的?

【问题讨论】:

【参考方案1】:

简单的解决方案是调用基础方法:

def clickEvent(self, event):
    print("click event")
    QtWidgets.QSpinBox.mousePressEvent(self.spinBox, event)

但它不是很优雅所以我不推荐它,而是有其他选择,例如:

1。促销

您可以创建自定义 QSpinBox 并通过 QtDesigner 将其包含在 .ui 中(参见 1 和 2):

spinbox.py

from PyQt5 import QtCore, QtWidgets


class SpinBox(QtWidgets.QSpinBox):
    clicked = QtCore.pyqtSignal()

    def mousePressEvent(self, event):
        super().mousePressEvent(event)
        self.clicked.emit()

spin.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>800</width>
    <height>600</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralwidget">
   <widget class="SpinBox" name="spinBox">
    <property name="geometry">
     <rect>
      <x>102</x>
      <y>60</y>
      <width>78</width>
      <height>28</height>
     </rect>
    </property>
   </widget>
  </widget>
  <widget class="QMenuBar" name="menubar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>800</width>
     <height>30</height>
    </rect>
   </property>
  </widget>
  <widget class="QStatusBar" name="statusbar"/>
 </widget>
 <customwidgets>
  <customwidget>
   <class>SpinBox</class>
   <extends>QSpinBox</extends>
   <header>spinbox.h</header>
  </customwidget>
 </customwidgets>
 <resources/>
 <connections/>
</ui>
pyuic5 spin.ui -o spin.py

ma​​in.py

from PyQt5 import QtWidgets, QtCore, QtGui
from spin import Ui_MainWindow


class MainWindow(Ui_MainWindow, QtWidgets.QMainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.spinBox.clicked.connect(self.clickEvent)

    def clickEvent(self):
        print("click event")

if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    ui = MainWindow()
    ui.show()
    sys.exit(app.exec_())

2。事件过滤器

既然你想要监听一个事件,那么你可以实现一个事件过滤器,这样就不再需要重写任何方法:

from PyQt5 import QtWidgets, QtCore, QtGui
from spin import Ui_MainWindow


class MainWindow(Ui_MainWindow, QtWidgets.QMainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.spinBox.installEventFilter(self)

    def eventFilter(self, obj, event):
        if obj is self.spinBox and event.type() == QtCore.QEvent.MouseButtonPress:
            self.clickEvent()
        return super().eventFilter(obj, event)

    def clickEvent(self):
        print("click event")


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    ui = MainWindow()
    ui.show()
    sys.exit(app.exec_())

【讨论】:

以上是关于PyQt5中实现widget的功能,无需修改pyuic5生成的文件的主要内容,如果未能解决你的问题,请参考以下文章

在PyQt5中实现泛洪填充

Android开发中实现桌面小部件

如何在 VBA 中实现 Zip 功能?

PyQt5代码在某个功能启动时失败

如何在 Flutter 中实现 CheckBox?

flutter 中实现磨砂玻璃效果