ItrayIcon.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. ////////////////////////////////////////////////////////////////
  2. // If this code works, it was written by Paul DiLascia.
  3. // If not, I don't know who wrote it.
  4. //
  5. #include "stdafx.h"
  6. #include "Itrayicon.h"
  7. #include <afxpriv.h> // for AfxLoadString
  8. IMPLEMENT_DYNAMIC(ITrayIcon, CCmdTarget)
  9. typedef BOOL (WINAPI *LPFN_ISWOW64PROCESS) (HANDLE, PBOOL);
  10. ITrayIcon::ITrayIcon(UINT uID)
  11. {
  12. // Initialize NOTIFYICONDATA
  13. memset(&m_nid, 0 , sizeof(m_nid));
  14. m_nid.cbSize = sizeof(m_nid);
  15. m_nid.uID = uID; // never changes after construction
  16. OpendThread();
  17. bShowMinibox = FALSE;
  18. //m_pWnd = AfxGetMainWnd();
  19. // Use resource string as tip if there is one
  20. AfxLoadString(uID, m_nid.szTip, sizeof(m_nid.szTip));
  21. }
  22. ITrayIcon::~ITrayIcon()
  23. {
  24. SetIcon(0); // remove icon from system tray
  25. TerminalThread();
  26. }
  27. //////////////////
  28. // Set notification window. It must created already.
  29. //
  30. void ITrayIcon::SetNotificationWnd(CWnd* pNotifyWnd, UINT uCbMsg)
  31. {
  32. // If the following assert fails, you're probably
  33. // calling me before you created your window. Oops.
  34. ASSERT(pNotifyWnd==NULL || ::IsWindow(pNotifyWnd->GetSafeHwnd()));
  35. m_nid.hWnd = pNotifyWnd->GetSafeHwnd();
  36. ASSERT(uCbMsg==0 || uCbMsg>=WM_USER);
  37. m_nid.uCallbackMessage = uCbMsg;
  38. }
  39. //////////////////
  40. // This is the main variant for setting the icon.
  41. // Sets both the icon and tooltip from resource ID
  42. // To remove the icon, call SetIcon(0)
  43. //
  44. BOOL ITrayIcon::SetIcon(UINT uID)
  45. {
  46. HICON hicon=NULL;
  47. if (uID)
  48. hicon = AfxGetApp()->LoadIcon(uID);
  49. UINT msg;
  50. m_nid.uFlags = 0;
  51. // Set the icon
  52. if (hicon)
  53. {
  54. // Add or replace icon in system tray
  55. msg = m_nid.hIcon ? NIM_MODIFY : NIM_ADD;
  56. m_nid.hIcon = hicon;
  57. m_nid.uFlags |= NIF_ICON;
  58. } else
  59. { // remove icon from tray
  60. if (m_nid.hIcon==NULL)
  61. return TRUE; // already deleted
  62. msg = NIM_DELETE;
  63. }
  64. // Use callback if any
  65. if (m_nid.uCallbackMessage && m_nid.hWnd)
  66. m_nid.uFlags |= NIF_MESSAGE;
  67. // Do it
  68. BOOL bRet = Shell_NotifyIcon(msg, &m_nid);
  69. if (msg==NIM_DELETE || !bRet)
  70. m_nid.hIcon = NULL; // failed
  71. return bRet;
  72. }
  73. BOOL ITrayIcon::SetIcon(UINT uID,LPCTSTR lpTip)
  74. {
  75. HICON hicon=NULL;
  76. if (uID)
  77. {
  78. AfxLoadString(uID, m_nid.szTip, sizeof(m_nid.szTip));
  79. hicon = AfxGetApp()->LoadIcon(uID);
  80. }
  81. return SetIcon(hicon, lpTip ? lpTip:NULL);
  82. }
  83. //////////////////
  84. // Common SetIcon for all overloads.
  85. //
  86. BOOL ITrayIcon::SetIcon(HICON hicon, LPCTSTR lpTip)
  87. {
  88. UINT msg;
  89. m_nid.uFlags = 0;
  90. // Set the icon
  91. if (hicon)
  92. {
  93. // Add or replace icon in system tray
  94. msg = m_nid.hIcon ? NIM_MODIFY : NIM_ADD;
  95. m_nid.hIcon = hicon;
  96. m_nid.uFlags |= NIF_ICON;
  97. } else
  98. { // remove icon from tray
  99. if (m_nid.hIcon==NULL)
  100. return TRUE; // already deleted
  101. msg = NIM_DELETE;
  102. }
  103. // Use the tip, if any
  104. #ifdef _UNICODE
  105. if (lpTip)
  106. wcsncpy(m_nid.szTip, lpTip, sizeof(m_nid.szTip));
  107. #else
  108. if (lpTip)
  109. strncpy(m_nid.szTip, lpTip, sizeof(m_nid.szTip));
  110. #endif
  111. if (m_nid.szTip[0])
  112. m_nid.uFlags |= NIF_TIP;
  113. // Use callback if any
  114. if (m_nid.uCallbackMessage && m_nid.hWnd)
  115. m_nid.uFlags |= NIF_MESSAGE;
  116. // Do it
  117. BOOL bRet = Shell_NotifyIcon(msg, &m_nid);
  118. if (msg==NIM_DELETE || !bRet)
  119. m_nid.hIcon = NULL; // failed
  120. return bRet;
  121. }
  122. BOOL ITrayIcon::SetTip(LPCTSTR lpTip)
  123. {
  124. UINT msg;
  125. m_nid.uFlags = 0;
  126. if (m_nid.hIcon == NULL)
  127. return TRUE; // already deleted
  128. msg = NIM_MODIFY;
  129. // Use the tip, if any
  130. //if (lpTip)
  131. #ifdef _UNICODE
  132. wcsncpy(m_nid.szTip, lpTip, sizeof(m_nid.szTip));
  133. #else
  134. strncpy(m_nid.szTip, lpTip, sizeof(m_nid.szTip));
  135. #endif
  136. if (m_nid.szTip[0])
  137. m_nid.uFlags |= NIF_TIP;
  138. // Use callback if any
  139. if (m_nid.uCallbackMessage && m_nid.hWnd)
  140. m_nid.uFlags |= NIF_MESSAGE;
  141. return Shell_NotifyIcon(msg, &m_nid);
  142. }
  143. BOOL ITrayIcon::SetInfo(LPCTSTR lpInof, LPCTSTR lpInfoTitle)
  144. {
  145. UINT msg;
  146. m_nid.uFlags = 0;
  147. if (m_nid.hIcon == NULL)
  148. return TRUE; // already deleted
  149. msg = NIM_MODIFY;
  150. #ifdef _UNICODE
  151. wcsncpy(m_nid.szInfo, lpInof, sizeof(m_nid.szInfo));
  152. if ( lpInfoTitle != NULL)
  153. {
  154. wcsncpy(m_nid.szInfoTitle, lpInfoTitle, sizeof(m_nid.szInfoTitle));
  155. m_nid.uFlags |= NIF_INFO|NIIF_INFO;
  156. m_nid.uTimeout = 1000;
  157. }
  158. else
  159. {
  160. m_nid.uFlags |= NIF_INFO;
  161. }
  162. #else
  163. strncpy(m_nid.szInfo, lpInof, sizeof(m_nid.szInfo));
  164. if ( lpInfoTitle != NULL)
  165. {
  166. strncpy(m_nid.szInfoTitle, lpInfoTitle, sizeof(m_nid.szInfoTitle));
  167. m_nid.uFlags |= NIF_INFO|NIIF_INFO;
  168. m_nid.uTimeout = 1000;
  169. }
  170. else
  171. {
  172. m_nid.uFlags |= NIF_INFO;
  173. }
  174. #endif //#ifdef _UNICODE
  175. // Use callback if any
  176. if (m_nid.uCallbackMessage && m_nid.hWnd)
  177. m_nid.uFlags |= NIF_MESSAGE;
  178. return Shell_NotifyIcon(msg, &m_nid);
  179. }
  180. /////////////////
  181. // Default event handler handles right-menu and doubleclick.
  182. // Call this function from your own notification handler.
  183. //
  184. LRESULT ITrayIcon::OnTrayNotification(WPARAM wID, LPARAM lEvent)
  185. {
  186. if (wID!=m_nid.uID || (lEvent!=WM_RBUTTONUP && lEvent!=WM_LBUTTONDBLCLK))
  187. return 0;
  188. // If there's a resource menu with the same ID as the icon, use it as
  189. // the right-button popup menu. CTrayIcon will interprets the first
  190. // item in the menu as the default command for WM_LBUTTONDBLCLK
  191. //
  192. #if 0
  193. CMenu menu;
  194. if (!menu.LoadMenu(m_nid.uID))
  195. return 0;
  196. CMenu* pSubMenu = menu.GetSubMenu(0);
  197. if (!pSubMenu)
  198. return 0;
  199. if (lEvent==WM_RBUTTONUP)
  200. {
  201. // Make first menu item the default (bold font)
  202. ::SetMenuDefaultItem(pSubMenu->m_hMenu, 0, TRUE);
  203. // Display the menu at the current mouse location. There's a "bug"
  204. // (Microsoft calls it a feature) in Windows 95 that requires calling
  205. // SetForegroundWindow. To find out more, search for Q135788 in MSDN.
  206. //
  207. CPoint mouse;
  208. GetCursorPos(&mouse);
  209. ::SetForegroundWindow(m_nid.hWnd);
  210. ::TrackPopupMenu(pSubMenu->m_hMenu, 0, mouse.x, mouse.y, 0,
  211. m_nid.hWnd, NULL);
  212. } else // double click: execute first menu item
  213. ::SendMessage(m_nid.hWnd, WM_COMMAND, pSubMenu->GetMenuItemID(0), 0);
  214. #endif
  215. return 1; // handled
  216. }
  217. // ------------------------------- [5/30/2013 Z.t]
  218. BOOL ITrayIcon::IsWow64()
  219. {
  220. BOOL bIsWow64 = FALSE;
  221. LPFN_ISWOW64PROCESS fnIsWow64Process = (LPFN_ISWOW64PROCESS)GetProcAddress(GetModuleHandle(_T("kernel32")),"IsWow64Process");
  222. if (NULL != fnIsWow64Process)
  223. {
  224. if (!fnIsWow64Process(GetCurrentProcess(),&bIsWow64))
  225. {
  226. // handle error
  227. }
  228. }
  229. return bIsWow64;
  230. }
  231. /************************************************************************/
  232. /*
  233. 函数:GetTrayRect,获取托盘图标区域;(有待完善,添加各种错误判断)
  234. 参数:None;
  235. 返回:返回指定托盘图标的区域;
  236. 注意:函数的功能有些太复杂,暂时没办法使用更简单的方法获取图标区域;
  237. */
  238. /************************************************************************/
  239. void ITrayIcon::GetTrayRect(RECT &rc)
  240. {
  241. HWND hWnd,hWndPaper;
  242. long ret;
  243. LPVOID lngAddress;
  244. // long lngTextAdr; //,lngHwndAdr,lngHwnd,lngButtonID;
  245. TCHAR strBuff[1024]={0};
  246. TBBUTTON btnData={0};
  247. hWnd = FindWindow(_T("Shell_TrayWnd"), NULL); // 获取任务栏句柄 [5/31/2013 Z.t]
  248. hWnd = FindWindowEx(hWnd, 0, _T("TrayNotifyWnd"), NULL); // 获取托盘区域句柄 [5/31/2013 Z.t]
  249. hWndPaper = FindWindowEx(hWnd, 0, _T("SysPager"), NULL); // 获取系统页句柄 [5/31/2013 Z.t]
  250. if(!hWndPaper)
  251. hWnd = FindWindowEx(hWnd, 0, _T("ToolbarWindow32"), NULL);
  252. else
  253. hWnd = FindWindowEx(hWndPaper, 0, _T("ToolbarWindow32"), NULL);
  254. DWORD dwProcessId = 0;
  255. GetWindowThreadProcessId(hWnd, &dwProcessId);//LOG4C((LOG_NOTICE,"进程ID%d",dwProcessId));进程其实就是explorer.exe
  256. HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS|PROCESS_VM_OPERATION|PROCESS_VM_READ|PROCESS_VM_WRITE,0,dwProcessId);
  257. if ( hProcess == NULL )
  258. {
  259. //LOG4C((LOG_NOTICE,"hProcess == NULL--"));
  260. return ;
  261. }
  262. lngAddress = VirtualAllocEx(hProcess,0, 0x4096, MEM_COMMIT, PAGE_READWRITE);
  263. if( lngAddress == NULL)
  264. {
  265. //LOG4C((LOG_NOTICE,"lngAddress == NULL--"));
  266. return;
  267. }
  268. DWORD lTextAdr = 0;
  269. BYTE buff[1024] = {0};
  270. CString strFilePath;
  271. CString strTile;
  272. HWND hMainWnd = NULL;
  273. int nDataOffset = sizeof(TBBUTTON) - sizeof(INT_PTR) - sizeof(DWORD_PTR);
  274. int nStrOffset = 18;
  275. if ( IsWow64() )
  276. {
  277. nDataOffset+=4;
  278. nStrOffset+=6;
  279. }
  280. // 获取托盘区域的所有图标数量 [5/31/2013 Z.t]
  281. int ibtnCount = SendMessage(hWnd, TB_BUTTONCOUNT, 0, 0);
  282. LPVOID lngRect = VirtualAllocEx(hProcess,0,sizeof(RECT), MEM_COMMIT, PAGE_READWRITE);
  283. CRect rect;
  284. for(int i=0 ;i< ibtnCount;i++)
  285. {
  286. int j = i;
  287. ret = SendMessage(hWnd,TB_GETBUTTON,j, (LPARAM)(lngAddress));
  288. // 读文本地址;
  289. ret = ReadProcessMemory(hProcess, LPVOID(long(lngAddress) + nDataOffset),&lTextAdr,4,0);
  290. if(lTextAdr != -1)
  291. {
  292. // 读文本;
  293. ret = ReadProcessMemory(hProcess, LPVOID(lTextAdr),buff,1024,0);
  294. hMainWnd = (HWND)(*((DWORD*)buff));
  295. strFilePath = (WCHAR *)buff + nStrOffset; // 获取托盘图标进程路径 [5/31/2013 Z.t]
  296. strTile = (WCHAR *)buff + nStrOffset + MAX_PATH; // 获取托盘图标tip标题 [5/31/2013 Z.t]
  297. //_tprintf(_T("%s %s\n"),strTile,strFilePath);
  298. if (strTile.Compare(m_nid.szTip) == 0)
  299. {
  300. ::SendMessage(hWnd,TB_GETITEMRECT,(WPARAM)j,(LPARAM)lngRect);
  301. ReadProcessMemory(hProcess,lngRect,&rc, sizeof(rc),0); // 获取托盘图标区域;
  302. CWnd::FromHandle(hWnd)->ClientToScreen(&rc);
  303. }
  304. }
  305. }
  306. VirtualFreeEx( hProcess, lngAddress, 0x4096, MEM_DECOMMIT);
  307. VirtualFreeEx( hProcess, lngAddress, 0, MEM_RELEASE);
  308. VirtualFreeEx( hProcess, lngRect, sizeof(RECT), MEM_DECOMMIT);
  309. VirtualFreeEx( hProcess, lngRect, 0, MEM_RELEASE);
  310. CloseHandle(hProcess);
  311. }
  312. int ITrayIcon::OpendThread()
  313. {
  314. //LOG4C((LOG_NOTICE,"创建TrayIcon"));
  315. m_hThreadCtrl = CreateEvent(NULL,TRUE,FALSE,NULL); // 无信号事件;
  316. if ( m_hThreadCtrl == NULL )
  317. {
  318. //LOG4C((LOG_NOTICE,"创建TrayIcon事件失败"));
  319. return -1;
  320. }
  321. m_hThreadObj = CreateThread(NULL,0,FlashingThread,this,CREATE_SUSPENDED,&m_dwThreadID);
  322. if ( m_hThreadObj == NULL )
  323. {
  324. // LOG4C((LOG_NOTICE,"创建TrayIcon线程失败"));
  325. return -1;
  326. }
  327. bSuspending = true;
  328. return 0;
  329. }
  330. void ITrayIcon::TerminalThread()
  331. {
  332. if ( m_hThreadCtrl )
  333. {
  334. SetEvent( m_hThreadCtrl );
  335. }
  336. if (WaitForSingleObject(m_hThreadCtrl,INFINITE) != WAIT_TIMEOUT)
  337. {
  338. CloseHandle(m_hThreadObj);
  339. m_hThreadObj = NULL;
  340. }
  341. CloseHandle( m_hThreadCtrl );
  342. m_hThreadCtrl = NULL;
  343. bSuspending = false;
  344. }
  345. BOOL ITrayIcon::StartFlashing()
  346. {
  347. if ( !bSuspending ) return FALSE;
  348. if(::ResumeThread(m_hThreadObj) == 0xFFFFFFFF)
  349. return FALSE;
  350. bSuspending = false;
  351. SetInfo(_T("环境监控系统产生新警报,请及时处理!"), _T("警报提示"));
  352. return TRUE;
  353. }
  354. BOOL ITrayIcon::StopFlashing()
  355. {
  356. if ( bSuspending ) return FALSE;
  357. if(::SuspendThread(m_hThreadObj) == 0xFFFFFFFF)
  358. return FALSE;
  359. bSuspending = true;
  360. SetIcon(_utIcon[0]);
  361. return TRUE;
  362. }
  363. /************************************************************************/
  364. /*
  365. 线程函数:闪烁线程;
  366. */
  367. /************************************************************************/
  368. DWORD WINAPI ITrayIcon::FlashingThread(LPVOID lpVoid)
  369. {
  370. ITrayIcon *pTrayIcon = (ITrayIcon*)lpVoid;
  371. CPoint pt;
  372. CRect IconRect;
  373. bool bChange = FALSE;
  374. do
  375. {
  376. if ( bChange)
  377. {
  378. pTrayIcon->SetIcon(pTrayIcon->_utIcon[1]);
  379. bChange = FALSE;
  380. }
  381. else
  382. {
  383. pTrayIcon->SetIcon(pTrayIcon->_utIcon[2]);
  384. bChange = TRUE;
  385. }
  386. if( pTrayIcon->bShowMinibox )
  387. {
  388. GetCursorPos(&pt);
  389. pTrayIcon->GetTrayRect(IconRect);
  390. if ( IconRect.PtInRect( pt) )
  391. ::PostMessage(pTrayIcon->m_nid.hWnd,MYWM_SHOWMINBOX,1,0);
  392. else
  393. ::PostMessage(pTrayIcon->m_nid.hWnd,MYWM_SHOWMINBOX,0,0);
  394. }
  395. }while (WaitForSingleObject(pTrayIcon->m_hThreadCtrl,300) == WAIT_TIMEOUT);
  396. return 0;
  397. }