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;
}