程序异常退出后,托盘残留图标清理方法(C#)
程序猿对于宕机再熟悉不过了,如果一个程序异常退出,那我们只能在Windows日志中找到它的踪迹;如果它是一个托盘程序,异常退出后,它会停留在托盘中久久不能离去,直到我们的鼠标从它身上划过;如果一个程序被另一个守护程序不断的杀掉、重启,那么它的结果会如下图所示,
请注意!!这不是扫雷!!没错,这正是残留的图标。解决方法其实很简单,在程序每次开启前,向托盘窗口发送鼠标经过消息WM_MOUSEMOVE,使其刷新。
首先要明确一点,托盘包括显示区域和溢出区域,所以需要同时刷新,防止遗漏。可以用Spy++鼠标右键选择“突出显示”,窗口轮廓就会闪烁,SysPager和NotifyIconOverflowWindow。
代码如下:
using System; using System.Runtime.InteropServices; public class TaskBarUtil { struct RECT { public int left, top, right, bottom; } public static void RefreshNotification() { var NotifyAreaHandle = GetNotifyAreaHandle(); if (NotifyAreaHandle != IntPtr.Zero) RefreshWindow(NotifyAreaHandle); var NotifyOverHandle = GetNotifyOverHandle(); if (NotifyOverHandle != IntPtr.Zero) RefreshWindow(NotifyOverHandle); } private static void RefreshWindow(IntPtr windowHandle) { const uint WM_MOUSEMOVE = 0x0200; RECT rect; GetClientRect(windowHandle, out rect); for (var x = 0; x < rect.right; x += 5) for (var y = 0; y < rect.bottom; y += 5) SendMessage(windowHandle, WM_MOUSEMOVE, 0, (y << 16) + x); } private static IntPtr GetNotifyAreaHandle() { var TrayWndHandle = FindWindowEx(IntPtr.Zero, IntPtr.Zero, "Shell_TrayWnd", string.Empty); var TrayNotifyWndHandle = FindWindowEx(TrayWndHandle, IntPtr.Zero, "TrayNotifyWnd", string.Empty); var SysPagerHandle = FindWindowEx(TrayNotifyWndHandle, IntPtr.Zero, "SysPager", string.Empty); var NotifyAreaHandle = FindWindowEx(SysPagerHandle, IntPtr.Zero, "ToolbarWindow32", string.Empty); return NotifyAreaHandle; } private static IntPtr GetNotifyOverHandle() { var OverHandle = FindWindowEx(IntPtr.Zero, IntPtr.Zero, "NotifyIconOverflowWindow", string.Empty); var NotifyOverHandle = FindWindowEx(OverHandle, IntPtr.Zero, "ToolbarWindow32", string.Empty); return NotifyOverHandle; } [DllImport("user32.dll", SetLastError = true)] static extern IntPtr FindWindowEx(IntPtr parentHandle, IntPtr childAfter, string className, string windowTitle); [DllImport("user32.dll")] static extern bool GetClientRect(IntPtr handle, out RECT rect); [DllImport("user32.dll", CharSet = CharSet.Auto)] static extern IntPtr SendMessage(IntPtr handle, UInt32 message, Int32 wParam, Int32 lParam); }
TaskBarUtil.RefreshNotification();
托盘又恢复往日简洁 ^_^