从c++中设置GridView模型属性而不设置上下文

Set GridView model property from C++ without setting Context

本文关键字:设置 上下文 属性 GridView c++ 模型      更新时间:2023-10-16

我试图通过调用
从c++设置QML GridViewmodel属性QQmlProperty::write(gridview, "model", QVariant::fromValue(objlist)); .

gridview设置正确,我可以从c++修改属性,当我将其设置为具有6个条目的QList并从QML打印它们时,我得到
qml: model = item(0x30617b50), Item(0x30617b90), Item(0x30617bd0), Item(0x30617c10), Item(0x30617c50), Item(0x30617cd0),虽然模型没有显示。

Qt文档建议调用

QQmlContext *ctxt = view->rootContext(); ctxt->setContextProperty("gridModel", QVariant::fromValue(objlist));

然后用model: gridModel从QML设置属性,但这并不真正适合我的需要。不过,它工作得很好,只要设置了属性,就会显示正确的数据。当我从QML打印变量时,输出是
qml: model = [object Object]所以设置上下文属性和设置对象属性之间肯定有区别,但我不知道如何解决这个问题。

与其尝试从c++访问QML对象或属性,我建议在QML端使用绑定,并从c++提供属性值。

如果通过setContextProperty暴露模型实例不太适合您的需要,例如,如果模型是在QML加载时间后实例化的,那么我建议使用以下方法:

  1. 通过setContextProperty()暴露QObject派生类的实例,i
  2. 类为你的模型获得一个Q_PROPERTY,包括一个NOTIFY信号
  3. 在QML中,你将该属性绑定到GridView的模型
  4. 当你在c++中创建模型或者当你需要创建一个模型的新实例时,你发出上面提到的NOTIFY信号

接口类看起来有点像这样

class MyInterface : public QObject
{
    Q_OBJECT
    Q_PROPERTY(MyModel* model READ model NOTIFY modelChanged)
public:
    MyModel *model() const { return m_model; }
    void setModel(MyModel *model) {
        m_model = model;
        emit modelChanged();
    }
private:
    MyModel *m_model = 0;
};
当然,m_model的改变可以在MyInterface的内部,而不是setter,等等。这使您可以在c++方面完全控制何时创建模型实例,何时更改它,何时删除它。如果您将类型更改为QAbstractItemModel*,或者您的一些常见的模型基类,您甚至可以在运行时更改模型的类型,因为您看到fit

如果你说QQmlProperty::write正确设置模型,那么你的问题是什么?无论如何,我建议一个解决方案:

假设你有一个GridView,如下所示:

GridView {
    anchors.fill: parent
    objectName: "grid"
    delegate: Rectangle {
        id: rect
        width: 100;
        height: 100
        color: Qt.rgba(Math.random(),Math.random(),Math.random(),1)
        Text {
            anchors.centerIn: rect
            text: modelData
        }
    }
}

objectName是必选项。

所以从c++访问可以是:

QStringList list;
list.append("String1");
list.append("String2");
list.append("String3");
QQmlApplicationEngine engine;
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
QObject *obj = engine.rootObjects()[0]->findChild<QObject *>("grid");
if(obj) {
    obj->setProperty("model",QVariant(list));
}

我想到的解决方案与张贴的答案略有不同,所以我想写一个明确的答案是最好的。

关键是子类化QAbstractItemModel(或者更准确地说是..ListModel,这样你就不必处理c++ QML中的行/列)。

这样做时,您不仅可以使用

简单地将模型设置为属性
QQuickItem *mainform = view->rootObject();
QQuickItem *grid = (QQuickItem *)mainform->findChild<QObject*>("GridView object name");
ItemModel itemmodel;
itemmodel.setItems(objlist):
QQmlProperty::write(grid, "model", QVariant::fromValue(&itemmodel));

当模型发生变化时,它也会通知QML,例如一个Item被删除了。(您必须正确地处理这些更改,参见示例中的removeRows())

这是我的ItemModel:
// itemmodel.h
#include <QAbstractListModel>
#include <item.h>
class ItemModel : public QAbstractListModel
{
    Q_OBJECT
public:
    explicit ItemModel(QObject *parent = 0);
    QHash<int, QByteArray> roleNames() const;
public slots:
    void setItems(QList<Item *> items);
    QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const;
    int rowCount(const QModelIndex & parent = QModelIndex()) const;
    bool removeRows(int row, int count, const QModelIndex &parent = QModelIndex());
private:
    QList<Item *> items;
};

// itemmodel.cpp
#include "itemmodel.h"
ItemModel::ItemModel(QObject *parent) : QAbstractListModel(parent)
{
}
// Column Names have to match all the Q_PROPERTYs defined in Item
const char* COLUMN_NAMES[] = {
    "property1",
    "property2",
    "...",
    NULL
};
QHash<int, QByteArray> makeRoleNames()
{
    int idx = 0;
    QHash<int, QByteArray> roleNames;
    while(COLUMN_NAMES[idx])
        roleNames[Qt::UserRole + idx + 1] = COLUMN_NAMES[idx++];
    return roleNames;
}
QHash<int, QByteArray> ItemModel::roleNames() const
{
    static const QHash<int, QByteArray> roleNames = makeRoleNames();
    return roleNames;
}
void ItemModel::setItems(QList<Item *> items)
{
    this->items = items;
}
int ItemModel::rowCount(const QModelIndex & /* parent */) const
{
    return items.count();
}
bool ItemModel::removeRows(int row, int count, const QModelIndex &parent)
{
    Q_UNUSED(parent);
    beginRemoveRows(QModelIndex(), row, row + count - 1);
    while (count--) delete items.takeAt(row);
    // example for custom deletion:
    //              items.takeAt(row)->removeFromRoot();
    endRemoveRows();
    return true;
}
QVariant ItemModel::data(const QModelIndex &index, int role) const {
    if (!index.isValid())
        return QVariant();
    if (index.row() >= items.size() || index.row() < 0)
        return QVariant();
    if (role == Qt::DisplayRole) {
        return QVariant::fromValue(this->items.at(index.row()));
    }
    if (role > Qt::UserRole)
        return this->items.at(index.row())->property(makeRoleNames()[role]);
}

来源:

  • [1]关于为QML创建QAbstractItemModel子类的Qt文档页
  • [2] QAbstractItemModel reference
  • [3] QAbstractItemModel子类化指南
  • [4] QAbstractListModel子类的论坛帖子
  • [5]工作(几乎data()略有改变)QAbstractItemModel的实现,我从中获得了rolnamames函数