Windows 7 任务栏开发 之 覆盖图标(Overlay Icon)

简介:

当第一次使用Windows 7 时,任务栏(Taskbar)的改变可以说让我眼前一亮。在以前版本Windows 桌面功能的基础上,Windows 7 任务栏为我们增添了许多其他功能:Jump Lists,Window Preview,Process Bar,Overlay Icon 等等。

 taskbar_thumb.png

       新任务栏的功能使我们的操作更加方便快捷,在参加北京.Net俱乐部举办的“Windows 7 发布”活动时初步了解到有关应用程序支持Windows 7 任务栏特性方面的内容。那么作为一名开发人员我们的应用程序能否实现这些功能呢?答案当然是“可以”,微软提供了方便的工具Windows API Code Pack for .NET Framework 来帮助我们完成这些开发。

       本篇主要演示Overlay Icon(MSDN上直译为“覆盖图标”)的功能,从字面上来看也许不太明白这是什么意思,在Windows 7 中UAC 功能为系统增加了安全性,当UAC 启动时有些应用程序的图标上方会显示一个盾牌标志(如下图),没错这个就是Overlay Icon 功能,下面就来看看它是如何实现的。

021_thumb.png

 

准备工作

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

 

单窗口示例

程序运行后的状态:

demo_thumb.png

  
在图标列表中选择Overlay Icon 后的不同效果:

icon1_thumb.png icon2_thumb.png icon3_thumb.png

 

多窗口示例

在默认情况下,如果从父窗口中调出子窗口,其任务栏图标是组合叠加在一起的(如下图):

multiwin_thumb_1.png 

 

       如果想为不同的窗口中实现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,两个窗口的任务栏图标才真正的完成了分离:

two_thumb.png 

通过SetOverlayIcon(IntPtr, Icon, String) 来设置指定窗口(TestWindow)的Overlay Icon:

TaskbarManager.Instance.SetOverlayIcon(ptr, icon, "Overlay Icon Demo");

 

分离后再来看看效果,只显示MainWindow 图标:

mainshow_thumb.png    mainshowres_thumb.png

两个窗口的图标都显示:

testshow_thumb.png    testshowres_thumb.png

 

相关参考资料

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

 

源代码下载





本文转自Gnie博客园博客,原文链接:http://www.cnblogs.com/gnielee/archive/2010/03/05/windows7-taskbar-overlayicon.html,如需转载请自行联系原作者

