Qt —— 细说自定义Tree View Model

Tree view与Tree Widget 相比而言,需要为tree view 设置一个model。使用tree view 能有效降低内存的使用率。但是需要自定义model。下面参考Qt官方提供的demo,讨论如何自定义Tree View Model。

依据名为simple tree model官方demo可知道:要自定义tree view model需要定义一个树形结构的底层数据结构,该类型结构的每一个对象对应一个view中的Item(如下图每个正方形表示一个Item)。每个对象保存自己的父Item指针、子Item指针列表、数据段列表。自定义的tree model包含一个root Item,以该Item为根建立起item树。对于view中的Item都有与之关联的QModelIndex,可以通过index获取该Item的位置(row)、父Item的index以及底层数据的指针等信息。

在 tree model中顶级项(上图中的 A、C以及同级的Item)对应的模型索引的父索引(QModelIndex::parent()函数获得)是无效的。各项的数据保存在QModelIndex中。

先看底层数据结构类:


#include <QList>
#include <QVariant>

class TreeItem
{
public:
	TreeItem(const QList<QVariant> &data, TreeItem *parent = nullptr);
	~TreeItem();

	// 构建子Item列表;
	void appendChild(TreeItem *child);
	// 获取该Item指定行号的子Item;
	TreeItem *child(int row);
	// 获取该Item的子Item个数;
	int childCount() const;
	// 获取该Item的列数(数据段数); 
	int columnCount() const;
	// 获取该Item指定段的数据;
	QVariant data(int column) const;
	// 获取该item所在parent的row;
	int row() const;
	// 该Item的父Item;
	TreeItem *parentItem();

private:
	TreeItem* m_parentItem;				// 该Item的父Item;
	QList<TreeItem*> m_childItems;		// 该Item的子Item列表;
	QList<QVariant>	m_itemData;			// 该Item的各列数据;
};
#include "tree_item.h"

TreeItem::TreeItem(const QList<QVariant> &data, TreeItem *parent)
	:m_itemData(data), 
	m_parentItem(parent)
{

}

TreeItem::~TreeItem()
{
	qDeleteAll(m_childItems);
}

void TreeItem::appendChild(TreeItem *child)
{
	m_childItems.append(child);
}

TreeItem *TreeItem::child(int row)
{
	// 无效的row;
	if (row < 0 || row >= m_childItems.size())
		return nullptr;

	return m_childItems.at(row);
}

int TreeItem::childCount() const
{
	return m_childItems.count();
}

int TreeItem::columnCount() const
{
	return m_itemData.count();
}

QVariant TreeItem::data(int column) const
{
	// 无效的索引返回空的QVariant;
	if (column < 0 || column >= m_itemData.size())
		return QVariant();

	return m_itemData.at(column);
}

int TreeItem::row() const
{
	if (m_parentItem)
		m_parentItem->m_childItems.indexOf(const_cast<TreeItem*>(this));

	// 尽管根项(没有父项)被自动分配了行号0,但模型从不使用此信息。
	return 0;
}

TreeItem *TreeItem::parentItem()
{
	return m_parentItem;
}

接下来子类化QAbstractItemModel是自定义tree model的关键。需要重写父类的函数(override标记)每个函数有较详细的解释。

#include <QAbstractItemModel>

class TreeItem;

class TreeModel : public QAbstractItemModel
{
	Q_OBJECT

public:
	TreeModel(const QString &data, QObject *parent = 0);
	~TreeModel();

	QVariant data(const QModelIndex &index, int role) const override;
	Qt::ItemFlags flags(const QModelIndex &index) const override;
	QVariant headerData(int section, Qt::Orientation orientation,
		int role = Qt::DisplayRole) const override;
	QModelIndex index(int row, int column,
		const QModelIndex &parent = QModelIndex()) const override;
	QModelIndex parent(const QModelIndex &index) const override;
	int rowCount(const QModelIndex &parent = QModelIndex()) const override;
	int columnCount(const QModelIndex &parent = QModelIndex()) const override;


private:
	void setupModelData(const QStringList &lines, TreeItem *parent);

private:
	TreeItem *rootItem;
};
TreeModel::TreeModel(const QString &data, QObject *parent)
	:QAbstractItemModel(parent)
{
	QList<QVariant> rootData;
	rootData << "Title" << "Summary";
	rootItem = new TreeItem(rootData);

	setupModelData(data.split(QString("\n")), rootItem);
}

TreeModel::~TreeModel()
{
	delete rootItem;
}

QVariant TreeModel::data(const QModelIndex &index, int role) const
{
	if (!index.isValid())
		return QVariant();

	if (role != Qt::DisplayRole)
		return QVariant();

	TreeItem *item = static_cast<TreeItem*>(index.internalPointer());

	return item->data(index.column());
}

// We use the flags() function to ensure that views know that the model is read-only
Qt::ItemFlags TreeModel::flags(const QModelIndex &index) const
{
	if (!index.isValid())
		return Qt::NoItemFlags;

	return QAbstractItemModel::flags(index);
}

QVariant TreeModel::headerData(int section, Qt::Orientation orientation, int role) const
{
	if (orientation == Qt::Horizontal && role == Qt::DisplayRole)
		return rootItem->data(section);

	return QVariant();
}

