一、前言

设备监控主要用来实时监测制造零件等使用的设备的工作运行状态,每个设备都有对应的需要、分组名称、分组编号、设备名称、文字1、文字2、工作状态(1-开机 2-待机 3-维护 4-空),不同的工作状态不同颜色显示,比如维护状态红色显示,待机状态黄色显示。右侧显示对应设备分组的稼动率、开机数量、待机数量、维护数量。

Qt源自c++,类的思想非常强大,比如这个设备监控模块,每个设备需要显示设备名称、多种文字等,还有不同状态不用背景颜色,这就需要单独写个设备类,记录存储这些值,并根据设定的值做出反应,最后有多少个设备就new多少个这个类,放入面板中。

子模块表名对应表:

子模块标题 子模块表名
设备运行状态 t_3_1_device_runtime
稼动率 t_3_2_oee

1 设备运行状态

表名:t_3_1_device_runtime

字段名 中文名 类型 长度 说明
internal_id 序号 INTEGER 11 主键自增
group_name 分组名称 VARCHAR 4 不为空
no_id 分组编号 INTEGER 11 不为空
name 名称 VARCHAR 255 不为空
text_1 文字1 VARCHAR 255
text_2 文字2 VARCHAR 255
status 状态 1-开机 2-待机 3-维护 4-空 INTEGER 1 不为空

默认数据:

internal_id group_name no_id name text_1 text_2 status
1 CNC 1 CNC1 190411 PID11 1
2 CNC 2 CNC2 190412 PID12 1
3 CNC 3 CNC3 190413 PID13 1
4 CNC 4 CNC4 190414 PID14 2
5 CNC 5 CNC5 190415 PID15 1
6 CNC 6 CNC6 190416 PID16 2
7 CNC 7 CNC7 190417 PID17 1
8 CNC 8 4
9 EDM 1 EDM1 190421 PID21 1
10 EDM 2 EDM2 190422 PID23 2
11 EDM 3 EDM3 190423 PID23 1
12 EDM 4 EDM4 190424 PID24 3
13 EDM 5 EDM5 190425 PID25 1
14 EDM 6 EDM6 190426 PID26 1
15 EDM 7 EDM7 190427 PID27 1
16 EDM 8 EDM8 190428 PID28 2
17 WEDM 1 WEDM1 190431 PID28 1
18 WEDM 2 WEDM2 190432 PID28 3
19 WEDM 3 WEDM3 190434 PID29 2
20 WEDM 4 WEDM4 190435 PID30 1
21 WEDM 5 WEDM5 190436 PID36 1
22 WEDM 6 WEDM6 190437 PID37 3
23 WEDM 7 4
24 WEDM 8 4

2 稼动率

表名:t_3_2_oee

字段名 中文名 类型 长度 说明
internal_id 序号 INTEGER 11 主键自增
cnc 类别-cnc INTEGER 3 不为空
edm 类别-edm INTEGER 3 不为空
wedm 类别-wedm INTEGER 3 不为空

默认数据:1 110 90 90

