资源在Windows编程中的应用

资源

加速键、位图、光标、对话框、菜单、字符串、工具条

1、菜单的创建

菜单由以下组成部分:

(1)窗口主菜单条

(2)下拉式菜单框

(3)菜单项热键标识

(4)菜单项加速键标识

(5)菜单项分隔线

1)定义菜单

menuID MENU [,载入特性选项]
{
菜单项列表
}

menulD:menulD是菜单资源名,用以标识特定的菜单,应用程序通过菜单资源名加载指定菜单,它可以是一个字符串,也可以是1~6535之间的任何一个整数。

MENU关键字:同来标识资源的性质。

载入特性选项:用以标识菜单所具有的载入特性

DISCARDABLE
当不再需要菜单时可丢弃
FIXED
将菜单保存在内存中的固定位置
LOADONCALL
需要时加载菜单
MOVEABLE
底单在内存中可移动
PRELOAI
立即加载菜单

菜单项:菜单项是菜单的组成部分。应用程序在资源描述文件中使用关键字POPUP和MENUITEM 定义菜单项。

①POPUP语句

POPUP语句定义弹出式菜单,其形式为:

POPUP "菜单项名" [,选项];

程序员还可在菜单项名中加人符号"&",以定义该菜单项的热键。例如定义弹出式

菜单项"编辑"的形式如下:

POPUP"编辑(&E)";

该菜单项使用Alt十E键作为热键。

POPUP定义的弹出式菜单项还可包含子菜单。

②MENUITEM语句

MENUITEM语句用于定义菜单项,其形式为:

MENUITEM "菜单项名" 菜单项标识(ID)[,选项]

ID为菜单项标识,在WM_COMMAND消息中字参数wParam中包含选中菜单项的标识。每个菜单项的标识必须唯一,标识值常在头文件中定义。

在VS2019中建立windows桌面应用程序后,菜单项标识在Resource.h中定义,如:

#define    IDM_NEW    		10
#define IDM_OPEN 11
#define IDM_CLOSE 12
#define IDM_SAVE 13
#define IDM_SAVEAS 14
#define IDM_EXIT 15
...

然后在资源文件项目名.rc中编辑菜单样式,如可以直接打开.rc文件,图形化添加菜单选项,也可以使用记事本打开.rc文件,像下面一样添加:

IDC_RESOURCESTUDY MENU
BEGIN
POPUP "文件(&F)"
BEGIN
MENUITEM "打开文件...", ID_32771
MENUITEM "关闭文件...", ID_32773
MENUITEM SEPARATOR
MENUITEM "创建统计计算菜单项(&P)\t Ctrl+P", IDM_ADDMENU
MENUITEM "删除统计计算菜单项(&D)\t Ctrl+D", IDM_DELMENU
MENUITEM SEPARATOR
MENUITEM "退出(&X)", IDM_EXIT
END
POPUP "帮助(&H)"
BEGIN
MENUITEM "关于(&A) ...", IDM_ABOUT
END
END

2、加载菜单资源

在窗口类中加载菜单

WNDCLASS  wndclass;

wndclass.lpszMenuName=lpszMenuName;

在创建窗口时加载菜单

HWND hwnd;HMENU hmenu;

hmenu=LoadMenu(hlnstance, "My_menu");
hwnd=CreateWindow( …,hmenu, …);

动态加载菜单

hmenu2=LoadMenu(hlnstance, "Menu2");
SetMenu(hwnd,hmenu2);

3、操作菜单项

1)禁止或激活菜单项

应用程序创建菜单时,通过在资源描述文件中设定菜单项的选项以指定该菜单项的

初始状态为禁止或激活,或调用函数EnableMenuItem改变其初始状态。该函数的原型为:

BOOL EnableMenultem(HMENU hmenu, UINT wIDEnableItem, UINT dwEnable)

其中:wIDEnableItem为被禁止或激活的菜单项标识,根据dwEnable的取值,可能为该菜单项的ID值,也可能为该菜单项在菜单中位置;dwEnable为菜单项操作标识

dwEnable参数的操作标识:
MF_BYCOMMAND 表明以ID值标识菜单项
MF_BYPOSITION 表明以位置标识菜单项
MF_DISABLED 禁止菜单项
MF_ENABLED 激活菜单项
MF_GRAYED 禁止菜单项并使其变灰显示

例如,禁止弹出式菜单“文件”中的“打开”项的形式如下:

