在QML QList <QPair>中接收带有参数的c ++信号

时间:2019-08-27 09:24:15

标签: qt qml qt5 signals-slots qt-signals

我试图弄清楚如何从QML中的C ++信号获取QList>,我只得到QVariant(RecordList,)或QVariant(QList,)。尝试使用其他受支持的序列类型,它们可以正常工作(QList。如果有人可以帮助我理解我的错误,我将不胜感激。

jsonreaderasync.h

typedef QPair<qreal,qreal>Record;
typedef QList<Record>RecordList;
class JsonReaderAsync : public QObject
{
    Q_OBJECT
public:
    explicit JsonReaderAsync(QObject *parent = nullptr);
    Q_INVOKABLE void start(const QString& fileName);

signals:
    void started();
    //void finished(QList<qreal> record);
    void finished(RecordList record);
    //void finished(QList<Record> record); 

};

jsonreaderasync.cpp

#include <QDebug>

#include "jsonreaderasync.h"

Q_DECLARE_METATYPE(Record)
Q_DECLARE_METATYPE(RecordList)
//Q_DECLARE_METATYPE(QList<Record>)


JsonReaderAsync::JsonReaderAsync(QObject *parent) : QObject(parent)
{
    qRegisterMetaType<Record>("Record");
    qRegisterMetaType<RecordList>("RecordList"); // qml prints QVariant(RecordList, )
    //qRegisterMetaType<QList<Record>>("QList<Record>"); //qml prints QVariant(QList<Record>, )

}

void JsonReaderAsync::start(const QString &fileName)
{

    QList<Record> record;
    record.append(qMakePair(1,1));
    record.append(qMakePair(1,2));
    record.append(qMakePair(1,3));

//    QList<qreal> foo;
//    foo.append(1);
//    foo.append(1);
//    foo.append(1);

    emit finished(record);

}

main.cpp

#include "jsonreaderasync.h"

typedef QPair<qreal,qreal>Record;
typedef QList<QPair<qreal,qreal>>RecordList;


int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    QQmlApplicationEngine *engine = new QQmlApplicationEngine;
    JsonReaderAsync* dataReaderAsync = new JsonReaderAsync();
    engine->rootContext()->setContextProperty("JsonReaderAsync", dataReaderAsync);
    engine->load(QUrl(QStringLiteral("main.qml")));

    return app.exec();
}

main.qml

import QtQuick 2.12
import QtQuick.Dialogs 1.2
import QtQuick.Layouts 1.13
import QtQuick.Window 2.13

import QtQuick.Controls 2.12
import QtCharts 2.3

Window {
  id: appWindow
  visible: true
  minimumWidth : 400
  minimumHeight: 400

  Connections
  {
    target: JsonReaderAsync
    onStarted:{
      console.log("onStarted")
    }
    onFinished:{
      console.log("onFinished")
      console.log(record)
      console.log(record[0])
    }
  }

  Button {
    width : 40
    height: 40
    anchors.centerIn: parent
    onClicked: {
      JsonReaderAsync.start("")
    }
  }

}

2 个答案:

答案 0 :(得分:1)

在QML中,元类型系统是QML引擎可以从QML环境访问C ++结构的唯一方法。

也就是说,只有Predefined C++ Types和具有Q_PROPERTY声明的自定义对象可以从QML环境访问。

这是我推荐的(最简单的)修改方法:

jsonreaderasync.h

#include <QObject>
#include <QVariant>

class Record : public QPair<qreal, qreal> {
    Q_GADGET

    Q_PROPERTY(qreal first  MEMBER first  CONSTANT FINAL)
    Q_PROPERTY(qreal second MEMBER second CONSTANT FINAL)

public:

    Record() = default;
    Record(qreal a, qreal b) : QPair<qreal, qreal>(a, b) {}
};

class JsonReaderAsync : public QObject
{
    Q_OBJECT
public:
    explicit JsonReaderAsync(QObject *parent = nullptr);
    Q_INVOKABLE void start(const QString& fileName);

signals:
    void started();

    void finished(QVariantList record);  // RecordList
};

jsonreaderasync.cpp

#include <QDebug>

#include "jsonreaderasync.h"

JsonReaderAsync::JsonReaderAsync(QObject *parent) : QObject(parent)
{
    qRegisterMetaType<Record>("Record");
}

void JsonReaderAsync::start(const QString &fileName)
{
    QVariantList record;
    record.append(QVariant::fromValue(Record(1,1)));
    record.append(QVariant::fromValue(Record(1,2)));
    record.append(QVariant::fromValue(Record(1,3)));

    emit finished(record);
}

现在您可以从QML访问记录了:

onFinished: {
    for (var i = 0; i < record.length; ++i)
        console.log(record[i].first + "->" + record[i].second);
}

请注意,在C ++对象和QML对象之间进行转换确实会产生一些开销,如果这些代码对性能敏感,请考虑使用C++ Data Models

答案 1 :(得分:0)

虽然@GPBeta解决方案有效,但我希望Qml支持的线对具有更大的灵活性。我尝试使用模板,但是Q_GADGET不支持它。我猜可能会有一个聪明的包装器(Template + QVariant)解决方案……尽管如此,这是我解决问题的方法:

class PairQml {
    Q_GADGET

    Q_PROPERTY(QVariant first MEMBER first CONSTANT FINAL)
    Q_PROPERTY(QVariant second MEMBER second CONSTANT FINAL)

public:
    PairQml() = default;
    PairQml(QVariant f, QVariant s): first(f), second(s) {}

    QVariant first;
    QVariant second;
};

别忘了注册:qRegisterMetaType<PairQml>("PairQml");