对通知区域(Win 系统托盘)窗口层次的分析

目录

前言

一、气球通知(非 Toast 应用通知)

二、通知区域 

1)用户通知区域(UserPrompt Notification Area)

2)系统升级的通知区域(System Upgrade Notification Area)

3)任务栏角溢出通知区域(Taskbar Corner Overflow Notification Area)

三、MS 对应用通知的设计

3.1 什么是应用通知(Toast 通知)?

3.2 应用通知是如何实现的?


本文出处链接:https://blog.csdn.net/qq_59075481/article/details/128538050。

前言

Windows 的通知区域也就是我们俗称的 系统托盘,该区域是为了显示应用程序或系统的重要通知以及支持应用程序的最小化操作而诞生的。关于他的历史由来有一些有趣的说法,有兴趣的可以搜索资料了解一下。

在前面的几篇文章中,我详细介绍了在新旧不同版本的 Windows 系统上获取通知区域图标信息的方法。后来,我考虑到:如果缺乏对通知区域界面的基本知识的了解,可能会影响通知区域图标信息的获取方法的理解。所以我就决定将之前整理的一篇介绍通知区域窗口层次的文章展示出来,由于 Windows 对 UI 设计的理念一直在变化,所以这篇文章的分析难免有疏忽之处。这一点还希望大家指点。

P.S. :这篇文章是早在 2022 年就编写的文章,因为当初时间紧,没有写完而一直没有发布。这一点从文章的 AID 其实就可以看出(AID: 128 开头大概是 12 月底的文章序号)。当初是准备先写窗口层次分析,然后再介绍 TB_BUTTON 消息机制的处理方法。当时 Win11 刚出来不久,传统的 TrayBar 的设计还没有被废除。后来,一次偶然间发现之前编写的工具在 Win11 上失效了,于是就有了前面几篇的研究文章。

 系列文章列表:

编号文章标题AID
1

获取系统托盘图标信息的尝试

128594435
2获取 Windows 系统托盘图标信息的最新方案(一)136134195
3获取 Windows 系统托盘图标信息的最新方案(二)136199670
4获取 Windows 系统托盘图标信息的最新方案(三)136240462
5对通知区域(Win 系统托盘)窗口层次的分析128538050

一、气球通知(非 Toast 应用通知)

通知通过简要显示通知区域中图标中的气球(见图示[1]、[2]、[3]、[4]),来通知用户与当前用户活动无关的事件。通知可能是用户操作或重大系统事件导致的,或者可能提供来自 Microsoft Windows 或应用程序的有用信息。

通知中的信息 非常有用且相关,但从不至关重要。 因此,通知不需要即时用户操作,用户可以自由忽略它们。

传统通知主要通过桌面右下角弹出气球窗口的方式提示用户一些有用的信息,下图展示了从 WinXP 到 Win11 系统的部分气球通知消息。

图[1] - WinXP中的气球通知样式
图[2] - Win 7中的气球通知样式

图[3] - Win 10中的气球通知样式

图[4] - Win 11中的气球通知样式

这种气球通知需要应用程序在通知发送期间暂时显示通知区域图标,或者长期显示通知区域图标。主要实现的方式是:1.通过 SHQueryUserNotificationState 函数确定用户何时处于活动状态;2.通过 Shell_NotifyIcon 函数向通知区域发送通知的设置。

二、通知区域 

在刚刚的描述中,我们一直谈到名为“通知区域”的部分,那么什么是通知区域呢?

通知区域是任务栏的一部分,提供通知和状态的临时源,默认位于任务栏的右侧(提示:本文中任务栏位置按照默认的屏幕下方模式,且部分的 Win11 操作系统可能无法调整任务栏位置)。

它还可用于显示桌面上没有状态的系统和程序功能的图标,例如电池电量、音量控制和网络状态。 通知区域以前称为系统托盘或状态区域。

通知区域中的项称为通知区域图标,或者如果已明确建立通知区域的上下文,则只是图标。

