使用多个视图查看,编辑和更新QML中的数据(来自C++),而数据保留在C++中(订阅数据)

Ber*_*ert 2 c++ qobject qlist qml

我有一些数据存储在C++类(Data.cpp)的实例中.现在我希望能够在QML中从2个单独的表示中查看和编辑这些数据,这样如果View1中的值发生变化,数据本身(C++)就会发生变化,并且View2也会显示值(因为它会得到通知)当C++数据发生变化时).

这是我到目前为止所得到的:

Data.h

class Data : public QObject
{
    Q_OBJECT
    Q_PROPERTY(QString name READ name WRITE setName NOTIFY nameChanged)

public:
    Data(std::string name);
    QString name();
    void setName(const QString &n);

signals:
    void nameChanged();

private:
    std::string _name;
};
Run Code Online (Sandbox Code Playgroud)

Parser.h(提供数据列表)

class Parser : public QObject
{
    Q_OBJECT
    Q_PROPERTY(QList<QObject*> list READ list NOTIFY listChanged)
    //QList<Data*> is not working with QML :(

public:
    Parser(QObject *parent = 0);
    QList<QObject*> list() //stuff below is implementd in Parser.cpp
    {
       _list.append(new Data("name 1"));
       _list.append(new Data("name 2"));
       _list.append(new Data("name 3"));
        return _list;
    }


signals:
    void listChanged();

private:
    QList<QObject*> _list;
};
Run Code Online (Sandbox Code Playgroud)

QML部分:

    ListView
    {
        id: view1
        anchors.fill: parent
        spacing: 5
        delegate: Text { text: name}
        model: ListModel{Component.onCompleted: getModel()}
    }
    ListView
    {
        id: list2
        anchors.fill: parent
        spacing: 5
        delegate: Text { text: name}
        model: ListModel{Component.onCompleted: getModel()}
    }
    function getModel()
    {
        var m = parser.list;
        for(var i=0; i<m.length; i++)
        {
            list.model.append(m[i]); //simply returning the list (m) does not work
        }
    }
Run Code Online (Sandbox Code Playgroud)

现在,如果我单击view1中的某个项目(例如),我希望更改相应数据的名称,并相应地显示在view2中的名称.如果我从C++修改了名称,则应在两个视图中显示新名称.

有没有办法做到这一点?我坚持了好几天......谢谢你的帮助.


编辑:

在这里向这个主题提出了一个更具体的问题.

ksi*_*ons 7

这很有可能,但你遇到了一些问题:

  1. 您希望将Data对象列表公开为QQmlListProperty.这是将列表导入QML的正确方法
  2. 如果您的列表被正确公开为QQmlListProperty,您可以将其设置为ListView的模型,而无需执行您现在正在执行的奇怪的getModel()hack
  3. 您不应该在getter中将项添加到列表中,否则每次QML尝试读取它时都会附加到它.

一旦解决了这个问题,您只需通过与代理中当前模型项的引用进行交互即可更新Data对象.

这是一个完整的工作示例.我在QML中添加了一个更改模型的MouseArea,还在C++中添加了一个也改变模型的计时器,以显示任何一方的更改都会立即反映在UI中.

main.cpp中:

#include <QGuiApplication>
#include <QtQuick>

class Data : public QObject
{
    Q_OBJECT
    Q_PROPERTY(QString name READ name WRITE setName NOTIFY nameChanged)

public:
    Data(const QString &n) : _name(n) { }
    QString name() const { return _name; }

    void setName(const QString &n) {
        if (_name == n)
            return;
        _name = n;
        emit nameChanged(n);
    }

signals:
    void nameChanged(const QString &n);

private:
    QString _name;
};

class Parser : public QObject
{
    Q_OBJECT
    Q_PROPERTY(QQmlListProperty<Data> list READ list CONSTANT)

public:
    Parser(QObject *parent = 0) {
        _list.append(new Data(QStringLiteral("name 1")));
        _list.append(new Data(QStringLiteral("name 2")));
        _list.append(new Data(QStringLiteral("name 3")));

        startTimer(5000);
    }

    QQmlListProperty<Data> list() {
        return QQmlListProperty<Data>(this, _list);
    }

    void timerEvent(QTimerEvent *) {
        _list[1]->setName(_list[1]->name() + QStringLiteral("C++"));
    }

private:
    QList<Data*> _list;
};

int main(int argc, char *argv[])
{
    QGuiApplication a(argc, argv);

    qmlRegisterType<Data>();

    QQuickView view;
    view.rootContext()->setContextProperty(QStringLiteral("parser"), new Parser);
    view.setSource(QUrl("qrc:///qml/main.qml"));
    view.showNormal();

    return a.exec();
}

#include "main.moc"
Run Code Online (Sandbox Code Playgroud)

main.qml:

import QtQuick 2.0

Rectangle {
    width: 800
    height: 600

    ListView {
        id: view1
        anchors { top: parent.top; left: parent.left; bottom: parent.bottom }
        width: parent.width / 2
        spacing: 5

        delegate: Item {
            height: 30
            width: parent.width

            Text { text: name }

            MouseArea {
                anchors.fill: parent
                onClicked: model.name += "1";
            }
        }
        model: parser.list
    }

    ListView {
        id: view2
        anchors { top: parent.top; right: parent.right; bottom: parent.bottom }
        width: parent.width / 2
        spacing: 5

        delegate: Item {
            height: 30
            width: parent.width

            Text { text: name }

            MouseArea {
                anchors.fill: parent
                onClicked: model.name += "2";
            }
        }
        model: parser.list
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 非常感谢您快速而详细的回答.我用QList,QQmlListProperty,QDeclarativeListProperty,<QObject>,<QObject*>,<Data>,<Data*>等搞乱了一整天,无法找到正确的组合.然而,你的解决方案对我来说非常完美,干得好! (2认同)