1,目的
之前介绍了枚举进程并关闭进程的方法,可是有一个问题,使用TerminateProcess关闭的进程如果生成过托盘图标,这个图标就会停留在右下角的托盘区(因为该进程来不及执行自己的图标清理代码),直到鼠标移上去时才消失。显然这个效果并不理想。
2,思路
这里我们可以自己给这个托盘区窗口发送鼠标经过消息,使其更新。
用vs附带的spy++工具可以很方便的查看当前所有窗口的属性。
如下:
我是WIN7系统,其中,Shell_TrayWnd类的窗口就是桌面下面这一整条任务栏。
“用户升级的通知区域”就是那个托盘图标窗口了。
你可能会说,我也不知道哪个是哪个窗口啊?
很容易,右键每条窗口,点“突出显示”,就可以闪烁该窗口的轮廓方便你找到它。
3,实现
完整的关闭QQ并退出QQ图标的代码:
#include "stdafx.h" #include <Windows.h> #include "tlhelp32.h" //刷新任务栏图标 void RefreshTaskbarIcon() { //任务栏窗口 HWND hShellTrayWnd = ::FindWindow("Shell_TrayWnd",NULL); //任务栏右边托盘图标+时间区 HWND hTrayNotifyWnd = ::FindWindowEx(hShellTrayWnd,0,"TrayNotifyWnd",NULL); //不同系统可能有可能没有这层 HWND hSysPager = ::FindWindowEx(hTrayNotifyWnd,0,"SysPager",NULL); //托盘图标窗口 HWND hToolbarWindow32; if (hSysPager) { hToolbarWindow32 = ::FindWindowEx(hSysPager,0,"ToolbarWindow32",NULL); } else { hToolbarWindow32 = ::FindWindowEx(hTrayNotifyWnd,0,"ToolbarWindow32",NULL); } if (hToolbarWindow32) { RECT r; ::GetWindowRect(hToolbarWindow32,&r); int width = r.right - r.left; int height = r.bottom - r.top; //从任务栏中间从左到右 MOUSEMOVE一遍,所有图标状态会被更新 for (int x = 1; x<width; x++) { ::SendMessage(hToolbarWindow32,WM_MOUSEMOVE,0,MAKELPARAM(x,height/2)); } } } void CloseQQ() { PROCESSENTRY32 pe; DWORD id=0; HANDLE hSnapshot=CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0); pe.dwSize=sizeof(PROCESSENTRY32); if(!Process32First(hSnapshot,&pe)) return; while(1) { pe.dwSize=sizeof(PROCESSENTRY32); if(Process32Next(hSnapshot,&pe)==FALSE) break; //是qq就关闭它 if(0 == stricmp(pe.szExeFile,"qq.exe")) { HANDLE hProcess = OpenProcess(PROCESS_TERMINATE,FALSE,pe.th32ProcessID); if(hProcess != NULL) { TerminateProcess(hProcess,0); } } } CloseHandle(hSnapshot); } int main(int argc, char* argv[]) { //关闭QQ CloseQQ(); //刷新任务栏图标 RefreshTaskbarIcon(); return 0; }
© 版权声明
文章版权归作者所有,未经允许请勿转载。
THE END