chenluhua1980
6 天以前 ceb64b6612309fe384e096dcdc8b5a5e0dfe6cce
SourceCode/Bond/Servo/CControlJobManagerDlg.cpp
@@ -6,7 +6,11 @@
#include "CControlJobManagerDlg.h"
#include "afxdialogex.h"
#include "ToolUnits.h"
#include "RecipeManager.h"
bool CControlJobManagerDlg::m_bHasState = false;
CControlJobManagerDlg::State CControlJobManagerDlg::m_state{};
// CControlJobManagerDlg 对话框
@@ -20,15 +24,22 @@
CControlJobManagerDlg::~CControlJobManagerDlg()
{
   for (auto pj : m_processJobs) {
      delete pj;
   }
   m_processJobs.clear();
   if (m_pControlJob != nullptr) {
      delete m_pControlJob;
      m_pControlJob = nullptr;
}
void CControlJobManagerDlg::FreeState()
{
   if (!m_bHasState) return;
   for (auto item : m_state.pjWarps) {
      delete (SERVO::CProcessJob*)item.pj;
   }
   m_state.pjWarps.clear();
   if (m_state.pControlJob != nullptr) {
      delete m_state.pControlJob;
      m_state.pControlJob = nullptr;
   }
   m_bHasState = false;
}
void CControlJobManagerDlg::DoDataExchange(CDataExchange* pDX)
@@ -41,9 +52,14 @@
BEGIN_MESSAGE_MAP(CControlJobManagerDlg, CDialogEx)
   ON_WM_SIZE()
   ON_WM_GETMINMAXINFO()
   ON_NOTIFY(TVN_SELCHANGED, IDC_TREE1, &CControlJobManagerDlg::OnTvnSelchangedTree1)
   ON_NOTIFY(TVN_ITEMCHANGED, IDC_TREE1, &CControlJobManagerDlg::OnTvnItemChangedTree)
   ON_NOTIFY(NM_CLICK, IDC_TREE1, &CControlJobManagerDlg::OnTreeClick)          // 新增
   ON_NOTIFY(TVN_KEYDOWN, IDC_TREE1, &CControlJobManagerDlg::OnTreeKeyDown)        // 新增
   ON_MESSAGE(WM_AFTER_TVCHECK, &CControlJobManagerDlg::OnAfterTvCheck)                // 新增
   ON_WM_DESTROY()
   ON_BN_CLICKED(IDC_BUTTON_APPLY, &CControlJobManagerDlg::OnBnClickedButtonApply)
   ON_NOTIFY(TVN_SELCHANGING, IDC_TREE1, &CControlJobManagerDlg::OnTvnSelchangingTree1)
   ON_BN_CLICKED(IDC_BUTTON_BATH_COMPLETION, &CControlJobManagerDlg::OnBnClickedButtonBathCompletion)
END_MESSAGE_MAP()
@@ -60,7 +76,10 @@
      }
      else if (1 == code) {
         if (contextType == 1) {
            UpProcessJobId((SERVO::CProcessJob*)pContext);
            UpProcessJobId((PJWarp*)pContext);
         }
         else if (contextType == 2) {
            UpControlJobId((SERVO::CControlJob*)pContext);
         }
      }
   };
