CValueWnd.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. #include "../pch/pch.h"
  2. #include "CValueWnd.h"
  3. #include "../tool/CComHelper.h"
  4. #include "../tool/CSerialPort.h"
  5. void CValueWnd::Notify(TNotifyUI& msg)
  6. {
  7. if(msg.sType == _T("click"))
  8. {
  9. if(msg.pSender->GetName() == _T("closebtn"))
  10. {
  11. PostQuitMessage(0);
  12. return;
  13. }
  14. }
  15. }
  16. void CValueWnd::Init()
  17. {
  18. SetWindowPos(m_hWnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE);
  19. BOOL isAdmin = CSystem::IsRunasAdmin();
  20. if(isAdmin)
  21. {
  22. //MessageBox(m_hWnd, L"管理员", L"111", MB_OK);
  23. }
  24. else
  25. {
  26. //MessageBox(m_hWnd, L"非管理员", L"111", MB_OK);
  27. }
  28. SetHwnd(m_hWnd);
  29. BOOL ret = InstallHook();
  30. LONG id = GetWindowLongW((HWND)0x00211108, GWL_ID);
  31. LONG b = id;
  32. TCHAR szTitle[MAX_PATH];
  33. LPWSTR lpWindowName = szTitle;
  34. ::SendMessageW((HWND)0x00081342, WM_GETTEXT, (WPARAM)(MAX_PATH), (LPARAM)lpWindowName);
  35. wstring a = lpWindowName;
  36. std::thread(&CValueWnd::ReadValue, this).detach();
  37. std::thread(&CValueWnd::WatchWnd, this).detach();
  38. }
  39. void CValueWnd::ReadValue()
  40. {
  41. //读取串口的模式
  42. //CComHelper helper;
  43. //std::vector<std::wstring> comVector = helper.getComPort();
  44. //for (std::vector<std::wstring>::iterator it = comVector.begin(); it != comVector.end(); it++)
  45. //{
  46. // std::wstring comName = *it;
  47. // LOG_INFO("comName:" << comName.c_str());
  48. //}
  49. //int nReadType = 1;
  50. //if (nReadType == 1)
  51. //{
  52. // std::wstring com_name = L"COM2";
  53. // CSerialPort serial(com_name.c_str(), 2400);
  54. // bool ret = serial.openComm();
  55. // if (!ret)
  56. // {
  57. // LOG_INFO("open com failed!");
  58. // return;
  59. // }
  60. // while (true)
  61. // {
  62. // //开始读取串口的数据
  63. // char a[1024] = { 0 };
  64. // serial.readFromComm(a, 1024);
  65. // LOG_INFO("a:" << a);
  66. // Sleep(200);
  67. // }
  68. // serial.closeComm();
  69. //}
  70. FindValueWnd(L"");
  71. }
  72. void CValueWnd::WatchWnd()
  73. {
  74. while(true)
  75. {
  76. if(m_watchWnd != NULL)
  77. {
  78. char szTitle[MAX_PATH] = { 0 };
  79. LPSTR lpWindowName = szTitle;
  80. std::string value;
  81. int ret = ::SendMessageA(m_watchWnd, WM_GETTEXT, (WPARAM)(MAX_PATH), (LPARAM)lpWindowName);
  82. if(ret == 0)
  83. {
  84. //说明m_watchWnd已经失效了
  85. value = "0.00";
  86. m_watchWnd = NULL;
  87. }
  88. else
  89. {
  90. value = lpWindowName;
  91. }
  92. if(value != m_watchValue)
  93. {
  94. m_watchValue = value;
  95. SendMessage(WM_WATCH_WND_UPDATE, NULL, NULL);
  96. }
  97. }
  98. else
  99. {
  100. //根据上次保存的记录,来查找符合条件的窗口
  101. if(g_valuewnd_vector.size() > 0)
  102. {
  103. ValueWnd lastWnd = g_valuewnd_vector[0];
  104. //获得进程名字
  105. std::wstring processName = lastWnd.processName;
  106. //根据进程名字,找到对应的主窗口
  107. HWND mainWnd = GetWndHwnd(processName.c_str(), lastWnd.wndTitle);
  108. if(mainWnd == NULL)
  109. {
  110. //说明这个进程还没有启动
  111. m_watchWnd = NULL;
  112. }
  113. else
  114. {
  115. //根据Z序查找目标窗口
  116. char szTitle[MAX_PATH] = { 0 };
  117. int nMaxCount = MAX_PATH;
  118. LPSTR lpWindowName = szTitle;
  119. GetWindowTextA(mainWnd, lpWindowName, nMaxCount);
  120. if (string(lpWindowName) != lastWnd.wndTitle)
  121. {
  122. //这次根据进程名字找到的主窗口,窗口名字和上次的不一样,无效
  123. m_watchWnd = NULL;
  124. }
  125. else
  126. {
  127. ValueWnd newWnd;
  128. newWnd.wndTitle = lpWindowName;
  129. newWnd.TopWnd = mainWnd;
  130. newWnd.z_order = lastWnd.z_order;
  131. newWnd.processName = lastWnd.processName;
  132. newWnd.selfWnd = NULL;
  133. g_zorder = 0;
  134. EnumChildWindows(mainWnd, EnumMainwndChildProc, (LPARAM)(&newWnd));
  135. if (newWnd.selfWnd != NULL)
  136. {
  137. //找到了
  138. m_watchWnd = newWnd.selfWnd;
  139. }
  140. }
  141. }
  142. }
  143. }
  144. Sleep(200);
  145. }
  146. }
  147. LRESULT CValueWnd::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  148. {
  149. LONG styleValue = ::GetWindowLong(*this, GWL_STYLE);
  150. styleValue &= ~WS_CAPTION;
  151. ::SetWindowLong(*this, GWL_STYLE, styleValue | WS_CLIPSIBLINGS | WS_CLIPCHILDREN);
  152. // 把自己的窗口句柄与窗口绘制管理器挂接在一起
  153. m_pm.Init(m_hWnd);
  154. m_pm.AddPreMessageFilter(this);
  155. CDialogBuilder builder;
  156. CControlUI* pRoot = builder.Create(_T("value.xml"), (UINT)0, NULL, &m_pm);
  157. ASSERT(pRoot && "Failed to parse XML");
  158. // 把这些控件绘制到本窗口上
  159. m_pm.AttachDialog(pRoot);
  160. // 把自己加入到CPaintManagerUI的m_aNotifiers数组中,用于处理Notify函数
  161. m_pm.AddNotifier(this);
  162. Init();
  163. return 0;
  164. }
  165. LRESULT CValueWnd::OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  166. {
  167. bHandled = FALSE;
  168. return 0;
  169. }
  170. /*
  171. *这个是窗口被销毁的时候调用的
  172. **/
  173. LRESULT CValueWnd::OnDestroy(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  174. {
  175. bHandled = FALSE;
  176. return 0;
  177. }
  178. LRESULT CValueWnd::OnNcActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  179. {
  180. if(::IsIconic(*this))
  181. {
  182. bHandled = FALSE;
  183. }
  184. return (wParam == 0) ? TRUE : FALSE;
  185. }
  186. LRESULT CValueWnd::OnNcCalcSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  187. {
  188. return 0;
  189. }
  190. LRESULT CValueWnd::OnNcPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  191. {
  192. //在这里设置焦点才有用
  193. CEditUI* pAccountEdit = static_cast<CEditUI*>(m_pm.FindControl(_T("accountedit")));
  194. if(pAccountEdit)
  195. {
  196. pAccountEdit->SetFocus();
  197. }
  198. return 0;
  199. }
  200. LRESULT CValueWnd::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  201. {
  202. POINT pt;
  203. pt.x = GET_X_LPARAM(lParam);
  204. pt.y = GET_Y_LPARAM(lParam);
  205. ::ScreenToClient(*this, &pt);
  206. RECT rcClient;
  207. ::GetClientRect(*this, &rcClient);
  208. RECT rcCaption = m_pm.GetCaptionRect();
  209. if(pt.x >= rcClient.left + rcCaption.left && pt.x < rcClient.right - rcCaption.right \
  210. && pt.y >= rcCaption.top && pt.y < rcCaption.bottom)
  211. {
  212. CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(pt));
  213. if(pControl && _tcscmp(pControl->GetClass(), DUI_CTR_BUTTON) != 0)
  214. {
  215. return HTCAPTION;
  216. }
  217. }
  218. return HTCLIENT;
  219. }
  220. LRESULT CValueWnd::OnKexianUpdate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  221. {
  222. //收到了客显数据消息,读取客显数据
  223. char data[MAX_DATA_LENGTH] = { 0 };
  224. int data_length = 0;
  225. HookGetData(data, &data_length);
  226. if(data_length == 0)
  227. {
  228. return true;
  229. }
  230. UpdateKexian(data, data_length);
  231. return true;
  232. }
  233. LRESULT CValueWnd::OnWatchWndUpdate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  234. {
  235. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  236. std::string money_show = CLewaimaiString::DoubleToString(atof(m_watchValue.c_str()), 2);
  237. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  238. return TRUE;
  239. }
  240. LRESULT CValueWnd::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
  241. {
  242. LRESULT lRes = 0;
  243. BOOL bHandled = TRUE;
  244. switch(uMsg)
  245. {
  246. case WM_CREATE:
  247. lRes = OnCreate(uMsg, wParam, lParam, bHandled);
  248. break;
  249. case WM_CLOSE:
  250. lRes = OnClose(uMsg, wParam, lParam, bHandled);
  251. break;
  252. case WM_DESTROY:
  253. lRes = OnDestroy(uMsg, wParam, lParam, bHandled);
  254. break;
  255. case WM_NCACTIVATE:
  256. lRes = OnNcActivate(uMsg, wParam, lParam, bHandled);
  257. break;
  258. case WM_NCCALCSIZE:
  259. lRes = OnNcCalcSize(uMsg, wParam, lParam, bHandled);
  260. break;
  261. case WM_NCPAINT:
  262. lRes = OnNcPaint(uMsg, wParam, lParam, bHandled);
  263. break;
  264. case WM_NCHITTEST:
  265. lRes = OnNcHitTest(uMsg, wParam, lParam, bHandled);
  266. break;
  267. case WM_HOOK_MESSAGE:
  268. lRes = OnKexianUpdate(uMsg, wParam, lParam, bHandled);
  269. break;
  270. case WM_WATCH_WND_UPDATE:
  271. lRes = OnWatchWndUpdate(uMsg, wParam, lParam, bHandled);
  272. break;
  273. default:
  274. bHandled = FALSE;
  275. }
  276. if(bHandled)
  277. {
  278. return lRes;
  279. }
  280. if(m_pm.MessageHandler(uMsg, wParam, lParam, lRes))
  281. {
  282. return lRes;
  283. }
  284. return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
  285. }
  286. LRESULT CValueWnd::MessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, bool& bHandled)
  287. {
  288. return false;
  289. }
  290. void CValueWnd::UpdateKexian(const char* data, int length)
  291. {
  292. std::string new_data = data;
  293. std::string show_command = "\x1bQA";
  294. std::string type_command = "\x1bs";
  295. std::string money;
  296. if(new_data.find(show_command) == new_data.npos)
  297. {
  298. return;
  299. }
  300. int nPos = new_data.find(show_command);
  301. int nTypePos = new_data.find(type_command, nPos + 1);
  302. if(nTypePos == new_data.npos)
  303. {
  304. //没有类型信息,有些收银机(比如思迅天店)就是不传类型信息,这种只有传什么显示什么了
  305. int nTabPos = new_data.find("\r", nPos + 1);
  306. if(nTabPos == new_data.npos)
  307. {
  308. //制表符也没有
  309. return;
  310. }
  311. money = new_data.substr(nPos + 3, nTabPos - nPos - 3);
  312. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  313. std::string money_show = CLewaimaiString::DoubleToString(atof(money.c_str()), 2);
  314. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  315. }
  316. else
  317. {
  318. std::string value_type = new_data.substr(nTypePos + 2, 1);
  319. money = new_data.substr(nPos + 3, nTypePos - nPos - 3 - 1);
  320. if(value_type == "2")
  321. {
  322. //这个表示是总计的,那么就要刷新金额显示
  323. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  324. std::string money_show = CLewaimaiString::DoubleToString(atof(money.c_str()), 2);
  325. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  326. }
  327. }
  328. return;
  329. }
  330. void CValueWnd::FindValueWnd(std::wstring value)
  331. {
  332. //先查找子窗口
  333. EnumWindows(EnumWindowsProc, NULL);
  334. //展示查找结果
  335. int nCount = (int)g_valuewnd_vector.size();
  336. if(nCount > 0)
  337. {
  338. //开始监听这个窗口的内容变化
  339. ValueWnd selectWnd = g_valuewnd_vector[0];
  340. m_watchWnd = selectWnd.selfWnd;
  341. }
  342. }