意思就是说:如果只是为了通知而临时添加的图标和为了更好的功能而本来就添加到任务栏的托盘图标都属于通知区域图标,但后者更趋向于是不显示在桌面的缩小版图标。

微软为了更好的用户体验,所以一直在更改 UI 的设计。在 XP 上,只有一个用户通知区域(旧称系统托盘),在图标过多的时候可以隐藏部分图标,但并不是重新划分区域收纳这些图标。从 Win 7 开始的系统中,改用分开的通知区域来实现更加清洁的任务栏,并且在以后的设计中,除了圆角设计和新增加的 Toast 通知以外,没有太大的变动,但有些细节并不一致。

下面展示显式通知区域(类名均为:ToolbarWindow32)的组成:

1)用户通知区域(UserPrompt Notification Area)

听名字就可以了解到这个窗口是面向用户的,如图 [5] 所示,红色框圈出的区域为用户提示的通知区域,这个区域在 Win7 至 Win10 是默认显示在任务栏上的,用于显示常用应用的通知栏图标(在 XP 中只有这一个显式通知区域)

图[5]  用户通知区域示意图

在 Win11 中,如果没有在 “设置>任务栏设置>其它系统托盘图标” 中标记显示通知区域图标,则该窗口矩形大小将为 0\times 0,但窗口始终为 Visible(可见)。

图[6]  任务栏图标设置页面

在 Win7 中,需要在任务栏属性菜单>通知区域>自定义>取消始终显示...的可选项,并将每一个图标均设置为不显示图标和通知,才能使得该通知区域隐藏(窗口大小为 0,且 Invisible 不可见)。

图[7]  通知区域图标设置页

可以看出,微软在全新的设计理念中,看似完全颠覆了以前的设置。同样的话官方也不是没讲过,在 Windows 7 设计指南中有这样一个粉标备注:

图[8]

这个区域对应的窗口标题为:用户提示的通知区域(Win10、Win11)\用户升级的通知区域( Win7 ),以下简称(用户通知区)。窗口类的类名为:ToolbarWindow32。属于三级子窗口( SysPager 的子窗口,该部分窗口结构从 WinXP 沿用至今),窗口结构为:

      #32769 (桌面)- Shell_TrayWnd- TrayNotifyWnd- SysPager- "用户通知区"ToolbarWindow32

2)系统升级的通知区域(System Upgrade Notification Area)

系统升级的通知区域(以下简称系统通知区)光看名字,对很多人来说一定很陌生,甚至很容易忽略。这个窗口一般是看不到的,因为它的 NormalSize 为 0(注意如果需要检测该窗口矩形,请使用高 DPI 设置,否则将得到意外的结果),一般是看不到的,但是窗口却是 Visible(可见),只是因为大小为 0,而 UI 界面不绘制它的边框(毕竟那样不美观),所以一般看不到。

那我们什么时候能看到呢?

我们需要先了解这个通知区域的基本功能,系统通知区域用于部分应用通知图标的临时显示,如果这个应用没有设置为始终显示(或者没有设置始终显示所有图标和通知,Win7 ),那么,当程序刚刚创建通知图标的时候(一般为程序启动时),系统会在其注册图标后约一分钟内持续将其显示在系统通知区内,然后会将其自动移动到溢出通知区域(后面将会讲到)。

系统通知区的位置一处通知区 Button 的右侧,在用户通知区的左侧,当它里面容纳临时图标的时候,他会并排显示在用户通知区左侧(紧挨着)。如图 [9,10] 所示。

图[9] - 任务管理器没有打开前
图[10] - 任务管理器打开后

比如在任务管理器打开前后任务栏的变化:这边的任务管理器图标没有设置为始终显示,图中 OneDrive 和 QQ 被设置为了始终显示,且当前正在运行中。正常设置了始终显示属性的图标,默认是在用户通知区向右排列,而新打开的任务管理器图标显示在用户通知区的左侧边缘外。

