首页 > 解决方案 > 为什么 PyQt5 向我显示我的第一个文件对话框而不是我的第二个?

问题描述

我正在尝试使用 PyQt5 为 python 项目制作一个基本的前端 gui。具体来说,我想从两件事开始。首先,我想提示用户选择一个输入数据文件。然后,我想提示用户选择一个可以保存文件的目录。

可以在下面找到代码的工作示例。我的问题是为什么当用户的代码中有两个提示时用户只收到一个提示(首先是输入文件,然后是保存目录)。代码不会抛出错误;注释掉输入文件例程(self.select_save_directory()在 的__init__方法中Interface)将允许保存目录例程运行。

首先,所有的进口。

import sys
from PyQt5 import QtWidgets # QtGui, QtCore

然后,提示用户选择输入 .csv 文件的小部件。

class FileSelectionWidget(QtWidgets.QWidget):

    def __init__(self, parent=None):
        super().__init__(parent)
        button = QtWidgets.QPushButton("Click here and select the data file you want to read")
        button.clicked.connect(self.on_clicked)
        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(button)
        self._fpath = None

    @property
    def fpath(self):
        return self._fpath

    def on_clicked(self):
        dialog = QtWidgets.QFileDialog(
            self,
            "Select input file",
            "path",
            "*.csv",
            supportedSchemes=["file"],
            options=QtWidgets.QFileDialog.DontUseNativeDialog)
        fpath = dialog.getOpenFileName(None, 'Open file', '/home')[0]
        self._fpath = fpath
        self.close()
        print(self.fpath) # verify

然后,提示用户选择保存目录的小部件。

class DirectorySelectionWidget(QtWidgets.QWidget):

    """
    This class allows the user to select the directory
    to save files into via gui.
    """

    def __init__(self, parent=None):
        super().__init__(parent)
        button = QtWidgets.QPushButton("Click here and select the directory \nin which you would like to save files")
        button.clicked.connect(self.on_clicked)
        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(button)
        self._savedir = None

    @property
    def savedir(self):
        return self._savedir

    def on_clicked(self):
        dialog = QtWidgets.QFileDialog()
        savedir = dialog.getExistingDirectory(None, "Select directory")
        self._savedir = savedir
        self.close()
        print(self.savedir) # verify

我计划添加我现有的后端。

class BackEnd():

    def __init__(self):
        super().__init__()

    # data processing functions

最后是从后端继承方法并且可以调用/实例化上面的UI方法的接口。

class Interface(BackEnd):

    def __init__(self):
        """

        """
        super().__init__()
        self.select_data_file()
        self.select_save_directory()
        ... # more back-end things

    def select_data_file(self):
        app = QtWidgets.QApplication(sys.argv)
        file_selection_widget = FileSelectionWidget()
        file_selection_widget.show()
        sys.exit(app.exec_())

    def select_save_directory(self):
        app = QtWidgets.QApplication(sys.argv)
        directory_selection_widget = DirectorySelectionWidget()
        directory_selection_widget.show()
        sys.exit(app.exec_())

if __name__ == '__main__':
    interface = Interface()

标签: python-3.xuser-interfacedialogpyqt5widget

解决方案


sys.exit([arg]):
退出 Python。

这与 PyQt 无关。问题是select_data_file和都以select_save_directory结尾sys.exit,导致您的程序立即退出,无论之后调用什么函数。

除此之外,如果您要重用对话框(或在程序的生命周期内显示其他内容),通常最好避免重新创建 QApplication。

您可以做的在您main的. 这允许保持与您现在相同的“阻塞”效果(使用),一旦关闭对话框,该效果就会被清除,然后您可以在需要时显示其他窗口。exec_()show()app.exec_()

请注意,为了防止进一步的程序退出,您应该使用setQuitOnLastWindowClosed(False),然后QApplication.quit()在您想要实际退出时手动调用。

最后,请注意getOpenFileNamegetExistingDirectory都是QFileDialog 的静态方法。它们返回一个的预先构建的文件对话框实例,因此在此之前创建一个实例是没有意义的。
您需要将参数添加到静态函数本身。

   def on_clicked(self):
        fpath, _ = QtWidgets.QFileDialog.getOpenFileName(
            None, 'Select input file', '/home', '*.csv', 
            options=QtWidgets.QFileDialog.DontUseNativeDialog)
        if fpath:
            self._fpath = fpath
            self.close()

推荐阅读