二、功能特点

  1. 采用分层设计,整体总共分三级界面,一级界面是整体布局,二级界面是单个功能模块,三级界面是单个控件。
  2. 子控件包括饼图、圆环图、曲线图、柱状图、柱状分组图、横向柱状图、横向柱状分组图、合格率控件、百分比控件、进度控件、设备状态面板、表格数据、地图控件、视频控件等。
  3. 二级界面可以自由拖动悬浮,支持最小化隐藏、最大化关闭、响应双击自定义标题栏。
  4. 数据源支持模拟数据(默认)、数据库采集、串口通信(需定制)、网络通信(需定制)、网络请求等,可自由设定每个子界面的采集间隔即数据刷新频率。
  5. 采用纯QWidget编写,亲测Qt4.6到Qt6.2任意版本,理论上支持后续其他Qt版本。
  6. 超强跨平台,亲测windows、linux、mac、国产uos、国产银河麒麟kylin等系统,效果完美,同时还支持嵌入式linux比如树莓派、香橙派、全志、imx6等。
  7. 同时集成了自定义控件、qchart饼图、echart地图等功能。
  8. 内置多套配色风格样式(紫色、蓝色、深蓝、黑色),默认紫色,自适应任意分辨率。
  9. 可设置系统标题、目标分辨率、布局方案,启动立即应用。
  10. 可设置主背景颜色、面板颜色、十字线游标颜色等各种颜色。
  11. 可设置多条曲线不同颜色,没有设置颜色的情况下内置多套精美颜色随机应用。
  12. 可设置标题栏背景颜色、文字颜色。
  13. 可设置曲线图表背景颜色、文字颜色、网格颜色。
  14. 可设置正常颜色、警戒颜色、报警颜色、禁用颜色、百分比进度颜色。
  15. 可分别设置各种字体大小,比如全局字体、软件名称、标题栏、子标题栏、加粗标签等。
  16. 可设置标题栏高度、表头高度、行高度。
  17. 曲线支持游标、定位线、悬停高亮数据点、悬停显示值。
  18. 柱状图支持顶部(可设置顶端、上部、中间、底部)显示数据,全部自适应计算位置。
  19. 支持平滑曲线,内置多种平滑曲线算法,还支持面积图平滑。
  20. 面积图填充颜色可选多种规则比如单色透明度填充、透明度渐变填充等。
  21. 数据库支持sqlite、mysql、postgresql、oracle、国产人大金仓等数据库。
  22. 主界面直接鼠标右键切换布局、配色方案、关闭开启某个二级窗体。
  23. 自动记忆所有子窗口的大小和位置,下次启动立即应用。
  24. 动态加载布局方案菜单,可以动态新建布局、恢复布局、保存布局、另存布局等,用户可以制造任意布局。
  25. 二级窗体,双击从主窗体分离出来浮动,可以自由调整大小。再次双击标题栏最大化,再次双击还原。
  26. 子模块也可以全屏显示作为一个大屏,这样就可以一个大屏拓展出多个子大屏,放大查看子模块的数据详情,适用多屏展示。
  27. 每个模块都可以自定义采集速度,如果是数据库采集会自动排队处理,后期还可以拓展每个子模块都独立的数据库采集。
  28. 提供系统设置模块进行整体的配置参数设置,效果立即应用。
  29. 提供精美炫酷的大屏地图模块,包括静态图片、闪烁效果、迁徙效果、世界地图、区域地图等,可指定点的经纬度坐标,识别单击响应,可以做地图跳转等,每个点都可以不同的颜色和提示信息。
  30. 除了提供大屏系统外,还将每个模块都做了独立的模块示例界面,每个模块都可以独立学习使用,里面用到的控件也单独做了控件示例界面,方便学习每个控件如何使用。
  31. 非常详细的开发和使用手册,其中包括数据库说明、模块对照图、控件对照图、项目结构、代码说明(精确到每个类)、演示demo、使用方法等。

三、体验地址

  1. 体验地址:https://pan.baidu.com/s/1d7TH_GEYl5nOecuNlWJJ7g 提取码:01jf 文件名:bin_bigscreen.zip。
  2. 国内站点:https://gitee.com/feiyangqingyun
  3. 国际站点:https://github.com/feiyangqingyun
  4. 个人主页:https://blog.csdn.net/feiyangqingyun
  5. 知乎主页:https://www.zhihu.com/people/feiyangqingyun/
  6. 在线文档:https://feiyangqingyun.gitee.io/qwidgetdemo/bigscreen/

四、效果图

五、核心代码

