我想做这样的事情
QML app:
{
signal qmlSignal
function qmlFunction
}
和
c++ Hnadler:
{
c++ slot
c++ signal
}
想要与同一个 qml 对象进行两种方式的通信。
我指的是 http://qt-project.org/doc/qt-4.8/qtbinding.html
要从 C++ 更改 qml 中的值,我们可以这样做
QDeclarativeEngine engine;
QDeclarativeComponent component(&engine, "MyItem.qml");
QObject *object = component.create();
QVariant returnedValue;
QVariant msg = "Hello from C++";
QMetaObject::invokeMethod(object, "myQmlFunction",
Q_RETURN_ARG(QVariant, returnedValue),
Q_ARG(QVariant, msg));
qDebug() << "QML function returned:" << returnedValue.toString();
delete object;
可以用来调用qml函数。但是有了这个我们不能使用 QT 方法
和
class MyClass : public QObject
{
Q_OBJECT
public slots:
void cppSlot(const QString &msg) {
qDebug() << "Called the C++ slot with message:" << msg;
}
};
int main(int argc, char *argv[]) {
QApplication app(argc, argv);
QDeclarativeView view(QUrl::fromLocalFile("MyItem.qml"));
QObject *item = view.rootObject();
MyClass myClass;
QObject::connect(item, SIGNAL(qmlSignal(QString)),
&myClass, SLOT(cppSlot(QString)));
view.show();
return app.exec();
}
这可用于 qmlsignal 和 c++ 插槽。
有没有一种方法可以在一个对象中完成这两项工作?
最佳答案
=== 更新问题 ===
您可以在 QML 中调用槽或 Q_INVOKABLE 方法来更改 QML 中的“C++ 属性”。不过,您需要将 C++ 对象公开为上下文属性。你需要在下面写这样的东西:
我的类.h
class MyClass : public QObject
{
Q_OBJECT
public:
MyClass(QObject *parent) : QObject(parent) { ... }
Q_INVOKABLE void myInvokable();
public slots:
void mySlot();
...
}
主程序
...
MyClass myClassObject;
QQuickView view;
view.rootContext()->setContextProperty("myClassContextProperty", &myClassObject;
view->setSource(QUrl::fromLocalFile("main.qml"));
view->show();
...
主文件
Button {
...
// You can replace onClicked with your own custom signal
onClicked: myClassContextProperty.myslot()
// or
onClicked: myClassContextProperty.myInvokable()
...
}
===评论中有更多问题===
您将在 C++ 中有一个 Q_PROPERTY,然后将该属性绑定(bind)到 QML 中的属性,或者捕获名为“foo”的属性的 onFooChanged 信号处理程序。
我的类.h
class MyClass : public QObject
{
Q_OBJECT
Q_PROPERTY(foo READ foo NOTIFY fooChanged)
public:
MyClass(QObject *parent) : QObject(parent) { ... }
int foo() const;
public signals:
void fooChanged();
public slots:
void mySlot() { foo = 5; emit fooChanged(); };
private:
int foo;
...
}
主程序
...
MyClass myClassObject;
QQuickView view;
view.rootContext()->setContextProperty("myClassContextProperty", &myClassObject;
view->setSource(QUrl::fromLocalFile("main.qml"));
view->show();
...
主文件
...
// Bind foo to bar
property int bar: myClassContextProperty.foo
=== 原始问题 ===
似乎您想询问是否可以同时在 C++ 和 QML 中声明相同的内容,即在 QML 中声明它的某些部分,然后在 QML 中声明其余部分。
我认为您需要使用 Qt 元类型系统注册(qmlRegisterMetaType 等)您的 C++ 类型,然后您可以将该类用作 QML 中的一个项目(可能与父类一起使用?)来实现您的愿望。
这是进一步阅读的相关文档:
Registering an Instantiable Object Type
为了您的方便,这里有一些内联代码:
消息.h
class Message : public QObject
{
Q_OBJECT
Q_PROPERTY(QString author READ author WRITE setAuthor NOTIFY authorChanged)
Q_PROPERTY(QDateTime creationDate READ creationDate WRITE setCreationDate NOTIFY creationDateChanged)
public:
// ...
};
qmlRegisterType<Message>("com.mycompany.messaging", 1, 0, "Message");
...
主文件
import com.mycompany.messaging 1.0
Message {
author: "Amelie"
creationDate: new Date()
}
关于c++ - 拥有 qml 函数和 c++ 插槽的最佳方法,反之亦然,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/20988224/