接受单个文件或单个目录的QFileDialog

时间:2014-12-17 07:29:14

标签: c++ qt qfiledialog

是否可以显示QFileDialog,用户可以在其中选择一个文件或目录?

QFileDialog::getOpenFileName()只接受文件,而QFileDialog::getExistingDirectory()只是目录,但我需要显示一个文件对话框,可以接受文件或目录(这对我的程序有意义)。 QFileDialog::​Options没有任何希望。

4 个答案:

答案 0 :(得分:4)

QFileDialog目前不支持此功能。我认为这里的主要问题是FileMode不是Q_FLAGS而且值也不是2的幂,所以,你不能写这个来解决这个问题。

setFileMode(QFileDialog::Directory|QFileDialog::ExistingFiles)

要解决这个问题,你需要相当多的摆弄,例如:

  • 覆盖打开按钮点击操作。

  • 为文件和目录正确获取“treeview”索引。

我下面的尝试演示了前者,但我并没有真正解决第二种问题,因为这似乎涉及到更多摆弄选择模型。

的main.cpp

#include <QFileDialog>
#include <QApplication>
#include <QWidget>
#include <QTreeWidget>
#include <QPushButton>
#include <QStringList>
#include <QModelIndex>
#include <QDir>
#include <QDebug>

class FileDialog : public QFileDialog
{
    Q_OBJECT
    public:
        explicit FileDialog(QWidget *parent = Q_NULLPTR)
            : QFileDialog(parent)
        {
            setOption(QFileDialog::DontUseNativeDialog);
            setFileMode(QFileDialog::Directory);
            // setFileMode(QFileDialog::ExistingFiles);
            for (auto *pushButton : findChildren<QPushButton*>()) {
                qDebug() << pushButton->text();
                if (pushButton->text() == "&Open" || pushButton->text() == "&Choose") {
                    openButton = pushButton;
                    break;
                }
            }
            disconnect(openButton, SIGNAL(clicked(bool)));
            connect(openButton, &QPushButton::clicked, this, &FileDialog::openClicked);
            treeView = findChild<QTreeView*>();
        }

        QStringList selected() const
        {
            return selectedFilePaths;
        }

    public slots:
        void openClicked()
        {
            selectedFilePaths.clear();
            qDebug() << treeView->selectionModel()->selection();
            for (const auto& modelIndex : treeView->selectionModel()->selectedIndexes()) {
                qDebug() << modelIndex.column();
                if (modelIndex.column() == 0)
                    selectedFilePaths.append(directory().absolutePath() + modelIndex.data().toString());
            }
            emit filesSelected(selectedFilePaths);
            hide();
            qDebug() << selectedFilePaths;
       }

    private:
        QTreeView *treeView;
        QPushButton *openButton;
        QStringList selectedFilePaths;
};

#include "main.moc"

int main(int argc, char **argv)
{
    QApplication application(argc, argv);
    FileDialog fileDialog;
    fileDialog.show();
    return application.exec();
}

main.pro

TEMPLATE = app
TARGET = main
QT += widgets
CONFIG += c++11
SOURCES += main.cpp

构建并运行

qmake && make && ./main

答案 1 :(得分:2)

一个很老的问题,但是我认为我有一个比大多数懒惰的解决方案更简单的解决方案。 您可以将QFileDialog的currentChanged信号与动态更改fileMode的函数连接。

//header
class my_main_win:public QMainWindow  
{
private:
    QFileDialog file_dialog;    
}

//constructor 

my_main_win(QWidget * parent):QMainWindow(parent)
{
    connect(&file_dialog,QFileDialog::currentChanged,this,[&](const QString & str)
        {
        QFileInfo info(str);
        if(info.isFile())
            file_dialog.setFileMode(QFileDialog::ExistingFile);
        else if(info.isDir())
            file_dialog.setFileMode(QFileDialog::Directory);
    });
}

然后只需按需调用file_dialog。

