欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 汽车 > 新车 > Qt进阶开发:QFileSystemModel的使用

Qt进阶开发:QFileSystemModel的使用

2025/4/5 20:38:25 来源:https://blog.csdn.net/TechNomad/article/details/146924430  浏览:    关键词:Qt进阶开发:QFileSystemModel的使用

文章目录

    • 一、QFileSystemModel的基本介绍
    • 二、QFileSystemModel的基本使用
      • 2.1 在 QTreeView 中使用
      • 2.2 在 QListView 中使用
      • 2.3 在 QTableView 中使用
    • 三、QFileSystemModel的常用API
      • 3.1 设置根目录
      • 3.2 过滤文件
        • 3.2.1 仅显示文件
        • 3.2.2 只显示特定后缀的文件
        • 3.2.3 只显示目录
    • 四、监听文件系统变化
      • 1. QFileSystemModel常见信号
      • 2. 信号的使用示例
        • 2.1 监听目录加载完成
        • 2.2 监听文件重命名
        • 2.3 监听根路径变化
        • 2.4 监听文件/目录添加
        • 2.5 监听文件/目录删除
      • 3. 动态更新QLabel显示选中文件
      • 4.右键菜单的实现
    • 五、继承QFileSystemModel并进行自定义

一、QFileSystemModel的基本介绍

  QFileSystemModel 是 Qt 框架中用于管理和显示本地文件系统的 MVC 结构中的 Model。它可以与 QTreeView、QListView 或 QTableView 结合使用,实现文件管理功能。相比 QDirModel,QFileSystemModel 只会加载需要的目录,支持懒加载,在处理大文件夹时更高效。

主要特点:

  • 继承自 QAbstractItemModel,提供标准的文件系统数据模型。
  • 懒加载(Lazy Loading):不会一次性加载整个文件系统,只会加载需要的部分,提升性能。
  • 支持文件过滤(setFilter()、setNameFilters())。
  • 自动监听文件系统变化,实时更新 UI。

二、QFileSystemModel的基本使用

2.1 在 QTreeView 中使用

#include <QApplication>
#include <QTreeView>
#include <QFileSystemModel>int main(int argc, char *argv[]) {QApplication app(argc, argv);QFileSystemModel model;model.setRootPath(QDir::homePath());  // 设置根目录QTreeView treeView;treeView.setModel(&model);treeView.setRootIndex(model.index(QDir::homePath())); // 绑定视图treeView.show();return app.exec();
}

在这里插入图片描述

2.2 在 QListView 中使用

QListView listView;
listView.setModel(&model);
listView.setRootIndex(model.index(QDir::homePath()));
listView.show();

在这里插入图片描述

2.3 在 QTableView 中使用

QTableView tableView;
tableView.setModel(&model);
tableView.setRootIndex(model.index(QDir::homePath()));
tableView.show();

在这里插入图片描述

三、QFileSystemModel的常用API

在这里插入图片描述

3.1 设置根目录

model.setRootPath(QDir::homePath());  // 设置为用户目录

注意:setRootPath() 只是设置数据模型的根目录,并不会影响 QTreeView 显示的目录。要限制 QTreeView 的显示范围,需使用 setRootIndex()。

3.2 过滤文件

3.2.1 仅显示文件
model.setFilter(QDir::Files | QDir::NoDotAndDotDot);

QDir::Files 只显示文件,QDir::NoDotAndDotDot 过滤掉 . 和 … 目录。

3.2.2 只显示特定后缀的文件
model.setNameFilters(QStringList() << "*.cpp" << "*.h");
model.setNameFilterDisables(false); // 过滤掉不符合的文件

这将只显示 .cpp 和 .h 文件。

3.2.3 只显示目录
model.setFilter(QDir::Dirs | QDir::NoDotAndDotDot);

只显示目录,隐藏所有文件。

四、监听文件系统变化

1. QFileSystemModel常见信号

在这里插入图片描述

2. 信号的使用示例

2.1 监听目录加载完成

当 QFileSystemModel 完成某个目录的加载时,可以使用 directoryLoaded() 信号:

