| | |
| | | } |
| | | } |
| | | |
| | | bool CAxisSettingsDlg::ParsePLCAddress(const CString& address, MC::SOFT_COMPONENT& component, int& addr) |
| | | { |
| | | if (address.GetLength() < 2) { |
| | | return false; |
| | | } |
| | | |
| | | // 提取组件类型(第一个字符) |
| | | TCHAR componentChar = address[0]; |
| | | if (address.Left(2) == _T("ZR")) { |
| | | component = MC::SOFT_COMPONENT::ZR; |
| | | // 提取数字部分(去除ZR前缀) |
| | | CString numericAddress = address.Mid(2); |
| | | addr = _ttoi(numericAddress); |
| | | return addr != 0 || numericAddress.CompareNoCase(_T("0")) == 0; // 如果是 "0",也认为有效 |
| | | } |
| | | |
| | | // 对于其他组件,按照常规规则处理 |
| | | CString hexAddress = address.Mid(1); |
| | | switch (componentChar) { |
| | | case 'D': |
| | | component = MC::SOFT_COMPONENT::D; |
| | | addr = _ttoi(hexAddress); |
| | | break; |
| | | case 'M': |
| | | component = MC::SOFT_COMPONENT::M; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'X': |
| | | component = MC::SOFT_COMPONENT::X; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'Y': |
| | | component = MC::SOFT_COMPONENT::Y; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'W': |
| | | component = MC::SOFT_COMPONENT::W; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'L': |
| | | component = MC::SOFT_COMPONENT::L; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'S': |
| | | component = MC::SOFT_COMPONENT::S; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'B': |
| | | component = MC::SOFT_COMPONENT::B; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | case 'F': |
| | | component = MC::SOFT_COMPONENT::F; |
| | | addr = _tcstoul(hexAddress, nullptr, 16); |
| | | break; |
| | | default: |
| | | return false; |
| | | } |
| | | |
| | | // 检查地址是否有效 |
| | | if (addr == 0 && hexAddress.CompareNoCase(_T("0")) != 0) { |
| | | return false; |
| | | } |
| | | |
| | | return true; |
| | | } |
| | | |
| | | void CAxisSettingsDlg::writeAxisDataToPLC(int nAxisId) |
| | | { |
| | | // 从 RecipeManager 获取轴数据 |
| | | RecipeManager& recipeManager = RecipeManager::getInstance(); |
| | | auto axisData = recipeManager.getAxis(nAxisId); |
| | | |
| | | // 去除非数字字符并转换起始地址 |
| | | std::string cleanAddress = axisData.startAddress; |
| | | cleanAddress.erase(std::remove_if(cleanAddress.begin(), cleanAddress.end(), |
| | | [](char c) { return !std::isdigit(c); }), cleanAddress.end()); |
| | | if (cleanAddress.empty()) { |
| | | int startAddress; |
| | | MC::SOFT_COMPONENT component; |
| | | if (!ParsePLCAddress(CString(axisData.startAddress.c_str()), component, startAddress)) { |
| | | AfxMessageBox(_T("无效的起始地址!")); |
| | | return; |
| | | } |
| | | int startAddress = std::stoi(cleanAddress); |
| | | |
| | | // 写入手动速度 |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, 5120, (int)axisData.manualSpeed, [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeDWord(component, startAddress + 82, (int)axisData.manualSpeed * 1000, [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 手动速度, 地址: %d, 值: %lu\n", addr, value); |
| | | } |
| | |
| | | }); |
| | | |
| | | // 写入自动速度 |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, startAddress + 2, (int)axisData.autoSpeed, [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeDWord(component, startAddress + 84, (int)(axisData.autoSpeed * 1000.0), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 自动速度, 地址: %d, 值: %lu\n", addr, value); |
| | | } |
| | |
| | | }); |
| | | |
| | | // 写入加速时间, 转换为毫秒 |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, startAddress + 4, (int)(axisData.accelerationTime * 1000), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeDWord(component, startAddress + 62, (int)(axisData.accelerationTime * 1000.0), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 加速时间, 地址: %d, 值: %lu\n", addr, value); |
| | | } |
| | |
| | | }); |
| | | |
| | | // 写入减速时间, 转换为毫秒 |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, startAddress + 6, (int)(axisData.decelerationTime * 1000), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeDWord(component, startAddress + 64, (int)(axisData.decelerationTime * 1000.0), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 减速时间, 地址: %d, 值: %lu\n", addr, value); |
| | | } |
| | |
| | | }); |
| | | |
| | | // 写入微动量 |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, startAddress + 8, (int)axisData.jogDistance, [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeWord(component, startAddress + 81, (int)(axisData.jogDistance * 1000.0), [](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 微动量, 地址: %d, 值: %lu\n", addr, value); |
| | | } |
| | |
| | | }); |
| | | |
| | | // 写入定位点数据 |
| | | int positionStartAddress = startAddress + 10; |
| | | /* |
| | | int positionStartAddress = startAddress + 100; |
| | | for (size_t i = 0; i < axisData.positions.size(); ++i) { |
| | | const auto& position = axisData.positions[i]; |
| | | int positionAddress = positionStartAddress + (i * 2); |
| | | unsigned int positionAddress = positionStartAddress + (i * 4); |
| | | |
| | | m_pPLC->writeWord(MC::SOFT_COMPONENT::D, positionAddress, (int)position.second, [i](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | m_pPLC->writeWord(component, positionAddress, (int)position.second, [i](IMcChannel* pChannel, int addr, DWORD value, int flag) { |
| | | if (flag == 0) { |
| | | TRACE("\n写入成功: 定位点 %d, 地址: %d, 值: %lu\n", i + 1, addr, value); |
| | | } |
| | |
| | | } |
| | | }); |
| | | } |
| | | */ |
| | | } |
| | | |
| | | void CAxisSettingsDlg::handleAxisOperation(AxisOperationType eOpType, bool bPressed) |
| | |
| | | RecipeManager& recipeManager = RecipeManager::getInstance(); |
| | | auto axisData = recipeManager.getAxis(nAxisId); |
| | | |
| | | std::string strCleanAddress = axisData.startAddress; |
| | | strCleanAddress.erase(std::remove_if(strCleanAddress.begin(), strCleanAddress.end(), |
| | | [](unsigned char c) { return !std::isdigit(c); }), strCleanAddress.end()); |
| | | if (strCleanAddress.empty()) { |
| | | int startAddress; |
| | | MC::SOFT_COMPONENT component; |
| | | if (!ParsePLCAddress(CString(axisData.startAddress.c_str()), component, startAddress)) { |
| | | AfxMessageBox(_T("无效的起始地址!")); |
| | | return; |
| | | } |
| | | |
| | | int nStartAddress = std::stoi(strCleanAddress); |
| | | |
| | | // 根据操作类型计算目标地址 |
| | | int nTargetAddress = nStartAddress; |
| | | int nTargetAddress = startAddress + 10; |
| | | switch (eOpType) { |
| | | case AxisOperationType::OPR: |
| | | nTargetAddress += 10; // OPR 信号地址 |
| | |
| | | } |
| | | |
| | | // 向 PLC 写入信号 |
| | | m_pPLC->writeBit(MC::SOFT_COMPONENT::D, nTargetAddress, bPressed, [eOpType, nTargetAddress, bPressed](IMcChannel* pChannel, int nAddr, DWORD nValue, int nFlag) { |
| | | m_pPLC->writeBit(component, nTargetAddress, bPressed, [eOpType, nTargetAddress, bPressed](IMcChannel* pChannel, int nAddr, DWORD nValue, int nFlag) { |
| | | if (nFlag == 0) { |
| | | TRACE("操作成功:类型=%d,地址=%d,值=%d\n", static_cast<int>(eOpType), nAddr, bPressed); |
| | | } |
| | |
| | | TRACE("操作失败:类型=%d,地址=%d,错误码=%d\n", static_cast<int>(eOpType), nAddr, nFlag); |
| | | } |
| | | }); |
| | | } |
| | | |
| | | void CAxisSettingsDlg::readPLCDataToUI(int nAxisId) |
| | | { |
| | | CBLLabel* pLabels[] = { &m_staticFLS, &m_staticDOG, &m_staticRLS, &m_staticReady, &m_staticBusy, &m_staticErr }; |
| | | |
| | | // 从 RecipeManager 获取轴数据 |
| | | RecipeManager& recipeManager = RecipeManager::getInstance(); |
| | | auto axisData = recipeManager.getAxis(nAxisId); |
| | | |
| | | MC::SOFT_COMPONENT component; |
| | | int startAddress, endAddress, readSize; |
| | | if (!ParsePLCAddress(CString(axisData.startAddress.c_str()), component, startAddress)) { |
| | | AfxMessageBox(_T("无效的起始地址!")); |
| | | } |
| | | |
| | | // 从 OPR 信号地址开始读取 |
| | | startAddress += 10; |
| | | endAddress = startAddress + 24; |
| | | readSize = endAddress - startAddress + 1; |
| | | |
| | | // 回调处理输入数据 |
| | | auto funOnReadData = [this, startAddress, &pLabels](IMcChannel* pChannel, int addr, char* pData, unsigned int nDataSize, int flag) { |
| | | if (flag == 0) { |
| | | int nOffset = 0; |
| | | for (auto pLabel : pLabels) { |
| | | int value = CToolUnits::toInt16(&pData[(startAddress + nOffset) * 2]); |
| | | if (value == 0) { |
| | | SetStatusColor(*pLabel, FALSE); |
| | | } else { |
| | | SetStatusColor(*pLabel, TRUE); |
| | | } |
| | | |
| | | nOffset++; |
| | | } |
| | | } |
| | | }; |
| | | |
| | | m_pPLC->readData(component, startAddress, readSize, funOnReadData); |
| | | } |
| | | |
| | | |
| | |
| | | if (TIMER_READ_PLC_DATA == nIDEvent) { |
| | | ASSERT(m_pPLC); |
| | | |
| | | int nAxisId = getCurrentSelectedAxisID(); |
| | | if (nAxisId == -1) { |
| | | return; |
| | | } |
| | | |
| | | int addr1, addr2, readSize; |
| | | addr1 = 5120; |
| | | addr2 = 5425; |
| | |
| | | SetDlgItemInt(IDC_EDIT_AXIS_CURR_ALARM_NUMBER, nAlarmCode); |
| | | } |
| | | }; |
| | | m_pPLC->readData(MC::SOFT_COMPONENT::D, addr1, readSize, funOnReadData); |
| | | //m_pPLC->readData(MC::SOFT_COMPONENT::D, addr1, readSize, funOnReadData); |
| | | |
| | | //readPLCDataToUI(nAxisId); |
| | | } |
| | | else if (nIDEvent == TIMER_JOG_ADD && m_bJogAddPressed) { |
| | | TRACE("持续发送 JOG+\n"); |