当前位置: 首页 > news >正文

QML与C++交互之QML端信号绑定C++端槽函数

qml与c++交互最重要的一点就是信号与槽的绑定,本篇博客将介绍在qml定义的信号如何与c++的槽函数绑定。

案例准备,新增自定义c++类MyObject,并且注册到qml中;

具体可以查看下方链接:

QML与C++交互之创建自定义对象-CSDN博客

myobject.h

#ifndef MYOBJECT_H
#define MYOBJECT_H#include <QObject>
#include <QDebug>class MyObject : public QObject
{Q_OBJECTpublic:MyObject(QObject *parent = nullptr);  // 构造函数~MyObject();static MyObject *getInstance();const int &iValue() const;void setIIValue(const int &newIValue);const QString &sString() const;void setSString(const QString &newSString);/*** @brief func  提供给qml直接调用的函数*/Q_INVOKABLE void func();signals:void iValueChanged();void sStringChanged();public slots:// 定义槽函数与qml的信号绑定void onQmlTestSig(QString name, int age);private:int m_iValue;QString m_sString;Q_PROPERTY(int iValue READ iValue WRITE setIIValue NOTIFY iValueChanged)
//    Q_PROPERTY(QString sString READ sString WRITE setSString NOTIFY sStringChanged)// 如果值是函数内部成员变量的值,可使用MEMBER去设置,与READ sString WRITE setSString实现效果一致Q_PROPERTY(QString sString MEMBER m_sString NOTIFY sStringChanged)
};#endif // MYOBJECT_H

myobject.cpp

#include "myobject.h"MyObject::MyObject(QObject *parent) : QObject(parent)
{}MyObject::~MyObject()
{
}MyObject *MyObject::getInstance()
{static MyObject *obj = nullptr;if (!obj) {obj = new MyObject;}return obj;
}const int &MyObject::iValue() const
{return m_iValue;
}void MyObject::setIIValue(const int &newIValue)
{if (m_iValue == newIValue) {return;}m_iValue = newIValue;emit iValueChanged();
}const QString &MyObject::sString() const
{return m_sString;
}void MyObject::setSString(const QString &newSString)
{if (m_sString == newSString) {return;}m_sString = newSString;emit sStringChanged();
}void MyObject::func()
{qDebug() << __FUNCTION__ << __func__;
}void MyObject::onQmlTestSig(QString name, int age)
{qDebug() << "name = " << name << "   age = " << age;
}

重点关注:MyObject::func()函数和MyObject::onQmlTestSig(QString name, int age)槽函数,其他可忽略。

在main函数中对MyObject进行注册:

#include <QGuiApplication>
#include <QQmlApplicationEngine>#include <QQmlContext>
#include "myobject.h"int main(int argc, char *argv[])
{QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);QGuiApplication app(argc, argv);QQmlApplicationEngine engine;// 获得全局对象,上下文对象QQmlContext *context = engine.rootContext();// 给qml设置一个全局变量;如果qml内部有定义重名变量,那么会优先使用qml内部定义的变量;另外,定义全局变量会有性能问题context->setContextProperty("SCREEN_WIDTH", 800);// 注册,在需要使用的地方 import MyObj 1.0qmlRegisterType<MyObject>("MyObj", 1, 0, "MyObject");engine.load(QUrl(QStringLiteral("qrc:/main.qml")));if (engine.rootObjects().isEmpty())return -1;return app.exec();
}

1 qml直接调用C++函数

在MyObject类中,我们定义了一个函数:Q_INVOKABLE void func();

在这个函数头部使用了 Q_INVOKABLE 宏进行修饰,当函数被这个宏就是后,那么就可以被qml直接调用了。

import MyObj 1.0    // 导入自定义模块Button {width: 100; height: 50objectName: "myButton"onClicked: {// 调用C++函数myObj.func()}
}MyObject {id: myObj
}

