当前位置:   article > 正文

【开源、应用】QT—TCP网络上位机的设计_qt上位机

qt上位机

本文设计一个终端控制的上位机软件(如“设计目标”下图所示),可以和STM32、Adruino等通信实现无线局域网控制系统。


本文的通信内容和图表内容可以参考作者之前的文章

STM32+ESP8266连接电脑Qt网络上位机——QT篇https://blog.csdn.net/qq_53734051/article/details/126706759?spm=1001.2014.3001.5501

QT—Qcharts绘制实时曲线https://blog.csdn.net/qq_53734051/article/details/126872728?spm=1001.2014.3001.5501


目录

一、设计目标

二、设计原理

三、通信部分

四、图表部分

五、数据刷新和UI设计


一、设计目标

        利用组件容器等,结合通信,发送相应格式的文本,在主控解析后做出相应的回应;同样在数据上报也需要主控发送指定的格式文本,上位机才能正确的解析。下方评论+邮箱发送源码或Git自行下载https://gitee.com/guo-lingran/tcp-sql-network-host-computer

百度网盘下载连接:https://pan.baidu.com/s/1AROPIT3big1puhzCKyRTSQ  提取码:pumk

哔哩哔哩视频演示链接 —>【开源、应用】QT—TCP+Sql网络上位机的设计_哔哩哔哩_bilibili

 

二、设计原理

 该软件主要分为以下部分:

        通信部分QTcpServer、QTcpSocket

        图表部分QChartView、QLineSeries、QValueAxis、QDateTimeAxis

        数据刷新QTimer

        UI槽函数设计QToolBar、QLabel、QLineEdit、QPushButton、QSlider、QCheckBox等)

三、通信部分

         之前的文章已经写过TCP通信实现过程以及部分代码,所以这里不再详写 —> 点击跳转STM32+ESP8266连接电脑Qt网络上位机—QT篇https://blog.csdn.net/qq_53734051/article/details/126706759?spm=1001.2014.3001.5501

这里有点小的变化,在工具栏里可以再调出一个窗口,可供调试使用

调用该窗口

        头文件需要定义一下该类 :

  1. private:
  2. Deb *deb = new Deb;

调出窗口即可

  1. //调出调试窗口
  2. void MainWindow::on_debb_triggered()
  3. {
  4. deb->show();
  5. }

 关闭窗口

  1. void MainWindow::on_exit_triggered()
  2. {
  3. this->close();
  4. }

容器接收

        当软件接收数据后,一部分传入界面二(调试界面),供调试使用,另一部分进行解析,并进入图表或文本(下面再讲解)。如下代码->

  1. //收到的数据放入接受框 解析
  2. void MainWindow::readyRead_Slot(){
  3. QByteArray receiveDate;
  4. QTextCodec *tc = QTextCodec::codecForName("GBK");
  5. while(!tcpSocket->atEnd()){
  6. receiveDate = tcpSocket->readAll();
  7. }
  8. if (!receiveDate.isEmpty())
  9. {
  10. QString strBuf=tc->toUnicode(receiveDate);
  11. //传入界面二数据
  12. deb->DisplayData(strBuf);
  13. //解析到可视化曲线图和文本
  14. BackDataParsing(strBuf);
  15. }
  16. receiveDate.clear();
  17. }

界面二中的数据显示

        如下图

  1. void Deb::DisplayData(QString qstring){
  2. ui->rec_edi->appendPlainText(qstring);
  3. }

容器发送

        在界面二直接进行数据发送后,会引起阻塞,所以为了保证发送成功,这里在界面一使用定时器来判断是否有数据发送,这样两个界面不会有冲突。

  1. /*构造函数*/
  2. timer_send = new QTimer(this);
  3. timer_send->start(20);
  4. connect(timer_send,SIGNAL(timeout()),this,SLOT(timer_send_Slot()));
  5. /*END*/
  6. void MainWindow::timer_send_Slot(){
  7. if(flag_Send==1){
  8. flag_Send=0;
  9. tcpSocket->write(strbuf.toLocal8Bit().data());
  10. }
  11. }

