使用C++实现QML的TreeView Model (一)
程序员文章站
2022-04-24 14:37:09
...
使用C++实现QML的TreeView Model (一)
QML中的数据访问组件如ListView、TableView、GridView通常使用ListModel做为数据提供者,这种应用有相当大局限性,如无法访问本地文件系统、无法连接到传统的SQL数据库,所以通常在使用中都是通过C++实现数据访问,通过QML进行数据展示和编辑,常用的数据模型组件有QAbstractItemModel、QAbstractTableModel、QSQLTableModel等。所有的高级Model组件都继承自QAbstractItemModel,只要了解QAbstractItemModel的接口函数和运作机理,就可以了解QT的Model/View机制实现方式。QAbstractItemModel是一个抽象类,要实例化QAbstractItemModel必须继承并至少实现以下5个方法:
int rowCount(const QModelIndex &parent=QModelIndex()) const;
int columnCount(const QModelIndex &parent=QModelIndex()) const;
QModelIndex index(int row, int column, const QModelIndex &parent=QModelIndex()) const;
QVariant data(const QModelIndex &index, int role=Qt::DisplayRole) const;
QModelIndex parent(const QModelIndex &child) const;
与 QWidget组件不同的是,在QML的数据模型中,并不通过列(Column)进行数据访问,而是通过Role进行数据访问,例如:
TableViewColumn是TableView的列定义,TableViewColumn通过role属性定义来向model获取数据,TableView会通过调用model的roleNames()方法来获取model可用的role。所以,除了必须要实现的5个虚函数外,还必须重新实现roleNames()来告诉View有哪些role是可用的,roleNames()的原型如下:
- TableView{
- id:tableView1
- anchors.fill: parent
- TableViewColumn{
- width:50
- title:""
- role:"tagging"
- }
- TableViewColumn{
- width:80
- title:"操作"
- role:"name"
- }
- }
- QHash
roleNames() const;
以下是一个完整的Model类定义:
roleNames()的实现相当简单:
- classSqlMenuEntry:public QAbstractItemModel,public QQmlParserStatus
- {
- Q_OBJECT
- public:
- explicit SqlMenuEntry(QObject *parent=0);
- ~SqlMenuEntry();
- enum MenuEntryRoles{idRole=Qt::UserRole+1,nameRole,defaultEntryRole,customEntryRole,iconRole,iconHoverRole};
- int rowCount(const QModelIndex &parent=QModelIndex()) const;
- int columnCount(const QModelIndex &parent=QModelIndex()) const;
- QModelIndex index(int row, int column, const QModelIndex &parent=QModelIndex()) const;
- QVariant data(const QModelIndex &index, int role=Qt::DisplayRole) const;
- QModelIndex parent(const QModelIndex &child) const;
- QHash
roleNames() const; - private:
- QHash
mRoleNames; - QList
> mRecords; //真正的数据保存在这里,QList只能保存二维数据没办法保存树状节点,这里仅仅是例子 - };
mRoleNames可以在类构造函数中进行初始化:
- QHash
SqlMenuEntry::roleNames() const - {
- return mRoleNames;
- }
在QML中就可以通过"name"、"menuid"、"icon"对数据进行访问:
- SqlMenuEntry::SqlMenuEntry(QObject *parent)
- :QAbstractItemModel(parent)
- {
- mRoleNames[nameRole] = "name";
- mRoleNames[idRole] = "menuid";
- mRoleNames[iconRole] = "icon";
- mRoleNames[defaultEntryRole] = "default";
- mRoleNames[iconHoverRole] = "iconHover";
- }
如果仅为二维表提供数据,那么根据以上几个接口函数的名称就可以简单的实现数据供给View视图,其中:
- ListView{
- model:MenuEntryModel{ }
- delegate:Item{
- Column{
- Text{text:name}
- Text{text:icon}
- }
- }
- }
- int SqlMenuEntry::rowCount(const QModelIndex &parent) const
- {
- return mRecords.size();
- }
- int SqlMenuEntry::columnCount(const QModelIndex &parent) const
- {
- return 1; //QML不使用列获取数据,默认返回一列,不返回1例的话,View控件会认为表是空表,不获取数据
- }
- QModelIndex SqlMenuEntry::index(int row, int column, const QModelIndex &parent) const
- {
- if((row >= 0)&&(row
- {
- return createIndex(row,column);
- }
- return QModelIndex(); //返回一个无效的空索引
- }
- QModelIndex SqlMenuEntry::parent(const QModelIndex &child) const
- {
- return QModelIndex(); //二维表中的行没有parent节点
- }
- QVariant SqlMenuEntry::data(const QModelIndex &index, int role) const
- {
- if(index.isValid)
- {
- return mRecords[index.row()][role];
- }
- }
mRecords中的数据可以按需求生成,如通过QSqlQuery组件从数据库服务器获取,获取添加数据:
实现后的model类可以通过
- QHash
row; - row[nameRole] = "name1";
- row[iconRole] = "icon1";
- mRecords.append(row);
进行注册,注册后的类可以在QML生成实例:
- qmlRegisterType
("com.limutech.tv",1,0,"MenuEntryModel");
View组件获取数据的流程大概如下:
- import com.limutech.tv 1.0
- MenuEntryModel{
- id:menuEntryModel
- }
- ListView{
- model:menuEntryModel
- ...
- }
1、View调用rowCount(constQModelIndex &parent)并传递一个空的parent获取根节点的行数;
2、View调用columnCount(const QModelIndex &parent)并传递一个空的parent获取根节点的列数;
3、View对各行列枚举调用index(int row, int column, const QModelIndex &parent)交传行号、列号和空的parent获取根节点QModelIndex;
4、继续以返回的modelIndex为parent获取每行的rowCount和columnCount,如果大于0则该节点还有子节点;
5、调用roleNames返回可用的role列表
6、以返回的modelIndex和role为参数,调用data获取数据并使用相应的delegate进行显示
所以要显示一个树状列表,需要对二维表模型进行完善,处理parent不为空的情况,同时,模型应该可以存储树状数据,在下一节我将继续分享层次模型的实现方式和涉及数据修改的一些实现。
推荐阅读
-
【输出一个静态“心形”图案、一个跳动的“心”——使用C++、C语言来实现】
-
使用C++实现QML的TreeView Model (二)
-
使用C++实现QML的TreeView Model (二)_PHP教程
-
使用C++实现QML的TreeView Model (一)
-
使用C++实现QML的TreeView Model (二)
-
微分差值 透明颜色计算 使用C++ 的模板函数 去实现 结合EasyX做一个简单示例
-
C++全排列:一种新的全排列方法(使用单向链表实现)
-
使用C++实现QML的TreeView Model (一)_PHP教程
-
(C++)输入10个整数,将其中最小的数与第一个数对换,把最大的数与最后一个数对换。要求用3个函数实现,分别为输入10个数、进行处理、输出10个数。要求使用指针的方法进行处理。