Qt/C++编写可视化大屏电子看板系统1-布局方案

2023-10-25 17:20

本文主要是介绍Qt/C++编写可视化大屏电子看板系统1-布局方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、前言

布局方案在整个数据可视化大屏界面电子看板系统中,是除了基础功能以外的核心功能之一,只有具备了布局方案这个功能,才能让用户随意调整自己想要的布局,保存成自定义名称的布局配置文件,这样就大大增加了灵活性,可以更好的适应各种分辨率,毕竟客户的电脑运行环境各种各样的都有,模块数量众多,有些不想展示,有些需要特别放大展示,有些需要偶尔全屏展示等,这些布局用户都可以自己定义好保存配置方案,存储到指定的目录下,下次启动会自动生成对应的布局菜单文件让用户自行选择切换。

二、电子看板介绍

电子看板是目视化管理的一种表现形式,即对数据的状况一目了然地表现,主要是对于管理项目,它通过利用形象直观而又色彩适宜的各种视觉感知信息来组织现场生产活动,目视管理依据人类的生理特征,在生产现场充分利用信号灯、标识牌、符号颜色等方式来发出视觉信号,鲜明准确地刺激人的神经末梢,快速地传递信息,形象直观地将潜在的问题和浪费现象都显现出来。以便任何人都可以及时掌握管理现状和必要的情报,从而能够快速制定并实施应对措施。因此,管理看板是发现问题、解决问题的非常有效且直观的手段,是优秀的现场管理必不可少的工具之一。

三、功能特点

  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)、基本配置参数

字段描述默认值
WorkMode工作模式 timer-模拟数据 db-数据库采集 tcp-网络采集 http-post请求timer
Title软件标题,显示在软件中间顶部数字化工厂信息中心
Ratio分辨率,目前无意义4096*216
Layout布局方案,每次切换布局方案以后都会保存完整布局
Theme配色方案,每次切换配色方案以后都会保存紫色风格
VideoAddr视频流地址,视频模块播放的视频地址凤凰卫视
AutoRun是否开机启动false
MoveEnable模块是否可以拖动,启用以后模块可以任意拖动true
CutLeftBottom底部布局左侧是否切掉true
CutRightBottom底部布局右侧是否切掉true
StaticLine是否绘制静态定位线,为假则绘制游标十字线true

(2)、颜色配置参数

字段描述默认值
ColorMainBg主背景颜色QColor(4, 7, 38)
ColorPanelBg面板背景颜色QColor(26, 29, 60)
ColorLine十字线定位线颜色QColor(255, 0, 0)
ColorLine1线条1颜色QColor(0, 176, 180)
ColorLine2线条2颜色QColor(32, 159, 223)
ColorLine3线条3颜色QColor(255, 192, 0)
ColorTitleBg标题栏背景颜色QColor(48, 48, 85)
ColorTitleText标题栏文字颜色QColor(255, 255, 255)
ColorChartBg曲线图表背景颜色QColor(38, 41, 74)
ColorChartText曲线图表文字颜色QColor(250, 250, 250)
ColorChartGrid曲线图表网格颜色QColor(180, 180, 180)
ColorOk正常颜色QColor(0, 176, 180)
ColorLow警戒颜色QColor(255, 192, 0)
ColorAlarm报警颜色QColor(214, 77, 84)
ColorDisable禁用背景颜色QColor(210, 210, 210)
ColorPercent环形百分比背景颜色QColor(0, 254, 254)

(3)、字体和尺寸配置参数

字段描述默认值
MainFont全局字号微软雅黑,12
NameFont软件名称字号19
LabFont加粗标签字号12
DeviceFont设备面板字号12
SubTitleFont模块子标题栏字号13
TitleFont模块标题栏字号15
TitleHeight模块标题栏高度23
HeadHeight表格表头高度28
RowHeight表格行高度25

(4)、采集速度配置参数

字段描述默认值
IntervalModule1模块1采集间隔5000
IntervalModule2模块2采集间隔5000
IntervalModule3模块3采集间隔5000
IntervalModule4模块4采集间隔5000
IntervalModule5模块5采集间隔5000
IntervalModule6模块6采集间隔5000
IntervalModule7模块7采集间隔5000
IntervalModule8模块8采集间隔5000

(5)、本地数据库配置参数

字段描述默认值
LocalDBType本地数据库类型,Sqlite、Mysql等Mysql
LocalDBIP本地数据库主机地址127.0.0.1
LocalDBPort本地数据库端口3306
LocalDBName本地数据库名称bigscreen
LocalUserName本地数据库用户名root
LocalUserPwd本地数据库密码,以密文存储root