@@ -174,8 +193,7 @@
void CControlJobManagerDlg::UpdateCtrlState()
{
   auto& master = theApp.m_model.getMaster();
   GetDlgItem(IDC_BUTTON_BATH_COMPLETION)->EnableWindow(false);
   GetDlgItem(IDC_BUTTON_BATH_COMPLETION)->EnableWindow(true);
}
void CControlJobManagerDlg::UpdateControlJob()
@@ -186,10 +204,10 @@
   HTREEITEM hRoot = m_tree.InsertItem(m_pControlJob->id().c_str(), 0, 0);
   m_tree.SetItemData(hRoot, (DWORD_PTR)m_pControlJob);
   m_tree.SetItemState(hRoot, 0, TVIS_STATEIMAGEMASK);
   for (auto pj : m_processJobs) {
      HTREEITEM hItem = m_tree.InsertItem(pj->id().c_str(), 0, 0, hRoot);
      m_tree.SetItemData(hItem, (DWORD_PTR)pj);
      m_tree.SetItemState(hItem, INDEXTOSTATEIMAGEMASK(false ? 2 : 1), TVIS_STATEIMAGEMASK);
   for (auto& item : m_pjWarps) {
      HTREEITEM hItem = m_tree.InsertItem(((SERVO::CProcessJob*)item.pj)->id().c_str(), 0, 0, hRoot);
      m_tree.SetItemData(hItem, (DWORD_PTR)&item);
      m_tree.SetItemState(hItem, INDEXTOSTATEIMAGEMASK(item.addToCj ? 2 : 1), TVIS_STATEIMAGEMASK);
   }
   m_tree.Expand(hRoot, TVE_EXPAND);
}
@@ -206,46 +224,134 @@
   return m_pControlJob->removePjPointer(pj->id());
}
void CControlJobManagerDlg::OnTvnSelchangedTree1(NMHDR* pNMHDR, LRESULT* pResult)
void CControlJobManagerDlg::OnTvnItemChangedTree(NMHDR* pNMHDR, LRESULT* pResult)
{
   auto* p = reinterpret_cast<LPNMTREEVIEW>(pNMHDR);
   UINT oldState = p->itemOld.state, newState = p->itemNew.state;
   HTREEITEM hItem = p->itemNew.hItem;
   if (((oldState ^ newState) & TVIS_STATEIMAGEMASK) != 0) {
      const int idx = (newState & TVIS_STATEIMAGEMASK) >> 12; // 1=未选,2=已选
      const bool checked = (idx == 2);
      PJWarp* pjWarp = (PJWarp*)m_tree.GetItemData(hItem);
      if (pjWarp != nullptr) {
         CString s; s.Format("%s %d", ((SERVO::CProcessJob*)pjWarp->pj)->id().c_str(),
            checked ? "" : "");
         AfxMessageBox(s);
      }
   }
   *pResult = 0;
}
// 命中复选框:用 NM_CLICK 做命中测试,然后“滞后”读取新状态
void CControlJobManagerDlg::OnTreeClick(NMHDR* pNMHDR, LRESULT* pResult)
{
   *pResult = 0;
   DWORD pos = ::GetMessagePos();
   CPoint pt(GET_X_LPARAM(pos), GET_Y_LPARAM(pos));
   m_tree.ScreenToClient(&pt);
   TVHITTESTINFO ht{}; ht.pt = pt;
   HTREEITEM hItem = m_tree.HitTest(&ht);
   if (hItem && (ht.flags & TVHT_ONITEMSTATEICON)) {
      // 让 TreeView 先切换,再异步读取最终状态
      PostMessage(WM_AFTER_TVCHECK, (WPARAM)hItem, 0);
   }
}
// 空格键也会切换复选框
void CControlJobManagerDlg::OnTreeKeyDown(NMHDR* pNMHDR, LRESULT* pResult)
{
   *pResult = 0;
   auto* p = reinterpret_cast<LPNMTVKEYDOWN>(pNMHDR);
   if (p->wVKey == VK_SPACE) {
      HTREEITEM hItem = m_tree.GetSelectedItem();
      if (hItem) PostMessage(WM_AFTER_TVCHECK, (WPARAM)hItem, 0);
   }
}
// 统一处理(读最终状态 + 你的业务)
LRESULT CControlJobManagerDlg::OnAfterTvCheck(WPARAM wParam, LPARAM /*lParam*/)
{
   HTREEITEM hItem = (HTREEITEM)wParam;
   if (!hItem) return 0;
   // 只处理第二层:根的直接子节点(可选)
   auto getLevel = [&](HTREEITEM h) {
      int lv = 0; for (HTREEITEM p = m_tree.GetParentItem(h); p; p = m_tree.GetParentItem(p)) ++lv; return lv;
   };
   if (getLevel(hItem) != 1) return 0;
   BOOL checked = m_tree.GetCheck(hItem);
   // 你的业务逻辑(修正了 CString::Format 的参数类型)
   auto* pjWarp = reinterpret_cast<PJWarp*>(m_tree.GetItemData(hItem));
   if (pjWarp) {
      pjWarp->addToCj = checked;
   }
   return 0;
}
void CControlJobManagerDlg::OnTvnSelchangingTree1(NMHDR* pNMHDR, LRESULT* pResult)
{
   LPNMTREEVIEW pNMTreeView = reinterpret_cast<LPNMTREEVIEW>(pNMHDR);
   bool allow = FALSE;
   HTREEITEM hOldSel = pNMTreeView->itemOld.hItem;
   HTREEITEM hSel = pNMTreeView->itemNew.hItem;
   if (hSel != nullptr) {
      HTREEITEM hParent = m_tree.GetParentItem(hSel);
      if (hParent == nullptr) {
         SERVO::CControlJob* cj = (SERVO::CControlJob*)m_tree.GetItemData(hSel);
         ASSERT(m_pages.size() == 3);
         m_pages[0]->ShowWindow(SW_HIDE);
         m_pages[1]->ShowWindow(SW_HIDE);
         m_pages[2]->ShowWindow(SW_SHOW);
         if (0 == ShowPage(2)) {
            SERVO::CControlJob* pControlJob = (SERVO::CControlJob*)m_tree.GetItemData(hSel);
            m_pages[2]->SetContext(pControlJob, 2);
         }
      }
      else if (m_tree.GetParentItem(hParent) == nullptr) {
         SERVO::CProcessJob* pj = (SERVO::CProcessJob*)m_tree.GetItemData(hSel);
         m_pages[0]->ShowWindow(SW_HIDE);
         m_pages[1]->ShowWindow(SW_SHOW);
         m_pages[2]->ShowWindow(SW_HIDE);
         m_pages[1]->SetContext(pj, 1);
         std::vector<std::string> names;
         for (auto item : m_processJobs) {
            if (item != pj) {
               names.push_back(item->id());
            }
         if (0 == ShowPage(1)) {
            PJWarp* pjWarp = (PJWarp*)m_tree.GetItemData(hSel);
            ((CCjPage2*)m_pages[1])->SetPjWarps(m_pjWarps);
            m_pages[1]->SetContext(pjWarp, 1);
         }
         ((CCjPage2*)m_pages[1])->SetExclusionNames(names);
         else {
            allow = TRUE;
         }
      }
      else {
         // 有祖先 → 第三层及以下 → Glass
      }
   }
   *pResult = 0;
   *pResult = allow;
}
int CControlJobManagerDlg::ShowPage(int index)
{
   ASSERT(0 <= index && index <= 2);
   for (int i = 0; i < 3; i++) {
      if (m_pages[i]->IsWindowVisible()) {
         int ret = m_pages[i]->OnApply();
         if (ret != 0) return -1;
      }
      m_pages[i]->ShowWindow(index == i ? SW_SHOW : SW_HIDE);
   }
   return 0;
}
void CControlJobManagerDlg::OnDestroy()
{
   CDialogEx::OnDestroy();
   SaveState();
   for (auto page : m_pages) {
      page->DestroyWindow();
@@ -255,26 +361,34 @@
void CControlJobManagerDlg::InitData()
{
   ASSERT(m_pControlJob == nullptr);
   LoadState();
   if (m_pControlJob != nullptr) return;
   m_pControlJob = new SERVO::CControlJob("CJ" + CToolUnits::NowStrSec());
   char szBuffer[256];
   for (int i = 0; i < 4; i++) {
      sprintf_s(szBuffer, 256, "PJ%03d", i + 1);
      SERVO::CProcessJob* pj = new SERVO::CProcessJob(std::string(szBuffer));
      m_processJobs.push_back(pj);
      PJWarp pjWarp = {};
      pjWarp.pj = pj;
      pjWarp.port = -1;
      m_pjWarps.push_back(pjWarp);
   }
}
void CControlJobManagerDlg::OnBnClickedButtonApply()
{
   for (auto item : m_pages) {
      item->OnApply();
      if (item->IsWindowVisible()) {
         if (0 == item->OnApply()) {
            GetDlgItem(IDC_BUTTON_APPLY)->EnableWindow(FALSE);
         }
      }
   }
   GetDlgItem(IDC_BUTTON_APPLY)->EnableWindow(FALSE);
}
void CControlJobManagerDlg::UpProcessJobId(SERVO::CProcessJob* pJob)
void CControlJobManagerDlg::UpProcessJobId(PJWarp* pjWarp)
{
   // 更新树控件
   // 遍历根节点
@@ -284,8 +398,9 @@
      HTREEITEM hChild = m_tree.GetChildItem(hRoot);
      while (hChild) {
         DWORD_PTR data = m_tree.GetItemData(hChild);
         if ((void*)data == pJob) {
            m_tree.SetItemText(hChild, pJob->id().c_str());
         if ((void*)data == pjWarp) {
            SERVO::CProcessJob* pj = (SERVO::CProcessJob*)pjWarp->pj;
            m_tree.SetItemText(hChild, pj->id().c_str());
            return; // 找到就返回
         }
         hChild = m_tree.GetNextSiblingItem(hChild);
@@ -294,3 +409,244 @@
      hRoot = m_tree.GetNextSiblingItem(hRoot);
   }
}
void CControlJobManagerDlg::UpControlJobId(SERVO::CControlJob* pControlJob)
{
   // 更新树控件
   // 遍历根节点
   HTREEITEM hRoot = m_tree.GetRootItem();
   if (hRoot != nullptr) {
      DWORD_PTR data = m_tree.GetItemData(hRoot);
      if ((void*)data == pControlJob) {
         m_tree.SetItemText(hRoot, pControlJob->id().c_str());
         return; // 找到就返回
      }
   }
}
void CControlJobManagerDlg::LoadState()
{
   if (!m_bHasState) return;
   // 把 s_state -> 成员变量
   m_pControlJob = m_state.pControlJob;
   m_pjWarps = m_state.pjWarps;
}
void CControlJobManagerDlg::SaveState()
{
   m_state.pControlJob = m_pControlJob;
   m_state.pjWarps = m_pjWarps;
   m_bHasState = true;
}
void CControlJobManagerDlg::OnBnClickedButtonBathCompletion()
{
   // 先检查当前master
   auto& master = theApp.m_model.getMaster();
   if (!master.canCreateControlJob()) {
      AfxMessageBox("当前Master有未结批的Job, 请先结批处理");
      return;
   }
   // 先应用
   for (int i = 0; i < 3; i++) {
      if (m_pages[i]->IsWindowVisible()) {
         int ret = m_pages[i]->OnApply();
         if (ret != 0) return ;
      }
   }
   GetDlgItem(IDC_BUTTON_APPLY)->EnableWindow(FALSE);
   // 先检查数据正确性
   int checkCount = 0;
   for (auto item : m_pjWarps) {
      if (!item.addToCj) continue;
      checkCount++;
   }
   if (checkCount == 0) {
      AfxMessageBox(_T("您没有选择要进行工艺处理的Process Job!\n请在要进行工艺处理的Process Job前打勾。"));
      return;
   }
   SERVO::CLoadPort* pPorts[4];
   pPorts[0] = (SERVO::CLoadPort*)master.getEquipment(EQ_ID_LOADPORT1);
   pPorts[1] = (SERVO::CLoadPort*)master.getEquipment(EQ_ID_LOADPORT2);
   pPorts[2] = (SERVO::CLoadPort*)master.getEquipment(EQ_ID_LOADPORT3);
   pPorts[3] = (SERVO::CLoadPort*)master.getEquipment(EQ_ID_LOADPORT4);
   bool bProcessStart[] = {false, false, false, false};
   std::vector<SERVO::CProcessJob*> pjs;
   for (auto item : m_pjWarps) {
      if (!item.addToCj) continue;
      if (item.port == -1) continue;
      BOOL bCheck = FALSE;
      for (int i = 0; i < 8; i++) {
         if (item.checkSlot[i]) {
            bCheck = TRUE;
            break;
         }
      }
      if (!bCheck) continue;
      SERVO::CProcessJob* pScr = (SERVO::CProcessJob*)item.pj;
      pScr->setPjWarp(item);
      pScr->setLotId("LotID1");
      pScr->setProductId("ProductId1");
      pScr->setOperationId("OperationId");
      pScr->setRecipe(SERVO::RecipeMethod::NoTuning, pScr->recipeSpec());
      SERVO::CProcessJob * pj = new SERVO::CProcessJob(pScr->id());
      pj->setPjWarp(item);
      pj->setLotId("LotID1");
      pj->setProductId("ProductId1");
      pj->setOperationId("OperationId");
      pj->setRecipe(SERVO::RecipeMethod::NoTuning, pScr->recipeSpec());
      std::vector<SERVO::CarrierSlotInfo> carriers;
      SERVO::CarrierSlotInfo csi;
      csi.carrierId = pPorts[item.port]->getCassetteId();
      for (int i = 0; i < 8; i++) {
         if (item.checkSlot[i]) {
            SERVO::CGlass* pGlass = pPorts[item.port]->getGlassFromSlot(i+1);
            if (pGlass != nullptr) {
               csi.slots.push_back(i + 1);
            }
         }
      }
      carriers.push_back(csi);
      pj->setCarriers(carriers);
      pjs.push_back(pj);
      bProcessStart[item.port] = true;
      m_pControlJob->addPJ(pScr->id());
   }
   if (pjs.empty()) {
      AfxMessageBox(_T("没有需要进行工艺处理的Process Job!\n可能未选择Port或选择任何物料。"));
      return;
   }
   m_pControlJob->setPJs(pjs);
   m_pControlJob->clearIssues();
   int nRet = master.setProcessJobs(pjs);
   if (nRet <= 0) {
      std::string msg("同步Process Job失败!");
      for (auto pj : pjs) {
         auto& issues = pj->issues();
         if (!issues.empty()) {
            msg.append("\n");
            msg.append(pj->id());
            msg.append(":\n");
            for (auto i : issues) {
               msg.append("[");
               msg.append(std::to_string(i.code));
               msg.append("]");
               msg.append(i.text);
               msg.append("\n");
            }
         }
         delete pj;
      }
      pjs.clear();
      AfxMessageBox(msg.c_str());
      return;
   }
   // 继续释放有问题的 ProcessJob
   for (auto pj : pjs) {
      if (!pj->issues().empty()) {
         delete pj;
      }
   }
   pjs.clear();
   nRet = master.setControlJob(*m_pControlJob);
   if (nRet != 0) {
      std::string msg("同步ControlJob失败!");
      auto& issues = m_pControlJob->issues();
      if (!issues.empty()) {
         msg.append("\n");
         for (auto i : issues) {
            msg.append("[");
            msg.append(std::to_string(i.code));
            msg.append("]");
            msg.append(i.text);
            msg.append("\n");
         }
      }
      AfxMessageBox(msg.c_str());
      return;
   }
   // 成功,要判断,同步到slot的glass中,类型等
   for (int p = 0; p < 4; p++) {
      if (m_pjWarps[p].port == -1) continue;
      ASSERT(0 <= m_pjWarps[p].port && m_pjWarps[p].port <= 3);
      SERVO::CLoadPort* pLoadPort = pPorts[m_pjWarps[p].port];
      for (int i = 0; i < SLOT_MAX; ++i) {
         SERVO::CSlot* pSlot = pLoadPort->getSlot(i);
         if (!pSlot) continue;
         SERVO::CGlass* pGlass = dynamic_cast<SERVO::CGlass*>(pSlot->getContext());
         if (pGlass == nullptr) continue;
         SERVO::CJobDataS* pJobDataS = pGlass->getJobDataS();
         if (pJobDataS == nullptr) continue;
         // 设置 Panel ID 和勾选框
         SERVO::CProcessJob* pj = (SERVO::CProcessJob*)m_pjWarps[p].pj;
         int nRecipeID = RecipeManager::getInstance().getIdByPPID(pj->recipeSpec());
         RecipeInfo stRecipeInfo = RecipeManager::getInstance().getRecipeByPPID(pj->recipeSpec());
         std::vector<DeviceRecipe> vecRecipeInfo = stRecipeInfo.vecDeviceList;
         pGlass->setScheduledForProcessing(m_pjWarps[p].checkSlot[i]);
         pGlass->setType(static_cast<SERVO::MaterialsType>(m_pjWarps[p].material[i]));
         pJobDataS->setLotId(pj->getLotId().c_str());
            pJobDataS->setProductId(pj->getProductId().c_str());
            pJobDataS->setOperationId(pj->getOperationId().c_str());
            pJobDataS->setMaterialsType(m_pjWarps[p].material[i]);
            pJobDataS->setMasterRecipe(nRecipeID);
         for (const auto& info : vecRecipeInfo) {
            const std::string& name = info.strDeviceName;
            short nRecipeID = (short)info.nRecipeID;
            if (name == EQ_NAME_EFEM) {
               pJobDataS->setDeviceRecipeId(0, nRecipeID);
            }
            else if (name == EQ_NAME_BONDER1) {
               pJobDataS->setDeviceRecipeId(1, nRecipeID);
            }
            else if (name == EQ_NAME_BONDER2) {
               pJobDataS->setDeviceRecipeId(2, nRecipeID);
            }
            else if (name == EQ_NAME_BAKE_COOLING) {
               pJobDataS->setDeviceRecipeId(3, nRecipeID);
            }
            else if (name == EQ_NAME_VACUUMBAKE) {
               pJobDataS->setDeviceRecipeId(4, nRecipeID);
            }
            else if (name == EQ_NAME_MEASUREMENT) {
               pJobDataS->setDeviceRecipeId(5, nRecipeID);
            }
         }
      }
   }
   // process start
   for (int p = 0; p < 4; p++) {
      if (bProcessStart[p]) {
         pPorts[p]->sendCassetteCtrlCmd(CCC_PROCESS_START, nullptr, 0, 0, 0, nullptr, nullptr);
         Sleep(100);
      }
   }
}