WINAPI - WM_COMMAND 与固定菜单和动态子菜单的 WM_MENUCOMMAND

WINAPI - WM_COMMAND vs. WM_MENUCOMMAND for fixed menu and dynamic submenu

尝试在 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 是菜单 header 样式,应用于单个子菜单时无效。

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

很遗憾,我找不到有关什么是 "menu header style" 的更多信息。您将需要进行试验。


编辑

我有一个管理动态菜单的通用模块。复制太多了(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);