4

我正在使用 QFileDialog 来选择一个目录。我遇到了无法解决的问题。我花了很多时间在谷歌上搜索这个,但我想出了 zilch。

我指定了起始目录(比如 /home/dhoti/downloads),并且我想禁用该目录上方的导航。例如,不应允许用户访问 /home/dhoti 或 /tmp 等。我该如何实现?

这是我的代码:

QFileDialog dlg(this, "Select Firmware Version");
dlg.setDirectory("/home/dhoti/downloads");
dlg.setFileMode(QFileDialog::DirectoryOnly);
dlg.setOption(QFileDialog::ReadOnly, true);
dlg.setOption(QFileDialog::HideNameFilterDetails, true);
dlg.setViewMode(QFileDialog::List);
dlg.setAcceptMode(QFileDialog::AcceptOpen);
dlg.exec();
qDebug() << "selected files: " << dlg.selectedFiles();

感谢您的帮助

4

4 回答 4

2

您可以检测当前目录何时更改,如果超出限制,请将目录设置回限制目录。

您可以通过非阻塞执行对话框并将QFileDialog::directoryEntered(const QString& directory)信号连接到您自己的插槽来执行此操作,您可以在其中进行检查。如果检查失败,请将当前目录设置为限制目录QFileDialog::setDirectory(const QString& directory)

免责声明我没有尝试过,但如果它不起作用,我会感到惊讶。

于 2012-08-29T08:14:03.277 回答
1

尝试以下操作:

文件对话框.h

#ifndef FILEDIALOG_H
#define FILEDIALOG_H

class QEvent;

#include <QFileDialog>
#include <QString>

class FileDialog : public QFileDialog
{
    Q_OBJECT
public:
    explicit FileDialog(QWidget *parent = 0);
public:
    bool eventFilter(QObject *o, QEvent *e);
    void setTopDir(const QString &path);
    QString topDir() const;
private:
    bool pathFits(const QString &path) const;
private slots:
    void checkHistory();
    void checkGoToParent();
    void checkLineEdit(const QString &text);
private:
    QString mtopDir;
};

#endif // FILEDIALOG_H

文件对话框.cpp

#include "filedialog.h"

#include <QString>
#include <QStringList>
#include <QFileDialog>
#include <QList>
#include <QToolButton>
#include <QDir>
#include <QLineEdit>
#include <QDialogButtonBox>
#include <QEvent>
#include <QKeyEvent>
#include <QAbstractButton>
#include <QCompleter>
#include <QAbstractItemView>
#include <QFileInfo>

FileDialog::FileDialog(QWidget *parent) :
    QFileDialog(parent)
{
    connect(this, SIGNAL(directoryEntered(QString)), this, SLOT(checkHistory()));
    connect(this, SIGNAL(directoryEntered(QString)), this, SLOT(checkGoToParent()));
    connect(findChild<QToolButton *>("backButton"), SIGNAL(clicked()), this, SLOT(checkGoToParent()));
    connect(findChild<QToolButton *>("forwardButton"), SIGNAL(clicked()), this, SLOT(checkGoToParent()));
    connect(findChild<QLineEdit *>("fileNameEdit"), SIGNAL(textChanged(QString)), this, SLOT(checkLineEdit(QString)));
    findChild<QLineEdit *>("fileNameEdit")->installEventFilter(this);
    findChild<QWidget *>("listView")->installEventFilter(this);
    findChild<QWidget *>("treeView")->installEventFilter(this);
    findChild<QLineEdit *>("fileNameEdit")->completer()->popup()->installEventFilter(this);
    setOption(DontUseNativeDialog, true);
}