四、图表部分

之前的文章也介绍过怎么绘制实时曲线,这里不再详写

  1. //创建chart
  2. void MainWindow::creatChart()
  3. {
  4. QChart *qchart = new QChart();
  5. //把chart放到容器里
  6. ui->graphicsView->setChart(qchart);
  7. ui->graphicsView->setRenderHint(QPainter::Antialiasing); //设置抗锯齿
  8. //创建两条线
  9. QLineSeries *series0 = new QLineSeries;
  10. QLineSeries *series1 = new QLineSeries;
  11. QLineSeries *series2 = new QLineSeries;
  12. //设置名字
  13. series0->setName("温度");
  14. series1->setName("湿度");
  15. series2->setName("光强");
  16. //把线条放到chart里
  17. qchart->addSeries(series0);
  18. qchart->addSeries(series1);
  19. qchart->addSeries(series2);
  20. //创建x 坐标
  21. QDateTimeAxis *axisX = new QDateTimeAxis;
  22. //格式
  23. axisX->setFormat("hh:mm:ss");
  24. //设置竖条数量
  25. axisX->setTickCount(5);
  26. //设置坐标名称
  27. axisX->setTitleText("time(sec)");
  28. qchart->setAxisX(axisX,series0);
  29. qchart->setAxisX(axisX,series1);
  30. qchart->setAxisX(axisX,series2);
  31. //创建y坐标
  32. QValueAxis *axisY = new QValueAxis;
  33. axisY->setRange(0,100);
  34. axisY->setTickCount(5);
  35. qchart->setAxisY(axisY,series0);
  36. qchart->setAxisY(axisY,series1);
  37. qchart->setAxisY(axisY,series2);
  38. qchart->setDropShadowEnabled(true);
  39. //初始化坐标
  40. //设置最大值坐标值 系统时间当前时间
  41. qchart->axisX()->setMin(QDateTime::currentDateTime().addSecs(0));
  42. //设置最大值坐标值 系统时间后5*30秒
  43. qchart->axisX()->setMax(QDateTime::currentDateTime().addSecs(5*30));
  44. }

QT—Qcharts绘制实时曲线https://blog.csdn.net/qq_53734051/article/details/126872728?spm=1001.2014.3001.5501        发送格式:Params{temp:39.3;humi:82.9;light:69.3;soil:38.3;mq2:22.2;rain:57.3;}

        当接收到数据后,进入BackDataParsing函数进行文本解析,在解析文本时使用mid函数取指定长度的字符串,.indexOf定位关键词,返回关键词的位置(int)。

解析原理:

.mid有两个参数,第一个为起始位置(int),第二个为取值长度。

起始位置:

        首先Params为起始,利用.indexof定位到第一个关键词的位置,并加上该关键词的长度,即取到:后面;

