首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何覆盖QSqlRelationalTableModel?

如何覆盖QSqlRelationalTableModel?
EN

Stack Overflow用户
提问于 2015-08-15 09:10:26
回答 1查看 435关注 0票数 1

我想在最后一栏中添加一列。我想把QPushButtons放在那个专栏里。不过,我不知道该怎么做。关注的问题如下。

  1. 当我重写数据(const QModelIndex &index,int角色)时,我必须重写所有数据,这是可以避免的吗?
  2. 当我以QVariant的形式返回新的QVariant时,它会抛出一条错误消息。我不知道怎么解决。C:\Qt\5.4\mingw491_32\include\QtCore\qvariant.h:462: error: 'QVariant::QVariant(void*)' is private inline QVariant(void *) Q_DECL_EQ_DELETE; QVariant ActionSqlRelationModel::data(const QModelIndex &index,int角色) const { if (角色== Qt::DisplayRole) {返回新QPushButton;}返回QVariant()};

我尝试使用QIdentityProxyModel,但它会出现错误消息。我只是试图覆盖列数。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2015-08-15 16:34:38

当我重写数据(const QModelIndex &index,int角色)时,我必须重写所有数据,这是可以避免的吗?

是。您需要转发到基类的实现:

代码语言:javascript
复制
void MyModel : public BaseModel {
public:
  QVariant data(const QModelIndex & index, int role) const Q_DECL_OVERRIDE {
    if (index.column() == BaseModel::columnCount()) {
      // extra column
      return "MyData";
    }
    return BaseModel::data(index, role);
  }
  int columnCount() const Q_DECL_OVERRIDE {
    return BaseModel::columnCount() + 1;
  }
  ...
};

您还可以使用QIdentitityProxyModel而不是派生基本模型,并重写它的方法--这样,您就可以轻松地添加额外的列,而不管使用什么模型来保持数据的其余部分。

具体地说:

代码语言:javascript
复制
void MyProxy : public QIdentityProxyModel {
public:
  QVariant data(const QModelIndex & index, int role) const Q_DECL_OVERRIDE {
    if (index.column() == BaseModel::columnCount()) {
      // extra column
      return "MyData";
    }
    return QIdentityProxyModel::data(index, role);
  }
  int columnCount() const Q_DECL_OVERRIDE {
    return BaseModel::columnCount() + 1;
  }
  MyProxy(QObject * parent = 0) : QIdentityProxyModel(parent) {}
};

然后,您将在视图上使用而不是原始模型。要让代理知道它应该用作代理的模型,可以使用setSourceModel方法。例如:

代码语言:javascript
复制
int main(int argc, char ** argv) {
  QApplication app(argc, argv);
  QTableView view;
  QSqlRelationalTableModel model;
  MyProxy proxy;
  ...
  proxy.setSourceModel(&model);
  view.setModel(&proxy);
  view.show();
  return app.exec();  
}

当我以QVariant的形式返回新的QVariant时,它会抛出一条错误消息。

首先,从data方法返回一个小部件,即使您设法这样做,也不是由视图处理的。视图不会将这样的小部件用于任何事情--它们将忽略它。这样就没用了。我在下面展示一个人是如何做到的,但这只是为了满足你的好奇心--这是一个完全无用的练习。这样的代码“工作”,但它返回的小部件将被忽略。

如果要将按钮显示为特定数据项的表示形式,则需要向视图中添加自定义委托。委托处理具有非标准需求的项的可视化和小部件。您必须重新实现一个QStyledItemDelegate (或QAbstractItemDelegate),并使用setItemDelegateForColumn在视图的给定列上设置它。

满足你的好奇心:

QObject及其派生的类是不可复制的,因此不能直接存储在QVariant中。相反,您可以存储指向对象的共享指针:

代码语言:javascript
复制
typedef QSharedPointer<QPushButton> QPushButtonPtr;
Q_DECLARE_METATYPE(QPushButtonPtr)

...

QVariant ActionSqlRelationModel::data(const QModelIndex &index, int role) const
{
    if (role == Qt::DisplayRole)
    {
       return QPushButtonPtr(new QPushButton);
    }
    return QSqlRelationModel::data(index, role); // underlying class's data
}

我不太知道你想通过在模型中存储按钮来达到什么目的。当然,这样做是可以的,只是没有一个视图知道如何处理您提供给他们的按钮。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/32023282

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档