#include "frmmodule3.h"
#include "ui_frmmodule3.h"
#include "quihelper.h"
#include "appinit.h" frmModule3::frmModule3(QWidget *parent) : QWidget(parent), ui(new Ui::frmModule3)
{
ui->setupUi(this);
this->initForm();
this->initDevice();
this->loadDevice();
} frmModule3::~frmModule3()
{
delete ui;
} void frmModule3::initForm()
{
//设置对应的属性应用特定样式
ui->labTitle1->setProperty("flag", "title");
ui->labTitle2->setProperty("flag", "title");
ui->labTitle3->setProperty("flag", "title");
ui->labTitle4->setProperty("flag", "title");
ui->labTitle5->setProperty("flag", "title");
ui->labTitle6->setProperty("flag", "title"); ui->widgetSub1->setProperty("flag", "sub");
ui->widgetSub2->setProperty("flag", "sub");
ui->widgetSub3->setProperty("flag", "sub");
ui->widgetSub4->setProperty("flag", "sub"); ui->widgetResult1->setProperty("flag", "lab");
ui->widgetResult2->setProperty("flag", "lab");
ui->widgetResult3->setProperty("flag", "lab"); //定时器模拟数据
timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), this, SLOT(loadDevice())); //绑定信号槽处理接收的数据+发送执查询语句
connect(DbData::DbLocal, SIGNAL(receiveData(QString, QStringList, int)),
this, SLOT(receiveData(QString, QStringList, int)));
connect(DbData::DbHttp, SIGNAL(receiveData(QString, QStringList, int)),
this, SLOT(receiveData(QString, QStringList, int))); //绑定样式改变信号重新设置颜色
connect(AppEvent::Instance(), SIGNAL(changeStyle()), this, SLOT(changeStyle()));
} void frmModule3::initStatus()
{
foreach (frmDevice *frm, device1) {
frm->setStatus(frm->getStatus());
} foreach (frmDevice *frm, device2) {
frm->setStatus(frm->getStatus());
} foreach (frmDevice *frm, device3) {
frm->setStatus(frm->getStatus());
}
} void frmModule3::changeStyle()
{
//延时处理
QTimer::singleShot(100, this, SLOT(initStatus()));
} void frmModule3::receiveData(const QString &tag, const QStringList &data, int mesc)
{
int count = data.count();
if (tag == "t_3_1_device_runtime") {
if (count == 24 * 6) {
QList<int> status1, status2, status3;
QStringList taskID1, personID1, deviceID1;
QStringList taskID2, personID2, deviceID2;
QStringList taskID3, personID3, deviceID3; int max1 = 8 * 6, max2 = 16 * 6, max3 = 24 * 6;
for (int i = 0; i < max1; i = i + 6) {
deviceID1 << data.at(i + 2);
taskID1 << data.at(i + 3);
personID1 << data.at(i + 4);
status1 << data.at(i + 5).toInt();
} for (int i = max1; i < max2; i = i + 6) {
deviceID2 << data.at(i + 2);
taskID2 << data.at(i + 3);
personID2 << data.at(i + 4);
status2 << data.at(i + 5).toInt();
} for (int i = max2; i < max3; i = i + 6) {
deviceID3 << data.at(i + 2);
taskID3 << data.at(i + 3);
personID3 << data.at(i + 4);
status3 << data.at(i + 5).toInt();
} loadDevice1(taskID1, personID1, deviceID1, status1);
loadDevice2(taskID2, personID2, deviceID2, status2);
loadDevice3(taskID3, personID3, deviceID3, status3);
}
} else if (tag == "t_3_2_oee") {
if (count == 3) {
int value1 = data.at(0).toDouble();
int value2 = data.at(1).toDouble();
int value3 = data.at(2).toDouble();
loadResult(value1, value2, value3);
}
}
} void frmModule3::loadResult(int value1, int value2, int value3)
{
ui->labcnc1->setText(QString("稼 动 率: %1%").arg(value1));
ui->labedm1->setText(QString("稼 动 率: %1%").arg(value2));
ui->labwedm1->setText(QString("稼 动 率: %1%").arg(value3));
} void frmModule3::initDevice()
{
device1 << ui->devicecnc1 << ui->devicecnc2 << ui->devicecnc3 << ui->devicecnc4;
device1 << ui->devicecnc5 << ui->devicecnc6 << ui->devicecnc7 << ui->devicecnc8; device2 << ui->deviceedm1 << ui->deviceedm2 << ui->deviceedm3 << ui->deviceedm4;
device2 << ui->deviceedm5 << ui->deviceedm6 << ui->deviceedm7 << ui->deviceedm8; device3 << ui->devicewedm1 << ui->devicewedm2 << ui->devicewedm3 << ui->devicewedm4;
device3 << ui->devicewedm5 << ui->devicewedm6 << ui->devicewedm7 << ui->devicewedm8; foreach (frmDevice *frm, device1) {
frm->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Expanding);
} foreach (frmDevice *frm, device2) {
frm->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Expanding);
} foreach (frmDevice *frm, device3) {
frm->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Expanding);
}
} void frmModule3::loadDevice()
{
QString tableName = "t_3_1_device_runtime";
QString columnName = "group_name,no_id,name,text_1,text_2,status";
QString tableName2 = "t_3_2_oee";
QString columnName2 = "cnc,edm,wedm"; if (AppConfig::WorkMode == "timer") {
QList<int> status1, status2, status3;
QStringList taskID1, personID1, deviceID1;
QStringList taskID2, personID2, deviceID2;
QStringList taskID3, personID3, deviceID3; for (int i = 0; i < 8; i++) {
taskID1 << QString("TID1%1").arg(i + 1);
personID1 << QString("PID1%1").arg(i + 1);
deviceID1 << QString("CNC%1").arg(i + 1);
status1 << QUIHelper::getRandValue(1, 5); taskID2 << QString("TID2%1").arg(i + 1);
personID2 << QString("PID2%1").arg(i + 1);
deviceID2 << QString("EDM%1").arg(i + 1);
status2 << QUIHelper::getRandValue(1, 5); taskID3 << QString("TID3%1").arg(i + 1);
personID3 << QString("PID3%1").arg(i + 1);
deviceID3 << QString("WEDM%1").arg(i + 1);
status3 << QUIHelper::getRandValue(1, 5);
} loadDevice1(taskID1, personID1, deviceID1, status1);
loadDevice2(taskID2, personID2, deviceID2, status2);
loadDevice3(taskID3, personID3, deviceID3, status3);
loadResult(85, 90, 95);
} else if (AppConfig::WorkMode == "db") {
DbData::DbLocal->select(tableName, columnName, true);
DbData::DbLocal->select(tableName2, columnName2, true);
} else if (AppConfig::WorkMode == "http") {
DbData::DbHttp->select(tableName, columnName, true);
DbData::DbHttp->select(tableName2, columnName2, true);
}
} void frmModule3::loadDevice1(const QStringList &taskID, const QStringList &personID,
const QStringList &deviceID, const QList<int> &status)
{
if (taskID.count() != 8 || personID.count() != 8 || deviceID.count() != 8 || status.count() != 8) {
return;
} //自动统计开机待机维护数量
int count1 = 0, count2 = 0, count3 = 0;
for (int i = 0; i < 8; i++) {
device1.at(i)->setTaskID(taskID.at(i));
device1.at(i)->setPersonID(personID.at(i));
device1.at(i)->setDeviceID(deviceID.at(i));
device1.at(i)->setStatus(status.at(i)); switch (status.at(i)) {
case 1:
count1++;
break;
case 2:
count2++;
break;
case 3:
count3++;
break;
}
} ui->labcnc2->setText(QString("开机数量: %1").arg(count1));
ui->labcnc3->setText(QString("待机数量: %1").arg(count2));
ui->labcnc4->setText(QString("维护数量: %1").arg(count3));
} void frmModule3::loadDevice2(const QStringList &taskID, const QStringList &personID,
const QStringList &deviceID, const QList<int> &status)
{
if (taskID.count() != 8 || personID.count() != 8 || deviceID.count() != 8 || status.count() != 8) {
return;
} //自动统计开机待机维护数量
int count1 = 0, count2 = 0, count3 = 0;
for (int i = 0; i < 8; i++) {
device2.at(i)->setTaskID(taskID.at(i));
device2.at(i)->setPersonID(personID.at(i));
device2.at(i)->setDeviceID(deviceID.at(i));
device2.at(i)->setStatus(status.at(i)); switch (status.at(i)) {
case 1:
count1++;
break;
case 2:
count2++;
break;
case 3:
count3++;
break;
}
} ui->labedm2->setText(QString("开机数量: %1").arg(count1));
ui->labedm3->setText(QString("待机数量: %1").arg(count2));
ui->labedm4->setText(QString("维护数量: %1").arg(count3));
} void frmModule3::loadDevice3(const QStringList &taskID, const QStringList &personID,
const QStringList &deviceID, const QList<int> &status)
{
if (taskID.count() != 8 || personID.count() != 8 || deviceID.count() != 8 || status.count() != 8) {
return;
} //自动统计开机待机维护数量
int count1 = 0, count2 = 0, count3 = 0;
for (int i = 0; i < 8; i++) {
device3.at(i)->setTaskID(taskID.at(i));
device3.at(i)->setPersonID(personID.at(i));
device3.at(i)->setDeviceID(deviceID.at(i));
device3.at(i)->setStatus(status.at(i)); switch (status.at(i)) {
case 1:
count1++;
break;
case 2:
count2++;
break;
case 3:
count3++;
break;
}
} ui->labwedm2->setText(QString("开机数量: %1").arg(count1));
ui->labwedm3->setText(QString("待机数量: %1").arg(count2));
ui->labwedm4->setText(QString("维护数量: %1").arg(count3));
} void frmModule3::start(int interval)
{
this->loadDevice(); //如果间隔太短表示不需要刷新,执行一次即可
if (interval > 2000) {
timer->start(interval);
}
} void frmModule3::stop()
{
if (timer->isActive()) {
timer->stop();
}
}