取值长度:

        再利用.indexof定位到第二个关键词的位置,并减去第一个关键词的位置和长度,以此算出该有效数据的长度。

  1. void MainWindow::BackDataParsing(QString strBuf){
  2. //查找是否为参数; -1表示没有该子串
  3. if(strBuf.startsWith("Params")){
  4. //表一数据
  5. QString str = strBuf.mid(strBuf.indexOf("temp:")+((QString)"temp:").length(),strBuf.indexOf("humi:")-strBuf.indexOf("temp:")-((QString)"temp:").length()-1);
  6. tcpSocket->write(str.toUtf8());
  7. tcpSocket->write("->");
  8. QString st2 = strBuf.mid(strBuf.indexOf("humi:")+((QString)"humi:").length(),strBuf.indexOf("light:")-strBuf.indexOf("humi:")-((QString)"humi:").length()-1);
  9. tcpSocket->write(st2.toUtf8());
  10. tcpSocket->write("->");
  11. QString st3 = strBuf.mid(strBuf.indexOf("light:")+((QString)"light:").length(),strBuf.indexOf("soil:")-strBuf.indexOf("light:")-((QString)"light:").length()-1);
  12. tcpSocket->write(st3.toUtf8());
  13. tcpSocket->write("->");
  14. //表二数据
  15. QString st4 = strBuf.mid(strBuf.indexOf("soil:")+((QString)"soil:").length(),strBuf.indexOf("mq2:")-strBuf.indexOf("soil:")-((QString)"soil:").length()-1);
  16. tcpSocket->write(st4.toUtf8());
  17. tcpSocket->write("->");
  18. QString st5 = strBuf.mid(strBuf.indexOf("mq2:")+((QString)"mq2:").length(),strBuf.indexOf("rain:")-strBuf.indexOf("mq2:")-((QString)"mq2:").length()-1);
  19. tcpSocket->write(st5.toUtf8());
  20. tcpSocket->write("->");
  21. QString st6 = strBuf.mid(strBuf.indexOf("rain:")+((QString)"rain:").length(),strBuf.indexOf("}")-strBuf.indexOf("rain:")-((QString)"rain:").length()-1);
  22. tcpSocket->write(st6.toUtf8());
  23. temp_data = str.toFloat();
  24. humi_data = st2.toFloat();
  25. light_data = st3.toFloat();
  26. soil_data = st4.toFloat();
  27. mq2_data = st5.toFloat();
  28. rain_data = st6.toFloat();
  29. ToUpdata_Lab(str,st2,st3,st4,st5,st6);
  30. }
  31. }

解析完后更显到表里和显示区域里 (ToUpdata_Lab)

文本区域显示

  1. //文本显示
  2. void MainWindow::ToUpdata_Lab(QString Stemp,QString Shumi,QString Slight,QString Ssoil,QString Smq2,QString Srain){
  3. ui->temp_la->setText(Stemp+"%");
  4. ui->humi_la->setText(Shumi+"%");
  5. ui->light_la->setText(Slight+"%");
  6. ui->soil_la->setText(Ssoil+"%");
  7. ui->mq2_la->setText(Smq2+"%");
  8. ui->rain_la->setText(Srain+"%");
  9. }

表格刷新

        动态变化详见上篇文章

  1. //表一刷新
  2. void MainWindow::DisplayChart1(){
  3. //获取当前时间
  4. QDateTime currentTime = QDateTime::currentDateTime();
  5. //获取初始化的qchart
  6. QChart *qchart =(QChart *)ui->graphicsView->chart();
  7. //获取初始化的series;
  8. QLineSeries *series0 = (QLineSeries *)ui->graphicsView->chart()->series().at(0);
  9. QLineSeries *series1 = (QLineSeries *)ui->graphicsView->chart()->series().at(1);
  10. QLineSeries *series2 = (QLineSeries *)ui->graphicsView->chart()->series().at(2);
  11. series0->append(currentTime.toMSecsSinceEpoch(),temp_data);
  12. series1->append(currentTime.toMSecsSinceEpoch(),humi_data);
  13. series2->append(currentTime.toMSecsSinceEpoch(),light_data);
  14. qchart->axisX()->setMin(QDateTime::currentDateTime().addSecs(-5*30));
  15. qchart->axisX()->setMax(QDateTime::currentDateTime().addSecs(5*30));
  16. }

 五、数据刷新和UI设计

