我想做这样的事情

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/

10-11 05:00
查看更多