菜单的使用

Masimaro
2015-08-05 / 0 评论 / 6 阅读 / 正在检测是否收录...
温馨提示:
本文最后更新于2024年11月25日,已超过171天没有更新,若内容或图片失效,请留言反馈。

​# Windows菜单的基本知识:

1)顶级菜单:紧贴在标题栏下面的菜单称为顶级菜单,也可以叫做程序的主菜单;
2)弹出式菜单:一般在顶级菜单上都有很多菜单项,单击这些菜单项时会弹出一个下拉式的菜单项,我们点击的这个菜单称为弹出式菜单
3)菜单项:每一个可选菜单项被赋予一个唯一的ID,当用户单击某个菜单项时Windows会将该菜单项的ID发送给父窗口,父窗口通过WM_COMMAND消息处理菜单的单击消息,但是弹出式菜单没有ID,WM_COMMAND消息也不处理弹出式菜单的点击信息
4)菜单加速键:主要是多个键的组合,当同时按下这些键的时候相当于点击了菜单的某个菜单项
5)菜单项一般具有“可用”(Enabled)、“不可用”(disabled)、“变灰”(gray)几种选项,其中变灰选项将菜单项变成不可用的同时也会将菜单项变成灰色,所以当我们需要禁用某个菜单项的时候最好将它变灰,以便提示用户;
6)菜单句柄:每一种菜单都有一个菜单句柄,包括弹出式菜单的菜单项,顶级菜单,弹出式菜单;

菜单的创建:

Windows中菜单有两种方式,一种是通过资源的方式通过可视化或者编写rc文件来创建一个菜单资源,并在代码中显示的加载,另一种是通过调用CreateMenu、AppendMenu、InsertMenu等函数创建菜单并插入相应的菜单项,下面对这两种方式一一进行说明:

1)采用rc文件的方式:可以在visual studio中利用可视化的方式编辑菜单,在这里就不在说明,而需要手工编写rc文件请参考我的另外一篇博文

当我们编辑好了rc文件之后有三种方法添加菜单:

通过在创建窗口类的时候在lpszMenuName项的后面添加一个用于标示菜单的字符串,若菜单使用的是ID号作为标示那么可以使用宏MAKEINTRESOURCE

在函数CreateWindow或者CreateWindowEx中的相应参数中填入菜单句柄,为了获取这个句柄需要提前使用LoadMenu函数加载菜单,这个函数的功能是将资源文件中的菜单加载到内存,并返回一个菜单句柄,函数的原型如下:

HMENU LoadMenu(
  HINSTANCE hInstance,  // 当前应用程序的实例句柄
  LPCTSTR lpMenuName    // 菜单唯一标示,可以是字符串或者用MAKEINTRESOURCE转化而来的字符串
);

第三种方式是先通过LoadMenu函数获取菜单句柄后在窗口创建后通过SetMenu函数设置菜单,该函数用于为指定窗口加载一个顶级菜单、该函数原型如下:

BOOL SetMenu(
  HWND hWnd,  // 需加载菜单的窗口句柄
  HMENU hMenu // 菜单句柄
);

各个方式的源代码如下:

WNDCLASS wd = {0};
wd.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
wd.hCursor = LoadCursor(NULL, IDC_ARROW);
wd.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wd.hInstance = hInstance;
wd.lpfnWndProc = WindowProc;
wd.lpszClassName = "MenuClass";
//第一种方式
//wd.lpszMenuName = MAKEINTRESOURCE(IDM_MENU);
wd.style = CS_HREDRAW | CS_VREDRAW;
HMENU hMenu = LoadMenu(hInstance, MAKEINTRESOURCE(IDM_MENU));
    
//加载加速键
HACCEL hAccelerator = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDA_MAIN));
if (!RegisterClass(&wd))
{
    int nErr = GetLastError();
    return nErr;
}

//第二种方式
//HWND hWnd = CreateWindow("MenuClass", "Menu", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, hMenu, hInstance, NULL);
    
//第三种方式
HWND hWnd = CreateWindow("MenuClass", "Menu", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hInstance, NULL);
SetMenu(hWnd, hMenu);

如果采用函数动态创建的方式,需要如下几个步骤:

1)通过函数CreateMenu()创建一个顶级菜单;
2)通过CreateMenu()创建一个弹出式菜单;
3)利用AppendMenu()或者InsertMenu()向弹出式菜单中插入菜单项;
4)利用AppendMenu()将弹出式菜单插入到顶级菜单中;
5)用SetMenu函数将创建好的菜单加到程序

下面分别说明这些函数的功能和用法:

CreateMenu()用于创建一个菜单(函数会将菜单初始化为空菜单),并返回一个菜单句柄,函数原型如下:

HMENU CreateMenu(VOID)
AppendMenu()用于在顶级菜单、弹出式菜单的最后面的菜单项后查入新菜单项,函数原型如下:

BOOL AppendMenu(
  HMENU hMenu,      // 菜单项的句柄
  UINT uFlags,      // 新菜单项的类型,主要使用的是MF_STRING、MF_POUP(弹出式菜单)
  UINT uIDNewItem,  // 新菜单项的ID,如果是弹出式菜单、则使用菜单的句柄
  LPCTSTR lpNewItem //该值取决于第二个参数,若为MF_STRING则应该是一个以0结尾的字符串
);
InsterMenu()函数作用与AppendMenu相同,函数原型如下:

BOOL InsertMenu(
  HMENU hMenu,      // 菜单项的句柄
  UINT uPosition,   // 新菜单项的识别方式,主要有两种MF_BYCOMMAND和MF_BYPOSITION,在以后我们取菜单项的句柄或者对菜单项做其他操作,需要辨认时会有一定的作用,主要表明是靠ID号辨别还是靠在菜单中的相对位置(以0为第一个菜单项)
  UINT uFlags,     // 新菜单项的类型,主要使用的是MF_STRING、MF_POUP(弹出式菜单)
  UINT uIDNewItem,  // 新菜单项的ID,如果是弹出式菜单、则使用菜单的句柄
  LPCTSTR lpNewItem //该值取决于第三个个参数,若为MF_STRING则应该是一个以0结尾的字符串

);

下面是一个使用这种方式的例子

#include <Windows.h>
LRESULT CALLBACK WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
#define IDM_FILE            100
#define IDM_ABOUT            200
#define IDM_CLOSE            300
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
{
    WNDCLASS wd = {0};
    wd.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
    wd.hCursor = LoadCursor(NULL, IDC_ARROW);
    wd.hIcon = LoadIcon(NULL, IDI_APPLICATION);
    wd.hInstance = hInstance;
    wd.lpfnWndProc = WindowProc;
    wd.lpszClassName = "MenuClass";
    wd.style = CS_HREDRAW | CS_VREDRAW;
    
    if (!RegisterClass(&wd))
    {
        int nErr = GetLastError();
        return nErr;
    }

    HWND hWnd = CreateWindow("MenuClass", "Menu", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hInstance, NULL);

    //创建主菜单
    HMENU hMenu = CreateMenu();
    
    //创建弹出式菜单
    HMENU hPopup = CreateMenu();
    
    //向弹出式菜单中插入菜单项
    AppendMenu(hPopup, MF_STRING, IDM_FILE, TEXT("文件"));
    AppendMenu(hPopup, MF_STRING, IDM_ABOUT, TEXT("关于"));
    InsertMenu(hPopup, MF_BYCOMMAND, MF_STRING, IDM_CLOSE, TEXT("关闭"));

    //将弹出式菜单插入到主菜单中
    AppendMenu(hMenu, MF_POPUP,(UINT_PTR)hPopup, TEXT("系统"));

    SetMenu(hWnd,hMenu);

    if (NULL == hWnd)
    {
        int nErr = GetLastError();
        return nErr;
    }

    ShowWindow(hWnd, nShowCmd);

    MSG msg;
    while (GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return msg.wParam;
}

LRESULT CALLBACK WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    switch (uMsg)
    {
    case WM_COMMAND:
        {
            if (IDM_ABOUT == LOWORD(wParam))
            {
                MessageBox(hWnd, TEXT("About"), TEXT("TEST"), MB_OK);
            }
        }
        break;
    case WM_CLOSE:
        DestroyWindow(hWnd);
        break;
    case WM_DESTROY:
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hWnd, uMsg, wParam, lParam);
    }
    return 0;
}

键菜单的使用:

创建一个右键菜单有如下步骤(在WM_RBUTTONDOWN消息下处理):

1)创建一个可用的菜单(一般是主菜单);
2)根据主菜单获取弹出式菜单的句柄,使用函数GetSubMenu()
2)加载菜单项
3)获取鼠标点击的位置
4)将客户区坐标转化为屏幕坐标(这一步千万别忘了)
5)调用TrackPopupMenu函数,该函数用来显示一个快捷菜单,这个函数中需要填入菜单显示的位置,这个位置值为屏幕坐标,这也就是我们为什么需要转化坐标的原因;该函数的原型为:

BOOL TrackPopupMenu(
  HMENU hMenu,         // 快捷菜单的句柄
  UINT uFlags,         // 快捷菜单显示的类型
  int x,               // 
  int y,               //菜单显示点的坐标,根据第二个参数确定如何显示,一般有左对齐(最左边顶点为该坐标)、右对齐(右上角坐标为该坐标)、中间对齐(上边线的中点坐标为该坐标);
  int nReserved,       // 该参数必须给0
  HWND hWnd,           // 显示快捷菜单的窗口句柄
  CONST RECT *prcRect  // 该参数被忽略,一般给NNULL
);

下面是一段例子代码:

HMENU hMenu = LoadMenu(GetModuleHandle(NULL), MAKEINTRESOURCE(IDM_MENU));
            hMenu = GetSubMenu(hMenu, 0);
            POINT ptChick = {LOWORD(lParam), HIWORD(lParam)};
            ClientToScreen(hWnd, &ptChick);
            TrackPopupMenu(hMenu, TPM_LEFTALIGN, ptChick.x, ptChick.y, 0, hWnd, NULL);

其他菜单操作的函数主要有:

  • GetSystemMenu()获取系统菜单句柄;
  • Deletemenu()从菜单中删除某一菜单项并销毁它
  • RemoveMenu()从菜单中移出某一菜单项但不销毁它
  • InsertMenu()在菜单中插入一个菜单项
  • NodifyMenu()修改一个已存在的菜单项

0

评论 (0)

取消