From bc7f1c4e028e69be51079b59dae4ae5c4d43f5bb Mon Sep 17 00:00:00 2001
From: chenluhua1980 <Chenluhua@qq.com>
Date: 星期六, 31 一月 2026 21:54:56 +0800
Subject: [PATCH] 1.状态指示图,目前灰色表示掉线,绿色表示在线。增加Slot的小点表示有没有料,及加工状态 。 2.增加图示
---
SourceCode/Bond/Servo/CPageGraph1.cpp | 232 ++++++++++++++++++++++++++++++++++++++++++++++++++++++---
1 files changed, 218 insertions(+), 14 deletions(-)
diff --git a/SourceCode/Bond/Servo/CPageGraph1.cpp b/SourceCode/Bond/Servo/CPageGraph1.cpp
index 84b7d0c..30d0811 100644
--- a/SourceCode/Bond/Servo/CPageGraph1.cpp
+++ b/SourceCode/Bond/Servo/CPageGraph1.cpp
@@ -6,6 +6,8 @@
#include "CPageGraph1.h"
#include "afxdialogex.h"
#include "Common.h"
+#include "CEquipment.h"
+#include "CGlass.h"
const std::map<SERVO::ROBOT_POSITION, RobotPositionMapping> g_positionMap = {
{ SERVO::ROBOT_POSITION::Port1, { SERVO::ROBOT_POSITION::Port1, 1.00f, 0.00f } },
@@ -23,6 +25,7 @@
// Image
#define IMAGE_ROBOT 2
+#define IMAGE_LEGEND 3
#define INDICATE_BONDER1 1
#define INDICATE_BONDER2 2
@@ -55,6 +58,7 @@
m_pObserver = nullptr; // 瑙傚療鑰呭璞★紙鍙兘鏄簨浠惰瀵熻�咃級
m_crBkgnd = PAGE_GRPAH1_BACKGROUND_COLOR; // 鑳屾櫙棰滆壊
m_hbrBkgnd = nullptr; // 鑳屾櫙鍒峰彞鏌�
+ m_slotBarTestMode = 0; // 0=off,1=has,2=processing
// ===== 鏈哄櫒浜哄姩鐢荤姸鎬佸垵濮嬪寲 =====
m_bIsRobotMoving = FALSE; // 褰撳墠鏄惁姝e湪鍔ㄧ敾绉诲姩
@@ -108,6 +112,28 @@
std::string configDir = exePath.substr(0, exePath.find_last_of("\\/")) + "\\Config";
CreateDirectoryA(configDir.c_str(), NULL);
return configDir + "\\robot_offset.ini";
+}
+
+void CPageGraph1::UpdateLegendPosition()
+{
+ if (!m_pGraph) return;
+ auto* pImage = m_pGraph->GetImage(IMAGE_LEGEND);
+ if (!pImage) return;
+
+ RECT rc = { 0 };
+ ::GetClientRect(m_pGraph->GetSafeWnd(), &rc);
+ std::string iniPath = GetConfigPath();
+ int cfgX = GetPrivateProfileIntA("Graph1", "LegendX", -1, iniPath.c_str());
+ int cfgY = GetPrivateProfileIntA("Graph1", "LegendY", -1, iniPath.c_str());
+
+ int x = (cfgX >= 0) ? cfgX : (rc.right - pImage->bmWidth - 8);
+ int y = (cfgY >= 0) ? cfgY : 6;
+ if (x < 0) x = 0;
+ if (y < 0) y = 0;
+ if (x > rc.right - pImage->bmWidth) x = rc.right - pImage->bmWidth;
+ if (y > rc.bottom - pImage->bmHeight) y = rc.bottom - pImage->bmHeight;
+ m_pGraph->UpdateImageCoordinates(IMAGE_LEGEND, x, y);
+ m_pGraph->Invalidata();
}
void CPageGraph1::InitRxWindows()
@@ -180,7 +206,7 @@
{
CDialogEx::OnInitDialog();
InitRxWindows();
- SetTimer(TIMER_ID_DEVICE_STATUS, 3000, nullptr);
+ SetTimer(TIMER_ID_DEVICE_STATUS, 800, nullptr);
SetTimer(TIMER_ID_ROBOT_STATUS, 1000, nullptr); // 姣� 1000ms 鏇存柊涓�娆$姸鎬�
// 鍥剧ず
@@ -192,78 +218,138 @@
strPath.Format(_T("%s\\res\\Robot001.bmp"), (LPTSTR)(LPCTSTR)theApp.m_strAppDir);
m_pGraph->AddImage(IMAGE_ROBOT, (LPTSTR)(LPCTSTR)strPath, 170, 270);
+ // Legend
+ strPath.Format(_T("%s\\res\\GraphLegend.bmp"), (LPTSTR)(LPCTSTR)theApp.m_strAppDir);
+ m_pGraph->AddImage(IMAGE_LEGEND, (LPTSTR)(LPCTSTR)strPath, 0, 0);
+ UpdateLegendPosition();
+
// 娣诲姞鎸囩ず鍣�
// Bonder
- m_pGraph->AddIndicateBox(INDICATE_BONDER1, 220, 172, 48, RGB(22, 22, 22),
+ // size config
+ std::string iniPath = GetConfigPath();
+ int boxSize = GetPrivateProfileIntA("Graph1", "BoxSize", 56, iniPath.c_str());
+ if (boxSize < 40) boxSize = 40;
+ if (boxSize > 80) boxSize = 80;
+ int slotSize = GetPrivateProfileIntA("Graph1", "SlotSize", 6, iniPath.c_str());
+ if (slotSize < 2) slotSize = 2;
+ if (slotSize > 12) slotSize = 12;
+ m_pGraph->SetSlotBarSize(slotSize);
+ int armBoxSize = GetPrivateProfileIntA("Graph1", "ArmBoxSize", boxSize, iniPath.c_str());
+ if (armBoxSize < 30) armBoxSize = 30;
+ if (armBoxSize > 80) armBoxSize = 80;
+
+ // ArmSpacing = edge-to-edge gap between the two robot arm boxes.
+ int armGap = GetPrivateProfileIntA("Graph1", "ArmSpacing", 6, iniPath.c_str());
+ if (armGap < 0) armGap = 0;
+ if (armGap > 100) armGap = 100;
+
+ int arm1X = 190;
+ int arm2X = 243;
+ int armY = 294;
+ int minArmSpacing = armBoxSize + armGap;
+ if (minArmSpacing > 0) {
+ int mid = (m_arm1Offset.x + m_arm2Offset.x) / 2;
+ int half = minArmSpacing / 2;
+ if (m_arm1Offset.x <= m_arm2Offset.x) {
+ m_arm1Offset.x = mid - half;
+ m_arm2Offset.x = mid + half;
+ } else {
+ m_arm2Offset.x = mid - half;
+ m_arm1Offset.x = mid + half;
+ }
+ }
+ {
+ int baseMid = (arm1X + arm2X) / 2;
+ int half = minArmSpacing / 2;
+ arm1X = baseMid - half;
+ arm2X = baseMid + half;
+ }
+
+ m_pGraph->AddIndicateBox(INDICATE_BONDER1, 220, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_BONDER1, "", "Bonder 1");
- m_pGraph->AddIndicateBox(INDICATE_BONDER2, 220, 516, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_BONDER2, 220, 516, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_BONDER2, "", "Bonder 2");
// 缈昏浆
- m_pGraph->AddIndicateBox(INDICATE_FLIPER, 338, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_FLIPER, 338, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_FLIPER, "", "Fliper");
// 瀵逛綅
- m_pGraph->AddIndicateBox(INDICATE_ALIGNER, 428, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_ALIGNER, 428, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_ALIGNER, "", "Aligner");
// Load port 4
- m_pGraph->AddIndicateBox(INDICATE_LPORT4, 518, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_LPORT4, 518, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_LPORT4, "", "LPort4");
// Load port 3
- m_pGraph->AddIndicateBox(INDICATE_LPORT3, 606, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_LPORT3, 606, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_LPORT3, "", "LPort3");
// Load port 2
- m_pGraph->AddIndicateBox(INDICATE_LPORT2, 690, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_LPORT2, 690, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_LPORT2, "", "LPort2");
// Load port 1
- m_pGraph->AddIndicateBox(INDICATE_LPORT1, 774, 172, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_LPORT1, 774, 172, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_LPORT1, "", "LPort1");
// Robot
- m_pGraph->AddIndicateBox(INDICATE_ROBOT_ARM1, 190, 294, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_ROBOT_ARM1, arm1X, armY, armBoxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_ROBOT_ARM1, "", "Robot");
- m_pGraph->AddIndicateBox(INDICATE_ROBOT_ARM2, 243, 294, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_ROBOT_ARM2, arm2X, armY, armBoxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_ROBOT_ARM2, "", "Robot");
// Vacuum bake
- m_pGraph->AddIndicateBox(INDICATE_VACUUM_BAKE, 396, 516, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_VACUUM_BAKE, 396, 516, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_VACUUM_BAKE, "", "Vacuum bake");
// Bake cooling
- m_pGraph->AddIndicateBox(INDICATE_BAKE_COOLING, 566, 516, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_BAKE_COOLING, 566, 516, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_BAKE_COOLING, "", "Bake cooling");
// 绮惧害妫�
- m_pGraph->AddIndicateBox(INDICATE_MEASUREMENT, 737, 516, 48, RGB(22, 22, 22),
+ m_pGraph->AddIndicateBox(INDICATE_MEASUREMENT, 737, 516, boxSize, RGB(22, 22, 22),
RGB(255, 127, 39), EQ_BOX_OFFLINE);
m_pGraph->SetBoxText(INDICATE_MEASUREMENT, "", "Measurement");
+ // slot bar positions (top row / bottom row)
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_BONDER1, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_FLIPER, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_ALIGNER, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_LPORT4, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_LPORT3, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_LPORT2, SlotBarPos::Top);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_LPORT1, SlotBarPos::Top);
+
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_BONDER2, SlotBarPos::Bottom);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_VACUUM_BAKE, SlotBarPos::Bottom);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_BAKE_COOLING, SlotBarPos::Bottom);
+ m_pGraph->SetIndicateBoxSlotBarPosition(INDICATE_MEASUREMENT, SlotBarPos::Bottom);
+
+ UpdateSlotBars();
return TRUE; // return TRUE unless you set the focus to a control
// 寮傚父: OCX 灞炴�ч〉搴旇繑鍥� FALSE
@@ -360,6 +446,7 @@
CRect rcClient;
GetClientRect(&rcClient);
GetDlgItem(IDC_SERVO_GRAPH1)->MoveWindow(0, 0, rcClient.Width(), rcClient.Height());
+ UpdateLegendPosition();
}
void CPageGraph1::UpdateRobotPosition(float percentage)
@@ -542,6 +629,111 @@
return pt;
}
+
+void CPageGraph1::UpdateSlotBars()
+{
+ if (m_pGraph == nullptr) return;
+
+ if (m_slotBarTestMode != 0) {
+ ApplySlotBarTestPattern(m_slotBarTestMode);
+ return;
+ }
+
+ struct SlotBarBind {
+ int eqId;
+ int indicateId;
+ };
+ static const SlotBarBind kSlotBars[] = {
+ { EQ_ID_Bonder1, INDICATE_BONDER1 },
+ { EQ_ID_FLIPER, INDICATE_FLIPER },
+ { EQ_ID_ALIGNER, INDICATE_ALIGNER },
+ { EQ_ID_LOADPORT4, INDICATE_LPORT4 },
+ { EQ_ID_LOADPORT3, INDICATE_LPORT3 },
+ { EQ_ID_LOADPORT2, INDICATE_LPORT2 },
+ { EQ_ID_LOADPORT1, INDICATE_LPORT1 },
+ { EQ_ID_Bonder2, INDICATE_BONDER2 },
+ { EQ_ID_VACUUMBAKE, INDICATE_VACUUM_BAKE },
+ { EQ_ID_BAKE_COOLING, INDICATE_BAKE_COOLING },
+ { EQ_ID_MEASUREMENT, INDICATE_MEASUREMENT },
+ };
+
+ for (const auto& item : kSlotBars) {
+ SERVO::CEquipment* pEq = theApp.m_model.m_master.getEquipment(item.eqId);
+ std::vector<COLORREF> colors;
+ BuildSlotColors(pEq, colors);
+ m_pGraph->SetIndicateBoxSlotColors(item.indicateId, colors);
+ }
+}
+
+void CPageGraph1::BuildSlotColors(SERVO::CEquipment* pEq, std::vector<COLORREF>& colors)
+{
+ colors.clear();
+ if (pEq == nullptr) return;
+
+ for (int i = 0; i < SLOT_MAX; ++i) {
+ SERVO::CSlot* pSlot = pEq->getSlot(i);
+ if (pSlot == nullptr || !pSlot->isEnable()) continue;
+ SERVO::CGlass* pGlass = (SERVO::CGlass*)pSlot->getContext();
+ BOOL isProcessing = FALSE;
+ if (pGlass != nullptr) {
+ const auto st = pEq->getProcessState(i + 1);
+ isProcessing = (st == SERVO::PROCESS_STATE::Processing);
+ }
+ colors.push_back(GetSlotColor(pGlass, isProcessing));
+ }
+}
+
+COLORREF CPageGraph1::GetSlotColor(SERVO::CGlass* pGlass, BOOL isProcessing)
+{
+ if (pGlass == nullptr) {
+ return EQ_SLOT_EMPTY;
+ }
+ const auto type = pGlass->getType();
+ const bool isG2 = (type == SERVO::MaterialsType::G2 || type == SERVO::MaterialsType::G1G2);
+ if (isProcessing) {
+ return isG2 ? EQ_SLOT_PROC_G2 : EQ_SLOT_PROC_G1;
+ }
+ return isG2 ? EQ_SLOT_G2 : EQ_SLOT_G1;
+}
+
+void CPageGraph1::ApplySlotBarTestPattern(int mode)
+{
+ if (m_pGraph == nullptr) return;
+
+ std::vector<COLORREF> colors;
+ colors.reserve(SLOT_MAX);
+ for (int i = 0; i < SLOT_MAX; ++i) {
+ if (mode == 2) {
+ colors.push_back((i % 2 == 0) ? EQ_SLOT_PROC_G1 : EQ_SLOT_PROC_G2);
+ } else if (mode == 1) {
+ colors.push_back((i % 2 == 0) ? EQ_SLOT_G1 : EQ_SLOT_G2);
+ } else {
+ colors.push_back(EQ_SLOT_EMPTY);
+ }
+ }
+
+ struct SlotBarBind {
+ int indicateId;
+ };
+ static const SlotBarBind kSlotBars[] = {
+ { INDICATE_BONDER1 },
+ { INDICATE_FLIPER },
+ { INDICATE_ALIGNER },
+ { INDICATE_LPORT4 },
+ { INDICATE_LPORT3 },
+ { INDICATE_LPORT2 },
+ { INDICATE_LPORT1 },
+ { INDICATE_BONDER2 },
+ { INDICATE_VACUUM_BAKE },
+ { INDICATE_BAKE_COOLING },
+ { INDICATE_MEASUREMENT },
+ };
+
+ for (const auto& item : kSlotBars) {
+ m_pGraph->SetIndicateBoxSlotColors(item.indicateId, colors);
+ }
+}
+
void CPageGraph1::SaveArmOffset(const std::string& armName, const POINT& pt)
{
std::string iniPath = GetConfigPath();
@@ -558,6 +750,13 @@
{
BYSERVOGRAPH_NMHDR* pGraphNmhdr = reinterpret_cast<BYSERVOGRAPH_NMHDR*>(pNMHDR);
+ if (GetKeyState(VK_SHIFT) & 0x8000) {
+ // m_slotBarTestMode = (m_slotBarTestMode + 1) % 3;
+ UpdateSlotBars();
+ *pResult = 0;
+ return;
+ }
+
// 绉诲姩鍒版寚瀹氫綅缃� (娴嬭瘯浣跨敤)
if (pGraphNmhdr->dwData == INDICATE_LPORT1) {
@@ -568,6 +767,8 @@
void CPageGraph1::OnTimer(UINT_PTR nIDEvent)
{
+ UpdateLegendPosition();
+
if (TIMER_ID_DEVICE_STATUS == nIDEvent) {
KillTimer(TIMER_ID_DEVICE_STATUS);
@@ -624,6 +825,7 @@
else if (nIDEvent == TIMER_ID_ROBOT_STATUS) {
SERVO::CEFEM* pEFEM = (SERVO::CEFEM*)theApp.m_model.m_master.getEquipment(EQ_ID_EFEM);
if (!pEFEM || !pEFEM->isAlive()) {
+ UpdateSlotBars();
return;
}
@@ -647,6 +849,8 @@
if (robotData.position != m_lastRobotPosition) {
StartRobotMoveToPosition(robotData.position);
}
+
+ UpdateSlotBars();
}
else if (nIDEvent == TIMER_ID_ROBOT_ANIMATION) {
if (!m_bIsRobotMoving) {
--
Gitblit v1.9.3