《Windows API每日一练》9.2.1 菜单

news2025/1/17 23:19:00

和菜单有关的概念

窗口的菜单栏紧挨着标题栏下面显示。这个菜单栏有时叫作程序的“主菜单”或“顶级菜单“(top-level menu)。顶级菜单中的菜单项通常会激活下拉菜单(drop-downmenu),也 叫“弹出菜单”(popup menu)或“子菜单”(submenu)。你可以定义多级嵌套的弹出菜单: 一个弹出菜单项可以激活另一个出菜单。有时弹出菜单项可以激活对话框来提供更多信息。(对话框在第十章讨论。)许多父窗口在标题栏的最左边显示程序的小图标。这个图标会激活系统菜单,该菜单实际上是另一种弹出菜单。

●弹出菜单的菜单项可以被“选中”(checked),亦即Windows在菜单文本左边显示一个 小的选中标记。选中标记的使用让用户可以选择不同的程序选项。这些选项可以是互相排斥的,当然不是必须要这样做。顶级菜单项不能被选中。

●顶级菜中或弹出菜单的菜单项可以被“启用”(enabled)、“禁用”(disabled)或“变灰”(grayed)。单词“活动”(Active)和“非活动” (Inactive)有时可以和“启用”和“禁用”同义使用。标记为启用或禁用的菜单项对用户来讲看起来一样,但变灰菜单项显示为灰色文本。

从用户的角度看,启用、禁用或变灰的菜单项都能被“选择”(加亮)。也就是说,用户可以在禁用的菜单项上单击鼠标,或者将反色显示(reverse-video)的光标条移动到一个禁 用的菜单项,或者使用菜单项的快捷键字母来触发禁用的菜单项。然而,从程序的角度,启用、 禁用或变灰的菜单项功能不同。Windows只向被启用的菜单项发送WM_COMMAND消息。对当前无效的菜单选项你可使用禁用或变灰的办法。如果你想让用户知道选项是无效的,最好将它变灰。

差别

主菜单(顶级菜单)

子菜单(弹出菜单)

被选中(checked)

不能

可以

启用/禁用  enabled/disabled

