Събитията с клавиши с цифрова клавиатура водят до заседнали клавиши от GetKeyboardState

Имам приложение на C++ (MFC), което трябва да провери състоянието на ключа на таймер. Ако потребителят държи натиснат клавиш, забавяме обработката на някакъв код.

Ето чека за keydown:

if (!GetKeyboardState(keyState)) 
{
    s_executeDeferredResult = e_executeDeferredButtonCheckFailed;
    return;
}   
s_executeDeferredStuckKeys.clear();
for (int index=0; index<arrsize(keyState); ++index)
{
    if (keyState[index] & 0x80)
    {
        s_executeDeferredStuckKeys.insert(index);
    }
}
if (!s_executeDeferredStuckKeys.empty())
{
    s_executeDeferredResult = e_executeDeferredButtonsActive;
    return;
}

Но има някои ключови комбинации, които се забиват:

  1. Включете NUMLOCK
  2. Натиснете SHIFT
  3. Натиснете NumPad8
  4. Пуснете SHIFT
  5. Освободете NumPad8
    (това е един пример, има и други, включително doozy с CTRL-ALT-DEL)

GetKeyboardState сега ще съобщи, че VK_UP е натиснат.

Събитията, които се случват са (съответстващи на действията по-горе).

  1. <None>
  2. WM_KEYDOWN, VK_SHIFT
  3. WM_KEYUP, VK_SHIFT
    WM_KEYDOWN, VK_UP
    WM_KEYDOWN, VK_SHIFT
  4. WM_KEYUP, VK_SHIFT
  5. WM_KEYUP, VK_NUMPAD8

И така, Windows не разпознава, че се е появил клавишът Up и сега GetKeyboardState е повреден.

Има ли някакъв добър начин да се провери реалното състояние на ключа? GetAsyncKeyState и GetKeyState съобщават, че ключът също е свален.


person Craig Steele    schedule 28.08.2013    source източник
comment
Пробвал ли си с друга клавиатура?   -  person Jonathan Potter    schedule 29.08.2013
comment
Да, случва се на всички клавиатури, които сме пробвали.   -  person StilesCrisis    schedule 29.08.2013
comment
Тъй като това е MFC приложение, не забравяйте да проверите внедряването на PreTranslateMessage. Там с времето се натрупват всякакви лоши неща. Обърнете внимание и на забележката в документацията GetKeyboardState: < i>Състоянието се променя, тъй като нишката премахва съобщенията от клавиатурата от своята опашка със съобщения. С други думи: Ако приложението ви направи нещо подозрително, има вероятност GetKeyboardState да се обърка.   -  person IInspectable    schedule 29.08.2013
comment
Това поведение е възпроизводимо в малко самостоятелно тестово приложение, както се оказва. Все пак добра мисъл.   -  person StilesCrisis    schedule 29.08.2013


Отговори (1)


Реших го.

Свързах се със събитията на клавиатурата в InitInstance и проследявам възходите и паденията чрез код за сканиране (карта с кода за сканиране като ключ и виртуалните клавиши като стойност).

m_keyboardHook = SetWindowsHookEx(WH_KEYBOARD, &KeyboardHook, NULL, GetCurrentThreadId());

static LRESULT CALLBACK KeyboardHook(
    __in int nCode,
    __in WPARAM wParam,
    __in LPARAM lParam
    )
{
    // According to the docs, we're not allowed to do any "further processing" if nCode is < 0.
    // According to the docs, we should process messages if we get code HC_ACTION. 
    // http://msdn.microsoft.com/en-us/library/windows/desktop/ms644984(v=vs.85).aspx
    // It doesn't specify what to do if another code comes in, so we will just ignore any other codes.
    if (nCode == HC_ACTION)
    {
        uint8 scanCode = (uint8)((lParam & 0x00FF0000) >> 16);
        uint8 virtKey = (uint8)wParam;
        if (lParam & 0x80000000) // key up
            KeyState::SetKeyUp(scanCode);
        else
            KeyState::SetKeyDown(scanCode, virtKey);
    }

    // We must call the next hook in the chain, according to http://msdn.microsoft.com/en-us/library/windows/desktop/ms644975%28v=vs.85%29.aspx
    // First param is ignored, according to http://msdn.microsoft.com/en-us/library/windows/desktop/ms644974%28v=vs.85%29.aspx )
    return CallNextHookEx(0, nCode, wParam, lParam);
}

И така, моята отложена проверка става:

// Similarly, don't process deferred events if there are keys or mouse buttons which are currently down.
s_executeDeferredStuckKeys.clear();
if (KeyState::AnyKeysDown(s_excludeKeys, arrsize(s_excludeKeys)))
{
    s_executeDeferredResult = e_executeDeferredButtonsActive;
    KeyState::GetDownKeys(s_executeDeferredStuckKeys);
    return;
}
person Craig Steele    schedule 29.08.2013