Qt 学习之路:QFileSystemModel

上一章我们详细了解了QStringListModel。本章我们将再来介绍另外一个内置模型:QFileSystemModel。看起来,QFileSystemModelQStringListModel要复杂得多;事实也是如此。但是,虽然功能强大,QFileSystemModel的使用还是简单的。

让我们从 Qt 内置的模型说起。实际上,Qt 内置了两种模型:QStandardItemModelQFileSystemModelQStandardItemModel是一种多用途的模型,能够让列表、表格、树等视图显示不同的数据结构。这种模型会将数据保存起来。试想一下,列表和表格所要求的数据结构肯定是不一样的:前者是一维的,后者是二维的。因此,模型需要保存有实际数据,当视图是列表时,以一维的形式提供数据;当视图是表格时,以二维的形式提供数据。QFileSystemModel则是另外一种方式。它的作用是维护一个目录的信息。因此,它不需要保存数据本身,而是保存这些在本地文件系统中的实际数据的一个索引。我们可以利用QFileSystemModel显示文件系统的信息、甚至通过模型来修改文件系统。

QTreeView是最适合应用QFileSystemModel的视图。下面我们看一段代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

FileSystemWidget::FileSystemWidget(QWidget *parent) :

QWidget(parent)

{

model = new QFileSystemModel;

model->setRootPath(QDir::currentPath());

treeView = new QTreeView(this);

treeView->setModel(model);

treeView->setRootIndex(model->index(QDir::currentPath()));

QPushButton *mkdirButton = new QPushButton(tr("Make Directory..."), this);

QPushButton *rmButton = new QPushButton(tr("Remove"), this);

QHBoxLayout *buttonLayout = new QHBoxLayout;

buttonLayout->addWidget(mkdirButton);

buttonLayout->addWidget(rmButton);

QVBoxLayout *layout = new QVBoxLayout;

layout->addWidget(treeView);

layout->addLayout(buttonLayout);

setLayout(layout);

setWindowTitle("File System Model");

connect(mkdirButton, SIGNAL(clicked()),

this, SLOT(mkdir()));

connect(rmButton, SIGNAL(clicked()),

this, SLOT(rm()));

}

构造函数很简单,我们首先创建了QFileSystemModel实例,然后将其作为一个QTreeView的模型。注意我们将QFileSystemModel的根目录路径设置为当前目录。剩下来的都很简单,我们添加了按钮之类,这些都不再赘述。对于 treeView 视图,我们使用了setRootIndex()对模型进行过滤。我们可以尝试一下,去掉这一句的话,我们的程序会显示整个文件系统的目录;而这一句的作用是,从模型中找到 QDir::currentPath()所对应的索引,然后显示这一位置。也就是说,这一语句的作用实际是设置显示哪个目录。我们会在后面的章节中详细讨论index()函数。

现在,我们可以运行以下程序看看界面:

虽然我们基本一行代码都没写(有关文件系统的代码都没有写),但是从运行截图可以看出,QFileSystemModel完全将所能想到的东西——名称、大小、类型、修改时间等全部显示出来,可见其强大之处。

下面是mkdir()槽函数:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

void FileSystemWidget::mkdir()

{

QModelIndex index = treeView->currentIndex();

if (!index.isValid()) {

return;

}

QString dirName = QInputDialog::getText(this,

tr("Create Directory"),

tr("Directory name"));

if (!dirName.isEmpty()) {

if (!model->mkdir(index, dirName).isValid()) {

QMessageBox::information(this,

tr("Create Directory"),

tr("Failed to create the directory"));

}

}

}

正如代码所示,首先我们获取选择的目录。后面这个isValid()判断很重要,因为默认情况下是没有目录被选择的,此时路径是非法的,为了避免程序出现异常,必须要有这一步判断。然后弹出对话框询问新的文件夹名字,如果创建失败会有提示,否则就是创建成功。这时候你会发现,硬盘的实际位置的确创建了新的文件夹。

下面则是rm()槽函数:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

void FileSystemWidget::rm()

{

QModelIndex index = treeView->currentIndex();

if (!index.isValid()) {

return;

}

bool ok;

if (model->fileInfo(index).isDir()) {

ok = model->rmdir(index);

} else {

ok = model->remove(index);

}

if (!ok) {

QMessageBox::information(this,

tr("Remove"),

tr("Failed to remove %1").arg(model->fileName(index)));

}

}

这里同样需要先检测路径是否合法。另外需要注意的是,目录和文件的删除不是一个函数,需要调用isDir()函数检测。这一步在代码中有很清楚的描述,这里就不再赘述了。

实际上,我们这里不需要十分担心QFileSystemModel的性能问题,因为它会启动自己的线程进行文件夹扫描,不会发生因扫描文件夹而导致的主线程阻塞的现象。另外需要注意的是,QFileSystemModel会对模型的结果进行缓存,如果你要立即刷新结果,需要通知QFileSystemWatcher类。

如果仔细查看就会发现,我们的视图不能排序不能点击列头。为此,我们可以使用下面代码:

1

2

3

4

5

6

7

8

treeView->header()->setStretchLastSection(true);

treeView->header()->setSortIndicator(0, Qt::AscendingOrder);

treeView->header()->setSortIndicatorShown(true);

#if QT_VERSION >= 0x050000

treeView->header()->setSectionsClickable(true);

#else

treeView->header()->setClickable(true);

#endif

这是 Qt 中视图类常用的一种技术:如果我们要修改有关列头、行头之类的位置,我们需要从视图类获取到列头对象,然后对其进行设置。正如代码中所显示的那样。注意上面代码片段的最后一部分,我们使用一个条件判断来确定 Qt4 与 Qt5 的不同。

