Click here to Skip to main content
15,867,568 members
Articles / Programming Languages / C#
Article

A Simple C# Global Low Level Keyboard Hook

Rate me:
Please Sign up or sign in to vote.
4.90/5 (98 votes)
30 May 2007CPOL2 min read 967.6K   68.1K   171   187
A simple description and sample of creating a global low level keyboard hook in C#
Screenshot - key_preview.jpg

Introduction

This article discusses a class that I wrote that wraps a global low level keyboard hook. The sample hooks the A and B keys, just for demonstration.

Background

I was trying to find a way for an application that I am writing to restore itself when a combination of keys was pressed. This was born from searching around for the answer.

Using the Code

First download the source, and add globalKeyboardHook.cs to your project. Then add...

C#
using Utilities;

... to the top of the file you are going to use it in. Next add an instance of globalKeyboardHook to your class:

C#
globalKeyboardHook gkh = new globalKeyboardHook() ;

When a globalKeyboardHook is constructed, it automatically installs the hook, so all there is left to do is add some keys for it to watch, and define some event handlers for the KeyDown and KeyUp events. I usually do this on the main form's load event handler, like this:

C#
private void Form1_Load(object sender, EventArgs e) {
    gkh.HookedKeys.Add(Keys.A);
    gkh.HookedKeys.Add(Keys.B);
    gkh.KeyDown += new KeyEventHandler(gkh_KeyDown);
    gkh.KeyUp += new KeyEventHandler(gkh_KeyUp);
} 

void gkh_KeyUp(object sender, KeyEventArgs e) {
    lstLog.Items.Add("Up\t" + e.KeyCode.ToString());
    e.Handled = true ;
}

void gkh_KeyDown(object sender, KeyEventArgs e) {
    lstLog.Items.Add("Down\t" + e.KeyCode.ToString());
    e.Handled = true ;
} 

Here I have chosen to watch for the A and B keys, and defined handlers for the KeyUp and KeyDown events that both log to a listbox called lstLog. So whenever the user presses the A or B keys, no matter what has focus, the application will be notified. Setting e.Handled to true makes it so no other notifications for this event go out, in the sample, this effectively stops the user from typing an A or B. This can be useful in ensuring that key combinations are not also typed out when used.

You can add hooks for as many keys as you would like, just add them like above. Don't get frustrated if you add a hook for a key and it doesn't work, many of them, like Keys.Shift show up as other more specific keys, like Keys.LShiftKey or Keys.RShiftKey. Keys.Alt shows up as Keys.LMenu or Keys.RMenu, Keys.Control shows up as Keys.LControl or Keys.RControl, just to name a few.

If you would like to hook or unhook the keyboard hook at any point, just call your globalKeyboardHook's hook and unhook methods, like so:

C#
//unhook
gkh.unhook() 
//set the hook again
gkh.hook() 

Points of Interest

The bulk of the work in this code is done in the globalKeyboardHook class, although it is a fairly simple piece of code itself. The hardest part of doing this was finding the correct parameters for SetWindowsHookEx.

C#
[DllImport("user32.dll")]
static extern IntPtr SetWindowsHookEx
    (int idHook, keyboardHookProc callback, IntPtr hInstance, uint threadId);

The parameters were worked out to be:

C#
IntPtr hInstance = LoadLibrary("User32");
hhook = SetWindowsHookEx(WH_KEYBOARD_LL, hookProc, hInstance, 0);

The first parameter WH_KEYBOARD_LL is just saying that we want to hook the low level keyboard events, hookProc is the callback for the event, hInstance is a handle to User32.dll, where this event is first processed (I think). The last parameter is if you want to hook a specific thread, then you would just pass a thread id instead of using the hInstance.

History

  • 5/30/07 - First version

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer
United States United States
I currently work as a Software Engineer for a company in North Carolina, mainly working with C#.

Comments and Discussions

 
GeneralRe: MDA error garbage collected delegate Pin
StormySpike13-Mar-09 14:45
StormySpike13-Mar-09 14:45 
GeneralRe: MDA error garbage collected delegate Pin
weso18-Aug-09 15:54
weso18-Aug-09 15:54 
GeneralRe: MDA error garbage collected delegate Pin
Member 25165064-Oct-09 10:31
Member 25165064-Oct-09 10:31 
GeneralRe: MDA error garbage collected delegate Pin
avihaydar1012-Oct-09 22:21
avihaydar1012-Oct-09 22:21 
GeneralRe: MDA error garbage collected delegate Pin
kendolew23-Oct-09 5:43
kendolew23-Oct-09 5:43 
QuestionHow to use this for capturing all the keys? Pin
Member 475432628-Jan-09 9:46
Member 475432628-Jan-09 9:46 
AnswerRe: How to use this for capturing all the keys? Pin
StormySpike13-Mar-09 14:41
StormySpike13-Mar-09 14:41 
AnswerRe: How to use this for capturing all the keys? Pin
jwainer6-Aug-09 14:33
jwainer6-Aug-09 14:33 
GeneralRe: How to use this for capturing all the keys? Pin
BuggingMe22-Mar-10 6:54
BuggingMe22-Mar-10 6:54 
AnswerRe: How to use this for capturing all the keys? Pin
Shrish Pandit25-Mar-11 15:51
Shrish Pandit25-Mar-11 15:51 
GeneralNot capturing after random number of key strokes ! Pin
Member 475432611-Jan-09 20:38
Member 475432611-Jan-09 20:38 
GeneralMy vote of 1 Pin
Member 407286520-Dec-08 8:29
Member 407286520-Dec-08 8:29 
GeneralMy vote of 1 Pin
Member 407286520-Dec-08 7:00
Member 407286520-Dec-08 7:00 
GeneralReplace keystroke with another key Pin
stormydaniels3-Nov-08 5:51
stormydaniels3-Nov-08 5:51 
GeneralRe: Replace keystroke with another key Pin
StormySpike10-Nov-08 9:07
StormySpike10-Nov-08 9:07 
GeneralClarion MDI child window in C# Pin
papacol22-Oct-08 23:37
papacol22-Oct-08 23:37 
GeneralRe: Clarion MDI child window in C# Pin
StormySpike10-Nov-08 9:12
StormySpike10-Nov-08 9:12 
GeneralRe: Clarion MDI child window in C# Pin
papacol13-Nov-08 22:52
papacol13-Nov-08 22:52 
GeneralNullReferenceException for SetWindowsHookEx Pin
Wayne Walter19-Oct-08 17:34
Wayne Walter19-Oct-08 17:34 
GeneralRe: NullReferenceException for SetWindowsHookEx Pin
Wayne Walter19-Oct-08 19:20
Wayne Walter19-Oct-08 19:20 
GeneralHooked Keys cannot be used Pin
sashdude8-Aug-08 8:16
sashdude8-Aug-08 8:16 
GeneralRe: Hooked Keys cannot be used Pin
Michaelwh8-Aug-08 10:32
Michaelwh8-Aug-08 10:32 
QuestionHow do I do it without forms? Pin
djcouture29-Jul-08 5:06
djcouture29-Jul-08 5:06 
AnswerRe: How do I do it without forms? Pin
StormySpike9-Aug-08 5:18
StormySpike9-Aug-08 5:18 
GeneralRe: How do I do it without forms? Pin
djcouture14-Aug-08 8:18
djcouture14-Aug-08 8:18 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.