使用 Microsoft Spy ++ 分析窗口堆叠树层结构(分析窗口位置),不难发现,任务管理器当前是显示在新的窗口中,而不是用户通知区窗口。这个窗口应该是标题为“系统升级的通知区域”,类名为 ToolbarWindow32 的窗口,属于二级子窗口(与 SysPager 同级)。微软使用该通知区域来强调一个不经常显示的图标的创建操作,以便于提醒用户新的通知程序被创建,同时对于那些真正因为临时通知而不得不显示图标的应用提供便捷的舞台。当预计的超时时间结束时,无论是否是临时显示的通知图标还是长期使用但不需要显示的图标,都会被移动到溢出通知区域(默认处于折叠状态),除非设置了始终可见通知图标;当它显示临时图标的时侯,系统通知区才会显示出来(矩形大小是按照当前排队的临时图标格式来约定的),否则它的大小为 0。 

图[11]  系统升级的通知区域窗口

【备注】:如果是临时图标,在超时前处理完通知,并且结束线程或者注销图标,之后不会再在溢出通知区域创建新图标。如果临时图标没有在超时前完成通知,或者不准备注销或者隐藏通知图标,那么会被移动到溢出通知区域。溢出通知区域在外壳程序重启前或者新的命令发布前,不会尝试刷新或者自动删除这些图标。

3)任务栏角溢出通知区域(Taskbar Corner Overflow Notification Area)

任务栏角溢出通知区域是在 Win7 就引入的一个通知区域。目的是解决通知区域图标个数过多时,使得任务栏臃肿的问题。系统通过设置的配置,选择将大多数不那么重要的通知图标放入到一个自动折叠起来的通知区域。用户通过任务栏上的一个角标按钮来打开这个通知区域:

图[12]  任务栏角溢出

任务栏角溢出通知区域窗口是一个只有窗口类名称(NotifyIconOverflowWindow)的窗口:

图[13]  NotifyIconOverflowWindow 窗口层次

用户可以通过手动操作鼠标左键,按住并拖动图标在“溢出通知区域”和“用户升级通知区域”间移动位置。或者使用设置的 “个性化>任务栏>其它系统托盘图标” (Win11)以及 “控制面板>通知区域图标” (Win10 或更早)等设置来修改图标可见性。

三、MS 对应用通知的设计

3.1 什么是应用通知(Toast 通知)?

应用通知 (以前称 Toast 通知) 是具有文本、图像和按钮/输入的灵活通知。这是更适应现代化 UI 设计的通知框架,基于 C++/WinRT 和 XML 语言的应用通知框架从设计之初就是面向对象的。自适应和交互式应用通知允许开发者创建带有更多内容的灵活弹出通知、可选的嵌入式图像和可选的用户交互。

我们把应用通知的可视部分称作通知内容的视觉对象,其包括包含文本和图像的泛型绑定。下面是应用通知内容的视觉对象的示意图:

图[14]  应用通知的视觉对象

应用通知还支持可选图像设置(图[] 蓝色框)。图像可来自于应用包、应用的本地存储或来自 Web。以及标准的输入框和多个自定义按钮以及按钮功能(图[] 红色框、绿色框)。

图[15]  应用通知的其他特性

以上只是应用通知框架的最基本的应用,应用通知还有更多的有关视觉、交互的元素。例如:选项卡、音频、上下文菜单、按钮图标和颜色等即时交互元素。

3.2 应用通知是如何实现的?

