| SourceCode/Bond/Servo/CEquipment.cpp | ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史 | |
| SourceCode/Bond/Servo/CPageGlassList.cpp | ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史 | |
| SourceCode/Bond/Servo/CPageGlassList.h | ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史 | |
| SourceCode/Bond/Servo/CProcessDataListDlg.cpp | ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史 | |
| SourceCode/Bond/Servo/CProcessDataListDlg.h | ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史 |
SourceCode/Bond/Servo/CEquipment.cpp
@@ -927,6 +927,27 @@ for (int i = 0; i < SLOT_MAX; i++) { CGlass* pGlass = (CGlass*)m_slot[i].getContext(); if (pGlass != nullptr && compareJobData(pJobDataB, pGlass->getJobDataS())) { // 取片成功后,回填当前路径节点的离开时间(优先匹配尚未离开的节点)。 CPath* pPath = pGlass->getPath(); if (pPath != nullptr) { CPath* pScan = pPath->getHeadPath(); CPath* pOpenNode = nullptr; const unsigned int slotNo = (unsigned int)m_slot[i].getNo(); const unsigned int unitNo = getSlotUnit((int)slotNo); while (pScan != nullptr) { if (pScan->getEqID() == m_nID && pScan->getUnit() == unitNo && pScan->getSlot() == slotNo && pScan->getOutTime() == 0) { pOpenNode = pScan; } pScan = pScan->getNext(); } if (pOpenNode != nullptr) { pOpenNode->setOutTime(CToolUnits::getTimestamp()); } } pContext = pGlass; if (pGlass != nullptr) pGlass->addRef(); m_slot[i].setContext(nullptr); SourceCode/Bond/Servo/CPageGlassList.cpp
@@ -24,6 +24,86 @@ static const COLORREF kWipParentBk = RGB(201, 228, 180); // 基础绿 static const COLORREF kWipChildBk = RGB(221, 241, 208); // 更浅一点 namespace { CString FormatPathTime(ULONGLONG ts) { if (ts == 0) return _T(""); return CString(CToolUnits::timeToString2(ts).c_str()); } void BuildPathRowsFromGlass(SERVO::CGlass& glass, std::vector<CProcessDataListDlg::PathRow>& rows) { rows.clear(); SERVO::CPath* pHead = glass.getPath(); if (pHead == nullptr) return; SERVO::CPath* p = pHead->getHeadPath(); int step = 1; while (p != nullptr) { CProcessDataListDlg::PathRow r; r.step.Format(_T("%d"), step++); std::string eqName = SERVO::CServoUtilsTool::getEqUnitName( (int)p->getEqID(), (int)p->getUnit(), (int)p->getSlot()); if (eqName.empty()) { CString tmp; tmp.Format(_T("Eq:%u Unit:%u Slot:%u"), p->getEqID(), p->getUnit(), p->getSlot()); r.chamber = tmp; } else { r.chamber = eqName.c_str(); } r.enterTime = FormatPathTime(p->getInTime()); r.leaveTime = FormatPathTime(p->getOutTime()); r.isArmStep = (p->getEqID() == EQ_ID_ARM || p->getEqID() == EQ_ID_ARM_TRAY1 || p->getEqID() == EQ_ID_ARM_TRAY2); rows.push_back(r); p = p->getNext(); } } void ParseNameValuePairs(const CString& text, std::vector<std::pair<CString, CString>>& outRows) { outRows.clear(); int idx = 0; CString token; while (AfxExtractSubString(token, text, idx, ',')) { ++idx; const int pos = token.Find(_T(':')); if (pos < 0) continue; CString name = token.Left(pos); CString value = token.Mid(pos + 1); name.Trim(); value.Trim(); if (!name.IsEmpty()) { outRows.emplace_back(name, value); } } } void BuildBasicRowsFromList(CListCtrl& list, int row, std::vector<std::pair<CString, CString>>& rows) { rows.clear(); CHeaderCtrl* pHdr = list.GetHeaderCtrl(); const int colCount = pHdr ? pHdr->GetItemCount() : 0; for (int col = 1; col < colCount; ++col) { if (col == 11 || col == 12) continue; // 这两列单独作为其他页 TCHAR buf[256] = { 0 }; LVCOLUMN lvc = {}; lvc.mask = LVCF_TEXT; lvc.pszText = buf; lvc.cchTextMax = _countof(buf); CString header; if (list.GetColumn(col, &lvc)) header = lvc.pszText; header.Trim(); if (header.IsEmpty()) continue; CString val = list.GetItemText(row, col); rows.emplace_back(header, val); } } } // ===== 放在 CPageGlassList.cpp 顶部的匿名工具(文件内静态) ===== // 把当前“已展开”的父行,用它们的 classId(第4列文本)做 key 记录下来 static std::unordered_set<std::string> SnapshotExpandedKeys(CExpandableListCtrl& lv) { @@ -479,6 +559,7 @@ ON_BN_CLICKED(IDC_BUTTON_PREV_PAGE, &CPageGlassList::OnBnClickedButtonPrevPage) ON_BN_CLICKED(IDC_BUTTON_NEXT_PAGE, &CPageGlassList::OnBnClickedButtonNextPage) ON_NOTIFY(ELCN_SHOWFULLTEXT, IDC_LIST_ALARM, &CPageGlassList::OnShowFullText) ON_NOTIFY(NM_DBLCLK, IDC_LIST_ALARM, &CPageGlassList::OnNMDblclkListAlarm) ON_BN_CLICKED(IDC_BUTTON_EXPORT_ROW, &CPageGlassList::OnBnClickedButtonExportRow) END_MESSAGE_MAP() @@ -1578,16 +1659,192 @@ { auto* p = reinterpret_cast<NMC_ELC_SHOWFULLTEXT*>(pNMHDR); // 对话框显示工艺参数 if (p->iSubItem == 12) { CProcessDataListDlg dlg; dlg.setRawText(p->text); dlg.DoModal(); } else { if (p->iSubItem != 11 && p->iSubItem != 12) { AfxMessageBox(p->text); *pResult = 0; return; } const int row = p->iItem; std::vector<std::pair<CString, CString>> basicRows; std::vector<std::pair<CString, CString>> processRows; std::vector<CProcessDataListDlg::PathRow> pathRows; bool pathOk = false; BuildBasicRowsFromList(m_listCtrl, row, basicRows); CString paramsText = m_listCtrl.GetItemText(row, 12); ParseNameValuePairs(paramsText, processRows); CString strId = m_listCtrl.GetItemText(row, 1); // DB id,WIP为空 if (!strId.IsEmpty()) { try { auto rec = GlassLogDb::Instance().queryById(_ttoi64(strId)); if (rec.has_value() && !rec->pretty.empty()) { SERVO::CGlass tempGlass; if (GlassJson::FromString(rec->pretty, tempGlass)) { BuildPathRowsFromGlass(tempGlass, pathRows); pathOk = !pathRows.empty(); } } } catch (...) { pathOk = false; } } if (!pathOk) { CString cls = m_listCtrl.GetItemText(row, 4); // Class ID #ifdef _UNICODE std::string classId = CT2A(cls); #else std::string classId = cls.GetString(); #endif if (!classId.empty()) { std::vector<SERVO::CGlass*> wipGlasses; theApp.m_model.m_master.getWipGlasses(wipGlasses); std::vector<SERVO::CGlass*> tempRetain = wipGlasses; SERVO::CGlass* found = nullptr; for (auto* g : wipGlasses) { if (g == nullptr) continue; if (g->getID() == classId) { found = g; break; } SERVO::CGlass* b = g->getBuddy(); if (b != nullptr && b->getID() == classId) { found = b; break; } } if (found != nullptr) { BuildPathRowsFromGlass(*found, pathRows); pathOk = !pathRows.empty(); } for (auto* item : tempRetain) { if (item != nullptr) item->release(); } } } if (!pathOk) { CString pathText = m_listCtrl.GetItemText(row, 11); if (!pathText.IsEmpty()) { int cur = 0; CString token = pathText.Tokenize(_T("->"), cur); int step = 1; while (!token.IsEmpty()) { token.Trim(); if (!token.IsEmpty()) { CProcessDataListDlg::PathRow r; r.step.Format(_T("%d"), step++); r.chamber = token; r.enterTime = _T(""); r.leaveTime = _T(""); CString upper = token; upper.MakeUpper(); r.isArmStep = (upper.Find(_T("ARM")) >= 0 || upper.Find(_T("TRAY")) >= 0); pathRows.push_back(r); } token = pathText.Tokenize(_T("->"), cur); } } } CProcessDataListDlg dlg; dlg.setUnifiedData(basicRows, pathRows, processRows); dlg.setInitialTab((p->iSubItem == 11) ? 1 : 2); dlg.DoModal(); *pResult = 0; } void CPageGlassList::OnNMDblclkListAlarm(NMHDR* pNMHDR, LRESULT* pResult) { LPNMITEMACTIVATE pItem = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR); if (pItem == nullptr || pItem->iItem < 0) { *pResult = 0; return; } const int row = pItem->iItem; std::vector<std::pair<CString, CString>> basicRows; std::vector<std::pair<CString, CString>> processRows; std::vector<CProcessDataListDlg::PathRow> pathRows; bool pathOk = false; BuildBasicRowsFromList(m_listCtrl, row, basicRows); CString paramsText = m_listCtrl.GetItemText(row, 12); ParseNameValuePairs(paramsText, processRows); CString strId = m_listCtrl.GetItemText(row, 1); // DB id,WIP为空 if (!strId.IsEmpty()) { try { auto rec = GlassLogDb::Instance().queryById(_ttoi64(strId)); if (rec.has_value() && !rec->pretty.empty()) { SERVO::CGlass tempGlass; if (GlassJson::FromString(rec->pretty, tempGlass)) { BuildPathRowsFromGlass(tempGlass, pathRows); pathOk = !pathRows.empty(); } } } catch (...) { pathOk = false; } } if (!pathOk) { CString cls = m_listCtrl.GetItemText(row, 4); // Class ID #ifdef _UNICODE std::string classId = CT2A(cls); #else std::string classId = cls.GetString(); #endif if (!classId.empty()) { std::vector<SERVO::CGlass*> wipGlasses; theApp.m_model.m_master.getWipGlasses(wipGlasses); std::vector<SERVO::CGlass*> tempRetain = wipGlasses; SERVO::CGlass* found = nullptr; for (auto* g : wipGlasses) { if (g == nullptr) continue; if (g->getID() == classId) { found = g; break; } SERVO::CGlass* b = g->getBuddy(); if (b != nullptr && b->getID() == classId) { found = b; break; } } if (found != nullptr) { BuildPathRowsFromGlass(*found, pathRows); pathOk = !pathRows.empty(); } for (auto* item : tempRetain) { if (item != nullptr) item->release(); } } } if (!pathOk) { CString pathText = m_listCtrl.GetItemText(row, 11); if (!pathText.IsEmpty()) { int cur = 0; CString token = pathText.Tokenize(_T("->"), cur); int step = 1; while (!token.IsEmpty()) { token.Trim(); if (!token.IsEmpty()) { CProcessDataListDlg::PathRow r; r.step.Format(_T("%d"), step++); r.chamber = token; r.enterTime = _T(""); r.leaveTime = _T(""); CString upper = token; upper.MakeUpper(); r.isArmStep = (upper.Find(_T("ARM")) >= 0 || upper.Find(_T("TRAY")) >= 0); pathRows.push_back(r); } token = pathText.Tokenize(_T("->"), cur); } } } CProcessDataListDlg dlg; dlg.setUnifiedData(basicRows, pathRows, processRows); dlg.DoModal(); *pResult = 0; } SourceCode/Bond/Servo/CPageGlassList.h
@@ -91,6 +91,7 @@ afx_msg void OnBnClickedButtonPrevPage(); afx_msg void OnBnClickedButtonNextPage(); afx_msg void OnShowFullText(NMHDR* pNMHDR, LRESULT* pResult); afx_msg void OnNMDblclkListAlarm(NMHDR* pNMHDR, LRESULT* pResult); virtual BOOL PreTranslateMessage(MSG* pMsg); DECLARE_MESSAGE_MAP() public: SourceCode/Bond/Servo/CProcessDataListDlg.cpp
@@ -6,6 +6,10 @@ #include "CProcessDataListDlg.h" #include "afxdialogex.h" namespace { static const UINT IDC_CHECK_SHOW_ARM_STEPS = 0x5001; static const UINT IDC_TAB_UNIFIED = 0x5002; } // CProcessDataListDlg 对话框 @@ -30,6 +34,8 @@ BEGIN_MESSAGE_MAP(CProcessDataListDlg, CDialogEx) ON_BN_CLICKED(IDC_BUTTON1, &CProcessDataListDlg::OnBnClickedButton1) ON_BN_CLICKED(IDC_CHECK_SHOW_ARM_STEPS, &CProcessDataListDlg::OnBnClickedCheckShowArm) ON_NOTIFY(TCN_SELCHANGE, IDC_TAB_UNIFIED, &CProcessDataListDlg::OnTcnSelchangeTabUnified) END_MESSAGE_MAP() @@ -40,7 +46,18 @@ { CDialogEx::OnInitDialog(); SetupListCtrlStyle(); SetWindowText(_T("Glass数据详情")); LoadArmStepOption(); InitUnifiedLayout(); RenderUnifiedTab(); return TRUE; // return TRUE unless you set the focus to a control // 异常: OCX 属性页应返回 FALSE } void CProcessDataListDlg::SetupListCtrlStyle() { DWORD dwStyle = m_listCtrl.GetExtendedStyle(); dwStyle |= LVS_EX_FULLROWSELECT; dwStyle |= LVS_EX_GRIDLINES; @@ -49,19 +66,146 @@ HIMAGELIST imageList = ImageList_Create(24, 24, ILC_COLOR24, 1, 1); ListView_SetImageList(m_listCtrl.GetSafeHwnd(), imageList, LVSIL_SMALL); m_listCtrl.InsertColumn(0, _T("名称"), LVCFMT_RIGHT, 188); m_listCtrl.InsertColumn(1, _T("值"), LVCFMT_LEFT, 128); InsertParamsToListCtrl(m_listCtrl, m_strRawText); return TRUE; // return TRUE unless you set the focus to a control // 异常: OCX 属性页应返回 FALSE } void CProcessDataListDlg::setRawText(CString& strRawText) { m_strRawText = strRawText; m_initialTab = 2; } void CProcessDataListDlg::setPathRows(const std::vector<PathRow>& rows) { m_pathRows = rows; m_initialTab = 1; } void CProcessDataListDlg::setUnifiedData(const std::vector<std::pair<CString, CString>>& basicRows, const std::vector<PathRow>& pathRows, const std::vector<std::pair<CString, CString>>& processRows) { m_basicRows = basicRows; m_pathRows = pathRows; m_processRows = processRows; } void CProcessDataListDlg::setInitialTab(int tabIndex) { if (tabIndex < 0) tabIndex = 0; if (tabIndex > 2) tabIndex = 2; m_initialTab = tabIndex; } void CProcessDataListDlg::LoadArmStepOption() { CString iniPath; iniPath.Format(_T("%s\\configuration.ini"), (LPTSTR)(LPCTSTR)theApp.m_strAppDir); m_showArmSteps = (GetPrivateProfileInt(_T("GlassPathDlg"), _T("ShowArmSteps"), 0, iniPath) != 0); } void CProcessDataListDlg::SaveArmStepOption() const { CString iniPath, val; iniPath.Format(_T("%s\\configuration.ini"), (LPTSTR)(LPCTSTR)theApp.m_strAppDir); val.Format(_T("%d"), m_showArmSteps ? 1 : 0); WritePrivateProfileString(_T("GlassPathDlg"), _T("ShowArmSteps"), val, iniPath); } void CProcessDataListDlg::BuildPathList() { m_listCtrl.DeleteAllItems(); int stepNo = 1; for (const auto& r : m_pathRows) { if (!m_showArmSteps && r.isArmStep) continue; CString step; step.Format(_T("%d"), stepNo++); int nItem = m_listCtrl.InsertItem(m_listCtrl.GetItemCount(), step); m_listCtrl.SetItemText(nItem, 1, r.chamber); m_listCtrl.SetItemText(nItem, 2, r.enterTime); m_listCtrl.SetItemText(nItem, 3, r.leaveTime); } } void CProcessDataListDlg::BuildNameValueList(const std::vector<std::pair<CString, CString>>& rows) { m_listCtrl.DeleteAllItems(); for (int i = 0; i < (int)rows.size(); ++i) { int nItem = m_listCtrl.InsertItem(i, rows[i].first); m_listCtrl.SetItemText(nItem, 1, rows[i].second); } } void CProcessDataListDlg::InitUnifiedLayout() { CRect rcList; m_listCtrl.GetWindowRect(&rcList); ScreenToClient(&rcList); CRect rcTab(rcList.left, rcList.top, rcList.right, rcList.top + 24); m_tabCtrl.Create(WS_CHILD | WS_VISIBLE | WS_TABSTOP, rcTab, this, IDC_TAB_UNIFIED); m_tabCtrl.SetFont(GetFont()); m_tabCtrl.InsertItem(0, _T("基础信息")); m_tabCtrl.InsertItem(1, _T("工艺Step")); m_tabCtrl.InsertItem(2, _T("ProcessData")); m_tabCtrl.SetCurSel(m_initialTab); CRect rcNewList(rcList.left, rcTab.bottom + 2, rcList.right, rcList.bottom); m_listCtrl.MoveWindow(&rcNewList); CRect rcChk(rcNewList.left, rcNewList.bottom + 6, rcNewList.left + 200, rcNewList.bottom + 24); m_chkShowArm.Create(_T("显示机械手相关Step"), WS_CHILD | BS_AUTOCHECKBOX, rcChk, this, IDC_CHECK_SHOW_ARM_STEPS); m_chkShowArm.SetFont(GetFont()); m_chkShowArm.SetCheck(m_showArmSteps ? BST_CHECKED : BST_UNCHECKED); } void CProcessDataListDlg::RenderUnifiedTab() { const int tab = m_tabCtrl.GetCurSel(); while (m_listCtrl.GetHeaderCtrl() && m_listCtrl.GetHeaderCtrl()->GetItemCount() > 0) { m_listCtrl.DeleteColumn(0); } if (tab == 0) { m_chkShowArm.ShowWindow(SW_HIDE); m_listCtrl.InsertColumn(0, _T("名称"), LVCFMT_RIGHT, 200); m_listCtrl.InsertColumn(1, _T("值"), LVCFMT_LEFT, 320); BuildNameValueList(m_basicRows); } else if (tab == 1) { m_chkShowArm.ShowWindow(SW_SHOW); m_listCtrl.InsertColumn(0, _T("Step"), LVCFMT_RIGHT, 56); m_listCtrl.InsertColumn(1, _T("腔体名称"), LVCFMT_LEFT, 180); m_listCtrl.InsertColumn(2, _T("进入时间"), LVCFMT_LEFT, 156); m_listCtrl.InsertColumn(3, _T("离开时间"), LVCFMT_LEFT, 156); m_listCtrl.DeleteAllItems(); int stepNo = 1; for (const auto& r : m_pathRows) { if (!m_showArmSteps && r.isArmStep) continue; CString step; step.Format(_T("%d"), stepNo++); int nItem = m_listCtrl.InsertItem(m_listCtrl.GetItemCount(), step); m_listCtrl.SetItemText(nItem, 1, r.chamber); m_listCtrl.SetItemText(nItem, 2, r.enterTime); m_listCtrl.SetItemText(nItem, 3, r.leaveTime); } } else { m_chkShowArm.ShowWindow(SW_HIDE); m_listCtrl.InsertColumn(0, _T("名称"), LVCFMT_RIGHT, 200); m_listCtrl.InsertColumn(1, _T("值"), LVCFMT_LEFT, 320); if (!m_processRows.empty()) { BuildNameValueList(m_processRows); } else { InsertParamsToListCtrl(m_listCtrl, m_strRawText); } } } void CProcessDataListDlg::InsertParamsToListCtrl(CListCtrl& listCtrl, const CString& data) @@ -199,3 +343,16 @@ AfxMessageBox(_T("复制失败!")); } } void CProcessDataListDlg::OnBnClickedCheckShowArm() { m_showArmSteps = (m_chkShowArm.GetCheck() == BST_CHECKED); SaveArmStepOption(); RenderUnifiedTab(); } void CProcessDataListDlg::OnTcnSelchangeTabUnified(NMHDR* pNMHDR, LRESULT* pResult) { RenderUnifiedTab(); *pResult = 0; } SourceCode/Bond/Servo/CProcessDataListDlg.h
@@ -1,5 +1,7 @@ #pragma once #include "ListCtrlEx.h" #include <vector> #include <utility> // CProcessDataListDlg 对话框 @@ -13,13 +15,42 @@ virtual ~CProcessDataListDlg(); public: struct PathRow { CString step; CString chamber; CString enterTime; CString leaveTime; bool isArmStep = false; }; void setRawText(CString& strRawText); void setPathRows(const std::vector<PathRow>& rows); void setUnifiedData(const std::vector<std::pair<CString, CString>>& basicRows, const std::vector<PathRow>& pathRows, const std::vector<std::pair<CString, CString>>& processRows); void setInitialTab(int tabIndex); void InsertParamsToListCtrl(CListCtrl& listCtrl, const CString& data); bool CopyListCtrlToClipboard(CListCtrl& listCtrl, bool includeHeader = false); private: CString m_strRawText; CListCtrlEx m_listCtrl; CTabCtrl m_tabCtrl; std::vector<PathRow> m_pathRows; std::vector<std::pair<CString, CString>> m_basicRows; std::vector<std::pair<CString, CString>> m_processRows; CButton m_chkShowArm; bool m_showArmSteps = false; int m_initialTab = 0; private: void LoadArmStepOption(); void SaveArmStepOption() const; void SetupListCtrlStyle(); void BuildNameValueList(const std::vector<std::pair<CString, CString>>& rows); void BuildPathList(); void InitUnifiedLayout(); void RenderUnifiedTab(); // 对话框数据 @@ -34,4 +65,6 @@ public: virtual BOOL OnInitDialog(); afx_msg void OnBnClickedButton1(); afx_msg void OnBnClickedCheckShowArm(); afx_msg void OnTcnSelchangeTabUnified(NMHDR* pNMHDR, LRESULT* pResult); };