程序异常退出后,托盘残留图标清理方法(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();

托盘又恢复往日简洁 ^_^

 

 

posted @ 2019-08-23 22:24  吴小波  阅读(3375)  评论(5编辑  收藏  举报