数据刷新部分包括表格的动态变化、时间显示和继电器自动关停(蓄水ui的动态变化

  1. //时间刷新
  2. void MainWindow::ReData_Slot(){
  3. static int timer=0;
  4. DisplayChart1();
  5. DisplayChart2();
  6. //当前时间
  7. ui->time_l->setText(QTime::currentTime().toString("hh:mm:ss"));
  8. //继电器控制
  9. if(relaySw){
  10. timer++;
  11. // boVal 设定值 timer 当前值
  12. int boVal = ui->spinBox->value();
  13. ui->progressBar->setValue((timer*100)/boVal);
  14. //超时后自动关闭
  15. if(timer>=boVal){
  16. timer=0;
  17. relaySw=false;
  18. ui->relay->setIcon(QIcon(":/relay_off.png"));
  19. tcpSocket->write("relay_off");
  20. ui->progressBar->setValue(100);
  21. }
  22. }
  23. else
  24. {
  25. timer=0;
  26. }
  27. }

阈值设定

        采用QCheckBox多选框完成一个阈值的设定,需要设定哪一个阈值勾选即可,然后设定。

发送格式: 状态+soil:12;状态+rain:20;状态+temp:5;状态+light:6

若设定为以下的阈值,将会发送这样的数据:enable soil:12;enable rain:20;enable temp:5;enable light:6   (如下图)

 enable soil:30;disable rain:52;disable temp:6;enable light:62

 

 最后用于单片机的解析,并作出相应的变化。

槽函数:

  1. void MainWindow::on_set_yu_bt_clicked()
  2. {
  3. QString sendThrshold;
  4. sendThrshold = EnsoilHumi + " " + "soil:"+ ui->soil_yu_la->text()+";"+
  5. Enrain + " " + "rain:"+ ui->rain_yu_la->text()+";"+
  6. Entemp + " " + "temp:"+ ui->temp_yu_la->text()+";"+
  7. Enlight + " " + "light:"+ui->light_yu_la->text();
  8. tcpSocket->write(sendThrshold.toLocal8Bit());
  9. }
  1. //复选框 2选中 0未选中
  2. void MainWindow::on_checkBox_stateChanged(int arg1)
  3. {
  4. qDebug()<<arg1;
  5. if(arg1==2)
  6. EnsoilHumi="enable";
  7. else
  8. EnsoilHumi="disable";
  9. }
  10. void MainWindow::on_checkBox_2_stateChanged(int arg1)
  11. {
  12. qDebug()<<arg1;
  13. if(arg1==2)
  14. Enrain="enable";
  15. else
  16. Enrain="disable";
  17. }
  18. void MainWindow::on_checkBox_3_stateChanged(int arg1)
  19. {
  20. qDebug()<<arg1;
  21. if(arg1==2)
  22. Entemp="enable";
  23. else
  24. Entemp="disable";
  25. }
  26. void MainWindow::on_checkBox_4_stateChanged(int arg1)
  27. {
  28. if(arg1==2)
  29. Enlight="enable";
  30. else
  31. Enlight="disable";
  32. }

光强设定 

        和上面一样,只不过用到了一个QSlider的类,设定26%后,发送格式:Pwm:+值,同样的单片机解析后做出相对的改变。

  1. //滑动改变
  2. void MainWindow::on_horizontalSlider_valueChanged(int value)
  3. {
  4. ui->setlightNumla->setNum(value);
  5. light_pwm = value;
  6. }
  7. //光强控制
  8. void MainWindow::on_set_light_bt_clicked()
  9. {
  10. tcpSocket->write(("Pwm:"+QString::number(light_pwm)).toLocal8Bit());
  11. }

 QToolBar工具栏

        工具栏包括网络的开启、LED开关、继电器开关、模式切换、调试窗口和退出软件。原理都一样,代码部分不再列出。

 蓄水状态

        当打开继电器开关后,会有一个倒计时自动关闭的设计,这里的时间可以设定 (如下图)。达到100%后自动关闭,并发送命令:relay_off 。

        这里使用了QProgressBar和QSpinBox的类,在定时器里1s发生改变

               

  1. //继电器控制
  2. if(relaySw){
  3. timer++;
  4. // boVal 设定值 timer 当前值
  5. int boVal = ui->spinBox->value();
  6. //进度条设定
  7. ui->progressBar->setValue((timer*100)/boVal);
  8. //超时后自动关闭
  9. if(timer>=boVal){
  10. timer=0;
  11. relaySw=false;
  12. ui->relay->setIcon(QIcon(":/relay_off.png"));
  13. tcpSocket->write("relay_off");
  14. ui->progressBar->setValue(100);
  15. }
  16. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/31811
推荐阅读
相关标签
  

闽ICP备14008679号