所有类型的应用(WPF、UWP、WinForms、控制台 C/C++/C#)都可以发送应用通知。

根据 MS 的文档实现会很复杂,一般情况下可以使用 Github 上的开源项目来轻松完成任务。

比如 WinToast 这个开源项目就非常好用,它包含 Console 终端模式和 Qt 6 窗口模式。

图[16] Console 版的 WinToast

文章出处链接:https://blog.csdn.net/qq_59075481/article/details/128538050。

本文发布于:2024.03.04,更新于:2024.03.04。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/508699.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QT多语言切换功能

一.目的 在做项目时,有时希望我们的程序可以在不同的国家使用,这样最好的方式是一套程序能适应于多国语言。 Qt提供了这样的功能,使得一套程序可以呈现出不同的语言界面。本文将介绍QT如何实现多语言,以中文和英文为例。 QT开发…

快速幂(求解原理+例题)

目录 反复平方法(快速幂): 代码: 例题:快速幂求逆元 作用: 快速求出 的结果。 时间复杂度: O(logk) 如果使用一般做法,从1循环到k,时间复杂度是O(k) 反复平方法&am…

那些壁纸,不只是背景

1、方小童在线工具集 网址: 方小童 该网站是一款在线工具集合的网站,目前包含PDF文件在线转换、随机生成美女图片、精美壁纸、电子书搜索等功能,喜欢的可以赶紧去试试!

kafka同步副本集及关键参数

上篇文章讲了副本机制是什么,一文读懂kafka内部怎么运行的-CSDN博客 这里深挖下同步副本集及里面的关键参数。副本会去leader副本拉去数据追加到自己日志中。 我们知道kafka副本的作用是提高系统的高可用。当leader副本挂了时,会从候选副本集中选者一个当…

ubuntu环境下docker容器详细安装使用

文章目录 一、简介二、ubuntu安装docker1.删除旧版本2.安装方法一3. 安装方法二(推荐使用)4.运行Docker容器5. 配置docker加速器 三、Docker镜像操作1. 拉取镜像2. 查看本地镜像3. 删除镜像4. 镜像打标签5. Dockerfile生成镜像 四、Docker容器操作1. 获取…

H12-821_113

113.如图所示是路由器现ATE输出的部分信息,以下关于这部分信息的描述,错误的是哪一项? A.display pim rp-info命令用来查看组播组对应的RP信息 B.RP地址是2.2.2.2 C.组地址是225.0.0.0 D.RP的优先级是0 答案:C 注释: …

STM32-BKP备份寄存器和RTC时钟

BKP介绍 BKP(Bckup Registers)备份寄存器 备份寄存器是42个16位的寄存器,可用来存储84个字节的用户应用程序数据。他们处在备份域里,当VDD电源被切断,他们仍然由VBAT(备用电池电源)维持供电。当系统在待机…

mprpc分布式RPC网络通信框架

mprpc 项目介绍 该项目是一个基于muduo、Protobuf和Zookeeper实现的轻量级分布式RPC网络通信框架。 可以把任何单体架构系统的本地方法调用,重构成基于TCP网络通信的RPC远程方法调用,实现同一台机器的不同进程之间的服务调用,或者不同机器…

Lua 篇(一)— 安装运行Hello World

目录 前言一、Lua 是什么?二、Lua和C#的区别三、安装 LuaLinux 系统上安装Mac OS X 系统上安装Window 系统上安装emmyluaRider 安装(推荐) 四、Lua学习资料 前言 Lua 是一种轻量级的嵌入式脚本语言,它可以与 C 语言无缝集成,提供了强大的编程…

关于Facebook,你真的足够了解吗?

一、前言 说到站外推广,各位跨境卖家都不陌生,当站内流量较难获取时,总会想着法子从外部获取流量,而Facebook形式的站外推广各位卖家肯定也耳熟能详,低投入高回报一直是服务商主打的“性价比首选”。 二、Facebook基…

PclSharp1.12.0--半径滤波

一、半径滤波 原理&#xff1a;对整个输入迭代一次&#xff0c;对于每个点进行半径R邻域搜索&#xff0c;如果邻域点的个数低于某一阈值&#xff0c;则该点将被视为噪声点并被移除。 二、代码部分 注&#xff1a;测试代码均使用PclSharp1.12.0库 /// <summary> …

【已解决】Word文档无法复制怎么办?

无法复制Word文档里面的内容&#xff0c;有可能是这3个原因造成的&#xff0c;一起来看看如何解决吧。 原因1&#xff1a;Word程序出现故障 Word程序出现故障&#xff0c;有可能会导致打不开Word文档&#xff0c;或者打开后无法进行复制粘贴等操作&#xff0c;可以试试关闭Wor…