通知图标接收WM_LBUTTONDBLCLK,但不接收WM_CONTEXTMENU

Notify Icon Receives WM_LBUTTONDBLCLK But Not WM_CONTEXTMENU

本文关键字:WM CONTEXTMENU LBUTTONDBLCLK 图标 通知      更新时间:2023-10-16

我在我的基于对话框的应用程序中添加了一个通知图标,当图标被双击时,它会收到WM_LBUTTONDBLCLK,但是当图标被右键单击或当图标被键盘突出显示并按下上下文菜单键时,它不会收到WM_CONTEXTMENU。我基于Windows 7.1 SDK示例中的示例来使用通知图标。所以,我不知道我哪里出错了,也不知道为什么这不起作用。

注意:如果我改变WM_CONTEXTMENU到WM_RBUTTONUP,它接收事件,但光标坐标是错误的。

/******************************************************************************/
/* Menu Resource                                                              */
/******************************************************************************/
LANGUAGE LANG_NEUTRAL, SUBLANG_NEUTRAL
IDR_TRAYMENU MENU
{
    POPUP ""
    {
        MENUITEM "&Show Status Window", IDM__SHOW_STATUS_WINDOW
        MENUITEM "&About", IDM__ABOUT
        MENUITEM SEPARATOR
        MENUITEM "&Exit", IDM__EXIT
    }
}
/******************************************************************************/
/* WinMain()                                                                  */
/******************************************************************************/
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow )
{
    // ... code unrelated to icon
    // Enable Visual Styles
    InitCommonControls();
    // create the main dialog
    if( NULL == (hWnd=CreateDialog(hInstance,MAKEINTRESOURCE(IDD_MAINDLG),NULL,(DLGPROC)WndProc)) )
    {
        MessageBox( NULL, "Error creating the main dialog!", NULL, MB_OK | MB_ICONERROR );
        return -1;
    }
    // ... code unrelated to icon
    MSG msg;
    while( GetMessage(&msg,NULL,0,0) && IsWindow(hWnd) )
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    return 0; 
}    
/******************************************************************************/
/* WndProc()                                                                  */
/******************************************************************************/
BOOL CALLBACK WndProc(HWND hWndDlg, UINT Message, WPARAM wParam, LPARAM lParam)
{
    switch(Message)
    {
        case WM_INITDIALOG:
        {
            // ... code unrelated to icon
            hIcon = (HICON)LoadImage( GetModuleHandle(NULL), MAKEINTRESOURCE(IDI_DDCMP), IMAGE_ICON, 16, 16, LR_DEFAULTSIZE );
            // Setup the system tray icon
            memset( &nid, 0, sizeof(NOTIFYICONDATA) );
            nid.cbSize = sizeof(NOTIFYICONDATA);
            nid.hWnd = hWndDlg;
            nid.uID = 0xDDC;
            nid.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP | NIF_SHOWTIP;
            nid.uCallbackMessage = WM_APP + 0xDDC;
            nid.hIcon = hIcon;
            strcpy( nid.szTip, "DDCMP Driver" );
            Shell_NotifyIcon( NIM_ADD, &nid );
            // ... code unrelated to icon
            return true;
        } break;
        case WM_APP + 0xDDC:
        {
            switch( LOWORD(lParam) )
            {
                case WM_CONTEXTMENU:
                {
                    MessageBox( hWndDlg, "This message box never shows up.", NULL, MB_OK | MB_SYSTEMMODAL );
                    HMENU hMenu = LoadMenu(GetModuleHandle(NULL),MAKEINTRESOURCE(IDR_TRAYMENU));
                    if( hMenu )
                    {
                        HMENU hSubMenu = GetSubMenu(hMenu,0);
                        if( hSubMenu )
                        {
                            SetForegroundWindow( hWndDlg );
                            POINT pt = { LOWORD(wParam), HIWORD(wParam) };
                            UINT uFlags = TPM_RIGHTBUTTON;
                            if( 0 != GetSystemMetrics(SM_MENUDROPALIGNMENT) )
                                uFlags |= TPM_RIGHTALIGN;
                            else
                                uFlags |= TPM_LEFTALIGN;
                            TrackPopupMenuEx( hSubMenu, uFlags, pt.x, pt.y, hWndDlg, NULL );
                        }
                        DestroyMenu( hMenu );
                    }
                } break;
                case WM_LBUTTONDBLCLK:
                    if( IsWindowVisible(hWndDlg) )
                        ShowWindow( hWnd, SW_HIDE );
                    else
                        ShowWindow( hWnd, SW_SHOW );
                    break;
            }
            return true;
        } break;
        case WM_CLOSE:
            ShowWindow( hWndDlg, SW_HIDE );
            break;
        case WM_DESTROY:
        case WM_QUIT:
        {
            Shell_NotifyIcon( NIM_DELETE, &nid );
            // ... code unrelated to icon
            return true;
        } break;
    }
    return false;
}

