问:如何在QAbstractItemModel中设置子表的标题?

时间:2015-05-13 17:20:42

标签: qt

QAbstractItemModel有一个setHeaderData(int section,.....)方法,该方法根据标题方向取一个行或列。我有一个包含几个表的模型,这些表都是顶级项目的子项。也就是说,我的模型层次结构的第一级(在不可见的根下)由8行组成,每行有一个子元素,它是一个表(当然总共8个表)。似乎setHeaderData将为第一级提供带有标题的视图,但是如何为这些子表指定标题数据? QTableView有一个setRootIndex()方法,因此它可以深入到模型层次结构并显示这些子表中的数据,我希望setHeaderData也可以采用根索引,但事实并非如此。我可以在QTableView上手动设置标题,但这样更麻烦 - 是否有更好的解决方案?

1 个答案:

答案 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));