当第一次使用Windows 7 时,任务栏(Taskbar)的改变可以说让我眼前一亮。在以前版本Windows 桌面功能的基础上,Windows 7 任务栏为我们增添了许多其他功能:Jump Lists,Window Preview,Process Bar,Overlay Icon 等等。
新任务栏的功能使我们的操作更加方便快捷,在参加北京.Net俱乐部举办的“Windows 7 发布”活动时初步了解到有关应用程序支持Windows 7 任务栏特性方面的内容。那么作为一名开发人员我们的应用程序能否实现这些功能呢?答案当然是“可以”,微软提供了方便的工具Windows API Code Pack for .NET Framework 来帮助我们完成这些开发。
本篇主要演示Overlay Icon(MSDN上直译为“覆盖图标”)的功能,从字面上来看也许不太明白这是什么意思,在Windows 7 中UAC 功能为系统增加了安全性,当UAC 启动时有些应用程序的图标上方会显示一个盾牌标志(如下图),没错这个就是Overlay Icon 功能,下面就来看看它是如何实现的。
准备工作
1. 添加一些Overlay Icon 图片(.ico)到项目Resources 中。
2. 下载Windows API Code Pack,增加Microsoft.WindowsAPICodePack.dll 和Microsoft.WindowsAPICodePack.Shell.dll。
3. 引用Microsoft.WindowsAPICodePack.Taskbar 命名空间。
TaskbarManager.SetOverlayIcon(Icon, String) 方法
TaskbarManager 类提供了一些常用的任务栏特性设置方法,其中SetOverlayIcon 方法有三种不同的方式来设置Overlay Icon:
//用于应用程序主窗口 public void SetOverlayIcon(System.Drawing.Icon icon, string accessibilityText) { CoreHelpers.ThrowIfNotWin7(); TaskbarList.SetOverlayIcon(OwnerHandle,
icon != null ? icon.Handle : IntPtr.Zero, accessibilityText); } //用于指定窗口 public void SetOverlayIcon(IntPtr windowHandle, System.Drawing.Icon icon,
string accessibilityText) { CoreHelpers.ThrowIfNotWin7(); TaskbarList.SetOverlayIcon(windowHandle,
icon != null ? icon.Handle : IntPtr.Zero, accessibilityText); } //用于指定的WPF窗口 public void SetOverlayIcon(System.Windows.Window window,
System.Drawing.Icon icon, string accessibilityText) { CoreHelpers.ThrowIfNotWin7(); TaskbarList.SetOverlayIcon( (new WindowInteropHelper(window)).Handle, icon != null ? icon.Handle : IntPtr.Zero, accessibilityText); }
在程序中通过TaskbarManager.Instance.SetOverlayIcon() 即可实现Overlay Icon 效果:
Icon icon = iconList.SelectedItem as Icon; TaskbarManager.Instance.SetOverlayIcon(icon, "Overlay Icon Demo");
如果将Icon 和 String 都设为Null 则取消Overlay Icon 效果:
TaskbarManager.Instance.SetOverlayIcon(null, null);
单窗口示例
程序运行后的状态:
在图标列表中选择Overlay Icon 后的不同效果:
多窗口示例
在默认情况下,如果从父窗口中调出子窗口,其任务栏图标是组合叠加在一起的(如下图):
如果想为不同的窗口中实现Overlay Icon 则首先需要通过修改TestWindow 的Application ID (AppID)将两个窗口的任务栏图标分离开。每个运行的窗口都会有各自的AppID,用来决定任务栏图标属于哪个窗口。这也就是为什么当我们打开多个Word 文档或IE 标签后任务栏图标都是自动叠加在一起的,所以我们可以通过修改窗口的AppID 使任务栏图标分开显示。
通过使用TaskbarManager.SetApplicationIdForSpecificWindow(IntPtr windowHandle,String appID) 方法可以修改窗口的AppID。但是目前下载的Windows API 1.0.1 版本有些小问题,使得SetApplicationIdForSpecificWindow 方法根本不起作用。根源就在TaskbarNativeMethods.cs 的SetWindowProperty 方法,没有对pv 进行任何赋值操作,导致propStore 根本没有值,所以在该方法中增加pv.SetString(value),重新编译并替换掉原来的Microsoft.WindowsAPICodePack.Shell.dll 即可:
internal static void SetWindowProperty(IntPtr hwnd, PropertyKey propkey,
string value) { // Get the IPropertyStore for the given window handle IPropertyStore propStore = GetWindowPropertyStore(hwnd); // Set the value PropVariant pv = new PropVariant(); pv.SetString(value); propStore.SetValue(ref propkey, ref pv); // Dispose the IPropertyStore and PropVariant Marshal.ReleaseComObject(propStore); pv.Clear(); }
程序改好后,就可以使用SetApplicationIdForSpecificWindow(IntPtr, String) 干活了:
Window newWindow = new TestWindow(); newWindow.Show(); WindowInteropHelper helper = new WindowInteropHelper(newWindow); IntPtr ptr = helper.Handle; TaskbarManager.Instance.SetApplicationIdForSpecificWindow(ptr, "AppID");
修改了TestWindow 的AppID,两个窗口的任务栏图标才真正的完成了分离:
通过SetOverlayIcon(IntPtr, Icon, String) 来设置指定窗口(TestWindow)的Overlay Icon:
TaskbarManager.Instance.SetOverlayIcon(ptr, icon, "Overlay Icon Demo");
分离后再来看看效果,只显示MainWindow 图标:
两个窗口的图标都显示:
相关参考资料
1. Windows 7 New Taskbar - An Overview
http://channel9.msdn.com/posts/yochay/Windows-7-New-Taskbar-an-overview/
2. The Windows 7 Taskbar
http://blogs.msdn.com/e7/archive/2008/11/20/happy-anniversary-windows-on-the-evolution-of-the-taskbar.aspx
3. Windows API Code Pack for .NET Framework
http://code.msdn.microsoft.com/WindowsAPICodePack
4. Introducing The Taskbar APIs
http://msdn.microsoft.com/en-us/magazine/dd942846.aspx
5. Windows 7 Taskbar Dynamic Overlay Icons and Progress Bars
http://windowsteamblog.com/blogs/developers/archive/2009/07/28/windows-7-taskbar-dynamic-overlay-icons-and-progress-bars.aspx
6. Coding 4 Fun - Windows 7 Taskbar
http://blogs.msdn.com/coding4fun/archive/2009/08/18/9874533.aspx