这是Windows 7.1 SDK样例中的WndProc

    LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    static HWND s_hwndFlyout = NULL;
    static BOOL s_fCanShowFlyout = TRUE;
    switch (message)
    {
    case WM_CREATE:
        // add the notification icon
        if (!AddNotificationIcon(hwnd))
        {
            MessageBox(hwnd,
                L"Please read the ReadMe.txt file for troubleshooting",
                L"Error adding icon", MB_OK);
            return -1;
        }
        break;
    case WM_COMMAND:
        {
            int const wmId = LOWORD(wParam);
            // Parse the menu selections:
            switch (wmId)
            {
            case IDM_LOWINK:
                ShowLowInkBalloon();
                break;
            case IDM_NOINK:
                ShowNoInkBalloon();
                break;
            case IDM_PRINTJOB:
                ShowPrintJobBalloon();
                break;
            case IDM_OPTIONS:
                // placeholder for an options dialog
                MessageBox(hwnd,  L"Display the options dialog here.", L"Options", MB_OK);
                break;
            case IDM_EXIT:
                DestroyWindow(hwnd);
                break;
            case IDM_FLYOUT:
                s_hwndFlyout = ShowFlyout(hwnd);
                break;
            default:
                return DefWindowProc(hwnd, message, wParam, lParam);
            }
        }
        break;
    case WMAPP_NOTIFYCALLBACK:
        switch (LOWORD(lParam))
        {
        case NIN_SELECT:
            // for NOTIFYICON_VERSION_4 clients, NIN_SELECT is prerable to listening to mouse clicks and key presses
            // directly.
            if (IsWindowVisible(s_hwndFlyout))
            {
                HideFlyout(hwnd, s_hwndFlyout);
                s_hwndFlyout = NULL;
                s_fCanShowFlyout = FALSE;
            }
            else if (s_fCanShowFlyout)
            {
                s_hwndFlyout = ShowFlyout(hwnd);
            }
            break;
        case NIN_BALLOONTIMEOUT:
            RestoreTooltip();
            break;
        case NIN_BALLOONUSERCLICK:
            RestoreTooltip();
            // placeholder for the user clicking on the balloon.
            MessageBox(hwnd, L"The user clicked on the balloon.", L"User click", MB_OK);
            break;

        // 
        //
        // As you can very plainly see, the Windows SDK Sample ONLY used WM_CONTEXTMNEU
        // 
        //
        case WM_CONTEXTMENU:
            {
                POINT const pt = { LOWORD(wParam), HIWORD(wParam) };
                ShowContextMenu(hwnd, pt);
            }
            break;
        }
        break;
    case WMAPP_HIDEFLYOUT:
        HideFlyout(hwnd, s_hwndFlyout);
        s_hwndFlyout = NULL;
        s_fCanShowFlyout = FALSE;
        break;
    case WM_TIMER:
        if (wParam == HIDEFLYOUT_TIMER_ID)
        {
            // please see the comment in HideFlyout() for an explanation of this code.
            KillTimer(hwnd, HIDEFLYOUT_TIMER_ID);
            s_fCanShowFlyout = TRUE;
        }
        break;
    case WM_DESTROY:
        DeleteNotificationIcon();
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hwnd, message, wParam, lParam);
    }
    return 0;
}

我认为你应该尝试将NOTIFYICONDATA结构的uVersion成员更改为NOTIFYICON_VERSION_4,文档指出,该成员值将告诉uCallbackMessage参数在传递给回调函数时如何解释。

你也可以看看这个:在Win32中Shell_NotifyIcon的基本使用

我做了一些研究,下面的方法应该对你有用:

memset(&nid, 0, sizeof(NOTIFYICONDATA));
nid.cbSize = sizeof(NOTIFYICONDATA);
nid.hWnd = hWndDlg;
nid.uID = 0xDDC;
nid.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP | NIF_SHOWTIP;
nid.uCallbackMessage = WM_APP + 0xDDC;
nid.hIcon = hIcon;
nid.uVersion = NOTIFYICON_VERSION_4;
strcpy(nid.szTip, "DDCMP Driver");
Shell_NotifyIcon(NIM_ADD, &nid);
Shell_NotifyIcon(NIM_SETVERSION, &nid);

NIM_SETVERSION (MSDN):

Shell32.dll版本5.0及更高版本。指示通知控件中指定的版本号执行操作lpdata所指向的结构的uVersion成员。的版本Number指定识别哪些成员。

通知图标多年来已经改变了行为。出于与已有代码兼容的原因,您必须选择加入新的行为。如果你不选择加入,那么你就不会收到WM_CONTEXTMENU消息。相反,你必须回应WM_RBUTTONUP。即使你从键盘上调用上下文菜单,系统仍然发送WM_RBUTTONUP。您必须通过调用GetCursorPos来获取光标位置,以便知道在哪里显示菜单。

你可以选择加入新的行为(和WM_CONTEXTMENU)在文档中描述,通过调用Shell_NotifyIcon传递NIM_SETVERSIONNIM_ADD调用之后。您正在查看的SDK示例可能在某个地方完成了此操作。我猜这就是你的代码中缺少的东西。

文档中的关键摘录在注释部分:

在Windows 2000 (Shell32.dll版本5.0)中,Shell_NotifyIcon鼠标和键盘事件的处理方式与在Microsoft Windows NT 4.0, Windows 95和Windows 98上发现的早期Shell版本不同。不同之处在于:

  • 如果用户用键盘选择通知图标的快捷菜单,Shell现在向相关的应用程序发送一个WM_CONTEXTMENU消息。早期版本发送WM_RBUTTONDOWN和WM_RBUTTONUP消息。
  • 如果用户用键盘选择一个通知图标,并使用空格键或ENTER键激活它,5.0版本Shell会向相关的应用程序发送一个NIN_KEYSELECT通知。早期版本发送WM_RBUTTONDOWN和WM_RBUTTONUP消息。
  • 如果用户用鼠标选择一个通知图标,并用ENTER键激活它,Shell现在向相关的应用程序发送一个NIN_SELECT通知。早期版本发送WM_RBUTTONDOWN和WM_RBUTTONUP消息。