События клавиатуры C ++ Win32

Я работаю над своим регистратором нажатия клавиш для личного интереса и вчера задал связанный с этим вопрос; Цикл while, использующий много ЦП.

Проблема с программой заключалась в том, что она потребляла слишком много ресурсов ЦП, и люди предлагали сделать входные данные на основе ключевых событий.

Поскольку я новичок в Win32 API, я пытаюсь найти ссылки и руководства, которые расскажут мне, как создавать ввод с клавиатуры на основе событий, а не на основе опросов. Но проблема в том, что я не смог найти никаких убедительных примеров или ссылок, так как это было довольно сложно понять для полного новичка.

Большинство из них упомянули, что программирование на основе событий осуществляется в приложении с графическим интерфейсом пользователя, но я хочу, чтобы это приложение для регистрации нажатий клавиш было консольным приложением.

Из всего этого у меня есть два основных вопроса:

  • Могу ли я написать регистратор нажатий клавиш консоли на основе событий с помощью Win32 API? Если нет, то какие у меня варианты?

и

  • Есть ли у кого-нибудь справочные веб-сайты, которые помогут мне понять, как фиксировать нажатия клавиш на основе событий?

Если требуется дополнительная информация, я использую блоки кода под Windows XP с компилятором GCC.


person Community    schedule 17.09.2009    source источник


Ответы (2)


Приложения регистратора ключей используют такие механизмы, как Перехватчики Win32. В частности, вам нужно установить хук WH_KEYBOARD.

Существуют продвинутые методы перемещения, такие как создание собственного драйвера клавиатуры, но для начала хорошим выбором будут хуки.

Изменить: чтобы получить представление о том, как выглядит процедура обработки, я публикую фрагмент из моей личной утилиты.

// ...
thehook = SetWindowsHookEx( WH_KEYBOARD_LL, hook_proc, hwnd, 0 );
// ...

/**
 *
 *  wParam, one of the: WM_KEYDOWN, WM_KEYUP, WM_SYSKEYDOWN, or WM_SYSKEYUP
    lParam: pointer to a KBDLLHOOKSTRUCT structure

    (*) "The hook procedure should process a message in less time than the
    data entry specified in the LowLevelHooksTimeout value in the following registry key: 
    HKEY_CURRENT_USER\Control Panel\Desktop 

    The value is in milliseconds. If the hook procedure does not 
    return during this interval, the system will pass the message to the next hook."

 *
 */
LRESULT CALLBACK
hook_proc( int code, WPARAM wParam, LPARAM lParam )
{
  static long ctrl_cnt = 0;
  static bool mmode = false;
  static DWORD time;

  KBDLLHOOKSTRUCT*  kbd = (KBDLLHOOKSTRUCT*)lParam;

  if (  code < 0
  ||   (kbd->flags & 0x10) // ignore injected events
     ) return CallNextHookEx( thehook, code, wParam, lParam );

  long ret = 1; // by default I swallow the keys
  if (  mmode  ) { // macro mode is ON
    if (  WM_KEYDOWN == wParam  )
      PostMessage(mainwnd, WM_MCR_ACCUM, kbd->vkCode, 0);

    if (  WM_KEYUP == wParam  )
      switch (kbd->vkCode) {
        case VK_ESCAPE:
          mmode = false;
          keys.removeall();
          PostMessage(mainwnd, WM_MCR_HIDE, 0, 0);
          break;

        case VK_RETURN:
          PostMessage(mainwnd, WM_MCR_EXEC, 0, 0);
          break;

        case VK_LCONTROL:
          mmode = false;
          PostMessage(mainwnd, WM_MCR_HIDE, 0, 0);
          PostMessage(mainwnd, WM_MCR_EXEC, 0, 0);
          break;
      }

    /* Which non printable keys allow passing? */
    switch( kbd->vkCode ) {
      case VK_LCONTROL:
      case VK_CAPITAL:
      case VK_LSHIFT:
      case VK_RSHIFT:
        ret = CallNextHookEx( thehook, code, wParam, lParam );
    }
  }
  else { // macro mode is OFF
    /* Ctrl pressed */
    if (  kbd->vkCode == VK_LCONTROL && WM_KEYDOWN == wParam  ) {
      ctrl_cnt = 1;
      time = kbd->time;
    }

    /* Prevent ctrl combinations to activate macro mode */
    if (  kbd->vkCode != VK_LCONTROL  )
      ctrl_cnt = 0;

    /* Ctrl released */
    if (  ctrl_cnt == 1 && WM_KEYUP == wParam  ) {
      if (  kbd->time - time > 40  ) {
        mmode = true;
        PostMessage(mainwnd, WM_MCR_SHOW, 0, 0);
      }
    }

    ret = CallNextHookEx( thehook, code, wParam, lParam ); // let it pass
  }

  return ret;
}
person Nick Dandoulakis    schedule 17.09.2009
comment
Как получить ПОДДЕРЖКУ другой программы? - person Matouš Vrba; 26.03.2014
comment
@ MatoušVrba, вам не нужно заводить ручки других программ. - person Nick Dandoulakis; 27.03.2014

Взгляните на API SetWindowsHookEx.

Вы можете сделать это как в консольном, так и в оконном приложении. Вам нужно будет поместить вашу клавиатуру в DLL, чтобы вы могли захватывать нажатия клавиш в процессах, отличных от ваших.

здесь есть пример кода.

person Ferruccio    schedule 17.09.2009
comment
Просто любопытно, что именно делает крючок? Я читал, что он внедряется в другую программу, или, возможно, это я понимаю. Кроме того, уменьшит ли это количество использования ЦП, которое использует регистратор нажатий клавиш? - person ; 17.09.2009
comment
Хук добавляет обратный вызов вашей функции, когда происходит выбранное событие. Вам не понадобится цикл, постоянно опрашивающий клавиатуру. Ваш код выполняется только при необходимости. - person Ferruccio; 17.09.2009
comment
См. Также это обсуждение stackoverflow.com/questions/3134183/ (я считаю, что exe может работать с сообщениями * _LL, но для всего остального требуется dll). - person rogerdpack; 11.09.2013
comment
Я отправил правку, чтобы добавить ваш комментарий к вашему ответу. Не стесняйтесь отклонить, если вам не нравится редактирование. :) - person Cypher; 18.07.2017