相关文章
|
11天前
如何隐藏windows10系统任务栏右下角的语言输入法图标?
如何隐藏windows10系统任务栏右下角的语言输入法图标?
|
1月前
|
数据可视化 数据库 C++
Qt 5.14.2揭秘高效开发:如何用VS2022快速部署Qt 5.14.2,打造无与伦比的Windows应用
Qt 5.14.2揭秘高效开发:如何用VS2022快速部署Qt 5.14.2,打造无与伦比的Windows应用
|
12天前
|
监控 安全 API
7.3 Windows驱动开发:内核监视LoadImage映像回调
在笔者上一篇文章`《内核注册并监控对象回调》`介绍了如何运用`ObRegisterCallbacks`注册`进程与线程`回调,并通过该回调实现了`拦截`指定进行运行的效果,本章`LyShark`将带大家继续探索一个新的回调注册函数,`PsSetLoadImageNotifyRoutine`常用于注册`LoadImage`映像监视,当有模块被系统加载时则可以第一时间获取到加载模块信息,需要注意的是该回调函数内无法进行拦截,如需要拦截则需写入返回指令这部分内容将在下一章进行讲解,本章将主要实现对模块的监视功能。
29 0
7.3 Windows驱动开发:内核监视LoadImage映像回调
|
4月前
|
监控 安全 API
7.2 Windows驱动开发:内核注册并监控对象回调
在笔者上一篇文章`《内核枚举进程与线程ObCall回调》`简单介绍了如何枚举系统中已经存在的`进程与线程`回调,本章`LyShark`将通过对象回调实现对进程线程的`句柄`监控,在内核中提供了`ObRegisterCallbacks`回调,使用这个内核`回调`函数,可注册一个`对象`回调,不过目前该函数`只能`监控进程与线程句柄操作,通过监控进程或线程句柄,可实现保护指定进程线程不被终止的目的。
29 0
7.2 Windows驱动开发:内核注册并监控对象回调
|
4月前
|
监控 安全 API
7.6 Windows驱动开发:内核监控FileObject文件回调
本篇文章与上一篇文章`《内核注册并监控对象回调》`所使用的方式是一样的都是使用`ObRegisterCallbacks`注册回调事件,只不过上一篇博文中`LyShark`将回调结构体`OB_OPERATION_REGISTRATION`中的`ObjectType`填充为了`PsProcessType`和`PsThreadType`格式从而实现监控进程与线程,本章我们需要将该结构填充为`IoFileObjectType`以此来实现对文件的监控,文件过滤驱动不仅仅可以用来监控文件的打开,还可以用它实现对文件的保护,一旦驱动加载则文件是不可被删除和改动的。
29 1
7.6 Windows驱动开发:内核监控FileObject文件回调
|
4月前
|
监控 安全 API
6.9 Windows驱动开发:内核枚举进线程ObCall回调
在笔者上一篇文章`《内核枚举Registry注册表回调》`中我们通过特征码定位实现了对注册表回调的枚举,本篇文章`LyShark`将教大家如何枚举系统中的`ProcessObCall`进程回调以及`ThreadObCall`线程回调,之所以放在一起来讲解是因为这两中回调在枚举是都需要使用通用结构体`_OB_CALLBACK`以及`_OBJECT_TYPE`所以放在一起来讲解最好不过。
41 1
6.9 Windows驱动开发:内核枚举进线程ObCall回调
|
4月前
|
监控 安全 API
6.8 Windows驱动开发:内核枚举Registry注册表回调
在笔者上一篇文章`《内核枚举LoadImage映像回调》`中`LyShark`教大家实现了枚举系统回调中的`LoadImage`通知消息,本章将实现对`Registry`注册表通知消息的枚举,与`LoadImage`消息不同`Registry`消息不需要解密只要找到`CallbackListHead`消息回调链表头并解析为`_CM_NOTIFY_ENTRY`结构即可实现枚举。
48 1
6.8 Windows驱动开发:内核枚举Registry注册表回调
|
4月前
|
存储 API 开发者
6.7 Windows驱动开发:内核枚举LoadImage映像回调
在笔者之前的文章`《内核特征码搜索函数封装》`中我们封装实现了特征码定位功能,本章将继续使用该功能,本次我们需要枚举内核`LoadImage`映像回调,在Win64环境下我们可以设置一个`LoadImage`映像加载通告回调,当有新驱动或者DLL被加载时,回调函数就会被调用从而执行我们自己的回调例程,映像回调也存储在数组里,枚举时从数组中读取值之后,需要进行位运算解密得到地址。
32 1
6.7 Windows驱动开发:内核枚举LoadImage映像回调
|
4月前
|
监控 安全 API
7.5 Windows驱动开发:监控Register注册表回调
在笔者前一篇文章`《内核枚举Registry注册表回调》`中实现了对注册表的枚举,本章将实现对注册表的监控,不同于32位系统在64位系统中,微软为我们提供了两个针对注册表的专用内核监控函数,通过这两个函数可以在不劫持内核API的前提下实现对注册表增加,删除,创建等事件的有效监控,注册表监视通常会通过`CmRegisterCallback`创建监控事件并传入自己的回调函数,与该创建对应的是`CmUnRegisterCallback`当注册表监控结束后可用于注销回调。
45 0
7.5 Windows驱动开发:监控Register注册表回调
|
4月前
|
存储 安全 数据安全/隐私保护
3.2 Windows驱动开发:内核CR3切换读写内存
CR3是一种控制寄存器,它是CPU中的一个专用寄存器,用于存储当前进程的页目录表的物理地址。在x86体系结构中,虚拟地址的翻译过程需要借助页表来完成。页表是由页目录表和页表组成的,页目录表存储了页表的物理地址,而页表存储了实际的物理页框地址。因此,页目录表的物理地址是虚拟地址翻译的关键之一。在操作系统中,每个进程都有自己的地址空间,地址空间中包含了进程的代码、数据和堆栈等信息。为了实现进程间的隔离和保护,操作系统会为每个进程分配独立的地址空间。在这个过程中,操作系统会将每个进程的页目录表的物理地址存储在它自己的CR3寄存器中。当进程切换时,操作系统会修改CR3寄存器的值,从而让CPU使用新的页
48 0
3.2 Windows驱动开发:内核CR3切换读写内存