清除异常关闭进程的任务栏托盘区残留图标

        使用TerminateProcess关闭的进程如果生成过托盘图标,这个图标就会停留在右下角的托盘区(因为该进程来不及执行自己的图标清理代码),直到鼠标移上去时才消失。显然这个效果并不理想。


 

        这里我们可以自己给这个托盘区窗口发送鼠标经过消息,使其更新。用vs附带的spy++工具可以很方便的查看当前所有窗口的属性。

如下:


我是WIN7系统,其中,Shell_TrayWnd类的窗口就是桌面下面这一整条任务栏。

“用户升级的通知区域”就是那个托盘图标窗口了。



你可能会说,我也不知道哪个是哪个窗口啊?

很容易,右键每条窗口,点“突出显示”,就可以闪烁该窗口的轮廓方便你找到它。




 

完整的关闭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;
}

你可能感兴趣的:(用户,include,清理,托盘,任务栏)