#include <QApplication>
#include <QTreeView>
#include <QFileSystemModel>
#include <QDebug>int main(int argc, char *argv[]) {QApplication app(argc, argv);QFileSystemModel model;QObject::connect(&model, &QFileSystemModel::directoryLoaded, [](const QString &path) {qDebug() << "目录加载完成:" << path;});model.setRootPath(QDir::homePath());QTreeView treeView;treeView.setModel(&model);treeView.setRootIndex(model.index(QDir::homePath()));treeView.show();return app.exec();
}

解释:

  • 当某个目录的内容加载完成时,会输出 “目录加载完成:/home/user”(Linux/macOS)或 “目录加载完成:C:/Users/xxx”(Windows)。
  • 适用场景:可以在目录加载完成后执行额外的操作,比如统计文件数量、更新 UI 状态等。
2.2 监听文件重命名

如果用户在文件管理器中修改了文件名,可以捕获 fileRenamed() 信号:

QObject::connect(&model, &QFileSystemModel::fileRenamed, [](const QString &path, const QString &oldName, const QString &newName) {qDebug() << "文件重命名:" << path + "/" + oldName << " -> " << newName;});

解释:

  • fileRenamed(path, oldName, newName) 提供了文件所在目录、旧文件名和新文件名。
  • 适用场景:可以记录日志、同步到数据库或通知 UI 更新。
2.3 监听根路径变化

当 setRootPath() 发生变化时,触发 rootPathChanged():

QObject::connect(&model, &QFileSystemModel::rootPathChanged, [](const QString &newPath) {qDebug() << "根目录更改为:" << newPath;});

适用场景:当用户更改文件浏览的根目录时,可以执行特定操作,比如清除旧数据、更新 UI 等。

2.4 监听文件/目录添加

当新文件或目录被创建时,会触发 rowsInserted() 信号:

QObject::connect(&model, &QFileSystemModel::rowsInserted, [&model](const QModelIndex &parent, int start, int end) {for (int row = start; row <= end; ++row) {QModelIndex index = model.index(row, 0, parent);qDebug() << "新文件/文件夹:" << model.filePath(index);}});

解释:

  • rowsInserted() 触发时,会输出新文件或目录的路径。
  • 适用场景:当用户创建新文件/文件夹时,自动执行相应的 UI 更新或日志记录。
2.5 监听文件/目录删除

当某个文件或文件夹被删除时,会触发 rowsRemoved() 信号:

QObject::connect(&model, &QFileSystemModel::rowsRemoved, [](const QModelIndex &parent, int start, int end) {qDebug() << "文件/文件夹删除:" << start << "到" << end;});

3. 动态更新QLabel显示选中文件

#include <QApplication>
#include <QTreeView>
#include <QFileSystemModel>
#include <QLabel>
#include <QVBoxLayout>
#include <QWidget>
#include <QItemSelectionModel>int main(int argc, char *argv[]) {QApplication app(argc, argv);QFileSystemModel model;model.setRootPath(QDir::homePath());QTreeView treeView;treeView.setModel(&model);treeView.setRootIndex(model.index(QDir::homePath()));QLabel label;label.setText("选中文件路径显示区域");// 监听选中项变化QObject::connect(treeView.selectionModel(), &QItemSelectionModel::selectionChanged, [&model, &treeView, &label](const QItemSelection &selected, const QItemSelection &) {if (!selected.indexes().isEmpty()) {QModelIndex index = selected.indexes().first();label.setText(model.filePath(index));}});// 设置 UI 布局QWidget window;QVBoxLayout layout;layout.addWidget(&treeView);layout.addWidget(&label);window.setLayout(&layout);window.show();return app.exec();
}

输出显示:
在这里插入图片描述

4.右键菜单的实现

可以为 QTreeView 添加右键菜单,实现文件操作(打开、删除等)。

#include <QMenu>
#include <QAction>
#include <QDesktopServices>
#include <QUrl>void onContextMenu(const QPoint &pos) {QModelIndex index = treeView.indexAt(pos);if (!index.isValid()) return;QMenu menu;QAction *openAction = new QAction("打开", &menu);QAction *deleteAction = new QAction("删除", &menu);connect(openAction, &QAction::triggered, [index, &model]() {QString path = model.filePath(index);QDesktopServices::openUrl(QUrl::fromLocalFile(path));});connect(deleteAction, &QAction::triggered, [index, &model]() {model.remove(index);});menu.addAction(openAction);menu.addAction(deleteAction);menu.exec(treeView.viewport()->mapToGlobal(pos));
}