Qt编写可视化大屏电子看板系统25-模块3设备监控的更多相关文章

  1. Qt编写数据可视化大屏界面电子看板13-基础版

    一.前言 之前发布的Qt编写的可视化大屏电子看板系统,很多开发者比较感兴趣,也收到了很多反馈意见,纵观市面上的大屏系统,基本上都是B/S结构的web版本,需要在后台进行自定义配置模块,绑定数据源等,其 ...

  2. Qt编写项目作品大全(自定义控件+输入法+大屏电子看板+视频监控+楼宇对讲+气体安全等)

    一.自定义控件大全 (一).控件介绍 超过160个精美控件,涵盖了各种仪表盘.进度条.进度球.指南针.曲线图.标尺.温度计.导航条.导航栏,flatui.高亮按钮.滑动选择器.农历等.远超qwt集成的 ...

  3. Qt编写数据可视化大屏界面电子看板2-配色方案

    一.前言 做完整个数据可视化大屏界面电子看板系统后,为了提升点逼格,需要提供好几套默认的风格样式以供选择,这样用户可以选择自己喜欢的配色方案来作为整个系统的颜色方案,去看了下市面上大部分的大屏电子看板 ...

  4. Qt编写安防视频监控系统9-自动隐藏光标

    一.前言 这个效果的灵感来自于大屏电子看板系统,在很多系统中尤其是上了大屏的时候,其实在用户不在操作的时候,是很不希望看到那个鼠标箭头指针的,只有当用户操作的时候才显示出来,这个就需要开个定时器定时计 ...

  5. Qt编写数据可视化大屏界面电子看板系统

    一.前言 目前大屏大数据可视化UI这块非常火,趁热也用Qt来实现一个,Qt这个一站式超大型GUI超市,没有什么他做不了的,大屏电子看板当然也不在话下,有了QSS和QPainter这两个无敌的工具组合, ...

  6. Qt编写数据可视化大屏界面电子看板12-数据库采集

    一.前言 数据采集是整个数据可视化大屏界面电子看板系统核心功能,没有数据源,这仅仅是个玩具UI,没啥用,当然默认做了定时器模拟数据,产生随机数据,这个可以直接配置文件修改来选择采用何种数据采集方法,总 ...

  7. Qt编写数据可视化大屏界面电子看板11-自定义控件

    一.前言 说到自定义控件,我是感觉特别熟悉的几个字,本人亲自原创的自定义控件超过110个,都是来自各个行业的具体应用真实需求,而不是凭空捏造的,当然有几个小控件也有点凑数的嫌疑,在编写整个数据可视化大 ...

  8. Qt编写数据可视化大屏界面电子看板8-调整间距

    一.前言 在数据可视化大屏界面电子看板系统中,前期为了使用目标客户机,调整间距是必不可少的工作,QMainWindow中的QDockWidget,会默认生成布局和QSplitter调整宽高大小,鼠标移 ...

  9. Qt编写数据可视化大屏界面电子看板4-布局另存

    一.前言 布局另存是数据可视化大屏界面电子看板系统中的额外功能之一,主要用于有时候用户需要在现有布局上做个微调,然后直接将该布局另存为一个布局配置文件使用,可以省略重新新建布局重新来一次大的调整的工作 ...

  10. Qt编写数据可视化大屏界面电子看板3-新建布局

    一.前言 能够新建布局,也是数据可视化大屏界面电子看板系统中的必备功能之一,新建布局这样的功能一般做到右键菜单中,单击新建布局菜单,弹出输入框要求输入新的布局的名称,为了更符合国情,直接支持中文名称, ...

