首页 > 解决方案 > 如何覆盖 Qt Designer 生成的 PySide2 小部件的方法?

问题描述

我正在尝试通过在 powertoys 中创建类似 powerrename 工具的模拟可执行文件来学习 PySide2。

成功加载 Ui_PowerRenameDialog 后,该对话框由 Qt Designer 创建的 .ui 文件生成。

我在尝试完成一项功能时遇到错误,该功能可以让用户将文件从 Windows 资源管理器拖放到.ui 文件中嵌入的 QTreeWidget。

这是我的完整代码:

import sys
from PySide6 import QtWidgets
from PySide6 import QtCore
from ui_powerRename import Ui_PowerRenameDialog

class MyPowerRenameDialog(QtWidgets.QDialog):
    def __init__(self,parent=None):
        super().__init__(parent)

        self.ui = Ui_PowerRenameDialog()
        self.ui.setupUi(self)

        self.modify_ui()
        self.create_connections()


    def modify_ui(self):
        self.ui.previewTreeWidget.setAcceptDrops(True)

        def dragEnterEvent(self,evt):
            if evt.mimeData().hasUrls():
                evt.accept()
            else:
                evt.ignore()

        def dropEvent(self,evt):
            for url in evt.mimeData().urls():
                newItem = QtWidgets.QTreeWidgetItem(url)
                self.addTopLevelItem(newItem)

        self.ui.previewTreeWidget.dragEnterEvent = dragEnterEvent
        self.ui.previewTreeWidget.dropEvent = dropEvent

        items = QtWidgets.QTreeWidgetItemIterator(self.ui.previewTreeWidget)
        for item in items:
            item.value().setFlags(item.value().flags()|QtCore.Qt.ItemIsUserCheckable)
            item.value().setCheckState(0,QtCore.Qt.Checked)

    def create_connections(self):
        self.ui.cancelButton.clicked.connect(self.close)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)

    window = MyPowerRenameDialog()
    window.show()

    sys.exit(app.exec_())

==================================================== ===============

这是我到目前为止所取得的成果(红色区域是 QTreeWidget): 可执行快照:
可执行快照

如您所见,我尝试重载 QTreeWidget 对象的dragEnterEventdropEvent方法。覆盖代码的最重要部分如下所示:

def dragEnterEvent(self,evt):
    if evt.mimeData().hasUrls():
        evt.accept()
    else:
        evt.ignore()
def dropEvent(self,evt):
    for url in evt.mimeData().urls():
        newItem = QtWidgets.QTreeWidgetItem(url)
        self.addTopLevelItem(newItem)

self.ui.previewTreeWidget.dragEnterEvent = dragEnterEvent
self.ui.previewTreeWidget.dropEvent = dropEvent

==================================================== ==============

问题是当试图将任何内容从资源管理器拖放到 QTreeWidget 实例时,我得到了一个 TypeError

TypeError: dragEnterEvent() missing 1 required positional argument: 'evt'.

我试图搜索类似的问题。一个潜在的解决方案是使用installEventFilter。但这只会导致我出现更多错误。

我的意思是 PySide2 文档真的很乱,我无法从他们格式错误的文档中获得任何有用的信息。

文档中的例子很少。而且大部分都是直接从C++版本复制过来的,甚至没有改成Python语法。

关于如何实现该功能的任何想法?

有什么建议的学习 PySide2 的方法吗?

标签: pythonpyside6

解决方案


错误来自dragEnterEventanddropEvent没有绑定到一个类的事实(因为它们嵌套在另一个函数中),这意味着它self不是自动的,这就是为什么当它被调用时它需要两个参数selfevt,但仅 Qt 框架通过evt

要修改Ui_PowerRenameDialog您的行为,应该创建一个子类Ui_PowerRenameDialog并在该子类中实现两个事件方法


推荐阅读