QAbstractItemModel有一个setHeaderData(int section,.....)方法,该方法根据标题方向取一个行或列。我有一个包含几个表的模型,这些表都是顶级项目的子项。也就是说,我的模型层次结构的第一级(在不可见的根下)由8行组成,每行有一个子元素,它是一个表(当然总共8个表)。似乎setHeaderData将为第一级提供带有标题的视图,但是如何为这些子表指定标题数据? QTableView有一个setRootIndex()方法,因此它可以深入到模型层次结构并显示这些子表中的数据,我希望setHeaderData也可以采用根索引,但事实并非如此。我可以在QTableView上手动设置标题,但这样更麻烦 - 是否有更好的解决方案?
答案 0 :(得分:6)
不幸的是,模型API没有提供这样做的好方法。最简单的解决方案是根据当前使用的根索引使模型返回不同的headerData
。但是,这意味着模型需要知道视图的状态(更具体地说,根索引),这不是您通常想要的。
我认为设置代理模型可能是解决此问题的优雅方案。以下是它的实现方式:
class ChildHeadersProxy : public QSortFilterProxyModel {
public:
static const int HorizontalHeaderRole = Qt::UserRole + 1;
static const int VerticalHeaderRole = Qt::UserRole + 2;
void setRootIndex(const QModelIndex& index) {
m_rootIndex = index;
if (sourceModel()) {
emit headerDataChanged(Qt::Horizontal, 0, sourceModel()->columnCount(m_rootIndex));
emit headerDataChanged(Qt::Vertical, 0, sourceModel()->rowCount(m_rootIndex));
}
}
QVariant headerData(int section, Qt::Orientation orientation,
int role = Qt::DisplayRole) const {
if (sourceModel() && m_rootIndex.isValid()) {
int role = orientation == Qt::Horizontal ? HorizontalHeaderRole : VerticalHeaderRole;
QStringList headers = sourceModel()->data(m_rootIndex, role).toStringList();
if (section >= 0 && section < headers.count()) {
return headers[section];
}
}
return QSortFilterProxyModel::headerData(section, orientation, role);
}
private:
QModelIndex m_rootIndex;
};
此代理模型使用源模型通过两个自定义角色提供的标头。例如,如果您使用QStandardItemModel
,则设置标头就像这样简单:
model.item(0, 1)->setData(QStringList() << "h1" << "h2",
ChildHeadersProxy::HorizontalHeaderRole);
model.item(0, 1)->setData(QStringList() << "vh1" << "vh2",
ChildHeadersProxy::VerticalHeaderRole);
其中model.item(0, 1)
是对应的根项。设置视图将如下所示:
QTableView view;
ChildHeadersProxy proxy;
proxy.setSourceModel(&model);
view.setModel(&proxy);
更改根索引将如下所示:
view.setRootIndex(proxy.mapFromSource(model.index(0, 1)));
proxy.setRootIndex(model.index(0, 1));