注意,如果qml端需要直接调用c++端函数,必须使用Q_INVOKABLE 宏进行修饰;c++的槽函数是可以直接被qml调用的,所以不需要加上Q_INVOKABLE 宏进行修饰。

2 在qml端实现qml信号与c++槽函数的绑定

首先在qml定义信号:

// 定义qml信号
signal qmlTestSig(string name, int age)

该信号绑定MyObject类的onQmlTestSig槽函数;

2.1 方式一,通过Connections进行绑定

通过在qml端qmlTestSig信号触发的槽函数中,直接调用c++端槽函数的方式,可以进行信号槽的绑定;

// qml信号绑定c++槽函数方式一
Connections {target: rootfunction onQmlTestSig(name, age) {myObj.onQmlTestSig(name, age)}
}

2.2 方式二,在初始化完成后进行绑定

在初始化完成后,可以直接使用信号的connect进行绑定; 

// qml信号绑定c++槽函数方式二
Component.onCompleted: {qmlTestSig.connect(myObj.onQmlTestSig)  // 在初始化完成后进行信号和槽的绑定
}

2.3 测试

定义一个按钮,在按钮的onClicked函数中发射信号;

Button {width: 100; height: 50objectName: "myButton"onClicked: {// 调用C++函数myObj.func()// 发射信号调用C++槽函数qmlTestSig("Jtom", 26)}
}

可以看到,因为上面方式一和方式二进行了两次绑定,所以这里会进行两次的打印;

2.4 main.qml

import QtQuick 2.9
import QtQuick.Window 2.2
import QtQuick.Controls 2.14import MyObj 1.0    // 导入自定义模块Window {id: rootvisible: truewidth: SCREEN_WIDTHheight: 500title: qsTr("Hello World")color: "white"objectName: "window"// 定义qml信号signal qmlTestSig(string name, int age)// qml信号绑定c++槽函数方式一Connections {target: rootfunction onQmlTestSig(name, age) {myObj.onQmlTestSig(name, age)}}// qml信号绑定c++槽函数方式二Component.onCompleted: {qmlTestSig.connect(myObj.onQmlTestSig)  // 在初始化完成后进行信号和槽的绑定}Button {width: 100; height: 50objectName: "myButton"onClicked: {// 调用C++函数myObj.func()// 发射信号调用C++槽函数qmlTestSig("Jtom", 26)}}MyObject {id: myObjiValue: 20sString: "this is a custom obj.";Component.onCompleted: {console.log("iValue:", 20, "  sString:", sString)}}
}

3 在c++端实现qml信号与c++槽函数的绑定

在engine加载完成后,可以通过rootObjects函数获得qml端的所有对象;

首先,需要在qml中给定义的控件加上objectName;

如上main.qml中,给Window加上了objectName: "window",给Button加上了objectName: "myButton";

在C++main函数中,在engine加载完成后,获取qml端所有对象;

QList<QObject*> list = engine.rootObjects();

list的首个元素就是main.qml文件中的Window对象,可以通过获取链表的首个QObject打印观察:

// list的首个元素就是window
QObject *windowObj = list.first();
qDebug() << windowObj << "     objectName = " << windowObj->objectName();

那么就说明,windowObj变量就是Window控件的对象了;

那么Window控件下的子控件Button如何获得呢?

可以通过获取到父控件后使用findChild模板函数去查找;

现在我们已经获取到Window控件的对象了,就可以直接该对象进行查找子控件对象了;

// 获得button对象
QObject *btnObject = windowObj->findChild<QObject *>("myButton");
qDebug() << btnObject << "     objectName" << btnObject->objectName();

那么,既然都可以获取到qml中的对象了,不就可以直接使用QObject::connect函数进行信号与槽的绑定了吗?

没错,是这么回事!

// 信号与槽的绑定
QObject::connect(windowObj, SIGNAL(qmlTestSig(QString, int)),MyObject::getInstance(), SLOT(onQmlTestSig(QString ,int)));