EnableMenuItem(hmenu, IDM_OPEN, MF_BYCOMMAND | MF_DISABLED) ;

2)设置或取消选中标志

在菜单旁显示一个选中标志,如"√"

DWORD CheckMenuItem(
HMENU hmenu,
UNIT wIDCheckItem, //wIDCheckItem为设置或取消选中标志的菜单项标识
UNIT dwCheck //dwCheck为操作标识,MF_CHECKED 添加选中标志;MF_UNCHECKED 删除选中标志
)

3)增加菜单项

程序员可在应用程序中通过两种形式动态地增加菜单项:

(1)在菜单的尾部增加菜单项

应用程序可调用函数AppendMenu在菜单的尾部增加菜单项,该函数的原型为:

BOOL AppendMenu(
HMENU hmenu,
UINT dwFlags,
//新加入的菜单项类型标识或其他信息
UINT dwIDNewItem,// 新加入菜单项的标识
LPCTSTR lpNewItem // 新加入的菜单项内容,取决于dwFlags参数
)

例如在弹出式菜单“文件”的末尾增加一项“关于”的形式如下:

AppendMenu(hmenu, MF_ENABLED, IDM_ABOUT, "关于(&A)");

(2)在菜单中插人菜单项

应用程序也可调用函数InsertMenu在菜单中插入新的菜单项,该函数的原型为:

BOOL InsertMenu
(
HMENU hmenu,
//菜单句柄
UINT wPosition,
//指定新菜单项插入的位置
UINT dwFlag,
//新加入的菜单项的信息及对wPosition的解释
UINT dwIDNweItem,
//新加入的菜单项的标识
LPCTSTR lpNewItem
//新插入的菜单项的内容
)

例如,在弹出式菜单“文件”的“退出”(其标识为IDM_EXIT)项之前加入新的菜单项“打印”(其标识为IDM_ PRINT)的语句如下:

InsertMenu (hmenu, IDM_EXIT, MF_BYCOMMAND |MF_ENABLED, IDM_PRINT,"打印(&P)");

4)删除菜单项

应用程序可调用函数DeleteMenu删除菜单项,该函数的原型为:

BOOL DeleteMenu
(
HMENU hmenu,
UINT wPosition,
//指定要删除的菜单项的位置
UINT dwFlag
//对wPosition的解释
)

对于wPosition,由参数dwFlag解释其意义,如果dwFlag为MF_BYCOMMAND,则该参数为菜单项的ID值;如果dwFlag为MF_BIYPOSITION,则该参数为菜单项的位置号。

例如,删除弹出式“文件”菜单中的“另存为”项的形式如下:

DeleteMenu(hmenu, IDM_SAVEAS, MF_BYCOMMAND)

值得注意的是,如果菜单项含有弹出式菜单,则删除该菜单项时该弹出式菜单也同时被删除。

5)修改菜单项

应用程序可调用函数ModifyMenu修改菜单中的某个项,该函数原型为:

BOOL ModifyMenu(
HMENU hmenu,
UINT wPosition, //指定需修改的菜单项位置
UINT dwFlag,
UINT dw IDNweItem, //-般为修改后菜单项的标识
LPCTSTR lpNewItem //-般为修改后的菜单项名
)

对于wPosition,如果dwFlag为MF_BYCOMMAND,则该参数为菜单项的ID值;如果dwFlag为MF_BYPOSITION ,则该参数为菜单项的位置号。

例如修改弹出式菜单“文件”中“打开”项为“加载”项的语句如下

ModifyMenu(hmenu, IDM_OPEN, MF_BYCOMMAND, IDM_LOAD,"加载(&L)");

4、菜单的动态创建

动态创建菜单可以使系统资源更加节省,在应用程序中动态创建菜单分两个步骤:

(1)调用函数CreateMenu创建空的弹出式菜单,CreateMenu函数的原型如下:

HMENU CreateMenu(void)

(2)调用函数AppendMenu或InsertMenu在该菜单中加入菜单项。

例如,在应用程序的窗口菜单中动态创建弹出式菜单“编辑"的过程如下:

HMENU hmenu, hPopupmenu;		//主窗口菜单句柄和新创建的菜单句柄
AppendMenu(lmenu, MF_POPUP ,(UINT)hmenuPopup, "编辑(&E)"); //将弹出式菜单“编辑”加人到菜单中