随机推荐

  1. 9. JS的数据类型,区别

    js 有2大数据类型分类 : 基本数据类型: 1. string 字符串 使用单.双引号包裹,或者使用反引号包裹 2. number 数字类型 3. boolean 布尔值 true false 4. ...

  2. kotlin更多语言结构——>类型安全的构建器

    通过使用命名得当的函数作为构建器,结合带有接收者的函数字面值,可以在 Kotlin 中创建类型安全.静态类型 的构建器 类型安全的构建器可以创建基于 Kotlin 的适用于采用半声明方式构建复杂层次数 ...

  3. KubeSphere 开源社区 2022 年度回顾与致谢

    2022 年,国内的云原生技术生态日趋完善,细分技术项目也不断涌现,形成了完整的支撑应用云原生化的全生命周期技术体系.基础设施即代码.微服务.Serverless 等技术,促使基础设施资源向更加灵活弹 ...

  4. HTML标签 b 和 strong 的区别

    <b>标签和<strong>标签都表示加粗,效果通常是一样的: <b>:义演丁真<strong>:义演丁真 类似的还有<em>和<i& ...

  5. My SQL 列转行操作

    原表结构如下,我们可以发现,"日运输量"和"车次"是在同一张表中相互独立的两个字段,即独立的两列数据,下面,我将系统中的测试数据以及代码全部放出来,以解释列转行 ...

  6. ESP8266 + MQTT + 土壤湿度传感器

    ESP8266 + MQTT + 土壤湿度传感器 连线 #include <Arduino.h> #include <Ticker.h> #include <ESP826 ...

  7. Redis数据结构:List类型全面解析

    文章目录 一.List数据类型 1.1 简介 1.2 应用场景 1.3 底层结构 二.数据结构 2.1 压缩列表ZipList 2.2 双向链表LinkedList(后续已废弃) 2.3 快速链表Qu ...

  8. debian大便系统配置国内软件源

    本例在debian:buster-slim docker镜像中实验通过 1.启动docker实例 docker run -it --name debian debian:buster-slim bas ...

  9. 网站免费https加密教程

    为网站实现HTTPS加密可以大大提高网站的安全性和用户信任度.以下是一个详细的免费HTTPS加密教程: 一.选择免费SSL证书提供商 JoySSL:这是目前国内为数不多的国产CA服务商打造的自主品牌S ...

  10. 2022 GDOI普及组游记

    2022 GDOI普及组游记 注:传送门均为校内网址 day -4 被年级主任集中开会,给我们免了亿堆作业,灌了亿壶鸡汤,宣布了为期一(亿)周的集训开始. day -3 中午一直在复习期中(4.21- ...