c++ qt qlistwidget清空_Qt编写控件属性设计器12-用户属性

一、前言

用户属性是后面新增加的一个功能,自定义控件如果采用的Q_PROPERTY修饰的属性,会自动识别到属性栏中,这个一般称为控件属性,在组态设计软件中,光有控件本身的控件属性还是不够的,毕竟这些属性仅仅是以外观为主,并不能表示某个设备的属性,所以需要除了这个控件属性以外增加用户属性来存储该控件关联的设备属性,比如设备编号、设备名称、地理位置等信息,而这些信息也要和控件属性一样,都能导入导出到xml文件,同时能支持多个用户属性,用户自己填写名字和值,名字和值都支持中文描述,在xml文件中为了区分用户属性和控件属性,特意在用户属性前面加上user-前缀来表示,这样在读取xml文件加载控件的时候,识别到user-开头的都存储到该控件的用户属性列表中。自从有了用户属性的机制,大大拓展了控件的现有功能,相当于可以绑定N个自定义的数据,而这些用户属性直接采用setProperty来设置即可,然后通过property来读取就行,为了支持中文的属性名称,需要设置属性的时候转换一下:widget->setProperty(name.toStdString().c_str(), value);

体验地址:[https://gitee.com/feiyangqingyun/QUCSDK](https://gitee.com/feiyangqingyun/QUCSDK)

[https://github.com/feiyangqingyun/qucsdk](https://github.com/feiyangqingyun/qucsdk)

二、实现的功能

1. 自动加载插件文件中的所有控件生成列表,默认自带的控件超过120个。

2. 拖曳到画布自动生成对应的控件,所见即所得。

3. 右侧中文属性栏,改变对应的属性立即应用到对应选中控件,直观简洁,非常适合小白使用。

4. 独创属性栏文字翻译映射机制,效率极高,可以非常方便拓展其他语言的属性栏。

5. 所有控件的属性自动提取并显示在右侧属性栏,包括枚举值下拉框等。

6. 支持手动选择插件文件,外部导入插件文件。

7. 可以将当前画布的所有控件配置信息导出到xml文件。

8. 可以手动选择xml文件打开控件布局,自动根据xml文件加载控件。

9. 可拉动滑动条、勾选模拟数据复选框、文本框输入,三种方式来生成数据应用所有控件。

10. 控件支持八个方位拉动调整大小,自适应任意分辨率,可键盘上下左右微调位置。

11. 打通了串口采集、网络采集、数据库采集三种方式设置数据。

12. 代码极其精简,注释非常详细,可以作为组态的雏形,自行拓展更多的功能。

13. 纯Qt编写,支持任意Qt版本+任意编译器+任意系统。

三、效果图

f5267690c137678cdcb423f4a7751b33.gif

四、核心代码

void frmMain::openFile(const QString &fileName){ //如果控件列表没有则不用继续 if (ui->listWidget->count() == 0) { return; } //打开文件 QFile file(fileName); if (!file.open(QFile::ReadOnly | QFile::Text)) { return; } //将文件填充到dom容器 QDomDocument doc; if (!doc.setContent(&file)) { file.close(); return; } file.close(); listSelect.clear(); listUserProperty.clear(); xmlName = fileName; //先清空原有控件 QList widgets = ui->centralwidget->findChildren(); qDeleteAll(widgets); widgets.clear(); //先判断根元素是否正确 QDomElement docElem = doc.documentElement(); if (docElem.tagName() == "canvas") { QDomNode node = docElem.firstChild(); QDomElement element = node.toElement(); while(!node.isNull()) { //控件名称 QString name = element.tagName(); //取出当前控件在控件列表中的索引,如果不存在则意味着配置文件中的该控件不存在了 int index = listNames.indexOf(name); if (index < 0) { continue; } //存储控件的坐标位置和宽度高度 int x, y, width, height; //存储自定义控件属性 QList > propertys; //存储控件自定义属性 QStringList userProperty; //节点名称不为空才继续 if (!name.isEmpty()) { //遍历节点的属性名称和属性值 QDomNamedNodeMap attrs = element.attributes(); for (int i = 0; i < attrs.count(); i++) { QDomNode node = attrs.item(i); QString nodeName = node.nodeName(); QString nodeValue = node.nodeValue(); //qDebug() << name << nodeName << nodeValue; //优先取出坐标+宽高属性,这几个属性不能通过设置弱属性实现 if (nodeName == "x") { x = nodeValue.toInt(); } else if (nodeName == "y") { y = nodeValue.toInt(); } else if (nodeName == "width") { width = nodeValue.toInt(); } else if (nodeName == "height") { height = nodeValue.toInt(); } else if (nodeName.startsWith("user-")) { //取出user-开头的自定义属性 nodeName = nodeName.split("-").last(); userProperty << QString("%1|%2").arg(nodeName).arg(nodeValue); } else { QVariant value = QVariant(nodeValue); //为了兼容Qt4,需要将颜色值的rgba分别取出来,因为Qt4不支持16进制字符串带透明度 //#6422a3a9 这种格式依次为 argb 带了透明度的才需要特殊处理 if (nodeValue.startsWith("#") && nodeValue.length() == 9) { bool ok; int alpha = nodeValue.mid(1, 2).toInt(&ok, 16); int red = nodeValue.mid(3, 2).toInt(&ok, 16); int green = nodeValue.mid(5, 2).toInt(&ok, 16); int blue = nodeValue.mid(7, 2).toInt(&ok, 16); value = QColor(red, green, blue, alpha); } propertys.append(qMakePair(nodeName, value)); } } } //qDebug() << name << x << y << width << height; //根据不同的控件类型实例化控件 int countWidget = listWidgets.count(); int countProperty = propertys.count(); for (int i = 0; i < countWidget; i++) { QString className = listWidgets.at(i)->name(); if (name == className) { //生成对应的控件 QWidget *widget = createWidget(i); //逐个设置自定义控件的属性 for (int j = 0; j < countProperty; j++) { QPair property = propertys.at(j); QString name = property.first; QVariant value = property.second; widget->setProperty(name.toStdString().c_str(), value); } //设置控件坐标及宽高 widget->setGeometry(x, y, width, height); //实例化选中窗体跟随控件一起 newSelect(widget, userProperty); break; } } //移动到下一个节点 node = node.nextSibling(); element = node.toElement(); } }}void frmMain::saveFile(const QString &fileName){ //如果控件列表没有则不用继续 if (ui->listWidget->count() == 0) { return; } QFile file(fileName); if (!file.open(QFile::WriteOnly | QFile::Text | QFile::Truncate)) { return; } //以流的形式输出文件 QTextStream stream(&file); //构建xml数据 QStringList list; //添加固定头部数据 list << "<?xml version="1.0" encoding="UTF-8"?>"; //添加canvas主标签,保存宽高和背景图片,还可以自行添加其他属性 list << QString("") .arg(ui->centralwidget->width()).arg(ui->centralwidget->height()).arg("bg.jpg"); //从容器中找到所有控件,根据控件的类名保存该类的所有属性 QList widgets = ui->centralwidget->findChildren(); foreach (QWidget *widget, widgets) { const QMetaObject *metaObject = widget->metaObject(); QString className = metaObject->className(); //如果当前控件的父类不是主窗体则无需导出,有些控件有子控件无需导出 if (widget->parent() != ui->centralwidget || className == "SelectWidget") { continue; } //逐个存储自定义控件属性 //metaObject->propertyOffset()表示当前控件的属性开始索引,0开始的是父类的属性 QStringList values; int index = metaObject->propertyOffset(); int count = metaObject->propertyCount(); for (int i = index; i < count; i++) { QMetaProperty property = metaObject->property(i); QString nodeName = property.name(); QVariant variant = property.read(widget); QString typeName = variant.typeName(); QString nodeValue = variant.toString(); //如果是颜色值则取出透明度一起,颜色值toString在Qt4中默认不转透明度 if (typeName == "QColor") { QColor color = variant.value(); if (color.alpha() < 255) { //Qt4不支持HexArgb格式的字符串,需要挨个取出来拼接 //nodeValue = color.name(QColor::HexArgb); QString alpha = QString("%1").arg(color.alpha(), 2, 16, QChar('0')); QString red = QString("%1").arg(color.red(), 2, 16, QChar('0')); QString green = QString("%1").arg(color.green(), 2, 16, QChar('0')); QString blue = QString("%1").arg(color.blue(), 2, 16, QChar('0')); nodeValue = QString("#%1%2%3%4").arg(alpha).arg(red).arg(green).arg(blue); } } //枚举值要特殊处理,需要以字符串形式写入,不然存储到配置文件数据为int if (property.isEnumType()) { QMetaEnum enumValue = property.enumerator(); nodeValue = enumValue.valueToKey(nodeValue.toInt()); } values << QString("%1="%2"").arg(nodeName).arg(nodeValue); //qDebug() << nodeName << nodeValue << variant; } //找到当前控件对应的索引 index = -1; count = listSelect.count(); for (int i = 0; i < count; i++) { if (listSelect.at(i)->getWidget() == widget) { index = i; break; } } //可以用下面方法列出所有的用户属性,然后取值,本程序已经用 listUserProperty 存储了 //qDebug() << widget->dynamicPropertyNames(); //逐个存储控件的用户属性 QStringList userProperty = listUserProperty.at(index); count = userProperty.count(); for (int i = 0; i < count; i++) { QStringList list = userProperty.at(i).split("|"); values << QString("user-%1="%2"").arg(list.at(0)).arg(list.at(1)); } //逐个添加界面上的控件的属性 QString geometry = QString("x="%1" y="%2" width="%3" height="%4"").arg(widget->x()).arg(widget->y()).arg(widget->width()).arg(widget->height()); QString str = QString("").arg(className).arg(geometry).arg(values.join(" ")); list << str; } //添加固定尾部数据 list << ""; //写入文件 QString data = list.join(""); stream << data; file.close();}

五、控件介绍

1. 超过160个精美控件,涵盖了各种仪表盘、进度条、进度球、指南针、曲线图、标尺、温度计、导航条、导航栏,flatui、高亮按钮、滑动选择器、农历等。远超qwt集成的控件数量。

2. 每个类都可以独立成一个单独的控件,零耦合,每个控件一个头文件和一个实现文件,不依赖其他文件,方便单个控件以源码形式集成到项目中,较少代码量。qwt的控件类环环相扣,高度耦合,想要使用其中一个控件,必须包含所有的代码。

3. 全部纯Qt编写,QWidget+QPainter绘制,支持Qt4.6到Qt5.12的任何Qt版本,支持mingw、msvc、gcc等编译器,支持任意操作系统比如windows+linux+mac+嵌入式linux等,不乱码,可直接集成到Qt Creator中,和自带的控件一样使用,大部分效果只要设置几个属性即可,极为方便。

4. 每个控件都有一个对应的单独的包含该控件源码的DEMO,方便参考使用。同时还提供一个所有控件使用的集成的DEMO。

5. 每个控件的源代码都有详细中文注释,都按照统一设计规范编写,方便学习自定义控件的编写。

6. 每个控件默认配色和demo对应的配色都非常精美。

7. 超过130个可见控件,6个不可见控件。

8. 部分控件提供多种样式风格选择,多种指示器样式选择。

9. 所有控件自适应窗体拉伸变化。

10. 集成自定义控件属性设计器,支持拖曳设计,所见即所得,支持导入导出xml格式。

11. 自带activex控件demo,所有控件可以直接运行在ie浏览器中。

12. 集成fontawesome图形字体+阿里巴巴iconfont收藏的几百个图形字体,享受图形字体带来的乐趣。

13. 所有控件最后生成一个动态库文件(dll或者so等),可以直接集成到qtcreator中拖曳设计使用。

14. 目前已经有qml版本,后期会考虑出pyqt版本,如果用户需求量很大的话。

15. 自定义控件插件开放动态库使用(永久免费),无任何后门和限制,请放心使用。

16. 目前已提供26个版本的dll,其中包括了qt5.12.3 msvc2017 32+64 mingw 32+64 的。

17. 不定期增加控件和完善控件,不定期更新SDK,欢迎各位提出建议,谢谢!

18. Qt入门书籍推荐霍亚飞的《Qt Creator快速入门》《Qt5编程入门》,Qt进阶书籍推荐官方的《C++ GUI Qt4编程》。

19. 强烈推荐程序员自我修养和规划系列书《大话程序员》《程序员的成长课》《解忧程序员》,受益匪浅,受益终生!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/472993.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

TensorFlow 2.0 - tf.saved_model.save 模型导出

文章目录1. tf.saved_model.save2. Keras API 模型导出学习于&#xff1a;简单粗暴 TensorFlow 2 1. tf.saved_model.save tf.train.Checkpoint 可以保存和恢复模型中参数的权值导出模型&#xff1a;包含参数的权值&#xff0c;计算图 无须源码即可再次运行模型&#xff0c;适…

机器人动力学与控制_力控制与位置控制的区别

1.背景介绍目前已经广泛落地的力控制方案是在机械臂末端安装多轴力矩传感器&#xff0c;用以检测机械臂对外界环境施加的力反馈值&#xff0c;并配合适当的控制策略&#xff0c;已达到控制机械臂与环境的作用力。这篇文章所要探讨的力控制&#xff08;上述力控制方案&#xff0…

基坑监测日报模板_基坑监测有多重要?实录基坑坍塌过程,不亲身经历,不知道现场有多恐怖!...

基坑整体坍塌不亲身经历&#xff0c;不知其恐怖▼前段时间&#xff0c;南宁绿地中央广场房地产项目D号地块(二期)基坑北侧约60米支护桩突然崩塌&#xff01;所幸无人伤亡。深基坑施工安全生产管理要点一、基坑开挖 1、 临边防护(1)基坑施工必须按要求进行&#xff0c;具体临边防…

[转]asp.net导出数据到Excel的三种方法

原文出处&#xff1a;asp.net导出数据到Excel的几种方法(1/3) 、asp.net导出数据到Excel的几种方法(2/3)、asp.net导出数据到Excel的几种方法(3/3) asp.net导出到Excel也是个老生常谈的问题&#xff0c;在此归纳一下。 第一种是比较常用的方法。是利用控件的RenderControl功能…

LintCode 378. 将二叉树转换成双链表(非递归遍历)

文章目录1. 题目2. 解题1. 题目 将一个二叉树按照中序遍历转换成双向链表。 样例 样例 1&#xff1a; 输入:4/ \2 5/ \1 3 输出: 1<->2<->3<->4<->5样例 2&#xff1a; 输入:3/ \4 1输出:4<->3<->1https://www.lintcode.com/pro…

js 将图片置灰_艾叶灰千万别扔——艾叶灰的神奇功效

请 点 上面“经络技巧”免费关注每晚9点准时免费更新点击下面图片阅读↓↓↓—— 以下是正文 ——艾灰的妙用1、宝宝经常会有红屁股&#xff0c;做妈妈的当然心疼&#xff0c;用了不少膏啊霜啊油啊&#xff0c;效果也是反反复复&#xff0c;尤其害怕会有依赖性。在妈妈的提醒下…

LintCode 434. 岛屿的个数II(并查集)

文章目录1. 题目2. 解题1. 题目 给定 n, m, 分别代表一个二维矩阵的行数和列数, 并给定一个大小为 k 的二元数组A. 初始二维矩阵全0. 二元数组A内的k个元素代表k次操作, 设第 i 个元素为 (A[i].x, A[i].y), 表示把二维矩阵中下标为A[i].x行A[i].y列的元素由海洋变为岛屿. 问在…

jqprintsetup已经安装还会提示_Windows 10更新将修复困扰用户已久的循环安装问题...

对于某些设备的用户来说&#xff0c;过去一年一直深受 Windows Update 陷入循环更新的问题困扰&#xff0c;尤其是那些使用英特尔驱动程序的设备。问题在于 Windows Update 会提示错误地提供不适配的驱动或版本&#xff0c;并且强行覆盖安装。此外即便用户已经安装了更新更好的…

springboot设置运行内存_Docker 如何运行多个 Springboot?

docker 如何运行多个Springboot &#xff1f;第一个&#xff1a;端口映射第二个&#xff1a;指定内存大小第三个&#xff1a;读取、写入物理文件第四个&#xff1a;日志文件第五个&#xff1a;多个容器内部网络访问第六个&#xff1a;遇到的问题第一个&#xff1a;端口映射Ngin…

LintCode 1915. 举重(01背包)

文章目录1. 题目2. 解题1. 题目 奥利第一次来到健身房&#xff0c;她正在计算她能举起的最大重量。 杠铃所能承受的最大重量为maxCapacity&#xff0c;健身房里有 n 个杠铃片&#xff0c;第 i 个杠铃片的重量为 weights[i]。 奥利现在需要选一些杠铃片加到杠铃上&#xff0c;使…

python实现简单线性回归和多元线性回归算法

1、问题引入 在统计学中&#xff0c;线性回归是利用称为线性回归方程的最小二乘函数对一个或多个自变量和因变量之间关系进行建模的一种回归分析。这种函数是一个或多个称为回归系数的模型参数的线性组合。一个带有一个自变量的线性回归方程代表一条直线。我们需要对线性回归结…

form表单通过checkbox_飞冰表单解决方案 - FormBinder

前言中后台业务场景中&#xff0c;表单是一种很常见的与用户交互的方式&#xff0c;从业务角度看&#xff0c;表单主要是收集用户的信息&#xff0c;而从技术角度看&#xff0c;作为一个通用型的组件&#xff0c;它要解决的问题无非就是三个&#xff1a;把一个初始数据对象扔给…

@data 重写set方法_C#中的类、方法和属性

这节讲C#中的类&#xff0c;方法&#xff0c;属性。这是面向对象编程中&#xff0c;我们最直接打交道的三个结构。类&#xff1a;类(class)是面向对象中最基本的单元&#xff0c;它是一种抽象&#xff0c;对现实世界中事物的抽象&#xff0c;在C#中使用class关键字声明一个类&a…

Docker安装+镜像拉取+容器+创建镜像+push to docker hub

文章目录1. 安装2. 镜像操作3. 容器4. docker hub本文参考&#xff1a;https://zhuanlan.zhihu.com/p/23599229 1. 安装 参考 https://www.runoob.com/docker/ubuntu-docker-install.html curl -fsSL https://get.docker.com | bash -s docker --mirror Aliyun测试&#xff…

css 百分比 怎么固定正方形_你未必知道的49个CSS知识点

本文的每一条&#xff0c;都是我曾经发过的掘金沸点&#xff0c;其中有很多条超过了百赞(窃喜)。鉴于时不时有童鞋翻我以前的沸点&#xff0c;因此&#xff0c;本文收集了个人目前发过的所有CSS知识点动图&#xff0c;以便阅读。需要说明的是&#xff0c;顺序仍是按当时发布顺序…

CSS 实现加载动画之五-光盘旋转

今天做的这个动画叫光盘旋转&#xff0c;名字自己取的。动画的效果估计很多人都很熟悉&#xff0c;就是微信朋友圈里的加载动画。做过前面几个动画&#xff0c;发现其实都一个原理&#xff0c;就是如何将动画的元素如何分离出来。这个动画的实现也很简单&#xff0c;关键点在于…

css hover变成手_web前端入门到实战:彻底掌握css动画「transition」

马上就2020年了&#xff0c;不知道小伙伴们今年学习了css3动画了吗&#xff1f;说起来css动画是一个很尬的事&#xff0c;一方面因为公司用css动画比较少&#xff0c;另一方面大部分开发者习惯了用JavaScript来做动画&#xff0c;所以就导致了许多程序员比较排斥来学习css动画(…

用Docker部署TensorFlow Serving服务

文章目录1. 安装 Docker2. 使用 Docker 部署3. 请求服务3.1 手写数字例子3.2 猫狗分类例子参考&#xff1a; https://tf.wiki/zh_hans/deployment/serving.html# https://tensorflow.google.cn/tfx/serving/docker 1. 安装 Docker 以下均为 centos7 环境 参考文章&#xff1a…

K-Means算法和K-Means++算法的聚类

在构成圆形的30000个随机样本点上&#xff0c;设置7个簇&#xff0c;使用K-Means算法聚类 from math import pi, sin, cos from collections import namedtuple from random import random, choice from copy import copy import matplotlib.pyplot as plt import numpy as np…

convert.todatetime指定日期格式_MATLAB的时间与日期

7 日期和时间MATLAB在2014b版本中将日期和时间独立成了一个数据类型&#xff0c;新的版本中对于时间数据的处理功能更为强大。比如datetime和duration等函数&#xff0c;可以支持对时间的高效计算、对比、格式化显示。对这类数组的操作方法和对普通数组的操作是基本一致的。下…