5、实例

// resource_study.cpp : 定义应用程序的入口点。
// #include "framework.h"
#include "resource_study.h"
#include "windows.h" #define MAX_LOADSTRING 100 // 全局变量:
HMENU hmenu, haddmenu; //定义菜单句柄
HINSTANCE hInst; // 当前实例
WCHAR szTitle[MAX_LOADSTRING]; // 标题栏文本
WCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口类名 // 此代码模块中包含的函数的前向声明:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM); int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine); // TODO: 在此处放置代码。 // 初始化全局字符串
LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadStringW(hInstance, IDC_RESOURCESTUDY, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance); // 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
} HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_RESOURCESTUDY)); MSG msg; // 主消息循环:
while (GetMessage(&msg, nullptr, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
} return (int) msg.wParam;
} //
// 函数: MyRegisterClass()
//
// 目标: 注册窗口类。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEXW wcex; wcex.cbSize = sizeof(WNDCLASSEX); wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_RESOURCESTUDY));
wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = MAKEINTRESOURCEW(IDC_RESOURCESTUDY);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL)); return RegisterClassExW(&wcex);
} //
// 函数: InitInstance(HINSTANCE, int)
//
// 目标: 保存实例句柄并创建主窗口
//
// 注释:
//
// 在此函数中,我们在全局变量中保存实例句柄并
// 创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
hInst = hInstance; // 将实例句柄存储在全局变量中 HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr); if (!hWnd)
{
return FALSE;
} ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd); return TRUE;
} //
// 函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// 目标: 处理主窗口的消息。
//
// WM_COMMAND - 处理应用程序菜单
// WM_PAINT - 绘制主窗口
// WM_DESTROY - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_COMMAND:
{
int wmId = LOWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case IDM_ADDMENU: //在主菜单中添加弹出式菜单
hmenu = GetMenu(hWnd); //获取主菜单句柄
haddmenu = CreateMenu(); //动态创建菜单
//在创建的菜单中增加菜单项
AppendMenu(haddmenu, MF_ENABLED, IDM_qiuhe, L"求和");
AppendMenu(haddmenu, MF_ENABLED, IDM_fangcha, L"方差");
AppendMenu(haddmenu, MF_ENABLED, IDM_pinjunzhi, L"平均值");
AppendMenu(haddmenu, MF_ENABLED, IDM_junfanggen, L"均方根");
//将创建的弹出式菜单插入主菜单中
InsertMenu(hmenu, 2, MF_POPUP | MF_BYPOSITION,
(UINT)haddmenu, L"统计计算(&C)");
//相应改变菜单中有关绘图统计计算菜单项的属性
EnableMenuItem(hmenu, IDM_ADDMENU, MF_GRAYED);
EnableMenuItem(hmenu, IDM_DELMENU, MF_ENABLED);
DrawMenuBar(hWnd); //重新显示窗口菜单
break;
case IDM_DELMENU:
DeleteMenu(hmenu, 2, MF_BYPOSITION); //删除统计计算菜单项
//相应改变“文件”菜单中有关统计计算菜单项的属性
EnableMenuItem(hmenu, IDM_ADDMENU, MF_ENABLED);
EnableMenuItem(hmenu, IDM_DELMENU, MF_GRAYED);
DrawMenuBar(hWnd); //重新显示窗口菜单
break; case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
}
break;
case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(hWnd, &ps);
// TODO: 在此处添加使用 hdc 的任何绘图代码...
EndPaint(hWnd, &ps);
}
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
} // “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE; case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}