五、特别说明

  1. 执行文件同级文件夹有layout+layout_1440+layout_1920,程序默认自动识别分辨率并加载对应的布局文件夹,比如1920分辨率则从layout_1920文件夹加载布局,并作为整体布局文件夹。
  2. 程序默认是模拟数据,如果需要从数据库采集则修改配置文件WorkMode=db即可。
  3. 如果发现布局拖动乱了,可以直接鼠标右键选择恢复布局即可,在保存布局以前。
  4. 在中间地图模块鼠标右键可以弹出菜单,切换布局和配色方案等。
  5. 在模块的标题栏上右键可以弹出默认的dock菜单,用来显示和隐藏各模块。
  6. 软件关闭过程中会自动保存布局,下次启动以后自动应用。
  7. 如果使用的默认的默认的配色方案比如紫色风格,则配置文件中的颜色全部无效,会自动应用代码中的颜色,如果需要启用自定义的颜色,则将配置文件的 Theme=\x81ea\x5b9a\x4e49\x98ce\x683c 即可。此时打开软件会应用配置文件中的颜色。
  8. 右键菜单可以截图保存,默认命名为 配色方案名称_布局方案名称.png 保存在snap目录下。
  9. 如果是XP系统请先执行fixff.cmd,用来修复ffmpeg在XP上不可用的BUG。
  10. 可执行文件下载地址:https://pan.baidu.com/s/1o97IGvZgTgDhlkuXQa4B0w 提取码:r2bv ,会不定期更新程序,欢迎各位提出批评和建议。

六、效果图

在这里插入图片描述

