c# — Keyboardhook на ленточном элементе управления Outlook

Я хочу подключить определенный элемент управления (поле со списком) и получить все клавиши, введенные в этом элементе управления. Поле со списком является частью ленты Outlook и не имеет таких событий, как нажатие клавиши или что-то в этом роде (только onChange, которое ведет себя очень странно).

Вот код:

    private const int WH_KEYBOARD_LL = 13;
    private const int WM_KEYDOWN = 0x0100;
    private LowLevelKeyboardProc _proc;
    private IntPtr _hookID = IntPtr.Zero;

    private void SetHook(IntPtr handle)
    {
        uint PID; //not needed
        _proc = HookCallback;
        uint threadid = GetWindowThreadProcessId(handle, out PID);
        _hookID = SetWindowsHookEx(WH_KEYBOARD_LL, _proc, IntPtr.Zero, threadid );

    }

    private delegate IntPtr LowLevelKeyboardProc(
            int nCode, IntPtr wParam, IntPtr lParam);

    private IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
    {
        if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN)
        {
            int vkCode = Marshal.ReadInt32(lParam);
            System.Diagnostics.Debug.WriteLine("Key: " + (Keys)vkCode);
        }
        return CallNextHookEx(_hookID, nCode, wParam, lParam);
    }

Имеющийся у меня дескриптор и полученный ThreadID верны (проверено с помощью Spy++), но ключ не захвачен. Прекрасно работает с «0» в качестве последнего параметра функции SetWindowsHookEx, но тогда, конечно, это глобальный хук.


person Burning86    schedule 29.11.2012    source источник
comment
Я только что узнал, что Marshal.GetLastWin32Error() возвращает ошибку 1429, что означает только глобальный хук, так что, в конце концов, нет?   -  person Burning86    schedule 29.11.2012


Ответы (1)


Я добавляю это для тех, у кого такая же проблема. Клавиатурные обработчики глобальны, их нельзя подключить к определенному элементу управления. Что вам нужно сделать, так это захватить сообщения данного дескриптора. Для этого вам нужно создать подкласс вашего окна/дескриптора.

    [DllImport("user32")]
    private static extern IntPtr SetWindowLong(IntPtr hWnd, int nIndex, Win32WndProc newProc);
    [DllImport("user32")]
    private static extern int CallWindowProc(IntPtr lpPrevWndFunc, IntPtr hWnd, int Msg, int wParam, int lParam);

    // A delegate that matches Win32 WNDPROC:
    private delegate int Win32WndProc(IntPtr hWnd, int Msg, int wParam, int lParam);

    // from winuser.h:
    private const int GWL_WNDPROC = -4;
    private const int WM_KEYDOWN = 0x0100;

    // program variables
    private IntPtr oldWndProc = IntPtr.Zero;
    private Win32WndProc newWndProc = null;

    private void SubclassHWnd(IntPtr hWnd)
    {
        // hWnd is the window you want to subclass..., create a new 
        // delegate for the new wndproc
        newWndProc = new Win32WndProc(MyWndProc);
        // subclass
        oldWndProc = SetWindowLong(hWnd, GWL_WNDPROC, newWndProc);
    }

    private const int ENTER_KEY = 1835009;

    // this is the new wndproc, just show a messagebox on left button down:
    private int MyWndProc(IntPtr hWnd, int Msg, int wParam, int lParam)
    {

        switch (Msg)
        {
            case WM_KEYDOWN:
                int vkCode = lParam;
                if (vkCode == ENTER_KEY)
                    doSomething();
                return 0;

            default:
                break;
        }

        return CallWindowProc(oldWndProc, hWnd, Msg, wParam, lParam);
    }
person Burning86    schedule 30.11.2012
comment
Не могли бы вы сообщить нам, какую ручку вы используете для Outlook? - person Ole K; 23.12.2013