Files
EJM_Display/Pages/P401_CuttingPage.cpp

257 lines
12 KiB
C++
Raw Permalink Normal View History

2025-08-20 23:06:28 +08:00
#include "P401_CuttingPage.h"
#include "ui_P401_CuttingPage.h"
2025-09-30 17:42:22 +08:00
#include <GlobalDefinitions/Variable.h>
#include <FileOperation/ConfigFiles.h>
2025-08-20 23:06:28 +08:00
P401_CuttingPage::P401_CuttingPage(QWidget *parent) :
QWidget(parent),
ui(new Ui::P401_CuttingPage)
{
ui->setupUi(this);
2025-09-30 17:42:22 +08:00
QTimer::singleShot(1000, this, &P401_CuttingPage::WinInit);
// QTimer::singleShot(2000, this, &P401_CuttingPage::Show3D);
qDebug()<<"P401_CuttingPage 初始化完成!";
2025-08-20 23:06:28 +08:00
}
P401_CuttingPage::~P401_CuttingPage()
{
delete ui;
}
2025-09-30 17:42:22 +08:00
void P401_CuttingPage::WinInit(){
2025-10-10 17:44:10 +08:00
for (int row = 1; row < CutConditionConfigFile.size(); ++row)
2025-09-30 17:42:22 +08:00
{
QString TextObjName = QString("Text_SSConditions_%1").arg(row);
QLabel* Label = findChild<QLabel*>(TextObjName);
if (Label) {
2025-10-10 17:44:10 +08:00
QString Text = CutConditionConfigFile.at(row)[1];
2025-09-30 17:42:22 +08:00
Text.replace("\\n","\n",Qt::CaseInsensitive);
Label->setText(Text);
}
2025-10-10 17:44:10 +08:00
TrueColor.append(CutConditionConfigFile.at(row)[2]);
FalseColor.append(CutConditionConfigFile.at(row)[3]);
2025-09-30 17:42:22 +08:00
}
// 现在 ui->CPW_1 已经是一个 CurvePlotWidget*
ui->CPW_1->addSeries("电机电流U", Qt::green);
ui->CPW_1->addSeries("电机电流V", Qt::blue);
ui->CPW_1->addSeries("电机电流W", Qt::darkCyan);
// 现在 ui->CPW_1 已经是一个 CurvePlotWidget*
ui->CPW_2->addSeries("电机温度", Qt::green);
//所用定时器初始化
QTimer* UIRefresh_Timer = new QTimer(this);
connect(UIRefresh_Timer, &QTimer::timeout, this, &P401_CuttingPage::UIRefreshTimeOut);
UIRefresh_Timer->setInterval(100); // 设置定时器间隔为 1000 毫秒1 秒)
UIRefresh_Timer->start();
//所用定时器初始化
QTimer* TrailRefresh_Timer = new QTimer(this);
connect(TrailRefresh_Timer, &QTimer::timeout, this, &P401_CuttingPage::TrailRefreshTimeOut);
TrailRefresh_Timer->setInterval(100); // 设置定时器间隔为 1000 毫秒1 秒)
TrailRefresh_Timer->start();
}
void P401_CuttingPage::Show3D()
{
2025-10-10 17:44:10 +08:00
DevName3D = "截割电机";
2025-09-30 17:42:22 +08:00
mObjLoader = new ObjLoader();
mObjLoader->setEnDebug(true);
mObjLoader->init3DScene(ui->Frame_Dev3D);
mObjLoader->setCameraSpeed(200.0, 200.0);
// 准备模型路径
QString filePath = "./3D模型文件/掘锚一体机/电机/" + DevName3D + ".obj";
m_currentDevName = DevName3D;
m_currentFilePath = filePath;
2025-10-10 17:44:10 +08:00
m_currentPosition = QVector3D(1052.00, 0.00273132, 0);
2025-09-30 17:42:22 +08:00
// 检查文件是否存在
if (!QFile::exists(filePath)) {
qCritical() << "模型文件不存在: " << filePath;
return;
}
// 创建工作线程和工作对象
m_workerThread = new QThread(this);
m_workerLoader = new ObjLoader(); // 专用预处理加载器
// 移动工作对象到工作线程
m_workerLoader->moveToThread(m_workerThread);
// 连接信号槽
connect(m_workerThread, &QThread::started, [=]() {
// 在工作线程中预处理模型(仅计算中心和半径)
QVector3D center = m_workerLoader->parseObjCenter(filePath);
float radius = m_workerLoader->calculateModelRadiusFromFile(filePath);
emit preprocessFinished(center, radius);
});
// 预处理完成后,在主线程加载模型
connect(this, &P401_CuttingPage::preprocessFinished, this, [=](QVector3D center, float radius) {
m_modelCenter = center;
m_modelRadius = radius;
// 使用定时器确保在主线程事件循环中执行
QTimer::singleShot(0, this, [=]() {
emit loadModelRequest(m_currentDevName, m_currentFilePath, center, radius);
});
// 清理工作线程
m_workerThread->quit();
m_workerThread->wait();
m_workerLoader->deleteLater();
});
// 主线程中实际加载模型
connect(this, &P401_CuttingPage::loadModelRequest, this, [=](const QString& devName, const QString& filePath, QVector3D center, float radius) {
bool LoaderOK = mObjLoader->loadModelAsync(devName, filePath, QVector3D(0, 0, 0), center, radius);
if (!LoaderOK) {
2025-10-10 17:44:10 +08:00
qCritical() << "截割电机加载:失败";
2025-09-30 17:42:22 +08:00
return;
}
mObjLoader->addAxisGizmo(nullptr, 5.0f);
mObjLoader->addAxisGizmo(mObjLoader->getDevice(devName)->entity, 3.0f);
2025-10-10 17:44:10 +08:00
SetPos(0.005,QVector3D(1052.00, 0.00273132, 0),QVector3D(0, 0, 0));
2025-09-30 17:42:22 +08:00
if (LoaderOK) {
QTimer *RotateDev_Timer = new QTimer(this);
connect(RotateDev_Timer, &QTimer::timeout, this, &P401_CuttingPage::RotateDev);
RotateDev_Timer->setInterval(33);
RotateDev_Timer->start();
}
});
// 启动工作线程
m_workerThread->start();
}
void P401_CuttingPage::SetPos(float Zoom,QVector3D Move,QVector3D Pos){
mObjLoader->scaleDevice(DevName3D,Zoom);
mObjLoader->moveArmLocalOriginTo(DevName3D, Move);//14.1, -1.4, 3.4
mObjLoader->setParentDevicePosition(DevName3D, Pos); // 世界坐标
// 6. 重置相机(确保能看到所有设备)
Qt3DRender::QCamera* cam = mObjLoader->getCamera();
if (cam) {
cam->setPosition(QVector3D(0, 3, 12)); // 相机在 Z 轴 30 位置,远离模型
cam->setViewCenter(QVector3D(0, 0, 0)); // 看向电控箱体的世界位置
}
}
2025-10-10 17:44:10 +08:00
void P401_CuttingPage::SetProgressBar(QProgressBar *PB,const uint16_t Value,const uint8_t ColorIndex,const QStringList Color){
2025-09-30 17:42:22 +08:00
PB->setValue(Value);
if(ColorIndex < Color.length()){
PB->setStyleSheet("QProgressBar {\n background-color: transparent;\n border: 2px solid #888888;\n border-radius: 3px;\n}\nQProgressBar::chunk {\n background-color:"+Color[ColorIndex]+";\n}");
}
}
2025-10-10 17:44:10 +08:00
void P401_CuttingPage::SetProgressBar(QProgressBar *PB,QLCDNumber *LCD,const QString NodeGroup,const QString NodeLCD,const QStringList Color){
// 从OPC节点获取油位显示组数据32位无符号整数
uint32_t PB_Group = gOPC_NodeValue["ns=6;s=::AsGlobalPV:" + NodeGroup].toUInt();
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
// 从32位数据中拆分出16位油位百分比值和颜色值
uint16_t Percentage = static_cast<uint16_t>(PB_Group >> 16); // 高16位百分比
uint16_t ColorIndex = static_cast<uint16_t>(PB_Group & 0xFFFF); // 低16位颜色值索引
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
// 设置油位进度条显示(百分比、颜色和样式)
SetProgressBar(PB, Percentage, ColorIndex, Color);
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
// 在LCD上显示实际油位测量值
LCD->display(gOPC_NodeValue["ns=6;s=::AsGlobalPV:" +NodeLCD].toReal());
}
void P401_CuttingPage::UIRefreshTimeOut()
{
if(!gPageIndexStr.contains("P401"))
return;
/************************************* 电机基本信息 *************************************/
// 从OPC节点获取切割设备电机相关的状态数据32位无符号整数
uint32_t PB_Group1 = gOPC_NodeValue["ns=6;s=::AsGlobalPV:PB_Curr_Cut.PageGroup1"].toUInt(); // 电流显示组1数据
uint32_t PB_Group2 = gOPC_NodeValue["ns=6;s=::AsGlobalPV:PB_Curr_Cut.PageGroup2"].toUInt(); // 电流显示组2数据
uint32_t SSConditions = gOPC_NodeValue["ns=6;s=::AsGlobalPV:StartStopConditions.Cut"].toUInt(); // 启停条件状态数据
uint32_t PB_MotorTemp = gOPC_NodeValue["ns=6;s=::AsGlobalPV:PB_Temp_Cut.PageGroup"].toUInt(); // 电机温度显示数据
// 从32位数据中提取16位百分比值高16位和低16位拆分
uint16_t PercentageCurU = static_cast<uint16_t>(PB_Group1 >> 16); // U相电流百分比高16位
uint16_t PercentageCurV = static_cast<uint16_t>(PB_Group1 & 0xFFFF); // V相电流百分比低16位
uint16_t PercentageCurW = static_cast<uint16_t>(PB_Group2 >> 16); // W相电流百分比高16位
uint16_t PercentageTmpColor = static_cast<uint16_t>(PB_Group2 & 0xFFFF); // 温度颜色百分比低16位
uint16_t PercentageMotorTemp= static_cast<uint16_t>(PB_MotorTemp >> 16); // 电机温度百分比高16位
uint16_t ColorTemp = static_cast<uint16_t>(PB_MotorTemp & 0xFFFF); // 温度颜色值低16位
// 解析颜色值(从百分比中提取个位、十位、百位)
uint8_t ColorU = (PercentageTmpColor / 10) % 10; // U相颜色值十位
uint8_t ColorV = (PercentageTmpColor / 100) % 10; // V相颜色值百位
uint8_t ColorW = PercentageTmpColor % 10; // W相颜色值个位
// 设置进度条显示(电流和温度)
SetProgressBar(ui->PB_Cur_1, PercentageCurU, ColorU, glMotorCurColor); // U相电流进度条
SetProgressBar(ui->PB_Cur_2, PercentageCurV, ColorV, glMotorCurColor); // V相电流进度条
SetProgressBar(ui->PB_Cur_3, PercentageCurW, ColorW, glMotorCurColor); // W相电流进度条
SetProgressBar(ui->PB_Temp, PercentageMotorTemp, ColorTemp, glMotorTempColor); // 电机温度进度条
// 在LCD上显示实际测量值电流和温度
ui->LCD_Cur_1->display(gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_U"].toReal()); // U相电流实际值
ui->LCD_Cur_2->display(gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_V"].toReal()); // V相电流实际值
ui->LCD_Cur_3->display(gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_W"].toReal()); // W相电流实际值
ui->LCD_Temp->display(gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered_Temp"].toReal()); // 电机温度实际值
// 更新32个启停条件状态标签的显示样式
for (int i = 0; i < 32; ++i)
{
// 根据索引查找对应的标签控件
QLabel* Label = findChild<QLabel*>(QString("Text_SSConditions_%1").arg(i));
// 如果找到标签
if (Label){
// 基础样式:默认背景图
QString StyleStr = "border-image: url(:/Frames/null.png);\n";
// 根据状态位设置不同颜色True/False对应不同样式
StyleStr += getBitOf32Data(SSConditions, i, false) ? TrueColor[i] + "\n" : FalseColor[i] + "\n";
// 应用样式到标签
Label->setStyleSheet(StyleStr);
}
}
/**************************************************************************************/
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
/************************************ 截割油缸压力信息 ***********************************/
SetProgressBar(ui->PB_1,ui->LCD_1,"PB_Pressure_Cut.PageGroup","IN_Sersor.Pressure_Cut");
/**************************************************************************************/
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
/************************************ 掏槽油缸压力信息 ***********************************/
SetProgressBar(ui->PB_2,ui->LCD_2,"PB_Pressure_Cutt.PageGroup","IN_Sersor.Pressure_Cutt");
/**************************************************************************************/
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
/************************************ 截割喷雾压力信息 ***********************************/
SetProgressBar(ui->PB_3,ui->LCD_3,"PB_Pressure_CutSpray.PageGroup","IN_Sersor.Pressure_CutSpray");
/**************************************************************************************/
2025-09-30 17:42:22 +08:00
2025-10-10 17:44:10 +08:00
/************************************ 截割喷雾流量信息 ***********************************/
SetProgressBar(ui->PB_4,ui->LCD_4,"PB_SprayFlow_Cut.PageGroup","IN_Sersor.SprayFlow_Cut");
/**************************************************************************************/
2025-09-30 17:42:22 +08:00
}
void P401_CuttingPage::TrailRefreshTimeOut(){
2025-10-10 17:44:10 +08:00
ui->CPW_1->appendPoint(0, gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_U"] .toReal());
ui->CPW_1->appendPoint(1, gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_V"] .toReal());
ui->CPW_1->appendPoint(2, gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered30_W"] .toReal());
ui->CPW_2->appendPoint(0,gOPC_NodeValue["ns=6;s=::AsGlobalPV:IN_PhaseCut.Filtered_Temp"] .toDouble());
2025-09-30 17:42:22 +08:00
}
void P401_CuttingPage::RotateDev(){
2025-10-10 17:44:10 +08:00
if(!gPageIndexStr.contains("P401"))
return;
2025-09-30 17:42:22 +08:00
mAngle = mAngle + 1;
if(mAngle > 360) mAngle =0;
mObjLoader->rotateArmToAbsoluteYAngle(DevName3D, mAngle);
}