现在我们简单介绍了有关两个常用的模型类:QStringListModel 和 QFileSystemModel。下一章,我们将在此基础上详细介绍模型的相关细节。

时间: 2024-10-12 07:00:20

Qt 学习之路:QFileSystemModel的相关文章

Qt学习之路

  Qt学习之路_14(简易音乐播放器) Qt学习之路_13(简易俄罗斯方块) Qt学习之路_12(简易数据管理系统) Qt学习之路_11(简易多文档编辑器) Qt学习之路_10(Qt中statusBar,MessageBox和Timer的简单处理) Qt学习之路_9(Qt中Item Widget初步探索) Qt学习之路_8(Qt中与文件目录相关操作) Qt学习之路_7(线性布局和网格布局初步探索) Qt学习之路_6(Qt局域网聊天软件) Qt学习之路_5(Qt TCP的初步使用) Qt学习之路

QT学习之路(1):彩票绝对不中模拟器

//============================================//绝对不中,彩票开奖模拟器#include "mainwindow.h"#include "ui_mainwindow.h"#include <QHash>#include <QDebug>MainWindow::MainWindow(QWidget *parent) :    QMainWindow(parent),    ui(new Ui::M

QT学习之路--创建一个对话框

Q_OBJECT:这是一个宏,凡是定义信号槽的类都必须声明这个宏. 函数tr()全名是QObject::tr(),被他处理过的字符串可以使用工具提取出来翻译成其他语言,也就是做国际化使用. 对于QT学习之路:Qt学习之路(7):创建一个对话框(上)这个程序.编译出现 invalid use of incomplete type ‘class QPushButton’ findButton->setEnabled(!text.isEmpty()); ^ In file included from

Qt 学习之路 2 --- 读书笔记

一.文章来由 来自豆子老师非常好的一本Qt教程,但是只有网络版,所以用这个做笔记了,不动笔墨不读书嘛~~ 二.读书笔记 1.Qt 学习之路 2(2):Qt 简介 1.1 关于 Qt 的一站式解决 Qt 是一个著名的 C++ 应用程序框架.但并不只是一个 GUI 库,因为 Qt 十分庞大,并不仅仅是 GUI 组件.使用 Qt,在一定程度上你获得的是一个"一站式"的解决方案:不再需要研究 STL,不再需要 C++ 的,不再需要到处去找解析 XML.连接数据库.访问网络的各种第三方库,因为

Qt学习之路1---软件下载安装及工程简介

1.下载安装目前最新版的qt,官网链接:https://www.qt.io/qt5-8/: 和qt4不同,qt5在线安装,轻巧快速,而且不用配置一些繁琐的东西,安装之后会出现Qt creator这就是我们之后使用的IDE. 2.Qt creator工程包含不同类型的文件 _ .pro项目描述文件 _ .pro.user 用户配置描述文件  _ .ui 界面描述文件  _ 资源文件(图片,音频等) 2.1 .pro项目描述文件的基本组成 _ .#  注释符 _ QT 模块声明 _ TARGET  

Qt学习之路(24): QPainter(改写paintEvent)

多些大家对我的支持啊!有朋友也提出,前面的几节有关event的教程缺少例子.因为event比较难做例子,也就没有去写,只是把大概写了一下.今天带来的是新的部分,有关Qt的2D绘图.这部分不像前面的内容,还是比较好理解的啦!所以,例子也会增加出来. 有人问豆子拿Qt做什么,其实,豆子就是在做一个Qt的画图程序,努力朝着Photoshop和GIMP的方向发展.但这终究要经过很长的时间.很困难的路程的,所以也放在网上开源,有兴趣的朋友可以来试试的呀… 好了,闲话少说,来继续我们的学习吧! Qt的绘图系

Qt 学习之路:QSortFilterProxyModel

从本章开始,我们将逐步了解有关自定义模型的相关内容.尽管前面我们曾经介绍过 Qt 提供的几个内置模型:QStringListModel和QFileSystemModel,但对于千变万化的需求而言,这些显然是远远不够的.于是,Qt 也允许我们对模型进行自定义. 在正式开始介绍自定义模形之前,我们先来了解一个新的类:QSortFilterProxyModel.之所以将这个类放在这里,是因为在一定程序上,我们可以使用QSortFilterProxyModel获得一些可能必须自定义才能达到的效果.QSo

Qt 学习之路 2笔记4

model/view 架构 MVC 是 Model-View-Controller 的简写,即模型-视图-控制器.在 MVC 中,模型负责获取需要显示的数据,并且存储这些数据的修改.视图用于将模型数据显示给用户.对于数量很大的数据,或许只显示一小部分,这样就能很好的提高性能.控制器是模型和视图之间的媒介,将用户的动作解析成对数据的操作,比如查找数据或者修改数据,然后转发给模型执行,最后再将模型中需要被显示的数据直接转发给视图进行显示.MVC 的核心思想是分层,不同的层应用不同的功能. 当 MVC

Qt 学习之路:Canvas

在 QML 刚刚被引入到 Qt 4 的那段时间,人们往往在讨论 Qt Quick 是不是需要一个椭圆组件.由此,人们又联想到,是不是还需要其它的形状?这种没玩没了的联想导致了一个最直接的结果:除了圆角矩形,Qt Quick 什么都没有提供,包括椭圆.如果你需要一个椭圆,那就找个图片,或者干脆自己用 C++ 写一个吧(反正 Qt Quick 是可以扩展的,不是么)! 为了使用脚本化的绘图机制,Qt 5 引入的Canvas元素.Canvas元素提供了一种与分辨率无关的位图绘制机制.通过Canvas,