CMainWnd.cpp 27 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019
  1. #include "../pch/pch.h"
  2. #include "CMainWnd.h"
  3. #include "CLoginWnd.h"
  4. #include "../control/ControlEx.h"
  5. #include "../page/CWaimaiOrderInfoPageUI.h"
  6. #include "../page/CDianneiOrderInfoPageUI.h"
  7. #include "../network/CMessagePush.h"
  8. #include "../print/CPosPrinter.h"
  9. #include <urlmon.h>
  10. #include "../wnd/CToastWnd.h"
  11. #include "../tool/CChengzhongWorker.h"
  12. void CMainWnd::Init()
  13. {
  14. //设置店铺名字
  15. CLabelUI* pShopname = static_cast<CLabelUI*>(m_pm.FindControl(_T("main_shopname")));
  16. pShopname->SetText(CLewaimaiString::UTF8ToUnicode(CShopinfo::m_shopinfo.m_shop_name).c_str());
  17. m_curOptionName = L"main_diandan";
  18. m_curOption = static_cast<CControlUI*>(m_pm.FindControl(_T("main_diandan")));
  19. m_pCloseBtn = static_cast<CButtonUI*>(m_pm.FindControl(_T("closebtn")));
  20. m_pMaxBtn = static_cast<CButtonUI*>(m_pm.FindControl(_T("maxbtn")));
  21. m_pRestoreBtn = static_cast<CButtonUI*>(m_pm.FindControl(_T("restorebtn")));
  22. m_pMinBtn = static_cast<CButtonUI*>(m_pm.FindControl(_T("minbtn")));
  23. //默认选择点单页面
  24. this->SwitchPage(DIANDAN);
  25. //开启键盘输入监听
  26. RegKeyboardRawInput();
  27. //登录成功,启动消息和任务处理
  28. m_push = new CMessagePush(m_hWnd);
  29. m_push->Start();
  30. //启动打印队列
  31. CPosPrinterQueue::GetInstance()->StartWork();
  32. //启动称重任务
  33. CChengzhongWorker::GetInstance()->StartWork();
  34. //启动一个线程,开始同步商品图片
  35. std::thread(&CMainWnd::UpdateFoodImage, this).detach();
  36. }
  37. void CMainWnd::UpdateFoodImage()
  38. {
  39. //先判断并创建临时目录
  40. string folderPath = CLewaimaiString::UnicodeToANSI(CSystem::GetProgramDir()) + "\\tmp\\image";
  41. std::wstring ws_folderPath = CLewaimaiString::ANSIToUnicode(folderPath);
  42. if (!CSystem::IsDirExist(ws_folderPath))
  43. {
  44. LOG_INFO("folderPath:" << folderPath.c_str() << ",没有找到对应的目录,即将创建");
  45. CSystem::CreateMultiLevel(folderPath);
  46. }
  47. CSqlite3 sqlite;
  48. std::vector<CFood> foodlist = sqlite.GetFoodByTypeid("0", true);
  49. for (std::vector<CFood>::iterator it = foodlist.begin(); it != foodlist.end(); it++)
  50. {
  51. CFood food = *it;
  52. if (food.goods_img.size() == 0)
  53. {
  54. //没有图片,直接跳过
  55. continue;
  56. }
  57. std::wstring ws_goods_img = food.getImageUrl();
  58. ws_goods_img += L"!max200"; //下载小图
  59. wstring imagePath = food.getImageTmpPath();
  60. if (CSystem::IsFileExist(imagePath))
  61. {
  62. //如果图片文件已经存在,直接跳过
  63. continue;
  64. }
  65. //图片还不存在,开始下载
  66. if (URLDownloadToFile(NULL, ws_goods_img.c_str(), imagePath.c_str(), 0, NULL) == S_OK)
  67. {
  68. //图片下载成功了,发个消息,更新图片
  69. }
  70. else
  71. {
  72. LOG_INFO("URLDownloadToFile Fail,Error"<<GetLastError());
  73. }
  74. }
  75. //再处理商品套餐的图片
  76. std::vector<CFoodpackage> foodpackagelist = sqlite.GetFoodpackages(true);
  77. for (std::vector<CFoodpackage>::iterator it = foodpackagelist.begin(); it != foodpackagelist.end(); it++)
  78. {
  79. CFoodpackage food = *it;
  80. if (food.goods_img.size() == 0)
  81. {
  82. //没有图片,直接跳过
  83. continue;
  84. }
  85. std::wstring ws_goods_img = food.getImageUrl();
  86. ws_goods_img += L"!max200"; //下载小图
  87. wstring imagePath = food.getImageTmpPath();
  88. if (CSystem::IsFileExist(imagePath))
  89. {
  90. //如果图片文件已经存在,直接跳过
  91. continue;
  92. }
  93. //图片还不存在,开始下载
  94. if (URLDownloadToFile(NULL, ws_goods_img.c_str(), imagePath.c_str(), 0, NULL) == S_OK)
  95. {
  96. //图片下载成功了,发个消息,更新图片
  97. }
  98. else
  99. {
  100. LOG_INFO("URLDownloadToFile Fail,Error" << GetLastError());
  101. }
  102. }
  103. }
  104. void CMainWnd::SwitchPage(MainPageName name)
  105. {
  106. if (m_curPageName == name)
  107. {
  108. return;
  109. }
  110. //先删除现在的子对象
  111. CContainerUI* pMainContentLayout = static_cast<CContainerUI*>(m_pm.FindControl(_T("main_content_layout")));
  112. pMainContentLayout->RemoveAll();
  113. //再创建一个对象
  114. CDialogBuilder builder;
  115. CDialogBuilderCallbackEx cb;
  116. CBasePageUI* pChildContainer = NULL;
  117. //注意事项:所有的子页面中的option不能带有selected="true"属性,如果带有这个属性,在下面的Create调用的时候就直接会生成一个selectchanged事件,但是这个时候
  118. //页面才刚刚Create,还没加入到窗口中,相关的变量对应关系也还没设置成功,会导致事件处理对象出现问题
  119. if (name == DIANDAN)
  120. {
  121. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("diandan_page.xml"), (UINT)0, &cb, &m_pm));
  122. }
  123. else if (name == DINGDAN)
  124. {
  125. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("waimaiorder_list_page.xml"), (UINT)0, &cb, &m_pm));
  126. }
  127. else if (name == SHEZHI)
  128. {
  129. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("setting_page.xml"), (UINT)0, &cb, &m_pm));
  130. }
  131. else if (name == HUIYUAN)
  132. {
  133. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("huiyuan_page.xml"), (UINT)0, &cb, &m_pm));
  134. }
  135. else if (name == WAIMAIINFO)
  136. {
  137. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("waimaiorder_info_page.xml"), (UINT)0, &cb, &m_pm));
  138. CWaimaiOrderInfoPageUI* pInfoPage = static_cast<CWaimaiOrderInfoPageUI*>(pChildContainer);
  139. pInfoPage->m_order_id = m_infopage_waimaiorder_id;
  140. pInfoPage->m_order_no = m_infopage_waimaiorder_no;
  141. }
  142. else if (name == DIANNEIDINGDAN)
  143. {
  144. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("dianneiorder_list_page.xml"), (UINT)0, &cb, &m_pm));
  145. }
  146. else if (name == DIANNEIINFO)
  147. {
  148. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("dianneiorder_info_page.xml"), (UINT)0, &cb, &m_pm));
  149. CDianneiOrderInfoPageUI* pInfoPage = static_cast<CDianneiOrderInfoPageUI*>(pChildContainer);
  150. pInfoPage->m_order_id = m_infopage_dianneiorder_id;
  151. }
  152. else if (name == BAOBIAO)
  153. {
  154. pChildContainer = static_cast<CBasePageUI*>(builder.Create(_T("baobiao_page.xml"), (UINT)0, &cb, &m_pm));
  155. }
  156. pChildContainer->SetMainWnd(this);
  157. pMainContentLayout->Add(pChildContainer);
  158. m_curPageName = name;
  159. m_curPageUI = pChildContainer;
  160. pChildContainer->InitShow();
  161. }
  162. LRESULT CMainWnd::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  163. {
  164. LONG styleValue = ::GetWindowLong(*this, GWL_STYLE);
  165. styleValue &= ~WS_CAPTION;
  166. ::SetWindowLong(*this, GWL_STYLE, styleValue | WS_CLIPSIBLINGS | WS_CLIPCHILDREN);
  167. m_pm.Init(m_hWnd);
  168. CDialogBuilder builder;
  169. CDialogBuilderCallbackEx cb;
  170. CControlUI* pRoot = builder.Create(_T("main.xml"), (UINT)0, &cb, &m_pm);
  171. ASSERT(pRoot && "Failed to parse XML");
  172. m_pm.AttachDialog(pRoot);
  173. m_pm.AddNotifier(this);
  174. Init();
  175. return 0;
  176. }
  177. void CMainWnd::Notify(TNotifyUI& msg)
  178. {
  179. if(msg.sType == _T("windowinit"))
  180. {
  181. }
  182. else if(msg.sType == _T("click"))
  183. {
  184. HandleClickMsg(msg);
  185. }
  186. else if(msg.sType == _T("selectchanged"))
  187. {
  188. HandleSelectChangeMsg(msg);
  189. }
  190. else if(msg.sType == _T("itemclick"))
  191. {
  192. }
  193. else if(msg.sType == _T("itemactivate"))
  194. {
  195. }
  196. else if(msg.sType == _T("itemselect"))
  197. {
  198. HandleItemSelectMsg(msg);
  199. }
  200. else if (msg.sType == _T("textchanged"))
  201. {
  202. //编辑框内容改变事件
  203. HandleTextChangedMsg(msg);
  204. }
  205. }
  206. void CMainWnd::HandleClickMsg(TNotifyUI& msg)
  207. {
  208. if(msg.pSender == m_pCloseBtn)
  209. {
  210. if(CSetting::GetParam("setting_is_close_min") == "1")
  211. {
  212. AddTrayIcon();
  213. }
  214. else
  215. {
  216. PostQuitMessage(0);
  217. }
  218. return;
  219. }
  220. else if(msg.pSender == m_pMinBtn)
  221. {
  222. SendMessage(WM_SYSCOMMAND, SC_MINIMIZE, 0);
  223. return;
  224. }
  225. else if(msg.pSender == m_pMaxBtn)
  226. {
  227. SendMessage(WM_SYSCOMMAND, SC_MAXIMIZE, 0);
  228. return;
  229. }
  230. else if(msg.pSender == m_pRestoreBtn)
  231. {
  232. SendMessage(WM_SYSCOMMAND, SC_RESTORE, 0);
  233. return;
  234. }
  235. CDuiString name = msg.pSender->GetName();
  236. if(name == _T("quitbtn"))
  237. {
  238. /*Close()*/
  239. PostQuitMessage(0); // 因为activex的原因,使用close可能会出现错误
  240. }
  241. else
  242. {
  243. this->m_curPageUI->HandleClickMsg(msg);
  244. }
  245. }
  246. /**
  247. * 这个处理Option的切换
  248. */
  249. void CMainWnd::HandleSelectChangeMsg(TNotifyUI& msg)
  250. {
  251. CDuiString name = msg.pSender->GetName();
  252. std::wstring ws_name = name;
  253. if (ws_name == _T("main_diandan") || ws_name == _T("main_dingdan") || ws_name == _T("main_huiyuan") || ws_name == _T("main_baobiao") || ws_name == _T("main_jiaoban") || ws_name == _T("main_setting") || ws_name == _T("main_gengduo"))
  254. {
  255. if (m_curOptionName != ws_name)
  256. {
  257. //表示切换了tab
  258. m_curOption->SetBkColor(0x00000000);
  259. msg.pSender->SetBkColor(0xFF3CB371);
  260. m_curOptionName = ws_name;
  261. m_curOption = static_cast<CControlUI*>(msg.pSender);
  262. }
  263. }
  264. //先判断主页面的tab
  265. if (name == _T("main_diandan"))
  266. {
  267. this->SwitchPage(DIANDAN);
  268. }
  269. else if(name == _T("main_dingdan"))
  270. {
  271. this->SwitchPage(DIANNEIDINGDAN);
  272. }
  273. else if (name == L"main_huiyuan")
  274. {
  275. this->SwitchPage(HUIYUAN);
  276. }
  277. else if (name == L"main_baobiao")
  278. {
  279. this->SwitchPage(BAOBIAO);
  280. }
  281. else if (name == L"main_jiaoban")
  282. {
  283. }
  284. else if(name == _T("main_setting"))
  285. {
  286. this->SwitchPage(SHEZHI);
  287. }
  288. else if (name == L"main_gengduo")
  289. {
  290. }
  291. else
  292. {
  293. //除了主界面固定区域的事件,其他的事件全部分发给子页面
  294. this->m_curPageUI->HandleSelectChangeMsg(msg);
  295. }
  296. }
  297. void CMainWnd::HandleItemSelectMsg(TNotifyUI& msg)
  298. {
  299. this->m_curPageUI->HandleItemSelectMsg(msg);
  300. }
  301. void CMainWnd::HandleTextChangedMsg(TNotifyUI& msg)
  302. {
  303. this->m_curPageUI->HandleTextChangedMsg(msg);
  304. }
  305. LRESULT CMainWnd::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
  306. {
  307. LRESULT lRes = 0;
  308. BOOL bHandled = TRUE;
  309. switch(uMsg)
  310. {
  311. case WM_CREATE:
  312. lRes = OnCreate(uMsg, wParam, lParam, bHandled);
  313. break;
  314. case WM_CLOSE:
  315. lRes = OnClose(uMsg, wParam, lParam, bHandled);
  316. break;
  317. case WM_DESTROY:
  318. lRes = OnDestroy(uMsg, wParam, lParam, bHandled);
  319. break;
  320. case WM_NCACTIVATE:
  321. lRes = OnNcActivate(uMsg, wParam, lParam, bHandled);
  322. break;
  323. case WM_NCCALCSIZE:
  324. lRes = OnNcCalcSize(uMsg, wParam, lParam, bHandled);
  325. break;
  326. case WM_NCPAINT:
  327. lRes = OnNcPaint(uMsg, wParam, lParam, bHandled);
  328. break;
  329. case WM_NCHITTEST:
  330. lRes = OnNcHitTest(uMsg, wParam, lParam, bHandled);
  331. break;
  332. case WM_SIZE:
  333. lRes = OnSize(uMsg, wParam, lParam, bHandled);
  334. break;
  335. case WM_GETMINMAXINFO:
  336. lRes = OnGetMinMaxInfo(uMsg, wParam, lParam, bHandled);
  337. break;
  338. case WM_SYSCOMMAND:
  339. lRes = OnSysCommand(uMsg, wParam, lParam, bHandled);
  340. break;
  341. case WM_SHOWTASK:
  342. lRes = OnTrayIcon(uMsg, wParam, lParam, bHandled);
  343. break;
  344. case WM_INPUT:
  345. lRes = OnInput(uMsg, wParam, lParam, bHandled);
  346. break;
  347. case WM_KEYDOWN:
  348. lRes = OnKeydown(uMsg, wParam, lParam, bHandled);
  349. break;
  350. case WM_KILLFOCUS:
  351. break;
  352. default:
  353. bHandled = FALSE;
  354. }
  355. if(bHandled)
  356. {
  357. return lRes;
  358. }
  359. bool ret = this->HandleCustomMessage(uMsg, wParam, lParam);
  360. if (ret)
  361. {
  362. //自定义消息已经处理
  363. return 0;
  364. }
  365. if(m_pm.MessageHandler(uMsg, wParam, lParam, lRes))
  366. {
  367. return lRes;
  368. }
  369. return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
  370. }
  371. bool CMainWnd::HandleCustomMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
  372. {
  373. if (uMsg == WM_LOGIN_AGAIN_OUT)
  374. {
  375. LoginOut(2);
  376. return true;
  377. }
  378. else
  379. {
  380. //走到这里,说明消息还没有被处理,那就不是MainWnd自己要处理的消息,应该是自定义的消息,交给各个子页面自己处理去
  381. if (m_curPageUI != NULL)
  382. {
  383. bool ret = this->m_curPageUI->HandleCustomMessage(uMsg, wParam, lParam);
  384. return ret;
  385. }
  386. }
  387. return false;
  388. }
  389. LRESULT CMainWnd::MessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, bool& bHandled)
  390. {
  391. return false;
  392. }
  393. LRESULT CMainWnd::OnSysCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  394. {
  395. // 有时会在收到WM_NCDESTROY后收到wParam为SC_CLOSE的WM_SYSCOMMAND
  396. if(wParam == SC_CLOSE)
  397. {
  398. ::PostQuitMessage(0L);
  399. bHandled = TRUE;
  400. return 0;
  401. }
  402. BOOL bZoomed = ::IsZoomed(*this);
  403. LRESULT lRes = CWindowWnd::HandleMessage(uMsg, wParam, lParam);
  404. if(::IsZoomed(*this) != bZoomed)
  405. {
  406. if(!bZoomed)
  407. {
  408. CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(_T("maxbtn")));
  409. if(pControl)
  410. {
  411. pControl->SetVisible(false);
  412. }
  413. pControl = static_cast<CControlUI*>(m_pm.FindControl(_T("restorebtn")));
  414. if(pControl)
  415. {
  416. pControl->SetVisible(true);
  417. }
  418. }
  419. else
  420. {
  421. CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(_T("maxbtn")));
  422. if(pControl)
  423. {
  424. pControl->SetVisible(true);
  425. }
  426. pControl = static_cast<CControlUI*>(m_pm.FindControl(_T("restorebtn")));
  427. if(pControl)
  428. {
  429. pControl->SetVisible(false);
  430. }
  431. }
  432. }
  433. return lRes;
  434. }
  435. LRESULT CMainWnd::OnGetMinMaxInfo(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  436. {
  437. int primaryMonitorWidth = ::GetSystemMetrics(SM_CXSCREEN);
  438. int primaryMonitorHeight = ::GetSystemMetrics(SM_CYSCREEN);
  439. MONITORINFO oMonitor = {};
  440. oMonitor.cbSize = sizeof(oMonitor);
  441. ::GetMonitorInfo(::MonitorFromWindow(*this, MONITOR_DEFAULTTOPRIMARY), &oMonitor);
  442. CDuiRect rcWork = oMonitor.rcWork;
  443. rcWork.Offset(-oMonitor.rcMonitor.left, -oMonitor.rcMonitor.top);
  444. if(rcWork.right > primaryMonitorWidth)
  445. {
  446. rcWork.right = primaryMonitorWidth;
  447. }
  448. if(rcWork.bottom > primaryMonitorHeight)
  449. {
  450. rcWork.right = primaryMonitorHeight;
  451. }
  452. LPMINMAXINFO lpMMI = (LPMINMAXINFO)lParam;
  453. lpMMI->ptMaxPosition.x = rcWork.left;
  454. lpMMI->ptMaxPosition.y = rcWork.top;
  455. lpMMI->ptMaxSize.x = rcWork.right;
  456. lpMMI->ptMaxSize.y = rcWork.bottom;
  457. bHandled = FALSE;
  458. return 0;
  459. }
  460. LRESULT CMainWnd::OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  461. {
  462. SIZE szRoundCorner = m_pm.GetRoundCorner();
  463. if(!::IsIconic(*this) && (szRoundCorner.cx != 0 || szRoundCorner.cy != 0))
  464. {
  465. CDuiRect rcWnd;
  466. ::GetWindowRect(*this, &rcWnd);
  467. rcWnd.Offset(-rcWnd.left, -rcWnd.top);
  468. rcWnd.right++;
  469. rcWnd.bottom++;
  470. HRGN hRgn = ::CreateRoundRectRgn(rcWnd.left, rcWnd.top, rcWnd.right, rcWnd.bottom, szRoundCorner.cx, szRoundCorner.cy);
  471. ::SetWindowRgn(*this, hRgn, TRUE);
  472. ::DeleteObject(hRgn);
  473. }
  474. bHandled = FALSE;
  475. return 0;
  476. }
  477. LRESULT CMainWnd::OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  478. {
  479. bHandled = FALSE;
  480. return 0;
  481. }
  482. /*
  483. *这个是窗口被销毁的时候调用的
  484. **/
  485. LRESULT CMainWnd::OnDestroy(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  486. {
  487. //直接退出程序,或者退出登录都会执行这个
  488. m_push->Stop();
  489. //停止打印任务
  490. CPosPrinterQueue::GetInstance()->StopWork();
  491. //停止称重任务
  492. CChengzhongWorker::GetInstance()->StopWork();
  493. bHandled = FALSE;
  494. return 0;
  495. }
  496. LRESULT CMainWnd::OnNcActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  497. {
  498. if(::IsIconic(*this))
  499. {
  500. bHandled = FALSE;
  501. }
  502. return (wParam == 0) ? TRUE : FALSE;
  503. }
  504. LRESULT CMainWnd::OnNcCalcSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  505. {
  506. return 0;
  507. }
  508. LRESULT CMainWnd::OnNcPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  509. {
  510. return 0;
  511. }
  512. LRESULT CMainWnd::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  513. {
  514. POINT pt;
  515. pt.x = GET_X_LPARAM(lParam);
  516. pt.y = GET_Y_LPARAM(lParam);
  517. ::ScreenToClient(*this, &pt);
  518. RECT rcClient;
  519. ::GetClientRect(*this, &rcClient);
  520. if(!::IsZoomed(*this))
  521. {
  522. RECT rcSizeBox = m_pm.GetSizeBox();
  523. if(pt.y < rcClient.top + rcSizeBox.top)
  524. {
  525. if(pt.x < rcClient.left + rcSizeBox.left)
  526. {
  527. return HTTOPLEFT;
  528. }
  529. if(pt.x > rcClient.right - rcSizeBox.right)
  530. {
  531. return HTTOPRIGHT;
  532. }
  533. return HTTOP;
  534. }
  535. else if(pt.y > rcClient.bottom - rcSizeBox.bottom)
  536. {
  537. if(pt.x < rcClient.left + rcSizeBox.left)
  538. {
  539. return HTBOTTOMLEFT;
  540. }
  541. if(pt.x > rcClient.right - rcSizeBox.right)
  542. {
  543. return HTBOTTOMRIGHT;
  544. }
  545. return HTBOTTOM;
  546. }
  547. if(pt.x < rcClient.left + rcSizeBox.left)
  548. {
  549. return HTLEFT;
  550. }
  551. if(pt.x > rcClient.right - rcSizeBox.right)
  552. {
  553. return HTRIGHT;
  554. }
  555. }
  556. RECT rcCaption = m_pm.GetCaptionRect();
  557. if(pt.x >= rcClient.left + rcCaption.left && pt.x < rcClient.right - rcCaption.right \
  558. && pt.y >= rcCaption.top && pt.y < rcCaption.bottom)
  559. {
  560. CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(pt));
  561. if(pControl && _tcscmp(pControl->GetClass(), DUI_CTR_BUTTON) != 0 &&
  562. _tcscmp(pControl->GetClass(), DUI_CTR_OPTION) != 0 &&
  563. _tcscmp(pControl->GetClass(), DUI_CTR_TEXT) != 0)
  564. {
  565. return HTCAPTION;
  566. }
  567. }
  568. return HTCLIENT;
  569. }
  570. LRESULT CMainWnd::OnInput(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  571. {
  572. UINT dwSize = 0;
  573. GetRawInputData((HRAWINPUT)lParam, (UINT)RID_INPUT, NULL, &dwSize, sizeof(RAWINPUTHEADER));//取数据,第一次调用函数将获取需要的字节大小
  574. LPBYTE lpbBuffer = new BYTE[dwSize];//分配指定的缓冲区大小
  575. GetRawInputData((HRAWINPUT)lParam, (UINT)RID_INPUT, (LPVOID)lpbBuffer, (PUINT)&dwSize, (UINT)sizeof(RAWINPUTHEADER));//第二次调用获取原始输入数据,读入lpbBuffer
  576. RAWINPUT * raw = (RAWINPUT *)lpbBuffer;
  577. if (raw->header.dwType == RIM_TYPEKEYBOARD)//这里可以截获所有键盘信息,如需区分不同的键盘输入信息,可以通过设备句柄判断。
  578. {
  579. if (raw->data.keyboard.Message == WM_KEYDOWN)
  580. {
  581. }
  582. else if (raw->data.keyboard.Message == WM_KEYUP)
  583. {
  584. char keytext[10] = { 0 };
  585. BYTE state[256] = { 0 };
  586. //通过虚拟键盘码得到名字
  587. ToAscii(raw->data.keyboard.VKey, raw->data.keyboard.MakeCode, state, (LPWORD)keytext, 0);
  588. LOG_INFO("vkey:" << raw->data.keyboard.VKey<<", keytext:"<<keytext);
  589. if (raw->data.keyboard.VKey >= 48 && raw->data.keyboard.VKey <= 122)
  590. {
  591. char keytext[10] = { 0 };
  592. BYTE state[256] = { 0 };
  593. //通过虚拟键盘码得到名字
  594. ToAscii(raw->data.keyboard.VKey, raw->data.keyboard.MakeCode, state, (LPWORD)keytext, 0);
  595. //这里是数字和英文字母之间的字符,包含部分标点
  596. if (m_is_start_catch == false)
  597. {
  598. //第一次点击,还无法判定是人工还是机器输入的,先保存下来
  599. m_is_start_catch = true;
  600. m_catch_string = "";
  601. m_is_next_daxie = false;
  602. m_catch_string += string(keytext);
  603. LOG_INFO("起始字符:" << keytext << ",累积:" << m_catch_string.c_str());
  604. m_last_catch_clock = clock();
  605. }
  606. else
  607. {
  608. //已经是在抓取过程中了
  609. clock_t now = clock();
  610. double total_t = (double)(now - m_last_catch_clock) / CLOCKS_PER_SEC;
  611. if (total_t > 0.05)
  612. {
  613. LOG_INFO("发现人工输入");
  614. //如果超过了50ms,说明是前面一个字符是人工输入的了,不能代表现在这个字符是人工输入的,把之前的字符清空,但是目前的这个字符还是要保存
  615. m_catch_string = "";
  616. m_is_next_daxie = false;
  617. //把当前字符保存下来
  618. m_catch_string += string(keytext);
  619. LOG_INFO("起始字符:" << keytext << ",累积:" << m_catch_string.c_str());
  620. m_last_catch_clock = clock();
  621. }
  622. else
  623. {
  624. //这种说明前一个和当前的字符,都是机器输入的,都保存下来
  625. std::string s_keytext = keytext;
  626. if (m_is_next_daxie == true)
  627. {
  628. //要把抓取的这个字符转为大写
  629. transform(s_keytext.begin(), s_keytext.end(), s_keytext.begin(), ::toupper);
  630. }
  631. //转换完了,再把状态还原
  632. m_is_next_daxie = false;
  633. m_catch_string += s_keytext;
  634. LOG_INFO("抓取字符:" << s_keytext.c_str() << ",累积:" << m_catch_string.c_str());
  635. m_last_catch_clock = clock();
  636. }
  637. }
  638. }
  639. else if (raw->data.keyboard.VKey == 13)
  640. {
  641. //输入了enter
  642. if (m_is_start_catch == true)
  643. {
  644. //正在抓取中,判断enter才有意义,否则没有意义
  645. clock_t now = clock();
  646. double total_t = (double)(now - m_last_catch_clock) / CLOCKS_PER_SEC;
  647. if (total_t > 0.05)
  648. {
  649. LOG_INFO("抓取enter人工");
  650. //如果超过了50ms,说明前面一个字符是人工输入的,那么相当于这次直接输入了一个enter,也没意义
  651. m_is_start_catch = false;
  652. m_catch_string = "";
  653. m_is_next_daxie = false;
  654. }
  655. else
  656. {
  657. LOG_INFO("抓取enter结束");
  658. //这种说明前一个字符和当前字符都是机器输入的,可以判定输入结束
  659. std::string last = m_catch_string;
  660. //对抓取结果,进行处理
  661. m_curPageUI->HandleTextCapture(last);
  662. //处理完成后,清空重来
  663. m_is_start_catch = false;
  664. m_catch_string = "";
  665. m_is_next_daxie = false;
  666. }
  667. }
  668. else
  669. {
  670. //还没开始抓取就输入了enter,没有意义,重新开始
  671. m_is_start_catch = false;
  672. m_catch_string = "";
  673. m_is_next_daxie = false;
  674. }
  675. }
  676. else if (raw->data.keyboard.VKey == 16)
  677. {
  678. //这个是大小写切换的按键,扫描大写字母的时候会用到,每个大写字母都是由2个组成,先来个16,再来个对应小写字母的asicc
  679. if (m_is_start_catch == false)
  680. {
  681. LOG_INFO("抓取第1个16");
  682. //说明第一个字符就是大写字母,暂时还无法判断是机器还是人工输入的,先开启抓取
  683. m_is_start_catch = true;
  684. m_catch_string = "";
  685. m_is_next_daxie = true;
  686. m_last_catch_clock = clock();
  687. }
  688. else
  689. {
  690. //已经在抓取中了
  691. clock_t now = clock();
  692. double total_t = (double)(now - m_last_catch_clock) / CLOCKS_PER_SEC;
  693. if (total_t > 0.05)
  694. {
  695. LOG_INFO("抓取16人工");
  696. //如果超过了50ms,只能说明之前的字符是人工输入的,不能代表目前的字符是人工输入的,把之前的清空
  697. m_catch_string = "";
  698. m_is_next_daxie = true;
  699. m_last_catch_clock = clock();
  700. }
  701. else
  702. {
  703. LOG_INFO("抓取16");
  704. //这个说明之前的和当前的都是机器输入的
  705. m_is_next_daxie = true;
  706. m_last_catch_clock = clock();
  707. }
  708. }
  709. }
  710. else
  711. {
  712. LOG_INFO("抓取结束,重新开始");
  713. //输入了无意义的字符,这种直接忽略,重新开始
  714. m_is_start_catch = false;
  715. m_catch_string = "";
  716. m_is_next_daxie = false;
  717. }
  718. }
  719. }
  720. delete[] lpbBuffer;
  721. bHandled = true;
  722. return 0;
  723. }
  724. LRESULT CMainWnd::OnKeydown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  725. {
  726. if (wParam == VK_RETURN)
  727. {
  728. //这里目前主要作用是,拦截按钮自带的焦点enter响应
  729. bHandled = true;
  730. return 0;
  731. }
  732. else if (wParam == VK_ESCAPE)
  733. {
  734. bHandled = true;
  735. return 0;
  736. }
  737. //其他的按键事件,不能返回0
  738. bHandled = false;
  739. return -1;
  740. }
  741. void CMainWnd::AddTrayIcon()
  742. {
  743. memset(&m_trayIcon, 0, sizeof(NOTIFYICONDATA));
  744. m_trayIcon.cbSize = sizeof(NOTIFYICONDATA);
  745. m_trayIcon.hIcon = ::LoadIcon(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON_DUILIB));
  746. m_trayIcon.hWnd = m_hWnd;
  747. lstrcpy(m_trayIcon.szTip, _T("智铺子收银系统"));
  748. m_trayIcon.uCallbackMessage = WM_SHOWTASK;
  749. m_trayIcon.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP;
  750. Shell_NotifyIcon(NIM_ADD, &m_trayIcon);
  751. ShowWindow(SW_HIDE);
  752. }
  753. LRESULT CMainWnd::OnTrayIcon(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  754. {
  755. //如果在图标中单击左键则还原
  756. if(lParam == WM_LBUTTONDOWN)
  757. {
  758. //删除托盘图标
  759. Shell_NotifyIcon(NIM_DELETE, &m_trayIcon);
  760. //显示主窗口
  761. ShowWindow(SW_SHOWNORMAL);
  762. //窗口最大化
  763. SendMessage(WM_SYSCOMMAND, SC_MAXIMIZE, 0);
  764. }
  765. //如果在图标中单击右键则弹出声明式菜单
  766. if(lParam == WM_RBUTTONDOWN)
  767. {
  768. //获取鼠标坐标
  769. POINT pt;
  770. GetCursorPos(&pt);
  771. //右击后点别地可以清除“右击出来的菜单”
  772. SetForegroundWindow(m_hWnd);
  773. //托盘菜单 win32程序使用的是HMENU,如果是MFC程序可以使用CMenu
  774. HMENU hMenu;
  775. //生成托盘菜单
  776. hMenu = CreatePopupMenu();
  777. //添加菜单,关键在于设置的一个标识符 WM_ONCLOSE 点击后会用到
  778. AppendMenu(hMenu, MF_STRING, WM_ONCLOSE, _T("退出"));
  779. //弹出菜单,并把用户所选菜单项的标识符返回
  780. int cmd = TrackPopupMenu(hMenu, TPM_RETURNCMD, pt.x, pt.y, NULL, m_hWnd, NULL);
  781. //如果标识符是WM_ONCLOSE则关闭
  782. if(cmd == WM_ONCLOSE)
  783. {
  784. m_trayIcon.hIcon = NULL;
  785. Shell_NotifyIcon(NIM_DELETE, &m_trayIcon);
  786. //退出程序
  787. ::PostQuitMessage(0);
  788. }
  789. }
  790. bHandled = true;
  791. return 0;
  792. }
  793. void CMainWnd::LoginOut(int mode)
  794. {
  795. CSetting::SetParam("setting_is_auto_login", "0", true);
  796. CLoginWnd* pLogin = new CLoginWnd();
  797. if(pLogin == NULL)
  798. {
  799. return;
  800. }
  801. //设置模式
  802. pLogin->SetMode(mode);
  803. pLogin->Create(NULL, _T("智铺子收银系统登录"), UI_WNDSTYLE_DIALOG, 0, 0, 0, 0, 0, NULL);
  804. pLogin->SetIcon(IDI_ICON_DUILIB);
  805. pLogin->CenterWindow();
  806. ::ShowWindow(*pLogin, SW_SHOWNORMAL);
  807. Close();
  808. }
  809. CMessagePush* CMainWnd::getMessagePush()
  810. {
  811. return m_push;
  812. }
  813. void CMainWnd::SetInfopageWaimaiorderParam(std::string order_id, std::string order_no)
  814. {
  815. m_infopage_waimaiorder_id = order_id;
  816. m_infopage_waimaiorder_no = order_no;
  817. }
  818. void CMainWnd::SetInfopageDianneiorderParam(std::string order_id)
  819. {
  820. m_infopage_dianneiorder_id = order_id;
  821. }
  822. void CMainWnd::ShowToast(std::wstring toast_value)
  823. {
  824. CToastWnd* pYouhuiDlg = new CToastWnd();
  825. if (pYouhuiDlg != NULL)
  826. {
  827. pYouhuiDlg->Create(m_pm.GetPaintWindow(), _T(""), UI_WNDSTYLE_DIALOG, WS_EX_WINDOWEDGE);
  828. pYouhuiDlg->SetIcon(IDI_ICON_DUILIB);
  829. pYouhuiDlg->CenterWindow();
  830. pYouhuiDlg->SetToast(toast_value);
  831. pYouhuiDlg->ShowWindow();
  832. }
  833. }
  834. //全局监听键盘输入
  835. void CMainWnd::RegKeyboardRawInput()
  836. {
  837. RAWINPUTDEVICE rawInputDevice[1];
  838. rawInputDevice[0].usUsagePage = 0x01;//设备类
  839. rawInputDevice[0].usUsage = 0x06;//设备类内的具体设备
  840. rawInputDevice[0].dwFlags = RIDEV_INPUTSINK;//意味着即使窗口失去焦点位置,仍然会一直接收输入消息
  841. rawInputDevice[0].hwndTarget = m_hWnd;
  842. if (RegisterRawInputDevices(rawInputDevice, 1, sizeof(rawInputDevice[0])) == FALSE)
  843. {
  844. printf("RegisterRawInputDevices failed");
  845. }
  846. }