五、继承QFileSystemModel并进行自定义

继承 QFileSystemModel 并进行自定义,可以用来修改文件显示方式、拦截用户操作、隐藏某些文件或添加额外的功能。
可以通过继承 QFileSystemModel来:

  • 自定义文件名显示(如:在文件名前加上 [文件] 或 [目录])。
  • 自定义文件图标。
  • 隐藏某些文件/目录。
  • 拦截文件删除或重命名操作。

示例 1:重写 data() 修改文件显示
默认情况下,QFileSystemModel 只显示文件名。如果我们想要修改显示内容(如:文件名前加上类型信息),可以重写 data() 方法:

#include <QFileSystemModel>
#include <QDebug>class CustomFileSystemModel : public QFileSystemModel {
public:explicit CustomFileSystemModel(QObject *parent = nullptr) : QFileSystemModel(parent) {}QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override {if (!index.isValid()) return QVariant();if (role == Qt::DisplayRole) {  // 修改显示文本QString fileName = QFileSystemModel::data(index, Qt::DisplayRole).toString();if (isDir(index))return "[目录] " + fileName;elsereturn "[文件] " + fileName;}return QFileSystemModel::data(index, role);}
};

示例 2:自定义文件图标
如果想要为特定类型的文件设置不同的图标(如 .txt 用文本图标,.cpp 用代码图标),可以重写 data() 的 Qt::DecorationRole:

#include <QFileIconProvider>
#include <QFileInfo>class CustomFileSystemModel : public QFileSystemModel {
public:explicit CustomFileSystemModel(QObject *parent = nullptr) : QFileSystemModel(parent) {}QVariant data(const QModelIndex &index, int role = Qt::DecorationRole) const override {if (!index.isValid()) return QVariant();if (role == Qt::DecorationRole) {QFileInfo fileInfo = this->fileInfo(index);if (fileInfo.suffix() == "txt")return QIcon(":/icons/text.png"); // 替换为你的图标else if (fileInfo.suffix() == "cpp" || fileInfo.suffix() == "h")return QIcon(":/icons/code.png"); // 替换为你的图标else if (fileInfo.isDir())return QIcon(":/icons/folder.png"); // 替换为你的文件夹图标}return QFileSystemModel::data(index, role);}
};

示例 2:自定义双击打开文件
如果希望在双击文件时执行特定操作(如:打开文件或显示信息),可以在 QTreeView 中监听 doubleClicked() 信号:

QObject::connect(&treeView, &QTreeView::doubleClicked, [&](const QModelIndex &index) {QString filePath = model.filePath(index);if (model.isDir(index)) {qDebug() << "双击打开目录:" << filePath;} else {qDebug() << "双击打开文件:" << filePath;QDesktopServices::openUrl(QUrl::fromLocalFile(filePath));}
});

下面是一个完整的示例,将以上功能整合到 QTreeView 中:

#include <QApplication>
#include <QTreeView>
#include <QFileSystemModel>
#include <QVBoxLayout>
#include <QWidget>
#include <QMessageBox>
#include <QDesktopServices>
#include <QUrl>class CustomFileSystemModel : public QFileSystemModel {
public:explicit CustomFileSystemModel(QObject *parent = nullptr) : QFileSystemModel(parent) {}QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override {if (!index.isValid()) return QVariant();if (role == Qt::DisplayRole) {QString fileName = QFileSystemModel::data(index, Qt::DisplayRole).toString();return isDir(index) ? "[目录] " + fileName : "[文件] " + fileName;}return QFileSystemModel::data(index, role);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);CustomFileSystemModel model;model.setRootPath(QDir::homePath());QTreeView treeView;treeView.setModel(&model);treeView.setRootIndex(model.index(QDir::homePath()));QObject::connect(&treeView, &QTreeView::doubleClicked, [&](const QModelIndex &index) {QString filePath = model.filePath(index);QDesktopServices::openUrl(QUrl::fromLocalFile(filePath));});treeView.resize(800, 600);treeView.show();return app.exec();
}

在这里插入图片描述

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词