一、概述
在Qt中,表格控件包含有水平和垂直表头,但是常规使用模式下都是只能实现一级表头,而树控件虽然包含有了branch分支,这也间接的削弱了他自身的表头功能,细心的同学可能会发现Qt自带的QTreeView树控件只包含有水平表头,没有了垂直表头。
既然Qt自带的控件中没有这个功能,那么我们只能自己去实现了。
要实现多级表头功能方式也多种多样,之前就看到过几篇关于实现多级表头的文章,总体可以分为如下两种方式
- 表头使用一个表格来模拟
- 通过给表头自定义Model
今天这篇文章我们是通过方式2来实现多级表头。如效果图所示,实现的是一个树控件的多级表头,并且他还包含了垂直列头,实现这个控件所需要完成的代码量还是比较多的。本篇文章可以算作是一个开头吧,后续会逐步把关键功能的实现方式分享出来。
二、效果展示
三、实现方式
本篇文章中的控件看起来是一个树控件,但是他又具备了表格控件该有的一些特性,比如垂直表头、多级水平表头等等。要实现这样的树控件,我们有两个大的方向可以去考虑
- 重写表格控件,实现branch
- 表格控件+树控件
方式1重写表格控件实行branch的工作量是比较大的,而且需要把Qt原本的代码迁出来,工作量会比加大,个人选择了发你。
方式2是表格控件+树控件的实现方式,说白了就是表格控件提供水平和垂直表头,树控件提供内容展示,听起来好像没毛病,那么还等什么,直接干呗。
既然大方向定了,那么接下来可能就是一些细节问题的确定。
- 多级水平表头
- 垂直列头拖拽时,实现树控件行高同步变动
- 自绘branch
以上三个问题都是实现表格树控件时遇到的一些比较棘手的问题,后续会分别通过单独的文章来进行讲解,今天这篇文章也是我们的第一讲,怎么实现水平多级表头
四、多级表头
第一节我们也说了,实现多级表头我们使用重写model的方式来实现,接下来就是贴代码的时候。
1、数据源
经常重写model的同学对如下代码应该不陌生,对于继承自QAbstractItemModel的数据源肯定是需要重写该类的所有纯虚方法,包括所有间接父类的纯虚方法。
除此之外自定义model应该还需要提供一个可以合并单元格的方法,为什么呢?因为我们多级表头需要。比如说我们一级表头下有3个二级表头,那这就说明一级表头合并了3列,使得本身的3列数据变成一列。
class HHeaderModel : public QAbstractItemModel
{
struct ModelData //模型数据结构
{
QString text;
ModelData() : text("")
{
}
};
Q_OBJECT
public:
HHeaderModel(QObject * parent = 0);
~HHeaderModel();
public:
void setItem(int row, int col, const QString & text);
QString item(int row, int col);
void setSpan(int firstRow, int firstColumn, int rowSpanCount, int columnSpanCount);
const CellSpan& getSpan(int row, int column);
public:
virtual QModelIndex index(int row, int column, const QModelIndex & parent) const over