c - WINAPI - WM_COMMAND 与 WM_MENUCOMMAND 用于固定菜单和动态子菜单

标签 c winapi menu notifications

尝试在 VS2019 中使用 WINAPI 用 C 创建托盘应用程序。有一个固定菜单和一个动态子菜单,其项目数量事先未知。主菜单来自 RC 文件,子菜单是动态生成的(示例如下)。

我设法对两个菜单的所有项目使用 MENUINFO 并触发 WM_MENUCOMMAND,或者不使用它并仅依赖 WM_COMMAND。

问题:是否可以在主菜单中触发并处理WM_COMMAND,同时从子菜单中触发WM_MENUCOMMAND来获取动态子菜单所选项目的id?否则识别动态添加的项目的正确方法是什么?

菜单创建代码:

hMenu = LoadMenu(g_hInst, MAKEINTRESOURCE(IDC_CONTEXTMENU));

if (hMenu)
{
    hSubMenu = GetSubMenu(hMenu, 0);

    if (hSubMenu)
    {
        // our window must be foreground before calling TrackPopupMenu
        // or the menu will not disappear when the user clicks away
        SetForegroundWindow(hwnd);

        hSubMenu2 = CreatePopupMenu();

        wchar_t warr[29];
        for (int i = 0; i < 10; i++) {
            swprintf(warr, sizeof warr / sizeof * warr, L"Item: %i", i);
            AppendMenu(hSubMenu2, MF_STRING, i + 1000, warr);
        }

        AppendMenu(hSubMenu, MF_STRING | MF_POPUP, (UINT_PTR)hSubMenu2, L"Sub-Sub Menu");

        MENUINFO mi;
        memset(&mi, 0, sizeof(mi));
        mi.cbSize = sizeof(mi);
        mi.fMask = MIM_STYLE;
        mi.dwStyle = MNS_NOTIFYBYPOS;
        SetMenuInfo(hSubMenu2, &mi);

        // respect menu drop alignment
        UINT uFlags = TPM_RIGHTBUTTON;
        if (GetSystemMetrics(SM_MENUDROPALIGNMENT) != 0)
            uFlags |= TPM_RIGHTALIGN;
        else 
            uFlags |= TPM_LEFTALIGN;

        TrackPopupMenuEx(hSubMenu, uFlags, pt.x, pt.y, hwnd, NULL);
    }

    DestroyMenu(hMenu);
}

消息处理代码:

case WM_MENUCOMMAND:;
    HMENU menu = (HMENU)lParam;
    int idx = wParam;
    wchar_t buffer[10];
    swprintf(buffer, sizeof buffer / sizeof * buffer, L"%i", idx);

    if (menu == hSubMenu2)
    {
        MessageBox(hwnd, buffer, L"Test", MB_OK);
    }
    else
        break;

case WM_COMMAND: // incomplete; for information
{
    int const wmId = LOWORD(wParam);
    // Parse the menu selections:
    switch (wmId)
    {
    case IDM_LOWINK:
        ShowLowInkBalloon();
        break;

    case IDM_NOINK:
        ShowNoInkBalloon();
        break;

// ...

最佳答案

Is it possible to trigger and process WM_COMMAND from the main menu, and at the same time WM_MENUCOMMAND from the submenu to get the id of the dynamic submenu selected item?

根据文档: WM_MENUCOMMAND 消息仅针对使用 MENUINFO 结构的 dwStyle 成员中设置的 MNS_NOTIFYBYPOS 标志定义的菜单发送。

并且:MNS_NOTIFYBYPOS 是一种菜单标题样式,应用于各个子菜单时无效。

据我了解,您可以为菜单设置此标志,然后单击一个项目将在 WM_MENUCOMMAND 消息中发送索引。现在,您要么记住索引以了解哪个项目被单击,要么您可以在创建项目、检索项目和使用项目时存储附加信息。例如,您现在可以将此索引转换为 WM_COMMAND 消息。

不幸的是,我找不到有关什么是“菜单标题样式”的更多信息。你需要进行实验。

<小时/> 编辑

我有一个管理动态菜单的通用模块。复制到这里有点太多了(超过 400 行),但我会给你提纲。

创建菜单项时,我将 MENUITEMINFO 结构的 wID 成员设置为负值,现在该值将与所有其他菜单 ID 区分开来。单击后,它会传递给我自己的处理程序。现在我假设这可以是一个WM_COMMAND菜单ID,并将在WM_COMMAND处理程序中处理(我有其他需求)。

基本的创建代码是(假设子菜单已经存在,所以只添加项目):

MENUITEMINFO MenuItemInfo = {sizeof(MENUITEMINFO), (MIIM_STATE|MIIM_TYPE|MIIM_ID|MIIM_DATA),
                            MFT_STRING, MFS_ENABLED, 0,0,0,0,0,0,0};

    // pMenuItem is my menu struct. replace with your data as needed
    MenuItemInfo.wID= IDM_MY_CMD; // the menu identifier; assumed here a WM_COMMAND id
    MenuItemInfo.fState= (pMenuItem->bEnabled?MFS_ENABLED:MFS_DISABLED);
    MenuItemInfo.cch= strlen(pMenuItem->szName);
    MenuItemInfo.dwTypeData= pMenuItem->szName;
    MenuItemInfo.dwItemData= (unsigned long)pMenuItem->pUser; // any data the user wants to store

    InsertMenuItem (hMenu, position, TRUE, &MenuItemInfo);

关于c - WINAPI - WM_COMMAND 与 WM_MENUCOMMAND 用于固定菜单和动态子菜单,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57646886/

相关文章:

php - 将特定的类/ID 应用到菜单上的当前页面 (PHP)

javascript - HTML 下拉选择值

c - 哪个更好? 32 位 MCU 中的 int8_t 与 int32_t

c - xv6: bootmain.c 读取eg() "round down to sector boundary"

c++ - 如何在 Win32 中等待窗口完成绘制?

菜单鼠标悬停的 CSS 修复

python - WAF——合并静态库

c - C语言的反射(reflection)?

c++ - 适用于 Windows 平台的 C/C++ 调用图实用程序

c - 如何在WM_PAINT中绘制Windows按钮?