可以看到,一共打印了三条,说明槽函数被触发了三次,因为信号与槽也绑定了三次嘛;

前两个是在qml中进行绑定的,第三个是在C++中进行绑定的;

main.cpp

#include <QGuiApplication>
#include <QQmlApplicationEngine>#include <QQmlContext>
#include "myobject.h"int main(int argc, char *argv[])
{QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);QGuiApplication app(argc, argv);QQmlApplicationEngine engine;// 获得全局对象,上下文对象QQmlContext *context = engine.rootContext();// 给qml设置一个全局变量;如果qml内部有定义重名变量,那么会优先使用qml内部定义的变量;另外,定义全局变量会有性能问题context->setContextProperty("SCREEN_WIDTH", 800);// 注册,在需要使用的地方 import MyObj 1.0qmlRegisterType<MyObject>("MyObj", 1, 0, "MyObject");engine.load(QUrl(QStringLiteral("qrc:/main.qml")));if (engine.rootObjects().isEmpty())return -1;// 在engine加载完成后,就可以获取qml的所有对象了QList<QObject*> list = engine.rootObjects();// list的首个元素就是windowQObject *windowObj = list.first();qDebug() << windowObj << "     objectName = " << windowObj->objectName();// 获得button对象QObject *btnObject = windowObj->findChild<QObject *>("myButton");qDebug() << btnObject << "     objectName" << btnObject->objectName();// 信号与槽的绑定QObject::connect(windowObj, SIGNAL(qmlTestSig(QString, int)),MyObject::getInstance(), SLOT(onQmlTestSig(QString ,int)));return app.exec();
}

到此,qml信号与c++槽函数的三种绑定方式已经介绍完毕,依据项目情况使用即可!

http://www.dtcms.com/a/268042.html

相关文章:

  • uniapp实现的多种时间线模板
  • jmm,`as - if - serial` 与 `happens - before` 原则
  • Dubbo 3.x源码(31)—Dubbo消息的编码解码
  • 容声W60以光水离子科技实现食材“主动养鲜”
  • 创客匠人深度剖析:家庭教育赛道创始人 IP 打造与知识变现的破局之道
  • 【算法刷题记录(简单题)003】统计大写字母个数(java代码实现)
  • 0704-0706上海,又聚上了
  • 【MyBatis】实现数据库的增、删、改、查
  • 深度解析命令模式:将请求封装为对象的设计智慧
  • 儿童趣味记忆配对游戏
  • LeetCode 75. 颜色分类(荷兰国旗问题)
  • 一次佳能iX6780彩色喷墨打印机报5B00维修的记录
  • 【网络协议安全】任务13:ACL访问控制列表
  • 牛客周赛Round 99(Go语言)
  • 《kubernetes》k8s实战之部署PHP/JAVA网站
  • 中级统计师-经济学基础知识-第四章 国民收入核算
  • 单片机物联网应用中的 Pogopin、串口与外围模组通信技术解析
  • Java 大视界 -- Java 大数据在智能教育在线课程学习效果影响因素分析与优化设计(334)
  • Zotero中进行文献翻译【Windows11】
  • SpiceMix enables integrative single-cell spatial modeling of cell identity 文章解读
  • 【kafka-python使用学习笔记1:Python操作Kafka之环境准备(1)】
  • 2、Connecting to Kafka
  • css模块化以及rem布局
  • linux/ubuntu日志管理--/dev/log 的本质与作用
  • arm 精准总线错误与非精准总线错误
  • C#使用Qdrant实现向量存储及检索
  • 基于ARM+FPGA的光栅尺精密位移加速度测试解决方案
  • 【精密测量】基于ARM+FPGA的多路光栅信号采集方案
  • 【PyTorch 当前版本不支持 NVIDIA GeForce RTX 5060 Ti处理办法】
  • 求医十年,病因不明,ChatGPT:你看起来有基因突变