// 该函数被view或delegate调用,用于获取某个item的特定row的子item的QModelIndex
QModelIndex TreeModel::index(int row, int column, const QModelIndex &parent) const
{
	// 判定指定的parent Item是否有(row, column) 的子Item;
	if (!hasIndex(row, column, parent))
		return QModelIndex(); 

	TreeItem *pParentItem = nullptr;

	if (!parent.isValid()) { // parent 是一个无效的索引,证明它是根Item;
		pParentItem = rootItem;
	} else {
		pParentItem = (TreeItem *)parent.internalPointer();
	}

	TreeItem *pChildItem = pParentItem->child(row);

	// 得到指定parent的第 row 个Item的 Model index;
	if (pChildItem)
		return createIndex(row, column, pChildItem);

	return QModelIndex();
}

// 返回指定索引的父Item的index;
QModelIndex TreeModel::parent(const QModelIndex &index) const
{
	if (!index.isValid())
		return QModelIndex();

	TreeItem *pChildItem = static_cast<TreeItem*>(index.internalPointer());
	TreeItem *pParentItem = pChildItem->parentItem();

	// 根Item有一个无效的index;
	if (pParentItem == rootItem)
		return QModelIndex();
	// 自己作为parent(相对 index), 但也是别人的child;
	return createIndex(pParentItem->row(), 0, pParentItem);
}

// 某个Item挂接多少子Item;
int TreeModel::rowCount(const QModelIndex &parent) const
{
	TreeItem *parentItem;
	if (parent.column() > 0)
		return 0;

	// 假如index 无效, 那就是根Item的子Item个数;
	if (!parent.isValid())
		parentItem = rootItem;
	else
		parentItem = static_cast<TreeItem*>(parent.internalPointer());

	return parentItem->childCount();
}

// 列数是固定的;
int TreeModel::columnCount(const QModelIndex &parent) const
{
	if (parent.isValid())
		return static_cast<TreeItem*>(parent.internalPointer())->columnCount();
	else
		return rootItem->columnCount();
}

void TreeModel::setupModelData(const QStringList &lines, TreeItem *parent)
{
	QList<TreeItem*> parents;
	QList<int> indentations;
	parents << parent;
	indentations << 0;

	int number = 0;

	while (number < lines.count()) {
		int position = 0;
		while (position < lines[number].length()) {
			if (lines[number].at(position) != ' ')
				break;
			position++;
		}

		QString lineData = lines[number].mid(position).trimmed();

		if (!lineData.isEmpty()) {
			// Read the column data from the rest of the line.
			QStringList columnStrings = lineData.split("\t", QString::SkipEmptyParts);
			QList<QVariant> columnData;
			for (int column = 0; column < columnStrings.count(); ++column)
				columnData << columnStrings[column];

			if (position > indentations.last()) {
				// The last child of the current parent is now the new parent
				// unless the current parent has no children.

				if (parents.last()->childCount() > 0) {
					parents << parents.last()->child(parents.last()->childCount() - 1);
					indentations << position;
				}
			}
			else {
				while (position < indentations.last() && parents.count() > 0) {
					parents.pop_back();
					indentations.pop_back();
				}
			}

			// Append a new item to the current parent's list of children.
			parents.last()->appendChild(new TreeItem(columnData, parents.last()));
		}

		++number;
	}
}

 

  • 3
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
Qt自定义Model/View模型,您可以继承QAbstractItemModel类,并实现其相关的虚拟函数来定义自己的数据模型。 下面是一个简单的示例,展示如何自定义一个简单的树形模型(Tree Model): ```cpp #include <QAbstractItemModel> #include <QModelIndex> #include <QVariant> class MyTreeModel : public QAbstractItemModel { public: explicit MyTreeModel(QObject *parent = nullptr); ~MyTreeModel(); // 重写父类的虚拟函数 QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override; QModelIndex parent(const QModelIndex &child) const override; int rowCount(const QModelIndex &parent = QModelIndex()) const override; int columnCount(const QModelIndex &parent = QModelIndex()) const override; QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; }; MyTreeModel::MyTreeModel(QObject *parent) : QAbstractItemModel(parent) { // 在构造函数中初始化数据模型 } MyTreeModel::~MyTreeModel() { // 在析构函数中清理数据 } QModelIndex MyTreeModel::index(int row, int column, const QModelIndex &parent) const { // 返回指定行、列和父索引的模型索引 } QModelIndex MyTreeModel::parent(const QModelIndex &child) const { // 返回指定子索引的父索引 } int MyTreeModel::rowCount(const QModelIndex &parent) const { // 返回指定父索引的子项数目 } int MyTreeModel::columnCount(const QModelIndex &parent) const { // 返回模型的列数 } QVariant MyTreeModel::data(const QModelIndex &index, int role) const { // 返回指定模型索引和角色的数据 } ``` 在这个示例中,我们自定义了一个名为MyTreeModel的树形模型,继承自QAbstractItemModel类,并重写了QAbstractItemModel的一些虚拟函数。在这些函数中,您可以根据自己的数据结构和需求来实现相应的逻辑。 要使用自定义的模型,您可以在视图中设置该模型,如: ```cpp MyTreeModel model; QTreeView view; view.setModel(&model); ``` 当您设置了自定义的模型后,视图将使用模型提供的数据来显示和管理数据。 当然,这只是一个简单的示例。在实际应用中,您可能需要更复杂的数据结构和逻辑来实现您的自定义模型。但是通过继承QAbstractItemModel类,并实现其虚拟函数,您可以很好地控制自己的数据模型和视图之间的交互。 希望这个示例能够帮助您开始自定义Model/View模型的开发!如果您有任何进一步的问题,请随时提问。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值