一、前言

设备监控主要用来实时监测制造零件等使用的设备的工作运行状态,每个设备都有对应的需要、分组名称、分组编号、设备名称、文字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. 谈一谈你对vue指令的理解

    vue指令的本质是给 html 标签新增一些属性 : vue的指令可以分为 3 中类型 : 1. 用于渲染数据的指令,比如 v-for ,v-if ,v-show : 2. 用来交互的指令 ,v-on ...

  2. .NET高级调试 - 3.7对象检查

    简介 在大多数调试会话中,首先需要检查的项目就是分析应用程序的状态.在确认程序的问题是某种无效状态造成的,我们便需要分析程序是如何变成无效状态的.那么在分析过程中,需要为我们深入了解对象的各种审查方法 ...

  3. day07-数据类型及标识符

    数据类型 强类型语言 要求变量的使用严格要严格符合规定,所有变量都必须先定义后才能使用 弱类型语言 java的数据类型分为两大类 基本类型(primitive type) 数值类型 整数类型 byte ...

  4. oracle中排序分析函数row_number()、rank()、dense_rank() 的区别,与rownum的注意事项

    row_number()产生的序号不会重复,即1.2.3... rank()产生的序号会重复,但是会跳号,出现1.2.2.4...的情况 dense_rank()产生的序号会重复,不会跳号,会出现1. ...

  5. 别人可以访问本项目的ip地址

    .markdown-body { line-height: 1.75; font-weight: 400; font-size: 16px; overflow-x: hidden; color: rg ...

  6. onethink自带编辑器内容无法修改

    楼主小白一个,之前有大神带着进入onethink框架 在一个编辑页里面放两个编辑框的时候,悲催了--- 一个用作文本编辑,一个用于多图上传 发现前面的文本编辑的内容无法实现,后来楼主想调试一下 之前提 ...

  7. Hugging Face 与 TruffleHog 合作,实现风险预警

    我们非常高兴地宣布与 Truffle Security 建立合作伙伴关系并在我们的平台集成 TruffleHog 强大的风险信息扫描功能.这些特性是 我们持续致力于提升安全性 的重要举措之一. Tru ...

  8. LLM应用实战: AI资讯的自动聚合及报告生成

    1.背景 花了整整两天时间,本qiang~开发了一个关于AI新闻资讯的自动聚合及报告生成工具. 本篇记录一下整体的框架和实现原理,并且本着它山之石可以攻玉,本qiang~开放了所有的源码,源码可见如下 ...

  9. 拯救php性能的神器webman-打包二进制

    看了看webman的官方文档,发现居然还能打包为二进制,这样太厉害了吧! 先执行这个  composer require webman/console ^1.2.24 安装这个console的包,然后 ...

  10. 黑盒视角下的RESTful API安全测试

    目录 前言 关于OWASP API TOP 10 REST API接口测试思路 接口权限测试 接口校验测试 接口滥用测试 总结 前言 RESTful API(或称RESTful Web API)在线开 ...