CValueWnd.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736
  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. //初始化窗口位置
  19. InitWndPos();
  20. //添加托盘图标
  21. AddTrayIcon();
  22. //初始化菜单窗口
  23. pMenu = new CMenuWnd();
  24. pMenu->Create(m_hWnd, L"智铺子收银插件菜单", WS_POPUP, WS_EX_TOOLWINDOW);
  25. pMenu->ShowWindow(false);
  26. //再安装钩子
  27. //BOOL ret = InstallHook();
  28. //开始启动监听
  29. RestartWatch();
  30. /*
  31. std::thread(&CValueWnd::WatchWnd, this).detach();
  32. //开始测试ocr
  33. tesseract::TessBaseAPI tess;
  34. if(tess.Init("./tessdata", "eng"))
  35. {
  36. std::cout << "OCRTesseract: Could not initialize tesseract." << std::endl;
  37. return ;
  38. }
  39. // setup
  40. tess.SetPageSegMode(tesseract::PageSegMode::PSM_AUTO);
  41. tess.SetVariable("save_best_choices", "T");
  42. // read image
  43. auto pixs = pixRead("D:\\3.png");
  44. if(!pixs)
  45. {
  46. std::cout << "Cannot open input file: " << "D:\1.png" << std::endl;
  47. return ;
  48. }
  49. // recognize
  50. tess.SetImage(pixs);
  51. tess.Recognize(0);
  52. // get result and delete[] returned char* string
  53. std::string aaa = tess.GetUTF8Text();
  54. // cleanup
  55. tess.Clear();
  56. pixDestroy(&pixs);*/
  57. }
  58. void CValueWnd::RestartWatch()
  59. {
  60. std::thread(&CValueWnd::StartWatchWork, this).detach();
  61. }
  62. /*
  63. *根据当前数据库的设置,重启监听
  64. **/
  65. void CValueWnd::StartWatchWork()
  66. {
  67. //先暂停已有的监控
  68. m_isWatchWork = false;
  69. //等待1秒,这1秒就是等已有的监控线程自动退出
  70. Sleep(1000);
  71. //然后开始工作
  72. m_isWatchWork = true;
  73. std::string system_setting_jinezhuaqu_setting_type = CSetting::GetParam("system_setting_jinezhuaqu_setting_type");
  74. m_nWatchType = atoi(system_setting_jinezhuaqu_setting_type.c_str());
  75. if(m_nWatchType == 1)
  76. {
  77. //更新监听的端口号
  78. std::string system_setting_jinezhuaqu_setting_chuankou_num = CSetting::GetParam("system_setting_jinezhuaqu_setting_chuankou_num");
  79. if(system_setting_jinezhuaqu_setting_chuankou_num == "")
  80. {
  81. //参数错误
  82. return;
  83. }
  84. std::wstring ws_system_setting_jinezhuaqu_setting_chuankou_num = CLewaimaiString::UTF8ToUnicode(system_setting_jinezhuaqu_setting_chuankou_num);
  85. //设置窗口句柄
  86. ::SetHwnd(m_hWnd);
  87. //设置串口端口号
  88. std::transform(system_setting_jinezhuaqu_setting_chuankou_num.begin(), system_setting_jinezhuaqu_setting_chuankou_num.end(), system_setting_jinezhuaqu_setting_chuankou_num.begin(), ::toupper);
  89. std::transform(ws_system_setting_jinezhuaqu_setting_chuankou_num.begin(), ws_system_setting_jinezhuaqu_setting_chuankou_num.end(), ws_system_setting_jinezhuaqu_setting_chuankou_num.begin(), ::toupper);
  90. ::SetComNum(system_setting_jinezhuaqu_setting_chuankou_num.c_str(), ws_system_setting_jinezhuaqu_setting_chuankou_num.c_str(), system_setting_jinezhuaqu_setting_chuankou_num.size());
  91. }
  92. else if(m_nWatchType == 2)
  93. {
  94. //虚拟串口的模式
  95. std::thread(&CValueWnd::ReadXunichuankouValue, this).detach();
  96. }
  97. else if(m_nWatchType == 3)
  98. {
  99. //窗口模式
  100. std::thread(&CValueWnd::WatchWnd, this).detach();
  101. }
  102. }
  103. /*
  104. *读取客显金额的线程,一旦m_nWatchType不为2,那么就直接退出这个线程,下次设置的时候再开启
  105. **/
  106. void CValueWnd::ReadXunichuankouValue()
  107. {
  108. //读取虚拟串口的模式
  109. std::string system_setting_jinezhuaqu_setting_xunichuankou_num_2 = CSetting::GetParam("system_setting_jinezhuaqu_setting_xunichuankou_num_2");
  110. std::string system_setting_jinezhuaqu_setting_xunichuankou_botelv = CSetting::GetParam("system_setting_jinezhuaqu_setting_xunichuankou_botelv");
  111. CSerialPort serial(CLewaimaiString::UTF8ToUnicode(system_setting_jinezhuaqu_setting_xunichuankou_num_2), atoi(system_setting_jinezhuaqu_setting_xunichuankou_botelv.c_str()));
  112. bool ret = serial.openComm();
  113. if(!ret)
  114. {
  115. LOG_INFO("open com failed!");
  116. return;
  117. }
  118. char a[1024] = { 0 };
  119. //检查监控模式和波特率,如果变了就关闭掉
  120. while(m_nWatchType == 2 && m_isWatchWork)
  121. {
  122. memset(a, 0, 1024);
  123. //开始读取串口的数据
  124. DWORD nReaded = 0;
  125. serial.readFromComm(a, 1024, &nReaded);
  126. if(nReaded > 0)
  127. {
  128. UpdateKexian(a, nReaded);
  129. }
  130. Sleep(200);
  131. }
  132. serial.closeComm();
  133. }
  134. void CValueWnd::WatchWnd()
  135. {
  136. std::string system_setting_jinezhuaqu_setting_chuangkou_process_name = CSetting::GetParam("system_setting_jinezhuaqu_setting_chuangkou_process_name");
  137. std::string system_setting_jinezhuaqu_setting_chuangkou_title = CSetting::GetParam("system_setting_jinezhuaqu_setting_chuangkou_title");
  138. std::string system_setting_jinezhuaqu_setting_chuangkou_zorder = CSetting::GetParam("system_setting_jinezhuaqu_setting_chuangkou_zorder");
  139. //如果模式变了,或者参数变了,就退出(因为会重新启动一个新的监控)
  140. while(m_nWatchType == 3 && m_isWatchWork)
  141. {
  142. if(m_watchWnd != NULL)
  143. {
  144. char szTitle[MAX_PATH] = { 0 };
  145. LPSTR lpWindowName = szTitle;
  146. std::string value;
  147. int ret = ::SendMessageA(m_watchWnd, WM_GETTEXT, (WPARAM)(MAX_PATH), (LPARAM)lpWindowName);
  148. if(ret == 0)
  149. {
  150. //说明m_watchWnd已经失效了
  151. value = "0.00";
  152. m_watchWnd = NULL;
  153. }
  154. else
  155. {
  156. value = lpWindowName;
  157. }
  158. if(value != m_watchValue)
  159. {
  160. m_watchValue = value;
  161. SendMessage(WM_WATCH_WND_UPDATE, NULL, NULL);
  162. }
  163. }
  164. else
  165. {
  166. //获得进程名字
  167. std::wstring processName = CLewaimaiString::UTF8ToUnicode(system_setting_jinezhuaqu_setting_chuangkou_process_name);
  168. //根据进程名字,找到对应的主窗口
  169. HWND mainWnd = GetWndHwnd(processName.c_str(), system_setting_jinezhuaqu_setting_chuangkou_title);
  170. if(mainWnd == NULL)
  171. {
  172. //说明这个进程还没有启动
  173. m_watchWnd = NULL;
  174. }
  175. else
  176. {
  177. //根据Z序查找目标窗口
  178. ValueWnd newWnd;
  179. newWnd.wndTitle = system_setting_jinezhuaqu_setting_chuangkou_title;
  180. newWnd.TopWnd = mainWnd;
  181. newWnd.z_order = atoi(system_setting_jinezhuaqu_setting_chuangkou_zorder.c_str());
  182. newWnd.processName = processName;
  183. newWnd.selfWnd = NULL;
  184. g_zorder = 0;
  185. EnumChildWindows(mainWnd, EnumMainwndChildProc, (LPARAM)(&newWnd));
  186. if(newWnd.selfWnd != NULL)
  187. {
  188. //找到了
  189. m_watchWnd = newWnd.selfWnd;
  190. }
  191. }
  192. }
  193. Sleep(200);
  194. }
  195. }
  196. LRESULT CValueWnd::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  197. {
  198. LONG styleValue = ::GetWindowLong(*this, GWL_STYLE);
  199. styleValue &= ~WS_CAPTION;
  200. ::SetWindowLong(*this, GWL_STYLE, styleValue | WS_CLIPSIBLINGS | WS_CLIPCHILDREN);
  201. // 把自己的窗口句柄与窗口绘制管理器挂接在一起
  202. m_pm.Init(m_hWnd);
  203. m_pm.AddPreMessageFilter(this);
  204. CDialogBuilder builder;
  205. CControlUI* pRoot = builder.Create(_T("value.xml"), (UINT)0, NULL, &m_pm);
  206. ASSERT(pRoot && "Failed to parse XML");
  207. // 把这些控件绘制到本窗口上
  208. m_pm.AttachDialog(pRoot);
  209. // 把自己加入到CPaintManagerUI的m_aNotifiers数组中,用于处理Notify函数
  210. //m_pm.AddNotifier(this);
  211. //进行一些初始化工作
  212. Init();
  213. return 0;
  214. }
  215. LRESULT CValueWnd::OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  216. {
  217. bHandled = FALSE;
  218. return 0;
  219. }
  220. /*
  221. *这个是窗口被销毁的时候调用的
  222. **/
  223. LRESULT CValueWnd::OnDestroy(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  224. {
  225. bHandled = FALSE;
  226. return 0;
  227. }
  228. LRESULT CValueWnd::OnNcActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  229. {
  230. if(::IsIconic(*this))
  231. {
  232. bHandled = FALSE;
  233. }
  234. return (wParam == 0) ? TRUE : FALSE;
  235. }
  236. LRESULT CValueWnd::OnNcCalcSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  237. {
  238. return 0;
  239. }
  240. LRESULT CValueWnd::OnNcPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  241. {
  242. //在这里设置焦点才有用
  243. CEditUI* pAccountEdit = static_cast<CEditUI*>(m_pm.FindControl(_T("accountedit")));
  244. if(pAccountEdit)
  245. {
  246. pAccountEdit->SetFocus();
  247. }
  248. return 0;
  249. }
  250. LRESULT CValueWnd::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  251. {
  252. POINT pt;
  253. pt.x = GET_X_LPARAM(lParam);
  254. pt.y = GET_Y_LPARAM(lParam);
  255. ::ScreenToClient(*this, &pt);
  256. RECT rcClient;
  257. ::GetClientRect(*this, &rcClient);
  258. RECT rcCaption = m_pm.GetCaptionRect();
  259. if(pt.x >= rcClient.left + rcCaption.left && pt.x < rcClient.right - rcCaption.right \
  260. && pt.y >= rcCaption.top && pt.y < rcCaption.bottom)
  261. {
  262. CControlUI* pControl = static_cast<CControlUI*>(m_pm.FindControl(pt));
  263. if(pControl && _tcscmp(pControl->GetClass(), DUI_CTR_BUTTON) != 0)
  264. {
  265. return HTCAPTION;
  266. }
  267. }
  268. return HTCLIENT;
  269. }
  270. LRESULT CValueWnd::OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  271. {
  272. /*
  273. //获取鼠标点击位置
  274. int xPos = LOWORD(lParam);
  275. int yPos = HIWORD(lParam);
  276. //將位置转换为屏幕坐标
  277. POINT pt = { xPos,yPos };
  278. ClientToScreen(m_hWnd, &pt);
  279. RECT menuWndRect = { 0 };
  280. ::GetWindowRect(pMenu->GetHWND(), &menuWndRect);
  281. if (pt.x <= menuWndRect.left || pt.x >= menuWndRect.right
  282. || pt.y <= menuWndRect.top || pt.y >= menuWndRect.bottom)
  283. {
  284. pMenu->ShowWindow(false);
  285. }
  286. bHandled = FALSE;
  287. return 0;*/
  288. bHandled = FALSE;
  289. return 0;
  290. }
  291. LRESULT CValueWnd::OnRButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  292. {
  293. /*
  294. int xPos = LOWORD(lParam);
  295. int yPos = HIWORD(lParam);
  296. POINT pt = { xPos,yPos };
  297. //可能是因为整个窗口都是标题区域,这里不需要做转换
  298. //ClientToScreen(m_hWnd, &pt);
  299. pMenu->ShowWindow(true);
  300. ::SetWindowPos(pMenu->GetHWND(), NULL, pt.x, pt.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  301. bHandled = FALSE;
  302. return 0;*/
  303. ShowMenu();
  304. bHandled = true;
  305. return 0;
  306. }
  307. LRESULT CValueWnd::OnKexianUpdate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  308. {
  309. //如果当前不是串口模式,直接忽略
  310. if(m_nWatchType != 1)
  311. {
  312. bHandled = TRUE;
  313. return true;
  314. }
  315. //收到了客显数据消息,读取客显数据
  316. char data[MAX_DATA_LENGTH] = { 0 };
  317. int data_length = 0;
  318. HookGetData(data, &data_length);
  319. if(data_length == 0)
  320. {
  321. bHandled = TRUE;
  322. return true;
  323. }
  324. UpdateKexian(data, data_length);
  325. bHandled = TRUE;
  326. return true;
  327. }
  328. LRESULT CValueWnd::OnWatchWndUpdate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  329. {
  330. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  331. std::string money_show = CLewaimaiString::DoubleToString(atof(m_watchValue.c_str()), 2);
  332. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  333. return TRUE;
  334. }
  335. LRESULT CValueWnd::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
  336. {
  337. LRESULT lRes = 0;
  338. BOOL bHandled = TRUE;
  339. switch(uMsg)
  340. {
  341. case WM_CREATE:
  342. lRes = OnCreate(uMsg, wParam, lParam, bHandled);
  343. break;
  344. case WM_CLOSE:
  345. lRes = OnClose(uMsg, wParam, lParam, bHandled);
  346. break;
  347. case WM_DESTROY:
  348. lRes = OnDestroy(uMsg, wParam, lParam, bHandled);
  349. break;
  350. case WM_NCACTIVATE:
  351. lRes = OnNcActivate(uMsg, wParam, lParam, bHandled);
  352. break;
  353. case WM_NCLBUTTONDOWN:
  354. lRes = OnLButtonDown(uMsg, wParam, lParam, bHandled);
  355. break;
  356. case WM_NCRBUTTONDOWN:
  357. lRes = OnRButtonDown(uMsg, wParam, lParam, bHandled);
  358. break;
  359. case WM_NCCALCSIZE:
  360. lRes = OnNcCalcSize(uMsg, wParam, lParam, bHandled);
  361. break;
  362. case WM_NCPAINT:
  363. lRes = OnNcPaint(uMsg, wParam, lParam, bHandled);
  364. break;
  365. case WM_NCHITTEST:
  366. lRes = OnNcHitTest(uMsg, wParam, lParam, bHandled);
  367. break;
  368. case WM_SHOWTASK:
  369. lRes = OnTrayIcon(uMsg, wParam, lParam, bHandled);
  370. break;
  371. case WM_HOOK_MESSAGE:
  372. lRes = OnKexianUpdate(uMsg, wParam, lParam, bHandled);
  373. break;
  374. case WM_WATCH_WND_UPDATE:
  375. lRes = OnWatchWndUpdate(uMsg, wParam, lParam, bHandled);
  376. break;
  377. default:
  378. bHandled = FALSE;
  379. }
  380. if(bHandled)
  381. {
  382. return lRes;
  383. }
  384. if(m_pm.MessageHandler(uMsg, wParam, lParam, lRes))
  385. {
  386. return lRes;
  387. }
  388. return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
  389. }
  390. LRESULT CValueWnd::MessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, bool& bHandled)
  391. {
  392. return false;
  393. }
  394. void CValueWnd::AddTrayIcon()
  395. {
  396. memset(&m_trayIcon, 0, sizeof(NOTIFYICONDATA));
  397. m_trayIcon.cbSize = sizeof(NOTIFYICONDATA);
  398. m_trayIcon.hIcon = ::LoadIcon(GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_ICON_DUILIB));
  399. m_trayIcon.hWnd = m_hWnd;
  400. lstrcpy(m_trayIcon.szTip, _T("智铺子收银插件"));
  401. m_trayIcon.uCallbackMessage = WM_SHOWTASK;
  402. m_trayIcon.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP;
  403. Shell_NotifyIcon(NIM_ADD, &m_trayIcon);
  404. }
  405. LRESULT CValueWnd::OnTrayIcon(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
  406. {
  407. //如果在图标中单击左键则还原
  408. if(lParam == WM_LBUTTONDOWN)
  409. {
  410. }
  411. //如果在图标中单击右键则弹出声明式菜单
  412. if(lParam == WM_RBUTTONDOWN)
  413. {
  414. ShowMenu();
  415. }
  416. bHandled = true;
  417. return 0;
  418. }
  419. void CValueWnd::UpdateKexian(const char* data, int length)
  420. {
  421. std::string new_data = data;
  422. //清屏命令
  423. std::string clear_command = "\f";
  424. //4个信号灯全灭
  425. std::string quanmie_command = "\x1bs0";
  426. std::string show_command = "\x1bQA";
  427. std::string type_command = "\x1bs";
  428. if(new_data == clear_command || new_data == quanmie_command)
  429. {
  430. //清屏命令,或者是熄灭灯的命令
  431. return;
  432. }
  433. if(new_data.find(show_command) == new_data.npos)
  434. {
  435. return;
  436. }
  437. std::string money;
  438. int nPos = new_data.find(show_command);
  439. int nTypePos = new_data.find(type_command, nPos + 1);
  440. if(nTypePos == new_data.npos)
  441. {
  442. //没有类型信息,有些收银机(比如思迅天店)就是不传类型信息,这种只有传什么显示什么了
  443. int nTabPos = new_data.find("\r", nPos + 1);
  444. if(nTabPos == new_data.npos)
  445. {
  446. //制表符也没有
  447. return;
  448. }
  449. money = new_data.substr(nPos + 3, nTabPos - nPos - 3);
  450. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  451. std::string money_show = CLewaimaiString::DoubleToString(atof(money.c_str()), 2);
  452. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  453. }
  454. else
  455. {
  456. std::string value_type = new_data.substr(nTypePos + 2, 1);
  457. money = new_data.substr(nPos + 3, nTypePos - nPos - 3 - 1);
  458. if(value_type == "2")
  459. {
  460. //这个表示是总计的,那么就要刷新金额显示
  461. CLabelUI* valueLabel = static_cast<CLabelUI*>(m_pm.FindControl(_T("value")));
  462. std::string money_show = CLewaimaiString::DoubleToString(atof(money.c_str()), 2);
  463. valueLabel->SetText(CLewaimaiString::UTF8ToUnicode(money_show).c_str());
  464. }
  465. else
  466. {
  467. //这种情况传输的是一些找零、单价之类的信息,不更新客显金额
  468. }
  469. }
  470. return;
  471. }
  472. void CValueWnd::InitWndPos()
  473. {
  474. ASSERT(::IsWindow(m_hWnd));
  475. ASSERT((GetWindowStyle(m_hWnd)&WS_CHILD) == 0);
  476. RECT rcDlg = { 0 };
  477. ::GetWindowRect(m_hWnd, &rcDlg);
  478. RECT rcArea = { 0 };
  479. RECT rcCenter = { 0 };
  480. HWND hWnd = *this;
  481. HWND hWndParent = ::GetParent(m_hWnd);
  482. HWND hWndCenter = ::GetWindowOwner(m_hWnd);
  483. if(hWndCenter != NULL)
  484. {
  485. hWnd = hWndCenter;
  486. }
  487. // 处理多显示器模式下屏幕居中
  488. MONITORINFO oMonitor = {};
  489. oMonitor.cbSize = sizeof(oMonitor);
  490. ::GetMonitorInfo(::MonitorFromWindow(hWnd, MONITOR_DEFAULTTONEAREST), &oMonitor);
  491. rcArea = oMonitor.rcWork;
  492. if(hWndCenter == NULL)
  493. {
  494. rcCenter = rcArea;
  495. }
  496. else
  497. {
  498. ::GetWindowRect(hWndCenter, &rcCenter);
  499. }
  500. int DlgWidth = rcDlg.right - rcDlg.left;
  501. int DlgHeight = rcDlg.bottom - rcDlg.top;
  502. // Find dialog's upper left based on rcCenter
  503. int xLeft = rcCenter.right - DlgWidth - 60;
  504. int yTop = 100;
  505. // The dialog is outside the screen, move it inside
  506. if(xLeft < rcArea.left)
  507. {
  508. xLeft = rcArea.left;
  509. }
  510. else if(xLeft + DlgWidth > rcArea.right)
  511. {
  512. xLeft = rcArea.right - DlgWidth;
  513. }
  514. if(yTop < rcArea.top)
  515. {
  516. yTop = rcArea.top;
  517. }
  518. else if(yTop + DlgHeight > rcArea.bottom)
  519. {
  520. yTop = rcArea.bottom - DlgHeight;
  521. }
  522. ::SetWindowPos(m_hWnd, HWND_TOPMOST, xLeft, yTop, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  523. }
  524. void CValueWnd::ShowMenu()
  525. {
  526. //获取鼠标坐标
  527. POINT pt;
  528. GetCursorPos(&pt);
  529. //右击后点别地可以清除“右击出来的菜单”
  530. SetForegroundWindow(m_hWnd);
  531. //托盘菜单,win32程序使用的是HMENU,如果是MFC程序可以使用CMenu
  532. HMENU hMenu;
  533. //生成托盘菜单
  534. hMenu = CreatePopupMenu();
  535. //添加菜单,关键在于设置的一个标识符 WM_ONCLOSE 点击后会用到
  536. AppendMenu(hMenu, MF_STRING, WM_MENU_SYSTEM_SETTING, _T("系统设置"));
  537. AppendMenu(hMenu, MF_STRING, WM_ONCLOSE, _T("退出"));
  538. //弹出菜单,并把用户所选菜单项的标识符返回
  539. int cmd = TrackPopupMenu(hMenu, TPM_RETURNCMD, pt.x, pt.y, NULL, m_hWnd, NULL);
  540. //如果标识符是WM_ONCLOSE则关闭
  541. if(cmd == WM_ONCLOSE)
  542. {
  543. //退出程序
  544. ::PostQuitMessage(0);
  545. }
  546. if(cmd == WM_MENU_SYSTEM_SETTING)
  547. {
  548. if(m_is_show_setting == true)
  549. {
  550. return;
  551. }
  552. CSystemSettingWnd* pSystemSettingFrame = new CSystemSettingWnd(this);
  553. if(pSystemSettingFrame != NULL)
  554. {
  555. m_is_show_setting = true;
  556. pSystemSettingFrame->Create(NULL, _T(""), UI_WNDSTYLE_DIALOG, WS_EX_TOOLWINDOW);
  557. pSystemSettingFrame->SetIcon(IDI_ICON_DUILIB);
  558. pSystemSettingFrame->CenterWindow();
  559. UINT ret = pSystemSettingFrame->ShowModal();
  560. m_is_show_setting = false;
  561. if(ret == IDOK)
  562. {
  563. }
  564. else if(ret == IDCANCEL)
  565. {
  566. }
  567. }
  568. }
  569. }