七、核心代码

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "quiwidget.h"
#include "appinit.h"
#include "customtitlebar.h"MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);this->initForm();this->changeLayout(App::Layout, true);
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::closeEvent(QCloseEvent *)
{closeAll();
}void MainWindow::initForm()
{this->setWindowTitle(App::Title);//this->setProperty("canMove", true);this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowSystemMenuHint | Qt::WindowMinMaxButtonsHint);//自动根据分辨率找合适的默认的布局文件夹layoutPath = QUIHelper::appPath() + "/layout";if (QUIHelper::deskWidth() == 1440) {QDir dir(QUIHelper::appPath() + "/layout_1440");if (dir.exists()) {layoutPath = QUIHelper::appPath() + "/layout_1440";}} else if (QUIHelper::deskWidth() == 1920) {QDir dir(QUIHelper::appPath() + "/layout_1920");if (dir.exists()) {layoutPath = QUIHelper::appPath() + "/layout_1920";}}module1 = new frmModule1;module2 = new frmModule2;module3 = new frmModule3;module4 = new frmModule4;module5 = new frmModule5;module6 = new frmModule6;module7 = new frmModule7;module8 = new frmModule8;moduleCenter = new frmModuleCenter;moduleVideo = new frmModuleVideo;//启动模拟数据或者数据采集module1->start(App::IntervalModule1);module2->start(App::IntervalModule2);module3->start(App::IntervalModule3);module4->start(App::IntervalModule4);module5->start(App::IntervalModule5);module6->start(App::IntervalModule6);module7->start(App::IntervalModule7);module8->start(App::IntervalModule8);//实例化停靠窗体newWidget(module1, "年度产量汇总");newWidget(module2, "当月计划达成率");newWidget(module3, "设备监控");newWidget(module4, "模具进度");newWidget(module5, "负荷分布");newWidget(module6, "送检一次合格率");newWidget(module7, "品质管理");newWidget(module8, "物料管理");newWidget(moduleVideo, "视频监控");QList<QWidget *> widgets;widgets << module1 << module2 << module3 << module4 << module5 << module6 << module7 << module8 << moduleCenter << moduleVideo;connect(moduleCenter, SIGNAL(changeLayout(QString)), this, SLOT(changeLayout(QString)));connect(moduleCenter, SIGNAL(saveLayout(QString, int)), this, SLOT(saveLayout(QString, int)));connect(moduleCenter, SIGNAL(changeTheme(QString)), this, SLOT(changeTheme(QString)));connect(moduleCenter, SIGNAL(closeAll()), this, SLOT(closeAll()));//设置拉伸策略for (int i = 0; i < widgets.count(); i++) {widgets.at(i)->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);}//设置中心窗体this->setCentralWidget(moduleCenter);//设置停靠参数,不允许重叠,只允许拖动this->setDockOptions(QMainWindow::AnimatedDocks);//将底部左侧作为左侧区域,底部右侧作为右侧区域,否则底部区域会填充拉伸if (App::CutLeftBottom) {setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);}if (App::CutRightBottom) {setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);}
}void MainWindow::clearWidget()
{for (int i = 0; i < widgets.count(); i++) {this->removeDockWidget(widgets.at(i));}
}void MainWindow::initWidget()
{//添加左侧窗体addWidget(widgets.at(0), 0);addWidget(widgets.at(7), 0);addWidget(widgets.at(3), 0);//添加右侧窗体addWidget(widgets.at(6), 1);addWidget(widgets.at(4), 1);addWidget(widgets.at(5), 1);//添加底部窗体addWidget(widgets.at(2), 3);addWidget(widgets.at(1), 3);addWidget(widgets.at(8), 3);
}void MainWindow::newWidget(QWidget *widget, const QString &title)
{//自定义停靠窗体标题栏QString objName = widget->objectName();CustomTitleBar *titleBar = new CustomTitleBar;titleBar->setObjectName("titleBar_" + objName);titleBar->setTitle(title);//实例化停靠窗体QDockWidget *dockWidget = new QDockWidget;dockWidget->setObjectName("dockWidget_" + objName);dockWidget->setWindowTitle(title);dockWidget->setTitleBarWidget(titleBar);dockWidget->setWidget(widget);//如果设置了不可移动则只允许关闭if (!App::MoveEnable) {dockWidget->setFeatures(QDockWidget::DockWidgetClosable);}//设置顶部不可停靠dockWidget->setAllowedAreas(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea | Qt::BottomDockWidgetArea);widgets << dockWidget;
}void MainWindow::addWidget(QDockWidget *widget, int position)
{//设置停靠位置Qt::DockWidgetArea area;if (position == 0) {area = Qt::LeftDockWidgetArea;} else if (position == 1) {area = Qt::RightDockWidgetArea;} else if (position == 2) {area = Qt::TopDockWidgetArea;} else if (position == 3) {area = Qt::BottomDockWidgetArea;}this->addDockWidget(area, widget);//如果是首次生成布局则需要全部可见QString file = QString("%1/%2.ini").arg(layoutPath).arg(App::Layout);if (!QFile(file).exists()) {widget->setVisible(true);}
}void MainWindow::changeLayout(const QString &layout, bool init)
{//首次加载不需要比较是否和配置文件一样bool needLoad = init;if (!init && App::Layout != layout) {needLoad = true;//先保存原有布局saveLayout(App::Layout, 2);}if (needLoad) {App::Layout = layout;App::writeConfig();this->clearWidget();this->initWidget();this->initLayout(App::Layout);//全屏+QWebEngineView控件一起会产生右键菜单无法弹出的BUG,需要上移一个像素QRect rect = qApp->desktop()->screenGeometry();
#if 1rect.setY(-1);rect.setHeight(rect.height());
#elserect.setX(30);rect.setY(40);rect.setWidth(1370);rect.setHeight(795);
#endifthis->setGeometry(rect);}
}void MainWindow::initLayout(const QString &layout)
{QString file = QString("%1/%2.ini").arg(layoutPath).arg(layout);QSettings set(file, QSettings::IniFormat);set.beginGroup("MainWindow");restoreState(set.value("State").toByteArray());set.endGroup();
}//type: 0-新建布局 1-恢复布局 2-保存布局 3-布局另存
void MainWindow::saveLayout(const QString &layout, int type)
{//如果为空则表示是恢复布局if (type == 0) {App::Layout = layout;this->changeLayout(App::Layout, true);return;} else if (type == 1) {this->changeLayout(App::Layout, true);return;}QString file = QString("%1/%2.ini").arg(layoutPath).arg(layout);QSettings set(file, QSettings::IniFormat);set.beginGroup("MainWindow");set.setValue("State", saveState());set.endGroup();App::Layout = layout;App::writeConfig();
}void MainWindow::changeTheme(const QString &theme)
{//必须是风格改变了才需要重新应用if (App::Theme != theme) {App::Theme = theme;App::writeConfig();AppInit::Instance()->initTheme();AppInit::Instance()->initStyle();}
}void MainWindow::closeAll()
{saveLayout(App::Layout, 2);QUIHelper::sleep(100);exit(0);
}

这篇关于Qt/C++编写可视化大屏电子看板系统1-布局方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

CSS弹性布局常用设置方式

《CSS弹性布局常用设置方式》文章总结了CSS布局与样式的常用属性和技巧,包括视口单位、弹性盒子布局、浮动元素、背景和边框样式、文本和阴影效果、溢出隐藏、定位以及背景渐变等,通过这些技巧,可以实现复杂... 一、单位元素vm 1vm 为视口的1%vh 视口高的1%vmin 参照长边vmax 参照长边re

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

c++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、