bool FileDialog::eventFilter(QObject *o, QEvent *e)
{
    if (e->type() != QEvent::KeyPress)
        return false;
    int key = static_cast<QKeyEvent *>(e)->key();
    if (o->objectName() == "listView" || o->objectName() == "treeView")
    {
        return (Qt::Key_Backspace == key && !pathFits(directory().absolutePath()));
    }
    else
    {
        if (Qt::Key_Return != key && Qt::Key_Enter != key)
            return false;
        QString text = findChild<QLineEdit *>("fileNameEdit")->text();
        QString path = QDir::cleanPath(directory().absolutePath() + (text.startsWith("/") ? "" : "/") + text);
        bool a = QDir(text).isAbsolute();
        return !((!a && pathFits(path)) || (a && pathFits(text)));
    }
}

void FileDialog::setTopDir(const QString &path)
{
    if (path == mtopDir)
        return;
    mtopDir = (!path.isEmpty() && QFileInfo(path).isDir()) ? path : QString();
    if (!pathFits(path))
    {
        setDirectory(mtopDir);
        checkHistory();
        checkLineEdit(findChild<QLineEdit *>("fileNameEdit")->text());
    }
    else
    {
        QLineEdit *ledt = findChild<QLineEdit *>("fileNameEdit");
        ledt->setText(ledt->text());
    }
    findChild<QWidget *>("lookInCombo")->setEnabled(mtopDir.isEmpty());
    findChild<QWidget *>("sidebar")->setEnabled(mtopDir.isEmpty());
    checkGoToParent();
}

QString FileDialog::topDir() const
{
    return mtopDir;
}

bool FileDialog::pathFits(const QString &path) const
{
    return mtopDir.isEmpty() || (path.startsWith(mtopDir) && path.length() > mtopDir.length());
}

void FileDialog::checkHistory()
{
    QStringList list = history();
    for (int i = list.size() - 1; i >= 0; --i)
        if (!pathFits(list.at(i)))
            list.removeAt(i);
    setHistory(list);
}

void FileDialog::checkGoToParent()
{
    findChild<QToolButton *>("toParentButton")->setEnabled(pathFits(directory().absolutePath()));
}

void FileDialog::checkLineEdit(const QString &text)
{
    QAbstractButton *btn = findChild<QDialogButtonBox *>("buttonBox")->buttons().first();
    QString path = QDir::cleanPath(directory().absolutePath() + (text.startsWith("/") ? "" : "/") + text);
    bool a = QDir(text).isAbsolute();
    btn->setEnabled(btn->isEnabled() && ((!a && pathFits(path)) || (a && pathFits(text))));
}

这段代码可能看起来很神奇,它并不完美,但它确实有效。我在 Qt 源代码中搜索了 QFileDialog 子对象名称并使用了

找孩子()

方法来访问它们。您只需要使用

设置顶部目录()

方法来指定一个不允许用户进入的目录。

这是使用此类的示例项目:https ://docs.google.com/file/d/0B3P3dwuDIZ1-Q19FbkFMY2puUE0/edit?usp=sharing

于 2013-07-17T22:10:06.757 回答
1

您可以使用@ololoepepe 的解决方案。并使用以下命令清除顶部组合框中不需要的条目:

connect(findChild<QComboBox *>("lookInCombo"), static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged), this, &FileDialog::checkComboBox);

void FileDialog::checkComboBox(int index) {
    int i;
    QComboBox *cb = findChild<QComboBox *>("lookInCombo");

    if (index == 0 && cb->model()->rowCount() > 1) {
        for (i = 0; i < cb->model()->rowCount(); ++i) {
            if (!pathFits(cb->model()->index(i, 0).data().toString() + "/")) {
                cb->model()->removeRow(i);
                --i;
            }
        }
    }
}
于 2015-06-23T07:46:59.767 回答
1

这是最简单的解决方案,使用限制目录遍历所需的最少步骤。

想法:当目录可能被更改时,使用公共信号directoryEntered(const QString &)QFileDialog获取通知,在你的一个类中为它实现插槽,并在那里放置一个逻辑来确保该目录是你需要的目录。

QFileDialog dialog(this);
    connect(&dialog, SIGNAL(directoryEntered(const QString &)), this, SLOT(onFileDialogDirectoryChanged(const QString &)));
于 2016-11-15T11:20:22.503 回答