Select QFileDialog pyqt5 中的一个文件或文件夹
Select a file or a folder in QFileDialog pyqt5
我的脚本目前正在使用 QtWidgets.QFileDialog.getOpenFileNames() 让用户 select 文件在 Windows 资源管理器中。现在我想知道是否有办法让它们 select 也可以是文件夹,而不仅仅是文件。有一些类似的帖子,但其中 none 提供了可行的解决方案。我真的不想使用 QFileDialog 文件浏览器来解决这个问题。
QFileDialog 本身不允许这样做。唯一的解决办法是创建自己的实例,做一些小的“修补”。
请注意,为了实现此目的,您不能使用 OS 的 原生 对话框,因为 Qt 几乎 没有 [=20] =] 控制他们;这就是 dialog.DontUseNativeDialog
标志的原因,它是强制性的。
以下代码与静态方法一样有效,returns 选定的项目(或 none,如果对话框被取消)。
def getOpenFilesAndDirs(parent=None, caption='', directory='',
filter='', initialFilter='', options=None):
def updateText():
# update the contents of the line edit widget with the selected files
selected = []
for index in view.selectionModel().selectedRows():
selected.append('"{}"'.format(index.data()))
lineEdit.setText(' '.join(selected))
dialog = QtWidgets.QFileDialog(parent, windowTitle=caption)
dialog.setFileMode(dialog.ExistingFiles)
if options:
dialog.setOptions(options)
dialog.setOption(dialog.DontUseNativeDialog, True)
if directory:
dialog.setDirectory(directory)
if filter:
dialog.setNameFilter(filter)
if initialFilter:
dialog.selectNameFilter(initialFilter)
# by default, if a directory is opened in file listing mode,
# QFileDialog.accept() shows the contents of that directory, but we
# need to be able to "open" directories as we can do with files, so we
# just override accept() with the default QDialog implementation which
# will just return exec_()
dialog.accept = lambda: QtWidgets.QDialog.accept(dialog)
# there are many item views in a non-native dialog, but the ones displaying
# the actual contents are created inside a QStackedWidget; they are a
# QTreeView and a QListView, and the tree is only used when the
# viewMode is set to QFileDialog.Details, which is not this case
stackedWidget = dialog.findChild(QtWidgets.QStackedWidget)
view = stackedWidget.findChild(QtWidgets.QListView)
view.selectionModel().selectionChanged.connect(updateText)
lineEdit = dialog.findChild(QtWidgets.QLineEdit)
# clear the line edit contents whenever the current directory changes
dialog.directoryEntered.connect(lambda: lineEdit.setText(''))
dialog.exec_()
return dialog.selectedFiles()
我的脚本目前正在使用 QtWidgets.QFileDialog.getOpenFileNames() 让用户 select 文件在 Windows 资源管理器中。现在我想知道是否有办法让它们 select 也可以是文件夹,而不仅仅是文件。有一些类似的帖子,但其中 none 提供了可行的解决方案。我真的不想使用 QFileDialog 文件浏览器来解决这个问题。
QFileDialog 本身不允许这样做。唯一的解决办法是创建自己的实例,做一些小的“修补”。
请注意,为了实现此目的,您不能使用 OS 的 原生 对话框,因为 Qt 几乎 没有 [=20] =] 控制他们;这就是 dialog.DontUseNativeDialog
标志的原因,它是强制性的。
以下代码与静态方法一样有效,returns 选定的项目(或 none,如果对话框被取消)。
def getOpenFilesAndDirs(parent=None, caption='', directory='',
filter='', initialFilter='', options=None):
def updateText():
# update the contents of the line edit widget with the selected files
selected = []
for index in view.selectionModel().selectedRows():
selected.append('"{}"'.format(index.data()))
lineEdit.setText(' '.join(selected))
dialog = QtWidgets.QFileDialog(parent, windowTitle=caption)
dialog.setFileMode(dialog.ExistingFiles)
if options:
dialog.setOptions(options)
dialog.setOption(dialog.DontUseNativeDialog, True)
if directory:
dialog.setDirectory(directory)
if filter:
dialog.setNameFilter(filter)
if initialFilter:
dialog.selectNameFilter(initialFilter)
# by default, if a directory is opened in file listing mode,
# QFileDialog.accept() shows the contents of that directory, but we
# need to be able to "open" directories as we can do with files, so we
# just override accept() with the default QDialog implementation which
# will just return exec_()
dialog.accept = lambda: QtWidgets.QDialog.accept(dialog)
# there are many item views in a non-native dialog, but the ones displaying
# the actual contents are created inside a QStackedWidget; they are a
# QTreeView and a QListView, and the tree is only used when the
# viewMode is set to QFileDialog.Details, which is not this case
stackedWidget = dialog.findChild(QtWidgets.QStackedWidget)
view = stackedWidget.findChild(QtWidgets.QListView)
view.selectionModel().selectionChanged.connect(updateText)
lineEdit = dialog.findChild(QtWidgets.QLineEdit)
# clear the line edit contents whenever the current directory changes
dialog.directoryEntered.connect(lambda: lineEdit.setText(''))
dialog.exec_()
return dialog.selectedFiles()