【QML COOK】- 006-用C++定义一个QML元素类型

2024-01-11 01:52
文章标签 c++ 类型 元素 定义 006 cook qml

本文主要是介绍【QML COOK】- 006-用C++定义一个QML元素类型,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Qt原本是一个C++图形框架,因此QML也少不了C++。QML通常只负责显示,而后台逻辑由C++实现,因此掌握C++和QML之间的交互非常必要。

本例实现一个最简单的例子,用C++定义一个QML的元素类型并在QML使用它。

需求是在窗口上显示鼠标点击的次数。用C++定义了一个Counter的QML元素类型来实现计数。

1. 创建C++类

选择【C++】 -> 【C++ Class】

起个名字,并且选上【Include QObject】【Add Q_OBJECT】【Add QML_ELEMENT】

点击完成

稍微解释一下。

QML是一种声明语言,想用C++定义声明语言的类型,就必须让C++具备反射能力

什么是反射能力

举个最简单的例子,在声明语言中声明一个Counter类型的对象,解释器将代码读到内存后需要根据字符串“Counter”去创建一个Counter类型的C++对象。C++本身显然是不具备这种能力。因此Qt实现了一套让C++实现反射能力的框架。这个框架由一系列类、宏及编译命令配合完成的,因此在创建对象时要选择继承Object类并且添加【Q_OBJECT】【Q_ELEMENT】这两个宏。

2. 查看并修改CMakeList.txt

在CMakeList.txt文件中查找【qt_add_qml_module】,看它是不是如下

qt_add_qml_module(appQT_COOKURI QT_COOKVERSION 1.0QML_FILESMain.qmlSOURCEScounter.cpp counter.h
)

通常添加完C++类后,Qt Create会自动将CMakeList.txt修改为上面那样。如果没有请手动修正,即添加以下内容

    SOURCEScounter.cpp counter.h

这也是为了让C++实现反射能力的操作。

3. 编辑count.h

#ifndef COUNTER_H
#define COUNTER_H#include <QObject>
#include <QQmlEngine>class Counter : public QObject
{Q_OBJECTQML_ELEMENTQ_PROPERTY(int count READ Count WRITE SetCount NOTIFY CountChanged)
public:explicit Counter(QObject *parent = nullptr);void SetCount(const int& count);const int& Count() const;signals:void CountChanged();private:int m_count {0};
};#endif // COUNTER_H

解释一下重要的点

  • 创建的QML元素类型名,同C++名一致。
  • 创建的QML具备哪些属性由Q_PROPERTY声明
Q_PROPERTY(int count READ Count WRITE SetCount NOTIFY CountChanged)

该宏的使用方法可以自行查看Qt帮助文档,这里把使用到的参数介绍一下。

  • int是属性的类型
  • count是属性名称
  • READ Count是告诉QML的解析器调用Count这个方法来获取count的值
  • WRITE SetCount是告诉QML的解析器用SetCount这个方法来更新count的值
  • NOTIFY CountChanged是告诉QML的解析器当count的值发生变化时会发送CountChanged信号

4. 编辑counter.cpp

#include "counter.h"Counter::Counter(QObject *parent): QObject{parent}
{}void Counter::SetCount(const int& count)
{m_count = count;emit CountChanged();
}const int& Counter::Count() const
{return m_count;
}

主要是实现Q_PROPERTY提到的Count和SetCount方法。一定要在SetCount中发送CountChanged这个信号。

5. 编辑Main.qml

import QtQuick
import QT_COOKWindow {width: 640height: 480visible: truetitle: qsTr("Hello World")Counter {id: myCounter}MouseArea {anchors.fill: parentonClicked: myCounter.count += 1}Text {id: nametext: myCounter.countanchors.centerIn: parentfont.pointSize: 60}
}
  • import QT_COOK是为了引入Counter类型,因为在CMakeList.txt我们将URL设置成了QT_COOK,所以这里就要引入QT_COOK
  • 创建对象,这里可以理解让C++ 创建一个Counter类型的变量名为myCounter
    Counter {id: myCounter}
  • 在鼠标点击是让myCounter的count属性的值加1
    MouseArea {anchors.fill: parentonClicked: myCounter.count += 1}
  • 让myCounter的count属性的值在窗口中间显示
    Text {id: nametext: myCounter.countanchors.centerIn: parentfont.pointSize: 60}

这里要区分【绑定】和【赋值】两个概念。“text: myCounter.count”的意思是将【text】和【myCounter.count】进行绑定而不是赋值。也就是说【myCounter.count】的值发生变化后【text】的显示也跟着变化。

至于QML的内部实现,猜测是当SetCount方法里发射了信号CountChanged后,QML解析器收到这个信号,就会再次调用Count方法获取更新后的值,然后让text重新渲染。可以在SetCount和Count方法中打印Log验证一下我说的是否正确。或者在SetCount中不再发送信号看看text会不会自动更新

6. 运行程序

每点击一下鼠标数字就会加1

这篇关于【QML COOK】- 006-用C++定义一个QML元素类型的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/592765

相关文章

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

Redis的Hash类型及相关命令小结

《Redis的Hash类型及相关命令小结》edisHash是一种数据结构,用于存储字段和值的映射关系,本文就来介绍一下Redis的Hash类型及相关命令小结,具有一定的参考价值,感兴趣的可以了解一下... 目录HSETHGETHEXISTSHDELHKEYSHVALSHGETALLHMGETHLENHSET

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

Python中异常类型ValueError使用方法与场景

《Python中异常类型ValueError使用方法与场景》:本文主要介绍Python中的ValueError异常类型,它在处理不合适的值时抛出,并提供如何有效使用ValueError的建议,文中... 目录前言什么是 ValueError?什么时候会用到 ValueError?场景 1: 转换数据类型场景

C# dynamic类型使用详解

《C#dynamic类型使用详解》C#中的dynamic类型允许在运行时确定对象的类型和成员,跳过编译时类型检查,适用于处理未知类型的对象或与动态语言互操作,dynamic支持动态成员解析、添加和删... 目录简介dynamic 的定义dynamic 的使用动态类型赋值访问成员动态方法调用dynamic 的

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