gpt4 book ai didi

c++ - 使用 C++ 对象作为参数从 QML/JS 调用 C++ 函数

转载 作者:行者123 更新时间:2023-12-04 04:14:58 27 4
gpt4 key购买 nike

我想从 QML/JS 调用 C++ 函数,它以 QML 中实例化的 C++ 类(数据对象)作为参数。
该程序编译并运行,但是当我尝试从 QML/JS 调用 C++ 函数时,出现以下错误:

“无法将参数 0 转换为”
“onClicked@qrc:/main.qml:26”
“将不兼容的参数从 JavaScript 传递给 C++ 函数是危险的,并且已被弃用。”
“这将在 Qt 的 future 版本中引发 JavaScript TypeError!”

我已经尝试了通过引用/值/指针调用的所有组合,以及 Q_GADGET 和 Q_OBJECT 的多种变体,但我根本无法让它工作。

我的基于 QObject 的 C++ 数据类型以及所有调用组合的最小示例如下:

C++ 数据类型


#include <QObject>
#include <QDebug>

//Data type which should be instantiated in QML
class CustomDataObject : public QObject{

Q_OBJECT
Q_PROPERTY(int exampleValue MEMBER exampleValue)

public:

int exampleValue;

explicit CustomDataObject(QObject * parent = nullptr) : QObject(parent) {}
~CustomDataObject() = default;
CustomDataObject(const CustomDataObject& blob) {}

};

Q_DECLARE_METATYPE(CustomDataObject);


具有调用函数的 C++ 类

//Class with function called from QML
class UniqueDataBackend : public QObject {

Q_OBJECT

public:

//Call with native data type (works)
Q_INVOKABLE void processNativeObject(int value) {
qDebug() << "C++: Nativ Value: " << value;
}

//Call by value (does not work)
Q_INVOKABLE void processDataObjectByValue(const CustomDataObject data) {
qDebug() << "C++: Data Object Value: " << data.exampleValue;
}

//Call by reference (does not work)
Q_INVOKABLE void processDataObjectByReference(const CustomDataObject & data) {
qDebug() << "C++: Data Object Value: " << data.exampleValue;
}

//Call by pointer (does not work)
Q_INVOKABLE void processDataObjectByPointer(const CustomDataObject * data) {
qDebug() << "C++: Data Object Value: " << data->exampleValue;
}


UniqueDataBackend(QObject * parent = nullptr) : QObject(parent) {}
~UniqueDataBackend() = default;
UniqueDataBackend(const UniqueDataBackend& blob) {}
};

Q_DECLARE_METATYPE(UniqueDataBackend);


main.cpp

#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QQmlContext>
#include <QQmlEngine>

#include "CustomStruct.h"

int main(int argc, char *argv[])
{
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);

QGuiApplication app(argc, argv);

QQmlApplicationEngine engine;

//Register types for use in QML
qRegisterMetaType<CustomDataObject>();
qmlRegisterType<CustomDataObject>("Custom.Types", 1, 0, "CustomDataObject");

//Register intances for use in QML
engine.rootContext()->setContextProperty("UniqueDataBackend", new UniqueDataBackend());

const QUrl url(QStringLiteral("qrc:/main.qml"));
QObject::connect(&engine, &QQmlApplicationEngine::objectCreated,
&app, [url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
}, Qt::QueuedConnection);
engine.load(url);

return app.exec();
}



QML 文件 (main.qml)

import QtQuick 2.12
import QtQuick.Controls 2.12
import Custom.Types 1.0

ApplicationWindow {
id: window
width: 400
height: 300
visible: true

CustomDataObject {
id: customData
exampleValue: 84
}

Button {
text: "Click me!"
anchors.fill: parent
onClicked: {
console.log("QML: Calling c++ function with native data type!")
UniqueDataBackend.processNativeObject(42)

console.log("QML: Checking value of custom data type in javascript: " + customData.exampleValue)

console.log("QML: Calling c++ function by value with custom data type!")
UniqueDataBackend.processDataObjectByValue(customData)

console.log("QML: Calling c++ function by reference with custom data type!")
UniqueDataBackend.processDataObjectByReference(customData)

console.log("QML: Calling c++ function by pointer with custom data type!")
UniqueDataBackend.processDataObjectByPointer(customData)
}
}


}



