Problems with creating a mouse hook using the Windows API

2.8k views Asked by At

I am trying to create a mouse hook using the Windows API. Currently, I have the following code, but I am struggling to make it work. Perhaps some sample code would help me to understand the concepts better.

LRESULT CALLBACK MouseProc(int nCode, WPARAM wParam, LPARAM lParam)
{
    HWND hWindow , hEntryWnd;
    HRESULT hResult;
    ztcEvents pEvent;
    ztcField *p_click_field;
    ztcInterfaceContext *p_context_interface;
    TCHAR      className [32];
    HWND hWnd;
    if (nCode < 0) // do not process message 
        return CallNextHookEx(hook, nCode, wParam, lParam); 
    MOUSEHOOKSTRUCT *msg = (MOUSEHOOKSTRUCT *)lParam;

    switch (nCode) 
    { 
    case HC_ACTION: 
        {
            hWnd                =  msg->hwnd;
            p_click_field       = pEvent.GetMatchField(hWnd);
            p_context_interface = pEvent.getMarkInterface();
            if(p_click_field == NULL || p_context_interface == NULL)
            {
                break;
            }
            GetClassName(p_click_field->Widget ()->WindowHandle() , className , sizeof (className ) - 1 );

            if(p_click_field->Widget()->IsKindOf(RUNTIME_CLASS(ztcOleWdgt)))
            {
                switch(wParam)
                {
                    case WM_LBUTTONDOWN:

                        hWindow = p_click_field->Widget()->Parent()->WindowHandle();
                        hEntryWnd = p_click_field->Widget()->WindowHandle();

                        ::PostMessage(hWindow , OLE_NOTIFY , (WPARAM)hEntryWnd , OLE_LCLICK);
                        /*pEvent.SetVariables(p_click_field , evClick ,etFormField , true , p_context_interface);
                        SetEvent(hEventWnd);*/
                        fprintf(trace_event , "buttonClick\t%x\n" , msg->hwnd );
                        fflush(trace_event);
                        break;
                    case WM_RBUTTONDOWN:
                        fprintf(trace_event , "RightClick\n");
                        fflush(trace_event);
                        break;
                    case WM_LBUTTONDBLCLK:
                        fprintf(trace_event , "ButtonDoubleClick\n");
                        fflush(trace_event);
                        break;
                }
            }
        }

        break;
      default:
        break;
    }
    return CallNextHookEx(hook, nCode, wParam, lParam); 
}    

While clicking the left mouse button, the code above went into the case WM_LBUTTONDOWN for more than 30 times. Is there something wrong with my code? If so, how can I fix it?

1

There are 1 answers

1
Cody Gray - on strike On BEST ANSWER

Are you clicking the mouse button more than 30 times? There will be an WM_LBUTTONDOWN message sent each the the mouse button is clicked, even if it is part of a double-click. That's a consequence of how Windows recognizes a double-click in the absence of a way to go back in time.

This sample looks trustworthy, and was written by a fairly reputable author:

     Hooks and DLLs