资源在windows编程中的应用----菜单的更多相关文章

  1. 资源在Windows编程中的应用

    学习目的 掌握菜单和对话框资源的创建和使用. 编写程序: 设计一个窗口应用程序, 其中有一个VC菜单, 该菜单下有"显示", "隐藏", "退出&qu ...

  2. 第七章 资源在Windows编程中的应用 P157 7-8

    资源在基于SDK的程序设计中的应用实验 一.实验目的 1.掌握各种资源的应用及资源应用的程序设计方法.   二.实验内容及步骤 实验任务 1.熟悉菜单资源的创建过程: 2.熟悉位图资源的创建: 3.熟 ...

  3. windows编程中的数据类型

    在windows编程中,有许多奇怪的数据类型,初学者不知道这些代表什么,下面就把一些数据类型列出如下: ATOM 原子(原子表中的一个字符串的参考) BOOL 布尔变量 BOOLEAN 布尔变量 BY ...

  4. Windows编程中的若干难点 - Windows程序设计(SDK)007

    Windows编程中的若干难点 让编程改变世界 Change the world by program 一个窗口的生与死 我记得有童鞋会问:如果我的程序需要在关闭前让用户判断是否确定要关闭窗口,我应该 ...

  5. Windows编程中UNICODE和_UNICODE定义问题

    Windows编程中UNICODE和_UNICODE定义问题 先转一篇文章: 初学Windows SDK编程时碰到过这个问题,相信很多初学Windows编程的人也都碰到过,后来慢慢搞明白些了,但有时别 ...

  6. Windows 编程中恼人的各种字符以及字符指针类型

    在Windows编程中,很容易见到这些数据类型:LPSTR,LPTSTR,LPCTSTR... 像很多童鞋一样,当初在学Windows编程的时候,对着些数据类型真的是丈二和尚,摸不着头脑,长时间不用就 ...

  7. Windows编程中char*转LPCWSTR解决的方法总结

    Windows编程中常常涉及到的一个问题是字符串之间的转换,开发过程总是遇到编译器提示无法格式转换的问题.于是自己总结了几种解决的方法. 1.通过T2W转换宏 char* szStr = " ...

  8. Windows编程中各种操作文件的方法

    windows编程中文件操作有以下几种常见方法:1.C语言中文件操作.2.C++语言中的文件操作.3.Win32 API函数文件操作.4.MFC CFile类文件操作.5.MFC CFileDialo ...

  9. Windows编程中回调函数的使用心得(MFC篇)

    回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数.回调函数不是由该函数的实现方直接调用,而是在特定 ...

随机推荐

  1. bootstrap 怎么制作好看的表格

    bootstrap 怎么制作表格 bootstrap 制作表格带有图文形式.主要知识点有以下几点 第一点肯定是写出一个普通的表格,这一点可以去菜鸟复制它的案例.添加tr和td就可以了 在表格放入图片加 ...

  2. Hadoop入门学习笔记-第一天 (HDFS:分布式存储系统简单集群)

    准备工作: 1.安装VMware Workstation Pro 2.新建三个虚拟机,安装centOS7.0 版本不限 配置工作: 1.准备三台服务器(nameNode10.dataNode20.da ...

  3. 能量项链(区间DP入门)

    题面:能量项链https://www.luogu.com.cn/problem/P1063 乍一看和石子合并差不多,可是多了头值和尾值,看起来十分麻烦 我们画一张图,紫色表示头值,蓝色表示尾值.规定西 ...

  4. springboot设置过滤器、监听器、拦截器

    其实这篇文章算不上是springboot的东西,我们在spring普通项目中也是可以直接使用的 设置过滤器: 以前在普通项目中我们要在web.xml中进行filter的配置,但是只从servlet 3 ...

  5. SpringBoot:模板引擎 thymeleaf、ContentNegotiatingViewResolver、格式化转换器

    目录 模板引擎 thymeleaf ContentNegotiatingViewResolver 格式化转换器 模板引擎 thymeleaf.ContentNegotiatingViewResolve ...

  6. Spring Cloud 学习 之 Spring Cloud Eureka(源码分析)

    Spring Cloud 学习 之 Spring Cloud Eureka(源码分析) Spring Boot版本:2.1.4.RELEASE Spring Cloud版本:Greenwich.SR1 ...

  7. 房价预测Task1

    pandas:简单的房价预测实例 我们使用pandas等工具,对于给出的.csv文件进行处理,完成要求的几个Task. 利用sklearn的线性回归,对于房价进行简单的预测. 所有的要求,数据集等文件 ...

  8. 数据库当中删除数据后主键id不连续的问题

    新建查询: ALTER TABLE `表名` DROP `主键名`;ALTER TABLE `表名` ADD `主键名` int NOT NULL FIRST;ALTER TABLE `表名` MOD ...

  9. aop面向切面编程的实现

    aop主要用于日志记录,跟踪,优化和监控 下面是来自慕课网学习的一些案例,复制黏贴就完事了,注意类和方法的位置 pom添加依赖: <dependency> <groupId>o ...

  10. angular js 页面修改数据存入数据库

    一.编写service,修改数据要根据ID回显数据 //根据ID查询 public Brand findById(Long id); //修改 public int update(Brand bran ...