gpt4 book ai didi

c++ - 如何覆盖 QSqlRelationalTableModel?

转载 作者:搜寻专家 更新时间:2023-10-30 23:43:35 26 4
gpt4 key购买 nike

我想在最后添加一列。我想把 QPushButtons 放到那一列。但是,我不知道该怎么做。问题如下。

  1. 当我覆盖 data(const QModelIndex &index, int role) 时,我必须重写所有数据,这是可以避免的吗?`

  2. 当我将新的 QPushButton 作为 QVariant 返回时,它会抛出一条错误消息。我不知道如何解决。C:\Qt\5.4\mingw491_32\include\QtCore\qvariant.h:462: 错误:'QVariant::QVariant(void*)' 是私有(private)内联 QVariant(void *) Q_DECL_EQ_DELETE;

    QVariant ActionSqlRelationModel::data(const QModelIndex &index, int role) const
    {
    if (role == Qt::DisplayRole)
    {
    return new QPushButton;
    }
    return QVariant();
    }

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

最佳答案

When I override data(const QModelIndex &index, int role), I have to rewrite all the data, is that avoidable?

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

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 而不是派生基本模型,而是覆盖它的方法 - 这样您就可以轻松添加额外的列,无论使用什么模型来保留其余数据.

具体来说:

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) {}
};

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

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();
}

When I return new QPushButton as QVariant, it throws an error message.

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

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

为了满足你的好奇心:

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

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
}

我不太清楚你想通过在模型中存储按钮来实现什么。这样做当然没问题,除非没有 View 知道如何处理您提供给它们的按钮。

关于c++ - 如何覆盖 QSqlRelationalTableModel?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32023282/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com