活动/非活动(Active/Inactive

可以

可以

变灰(grayed

可以

可以

WM_COMMAND消息

启用时,可发送。禁用或变灰里不能

句柄

有独立句柄

有独立句柄

菜单结构

在程序中创建或修改菜单时,将顶级菜单和每个弹出菜单想象成独立的菜单会有利于理解。顶级菜单有一个菜单句柄,在顶级菜单中的每个弹出菜单也都有自己的菜单句柄,系统菜单(也是一个弹出菜单)也有一个菜单句柄。

每个菜单项由三个特征定义。第一个特征是菜单显示什么。这可以是一个文本字符串或是一个位图。第二个特征是一个ID号或一个指向弹出菜单的句柄,Windows会在 WM_COMMAND消息中把ID号发送给你的程序,而弹出菜单则在用户选择该菜单项时由 Windows显示出来。第三个特征描述了菜单项的属性,包括该菜单项是否被禁用、变灰或选中。

定义菜单

要使用VS给程序的资源脚本加入菜单,应从Insert菜单中选择Resource,然后选择Menu。然后你就可以交互式定义菜单。菜单中的每一项都有一个相关联的Menu Item Properties对话框,用来指示菜单项的文本字符串。 如果Pop-up框被选中,则该菜单项会激活一个弹出菜单,这时它没有相关联的ID。如果 Pop-up框没被选中,那么该菜单项会生成一个带有特定ID的WM_COMMAND消息。这两种类型的菜单项会分别以POPUP和MENUITEM语句的形式在资源脚本中出现。

图9-10 创建菜单项

在为菜单中的一项输入文本时,可以输入一个符号&来指示Windows在显示菜单时给 紧接着&的下一个字符显示下划线。用Alt键选择一个菜单项时,Windows就寻找这样一个 带下划线的字符。如果不在文本中包含字符&,下划线不会出现,Windows将会用菜单项文本的第一个字母来进行Alt键搜索。

如果在Menu Items Properties对话框中选择了 Grayed选项,则表示该菜单项是非活动

的,它的文本会变灰,并且不会产生WM_COMMAND消息。如果选择Inactive选项,则表示该菜单项是非活动的,不会产生WM_COMMAND消息,但是它的文本会被正常显示。 Checked选项会在菜单项的旁边加一个复选标记。Separator选项会在弹出菜单上绘制一条 水平的分隔线。

对弹出菜单中的菜单项,可以在字符串中使用分栏制表符\t。即使弹出菜单第一栏的字 符串很长,\t后面的文本也会被放置在右边足够远的新一栏中。当我们学习键盘加速键时,我们会看到它是如何工作的。字符串中的\a会将它后面的文本进行右对齐。

指定的ID值是Windows在菜单消息中发给窗口过程的数字。ID值在一个菜单中应该 是唯一的。按照惯例,我们 使用以IDM(ID for a Menu)开头的标识符。                       

图9-11 设置菜单属性

                                                              

●菜单的单个特征:

特征

说明

①显示内容

1、表示文本字符串或位图。

2、带&指示紧接的字符显示下划线,配合Alt键搜索

3Spparator绘制水平分隔线

4\t制表符后面的文本在新一栏中

5\a后面的文本进行右对齐

ID

该项为MENUITEM(菜单项):则为菜单ID,会发送WM_COMMAND

POPUP(弹出菜单): 为菜单句柄

③属性

是否被禁用、变灰或选中等。非活动时,不会产生WM_COMMAND消息。

                                  

在程序中引用菜单

大多数Windows应用程序在资源脚本中只有一个菜单。可以给该菜单指定一个与程序 名一样的文本名字。程序员经常使用程序名作为菜单名,这样同一字符串可以作为窗口类名、程序图标名和菜单名。然后程序可以在窗口类定义中引用这个菜单:

wndclass.IpszMenuName = szAppName;

虽然在窗口类中指定菜单是引用菜单资源的最通常的方法,但它不是唯一的方法。 Windows应用程序可以用LoadMenu函数把菜单资源加载到内存,这和前面描述的Loadlcon 和LoadCursor函数非常类似。LoadMenu返回一个菜单的句柄。如果在资源脚本中为菜单设定了一个名字,那么该语句看起来会像这样:

hMenu = LoadMenu (hlnstance, TEXT ("MyMenu"));

如果使用数字,那么LoadMenu调用格式如下:

hMenu = LoadMenu (hlnstance, MAKEINTRESOURCE (ID_MENU))

之后便可以把这个菜单句柄指定为CreateWindow的第9个参数:

hvmd = CreateWindow (TEXT ("MyClass"), TEXT ("Window Caption"),

WS_OVERLAPPEDWINDOW,

                            CW_USEDEFAULT, CW_USEDEFAULT,                    

                            CW_USEDEFAULT, CW_USEDEFAULT,                    

                            NULL, hMenu, hlnstance, NULL);

这种情况下,CreateWindow中指定的菜单会覆盖窗口类中指定的任何菜单。如果 CreateWindow的第9个参数是NULL,那么你可以认为窗口类中的菜单是基于该窗口类的 所有窗口的默认菜单。因此,你可以对基于同一窗口类的几个不同窗口使用不同的菜单。

还可以在窗口类中指定一个NULL菜单名,并在CreateWindow调用中使用一个NULL 菜单句柄,然后在窗口创建之后再给它指派一个菜单:

SetMenu (hwnd, hMenu);

这种形式可以让你动态地改变窗口的菜单。我们会在本章后面的NOPOPUPS程序中看到 这样的一个例子。

当窗口被销毁时,附加到该窗口的任何菜单也将被销毁。而在程序结束前,任何没有附加到窗口的菜单应该通过DestroyMenu调用被显式地销毁。

其他菜单命令

除了之前已经介绍的菜单函数,还有更多和菜单相关的有用函数。

●当你改变一个顶级菜单项时,该改动直到Windows重绘菜单栏时才会被显示出来。你 可以调用下面的语句来强制重绘:

DrawMenuBar (hwnd);

【注意】DrawMenuBar的参数是一个指向窗口的句柄,而非菜单句柄。

●可以使用如下语句来获得弹出菜单的句柄:

hMenuPopup = GetSubMenu (hMenu, iPosition);

其中iPosition是弹出菜单在顶级菜单中的索引(从0开始),hMenu代表顶级菜单。然后便可以在其他函数(例如AppendMenu)中使用该弹出菜单句柄。

●使用如下语句,可以获得顶级菜单或弹出菜单中现有菜单项的数目:

iCount = GetMenuICemCount (hMenu);

●可以使用下面的语句来获得弹出菜单中某个菜单项的菜单ID:

 id = GetMenuItemID (hMenuPopup, iPosition);

其中iPosition是该菜单项在弹出菜单中的位置(从0开始)。

●在MENUDEMO中,展示了如何使用下面的语句在弹出菜单中“选中”和“取消选中” 某个菜单项:

CheckMenuItem (hMenu, id, iCheck);

在MENUDEMO中,hMenu是顶级菜单的句柄,id是菜单ID,iCheck的值是MF_CHECKED 或MF_UNCHECKED。如果hMenu是弹出菜单的句柄,那么id参数可以是位置索引而非菜单ID。如果使用索引更加方便,那么你可以在第三个参数中包含MF_BYPOSmON:

CheckMenuItem (hMenu, iPosition, MF_CHECKED | MF_BYP0SITI0N);

EnableMenuItem和CheckMenuItem函数类似,不同之处在于第三个参数是 MF_ENABLED、MF_DISABLED、或MF_GRAYED。如果你在顶级菜单项上使用 EnableMenuItem,而该菜单项还有弹出菜单,那么你必须在第三个参数中使用 MF_BYPOSITION标识符,因为该菜单项没有菜单ID。我们会在本章后面的POPPAD2程序中看到EnableMenuItem函数的示例。HiliteMenuItem函数和CheckMenuItem以及 EnableMenuItem类似,但它使用标志MF_HILITE和MF_UNHILITE。这种加亮使用的是 反色显示(Reverse Video),在你在菜单项之间移动时Windows使用的就是这种加亮形式。 通常情况下不需要使用HiliteMenuItem。

●获取菜单中使用的是什么字符串?你可以调用下面的语句:

iCharCount = GetMenuScring (hMenu, id, pString, iMaxCount, iFlag);

iFlag可以是MF_BYCOMMAND(此时id是一个菜单ID)或者MF_BYPOSITION(此时id是 一个位置索引)。该函数复制至多iMaxCount个字符到pString,并返回复制的字符数。

●或者你想知道一个菜单项的当前标志,可以调用:

iFlags = GeCMenuSCaCe (hMenu, id, iFlag);

同样,iFlag是MF_BYCOMMAND或MF_BYPOSmON之一。iFlag参数是所有当前标志 的组合值。你可以针对 MF_DISABLED、MF_GRAYED、MF_CHECKED、 MF_MENUBREAK、MF_MENUBARBREAK 和 MF_SEPARATOR 标识符进行检测,以确定当前的标志。

●当应用程序不再需要菜单时,通过下面的语句可以销毀它:

DestroyMenu (hMenu) ;

这个函数使该菜单句柄无效。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

头歌资源库(25)地图着色

一、 问题描述 任何平面区域图都可以用四种颜色着色,使相邻区域颜色互异。这就是四色定理。要求给定区域图,排出全部可能的着色方案。例如,区域图如下图所示: 要求用四种颜色着色。 则输入: 10 4 (分别表示…

什么是敏捷本地化

快速、敏捷的多语言产品和服务交付正逐渐成为众多行业的常态。在这种情况下,重点从传统的期望(即在合理的时间框架内翻译大量内容)转变为翻译工作量非常大的小片段,通常在2-3到12-24小时之间,通常在周末或假期。 Logr…

如何做好漏洞扫描工作提高网络安全

在数字化浪潮席卷全球的今天,企业数字化转型已成为提升竞争力、实现可持续发展的关键路径。然而,这一转型过程并非坦途,其中网络安全问题如同暗礁般潜伏,稍有不慎便可能引发数据泄露、服务中断乃至品牌信誉受损等严重后果。因此&a…

usbserver工程师手记(三)手工开通 OTP功能

1、设定密钥,用户自行选择一个密钥,以下以密钥为 EAZAYOKNGETBOPC5 为例说明 2、usb server 配置otp 密钥,目前还没有UI 界面开通,后续版本会支持从管理界面开通 curl -X POST -H Content-Type: application/json -H Accept: app…

mysql高可用解决方案:MHA原理及实现

MHA:Master High Availability。对主节点进行监控,可实现自动故障转移至其它从节点;通过提升某一从节点为新的主节点,基于主从复制实现,还需要客户端配合实现,目前MHA主要支持一主多从的架构,要…

应力平衡方程的推导

应力平衡方程的推导 对于一点,已知其应力状态有: σ x , τ x y , τ x z \sigma_x,\tau_{xy},\tau_{xz} σx​,τxy​,τxz​ 则其附近点的应力状态为: σ x ∂ σ x ∂ x d x , τ x y ∂ τ x y ∂ x d x , τ x z ∂ τ x z ∂ x d …

【JavaScript 报错】未捕获的范围错误:Uncaught RangeError

🔥 个人主页:空白诗 文章目录 一、错误原因分析1. 递归调用次数过多2. 数组长度超出限制3. 数值超出允许范围 二、解决方案1. 限制递归深度2. 控制数组长度3. 检查数值范围 三、实例讲解四、总结 Uncaught RangeError 是JavaScript中常见的一种错误&…

2024年06月CCF-GESP编程能力等级认证C++编程三级真题解析

本文收录于专栏《C等级认证CCF-GESP真题解析》,专栏总目录:点这里。订阅后可阅读专栏内所有文章。 一、单选题(每题 2 分,共 30 分) 第 1 题 小杨父母带他到某培训机构给他报名参加CCF组织的GESP认证考试的第1级&…

IO模型理论学习

1、什么是IO 计算机视角下的io AIO

Redis命令详解以及存储原理

Redis是什么 远程字典服务 分布式场景重的一个单独的节点。请求回应的模式:发起请求,处理之后得到回应的结果。字典的形式存储&索引数据。 内存数据库 数据在内存中,不可以出现需要的内存不在内存中而在磁盘中速度快,内存100…

智能家居开发新进展:乐鑫 ESP-ZeroCode 与亚马逊 ACK for Matter 实现集成

日前,乐鑫 ESP-ZeroCode 与亚马逊 Alexa Connect Kit (ACK) for Matter 实现了集成。这对智能家居设备制造商来说是一项重大进展。开发人员无需编写固件或开发移动应用程序,即可轻松设计符合 Matter 标准的产品。不仅如此,开发者还可以在短短…

goaccess分析json格式日志

一.安装使用yum安装,yum install goaccess 二.主要介绍格式问题 1.nginx日志格式如下: log_format main escapejson {"time_local":"$time_local", "remote_addr":"$remote_addr", "r…

C:数据结构---算法

1.1排序算法 稳定排序 不稳定排序 ①冒泡排序(稳定) 比较相邻的元素。如果第一个比第二个大,就交换他们两个。对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对 ②选择排序 在未排序序列中找到最小(大…

2.The DispatcherServlet

The DispatcherServlet Spring的Web MVC框架与许多其他Web MVC框架一样,是请求驱动的,围绕一个中央Servlet(即DispatcherServlet)设计,该Servlet将请求分派给控制器,并提供其他功能以促进Web应用程序的开发…

sqlmap使用之-post注入、head注入(ua、cookie、referer)

1、post注入 1.1、方法一,通过保存数据包文件进行注入 bp抓包获取post数据 将数据保存到post.txt文件 加上-r指定数据文件 1.2、方法二、通过URL注入 D:\Python3.8.6\SQLmap>python sqlmap.py -u "http://localhost/login.php" --data "userna…

《C语言程序设计 第4版》笔记和代码 第十一章 指针和数组

第十一章 指针和数组 11.1 指针和一维数组间的关系 1 由于数组名代表数组元素的连续存储空间的首地址,因此,数组元素既可以用下标法也可以用指针来引用。 例11.1见文末 2 p1与p在本质上是两个不同的操作,前者不改变当前指针的指向&#xf…

C++ | Leetcode C++题解之第230题二叉搜索树中第K小的元素

题目: 题解: class MyBst { public:MyBst(TreeNode *root) {this->root root;countNodeNum(root);}// 返回二叉搜索树中第k小的元素int kthSmallest(int k) {TreeNode *node root;while (node ! nullptr) {int left getNodeNum(node->left);if…

htb_PermX

PermX 端口开放 80,22 子域名扫描 ffuf -u http://permx.htb -H host: FUZZ.permx.htb -w /usr/share/seclists/Discovery/DNS/subdomains-top1million-5000.txt -fc 301,302 -mc allwww lms 访问lms.permx.htb chamilo 查找cve CVE-2023-4220 Chamilo LMS 未经身份验证的…

代码随想录算法训练营第三十一天|动态规划:01背包理论基础、01背包理论基础(滚动数组)

动态规划:01背包理论基础 1. dp[i][j]: 表示0到i个物品放入容量为j的背包中,价值总和最大是多少 2. dp[i][j]的状态取决于,第i个物品要不要放入这个背包。 不放物品i:dp[i-1][j] (其实就是当物品i的重量大于背包j的重量时&…

Golang | Leetcode Golang题解之第229题多数元素II

题目: 题解: func majorityElement(nums []int) (ans []int) {cnt : map[int]int{}for _, v : range nums {cnt[v]}for v, c : range cnt {if c > len(nums)/3 {ans append(ans, v)}}return }