if(file_dialog.exec()){
    QStringList dir_names=file_dialog.selectedFiles():
}

答案 2 :(得分:1)

对我有用的是:

file_dialog.setProxyModel(nullptr);

建议here

class FileFilterProxyModel : public QSortFilterProxyModel
{
protected:
    virtual bool filterAcceptsRow(int sourceRow, const QModelIndex& sourceParent) const
    {
        QFileSystemModel* fileModel = qobject_cast<QFileSystemModel*>(sourceModel());
        return (fileModel!=NULL && fileModel->isDir(sourceModel()->index(sourceRow, 0, sourceParent))) || QSortFilterProxyModel::filterAcceptsRow(sourceRow, sourceParent);
    }
};
...
FileFilterProxyModel* proxyModel = new FileFilterProxyModel;
file_dialog.setProxyModel(proxyModel);

建议here

class FileDialog : public QFileDialog
{
    Q_OBJECT
public:
    explicit FileDialog(QWidget *parent = Q_NULLPTR)
        : QFileDialog(parent)
    {
        m_btnOpen = NULL;
        m_listView = NULL;
        m_treeView = NULL;
        m_selectedFiles.clear();

        this->setOption(QFileDialog::DontUseNativeDialog, true);
        this->setFileMode(QFileDialog::Directory);
        QList<QPushButton*> btns = this->findChildren<QPushButton*>();
        for (int i = 0; i < btns.size(); ++i) {
            QString text = btns[i]->text();
            if (text.toLower().contains("open") || text.toLower().contains("choose"))
            {
                m_btnOpen = btns[i];
                break;
            }
        }

        if (!m_btnOpen) return;

        m_btnOpen->installEventFilter(this);
        //connect(m_btnOpen, SIGNAL(changed()), this, SLOT(btnChanged()))
        m_btnOpen->disconnect(SIGNAL(clicked()));
        connect(m_btnOpen, SIGNAL(clicked()), this, SLOT(chooseClicked()));


        m_listView = findChild<QListView*>("listView");
        if (m_listView)
            m_listView->setSelectionMode(QAbstractItemView::ExtendedSelection);

        m_treeView = findChild<QTreeView*>();
        if (m_treeView)
            m_treeView->setSelectionMode(QAbstractItemView::ExtendedSelection);
    }

    QStringList selectedFiles()
    {
        return m_selectedFiles;
    }

    bool eventFilter( QObject* watched, QEvent* event )
    {
        QPushButton *btn = qobject_cast<QPushButton*>(watched);
        if (btn)
        {
            if(event->type()==QEvent::EnabledChange) {
                if (!btn->isEnabled())
                    btn->setEnabled(true);
            }
        }

        return QWidget::eventFilter(watched, event);
    }

public slots:
    void chooseClicked()
    {
        QModelIndexList indexList = m_listView->selectionModel()->selectedIndexes();
        foreach (QModelIndex index, indexList)
        {
            if (index.column()== 0)
                m_selectedFiles.append(this->directory().absolutePath() + "/" + index.data().toString());
        }
        QDialog::accept();
    }

private:
    QListView *m_listView;
    QTreeView *m_treeView;
    QPushButton *m_btnOpen;
    QStringList m_selectedFiles;
};
建议here

。原作者和我的学分。

答案 3 :(得分:1)

连接到currentChanged信号,然后将文件模式设置为当前选定的项目(目录或文件)。这是一个Python3实现:

class GroupFileObjectDialog(QFileDialog):

    def __init__(self, parent):
        super().__init__(parent)
        self.setOption(QFileDialog.DontUseNativeDialog)
        self.setFileMode(QFileDialog.Directory)
        self.currentChanged.connect(self._selected)

    def _selected(self,name):
        if os.path.isdir(name):
            self.setFileMode(QFileDialog.Directory)
        else:
            self.setFileMode(QFileDialog.ExistingFile)

在Linux / Ubuntu18.04上运行的PyQt 5.14上进行了测试。