将小部件QCheckBox添加到 PyQt5Python中的 QFileDialog 不起作用

问题描述

我想将 QCheckBox 添加到 QFileDialog。我想使用静态方法 QFileDialog.getSaveFileName() 来显示对话框。

我发现了几个类似的问题,都在 c++ 中:

  1. How to add checkbox to QFileDialog window in QT3?
  2. Adding a widget in QFileDialog
  3. https://www.qtcentre.org/threads/42858-Creating-a-Custom-FileOpen-Dialog
  4. https://forum.qt.io/topic/103964/add-checkbox-to-qfiledialog/7

我已尽力将这些讨论转换为 Python,但尚未找到解决方案。我的代码运行,但复选框不显示,即使我使用 QFileDialog.DontUseNativeDialog。

这就是我继承 QFileDialog 的方式:

from PyQt5.QtWidgets import QFileDialog
from PyQt5.QtWidgets import QCheckBox

class ChkBxFileDialog(QFileDialog):
    def __init__(self,chkBxTitle=""):
        super().__init__()
        self.setoption(QFileDialog.DontUseNativeDialog)
        chkBx = QCheckBox(chkBxTitle)
        self.layout().addWidget(chkBx)
    #end __init__
#end ChkBxFileDialog

我以两种方式运行它。

选项 1(带有额外的 QFileDialog.DontUseNativeDialog):

import sys    
from PyQt5.QtWidgets import QApplication
if __name__ == "__main__":
        app = QApplication(sys.argv)
        fileDialog = ChkBxFileDialog(chkBxTitle="Chkbx")
        fileName = fileDialog.getSaveFileName(filter='*.txt',initialFilter='*.txt',options=QFileDialog.DontUseNativeDialog)[0]
        sys.exit(app.exec_())

选项 2(没有额外的 QFileDialog.DontUseNativeDialog):

import sys    
from PyQt5.QtWidgets import QApplication    
if __name__ == "__main__":
        app = QApplication(sys.argv)
        fileDialog = ChkBxFileDialog(chkBxTitle="Chkbx")
        fileName = fileDialog.getSaveFileName(filter='*.txt',initialFilter='*.txt')[0]
        sys.exit(app.exec_())

该复选框不会与任一选项一起显示。选项 1 使用不同的窗口样式。选项 2 显示了典型的 PyQt QFileDialog。

有人知道我错过了什么吗?

解决方法

问题在于 getSaveFileName 是一个静态方法,因此它们不从 ChkBxFileDialog 继承,因此没有自定义行为。

有两个选项:

  • 不要使用 getSaveFileName 而是直接使用 QFileDialog 实现逻辑:

    import sys
    
    from PyQt5.QtWidgets import QApplication,QCheckBox,QDialog,QFileDialog
    
    
    class ChkBxFileDialog(QFileDialog):
        def __init__(self,chkBxTitle="",filter="*.txt"):
            super().__init__(filter=filter)
            self.setSupportedSchemes(["file"])
            self.setOption(QFileDialog.DontUseNativeDialog)
            self.setAcceptMode(QFileDialog.AcceptSave)
            self.selectNameFilter("*.txt")
            chkBx = QCheckBox(chkBxTitle)
            self.layout().addWidget(chkBx)
    
    
    def main():
        app = QApplication(sys.argv)
        dialog = ChkBxFileDialog()
        if dialog.exec_() == QDialog.Accepted:
            filename = dialog.selectedUrls()[0].toLocalFile()
            print(filename)
    
    
    if __name__ == "__main__":
        main()
    
  • 使用一些技巧来获取 QFileDialog 实例,例如获取所有 topLevel 并验证它是 QFileDialog。

    import sys
    from functools import partial
    
    from PyQt5.QtCore import QTimer
    from PyQt5.QtWidgets import QApplication,QFileDialog
    
    
    def add_checkbox(chkBxTitle):
        for tl in QApplication.topLevelWidgets():
            if isinstance(tl,QFileDialog):
                tl.setOption(QFileDialog.DontUseNativeDialog)
                chkBx = QCheckBox(chkBxTitle)
                tl.layout().addWidget(chkBx)
    
    
    def main():
        app = QApplication(sys.argv)
    
        QTimer.singleShot(1000,partial(add_checkbox,""))
        fileName,_ = QFileDialog.getSaveFileName(
            filter="*.txt",initialFilter="*.txt",options=QFileDialog.DontUseNativeDialog
        )
    
    
    if __name__ == "__main__":
        main()
    
,

我想使用静态方法 QFileDialog.getSaveFileName() 来显示对话框

这不可能。 C++ 代码中定义的静态方法对您的派生类一无所知,因此它将创建基类的一个实例,其中不包含您的修改。您必须显式实例化您的派生类,在实例上调用 exec(),检查返回代码并可能调用其 selectedFiles() 方法以查看选择了哪些文件。