| | |
| | | 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() |
| | | |
| | | |
| | |
| | | |
| | | 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() |
| | |
| | | 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(false ? 2 : 1), TVIS_STATEIMAGEMASK); |
| | | m_tree.SetItemState(hItem, INDEXTOSTATEIMAGEMASK(item.addToCj ? 2 : 1), TVIS_STATEIMAGEMASK); |
| | | } |
| | | m_tree.Expand(hRoot, TVE_EXPAND); |
| | | } |
| | |
| | | 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)) { |
| | | |
| | | } |
| | | } |
| | | else if (m_tree.GetParentItem(hParent) == nullptr) { |
| | | if (0 == ShowPage(1)) { |
| | | PJWarp* pjWarp = (PJWarp*)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(pjWarp, 1); |
| | | |
| | | ((CCjPage2*)m_pages[1])->SetPjWarps(m_pjWarps); |
| | | m_pages[1]->SetContext(pjWarp, 1); |
| | | } |
| | | 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() |
| | |
| | | for (int i = 0; i < 4; i++) { |
| | | sprintf_s(szBuffer, 256, "PJ%03d", i + 1); |
| | | SERVO::CProcessJob* pj = new SERVO::CProcessJob(std::string(szBuffer)); |
| | | PJWarp pjWarp; |
| | | 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); |
| | | } |
| | | } |
| | | } |
| | | } |
| | | |
| | | void CControlJobManagerDlg::UpProcessJobId(PJWarp* pjWarp) |
| | |
| | | m_state.pjWarps = m_pjWarps; |
| | | m_bHasState = true; |
| | | } |
| | | |
| | | void CControlJobManagerDlg::OnBnClickedButtonBathCompletion() |
| | | { |
| | | // 先应用 |
| | | 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; |
| | | } |
| | | |
| | | |
| | | |
| | | auto& master = theApp.m_model.getMaster(); |
| | | |
| | | 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; |
| | | SERVO::CProcessJob * pj = new SERVO::CProcessJob(pScr->id()); |
| | | pj->setRecipe(SERVO::RecipeMethod::NoTuning, pScr->recipeSpec()); |
| | | |
| | | std::vector<SERVO::CarrierSlotInfo> carriers; |
| | | SERVO::CarrierSlotInfo csi; |
| | | csi.carrierId = "Port" + std::to_string(item.port + 1); |
| | | for (int i = 0; i < 8; i++) { |
| | | if (item.checkSlot[i]) { |
| | | csi.slots.push_back(i); |
| | | } |
| | | } |
| | | carriers.push_back(csi); |
| | | pj->setCarriers(carriers); |
| | | pjs.push_back(pj); |
| | | |
| | | m_pControlJob->addPJ(pScr->id()); |
| | | } |
| | | |
| | | |
| | | if (pjs.empty()) { |
| | | AfxMessageBox(_T("没有需要进行工艺处理的Process Job!\n可能未选择Port或选择任何物料。")); |
| | | return; |
| | | } |
| | | |
| | | |
| | | m_pControlJob->setPJs(pjs); |
| | | int nRet = master.setProcessJobs(pjs); |
| | | master.setControlJob(*m_pControlJob); |
| | | } |