输出

qml: QML: Calling c++ function with native data type!
C++: Nativ Value: 42

qml: QML: Checking value of custom data type in javascript: 84

qml: QML: Calling c++ function by value with custom data type!
"Could not convert argument 0 at"
"onClicked@qrc:/main.qml:26"
"Passing incompatible arguments to C++ functions from JavaScript is dangerous and deprecated."
"This will throw a JavaScript TypeError in future releases of Qt!"
C++: Data Object Value: 0

qml: QML: Calling c++ function by reference with custom data type!
"Could not convert argument 0 at"
"onClicked@qrc:/main.qml:29"
"Passing incompatible arguments to C++ functions from JavaScript is dangerous and deprecated."
"This will throw a JavaScript TypeError in future releases of Qt!"
C++: Data Object Value: -842150451

qml: QML: Calling c++ function by pointer with custom data type!
qrc:/main.qml:32: Error: Unknown method parameter type: const CustomDataObject*


我认为这是自定义数据类型的问题,因为我可以毫无问题地调用具有 native 数据类型的函数,但在我的一生中,我无法弄清楚。

非常感谢您的帮助/评论!

最佳答案

通过注册 Error: Unknown method parameter type: const CustomDataObject* 可以修复报告未知类型( const )的错误也输入:

qRegisterMetaType<CustomDataObject*>("const CustomDataObject*");
无法转换的错误 ( "Could not convert argument 0 at..." ) 更棘手。 QML 中的类型转换使用 QT 的元类型系统,它基于 QVariant .内置转换在 QVariant.canConvert() 中列出。文档。此外,您还可以阅读指向 QObject 的子类型的指针之间的转换。是自动的。这就是为什么您的 processDataObjectByPointer有效,如果您注册 const类型如上所述。
但除了错误消息 "Passing incompatible arguments to C++ functions from JavaScript is dangerous and deprecated." 之外,我找不到关于将对象作为值或引用传递的提及。显然这些类型是兼容的,但消息提示我正确的方法是将值和引用作为 QJSValue 传递。 .下面的方法可以作为引用。同样适用于“按值(value)”等价物:
void UniqueDataBackend::processDataObjectByQJSValue(const QJSValue& data) {
qDebug() << "C++: Data Object Value: " << qobject_cast<CustomDataObject*>(data.toQObject())->getExample();
}
出于好奇,我检查是否可以使用 QMetaType.registerConverter<From,CustomDataObject> 注册函数转换器因此您可以调用通过值和引用传递对象的方法。这里的难点是要知道哪个是 From Qml 内部使用的类型。安装了QT的调试符号、源码,在几个地方放了断点,发现它的ID是39。通过查看 table of built-in typesQMetaType ,是 QObject* .转换器的使用需要实现赋值运算符:
CustomDataObject &CustomDataObject::operator=(const CustomDataObject &other) {
exampleValue = other.exampleValue;
return *this;
}
以下代码安装了一个转换器,该转换器执行按值和常量引用调用的技巧。但不是通过引用,因为实际参数是在转换器处创建的 QML 对象的拷贝;因此更改不会反射(reflect)在原始文件中。请记住,这不是官方建议,并且完全依赖于内部表示(它在 QT 5.12.8 中对我有用):
QMetaType::registerConverter<QObject*,CustomDataObject>( [] (QObject* qObjPtr) {
CustomDataObject* dataPtr = qobject_cast<CustomDataObject*>( qObjPtr );
return (dataPtr == nullptr) ? CustomDataObject() : CustomDataObject( *dataPtr ) ;
});

关于c++ - 使用 C++ 对象作为参数从 QML/JS 调用 C++ 函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60915460/

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