Click here to Skip to main content
15,860,859 members
Articles / Desktop Programming / Windows Forms

InputManager library - Track user input and simulate input (mouse and keyboard)

Rate me:
Please Sign up or sign in to vote.
4.98/5 (39 votes)
13 Oct 2010CPOL5 min read 287.7K   24K   148   128
A library that contains classes to simulate user input and track user input (mouse and keyboard), in C# and VB.NET.

Introduction

The InputManager library allows you to take control over the whole user common input. From tracking (hooking) user input to simulating user input. It allows you to send and receive global mouse and keyboard messages, in one easy to use library. It supports DirectX input simulation, which means you can send mouse or keyboard messages to any DirectX game you want, and the game will register it correctly.

With this library, you can create a complete macro recording application; it's well documented, and can be easily used with any application.

Background

Starting my way to create my own Macro Recording application, today known as Mouse Recorder Pro, I needed to find a way to register mouse and keyboard messages and track them down. After a number of version updates for Mouse Recorder Pro, and after improving my own coding skills, I made this library to help others.

The hook classes in this library were not written by me (but they were modified by me) to be honest. I don't remember the person who helped with it, but if I am able to track him, I will update this article.

The other classes were written by me, with some research and a lot of reading. The main problem I had was to be able to register user input in DirectX games, which I managed to accomplish afterwards.

The Concept (Or, How it Works)

The library was written in VB.NET, and tested on many systems (as part of its use in the Mouse Recorder Pro commercial application).

The keyboard and mouse messages are being sent with the SendInput API, which is a bit more flexible than other APIs. With the SendInput API, your application can send mouse and keyboard messages in a low level (as DirectX is). This allows you to register those messages in any DirectX game.

The mouse and keyboard hook classes allow you to track the user input. The classes use the SetWindowsHookEx API. This API "binds" any new mouse or keyboard message to a specific method in the hook classes. Every time a new message arrives, Windows calls that method. Within that method, an event is raised to inform you that a new input message has arrived.

Screenshot.jpg

Mouse Recorder Pro 2 - it uses the InputManager library!

Using the code

To use this library, we first need to add a reference for it in our project; download the InputManager library from the link above, and extract it. Now do the following:

  1. Open up your project.
  2. Right click on "References" and "Add Reference".
  3. Select the "Browse" tab and select the extracted "InputManager.dll".
  4. Accept all dialogs.
  5. To make this project work in Debug mode, double click on Properties, go to "Debug", and uncheck "Enable the Visual Studio hosting process".

I wasn't able to use the class hook classes (used for tracking) with .NET Framework 4, so if you are using .NET Framework 4, please change the target framework to 3.5 or any framework below by going to the project properties and selecting a different version of the .NET Framework.

Now, let's add the "using" statement to make it a bit easier for coding. Add the following using statement:

C#
using InputManager;

OK, let's get to know how to use this library.

Tracking User Mouse and Keyboard Messages

InputManagerHookExample.jpg

Download this hook example at the top

To track the user mouse and keyboard messages, we need to set up a hook. How do we do that? With InputManager, it's pretty easy.

To track user keyboard messages, in your Form_Load event handling method, add the following code:

C#
//Adding keyboard event handlers and installing the hook
KeyboardHook.KeyDown += new KeyboardHook.KeyDownEventHandler(KeyboardHook_KeyDown);
KeyboardHook.KeyUp += new KeyboardHook.KeyUpEventHandler(KeyboardHook_KeyUp);
KeyboardHook.InstallHook();

The first two lines allow us to track the keyboard messages. This is done by two methods we will create to handle those events. The third line tells the InputManager library to set up a hook and start reading the keyboard messages.

Now let's add the two methods we talked about earlier:

C#
void KeyboardHook_KeyUp(int vkCode)
{
    //Everytime the users releases a certain key up,
    //your application will go to this line
    //Use the vKCode argument to determine which key has been released
}

void KeyboardHook_KeyDown(int vkCode)
{
    //Everytime the users holds a certain key down,
    //your application will go to this line
    //Use the vKCode argument to determine which key is held down
}

To convert those keys to a string instead of a virtual key code, use this simple technique:

C#
string keyString = ((Keys)vkCode).ToString();
//Using the Keys Enum ToString method for translating

To track user mouse messages, in your Form_Load event handling method, add the following code:

C#
//Adding mouse event handlers and installing the hook
MouseHook.MouseEvent += new MouseHook.MouseEventEventHandler(MouseHook_MouseEvent);
MouseHook.MouseMove += new MouseHook.MouseMoveEventHandler(MouseHook_MouseMove);
MouseHook.WheelEvent += new MouseHook.WheelEventEventHandler(MouseHook_WheelEvent);
MouseHook.InstallHook();

The first three lines tell the InputManager library to add event handlers for each mouse message type (mouse button events, mouse move event, and wheel scrolled event). We will add the methods that handle those messages below. The fourth line tells the InputManager class to install the hook and start receiving the messages.

Let's add the methods that handle those messages:

C#
void MouseHook_WheelEvent(MouseHook.MouseWheelEvents wEvent)
{
    //Event time the wheel was scrolled the application will go to this line
    //Using the wEvent argument we can detect 
    //if the mouse was scrolled forward or backward
}

void MouseHook_MouseMove(MouseHook.POINT ptLocat)
{
    //Everytime the mouse moved, the application will go to this line
    //Using the ptLocat arguments you can detect 
    //which point the mouse cursor moved to
}

void MouseHook_MouseEvent(MouseHook.MouseEvents mEvent)
{
    //Everytime a mouse button changed it's state 
    //(from up to down or down to up) the application will go to this line
    //Using the mEvent argument we can detect which button changed it's state
}

To detect if the wheel was scrolled up or down within the MouseHook_WheelEvent method, we use the MouseHook.MouseWheelEvents enum:

C#
string scrollDirection;
if (wEvent == MouseHook.MouseWheelEvents.ScrollDown)
    scrollDirection = "scrolled down";
else scrollDirection = "scrolled up";

To detect the position (X and Y) of the mouse every time it moves, we use the ptLocat argument within the MouseHook_MouseMove method:

C#
textBox1.Text = "The mouse cursor moved to: " + ptLocat.x + ", " + ptLocat.y;
//Updating a textbox text to the current mouse position

To detect which button changed its state, we use the the mEvent argument within the MouseHook_MouseEvent method:

C#
textBox1.Text = "Mouse button changed state to: " + mEvent.ToString();
//Updates the textbox to show which button changed it's state last time

You can use the switch statement to handle each mouse button/state change differently:

C#
switch (mEvent)
{
   case MouseHook.MouseEvents.LeftDown:
       //Left button is down, do something here
       break;

   case MouseHook.MouseEvents.LeftUp:
        //Left button is up, do something here
         break;
}

Sending Mouse and Keyboard Messages (With DirectX Support)

To send keyboard messages (key strokes), we use the Keyboard class in the InputManager library.

To send key down/key up messages, we use the Keyboard.KeyDown or Keyboard.KeyUp methods. An example of writing "hello" in a textbox:

C#
ExampleText.Focus(); //Sets focus on the example text

//Writing Hello with keys sending
Keyboard.KeyDown(Keys.H);
Keyboard.KeyUp(Keys.H);

Keyboard.KeyDown(Keys.E);
Keyboard.KeyUp(Keys.E);

Keyboard.KeyDown(Keys.L);
Keyboard.KeyUp(Keys.L);

Keyboard.KeyDown(Keys.L);
Keyboard.KeyUp(Keys.L);

Keyboard.KeyDown(Keys.O);
Keyboard.KeyUp(Keys.O);

To send shortcut keys, we use the Keyboard.ShortcutKeys method, with an array of Keys:

C#
ExampleText.Focus(); //Sets focus on the example textbox

//Sends Shift + Control + Left to select a word
Keys[] keys = new Keys[] { Keys.RShiftKey, Keys.RControlKey, Keys.Left };
Keyboard.ShortcutKeys(keys);

To send mouse messages (mouse key strokes, mouse movements, wheel scroll), we use the Mouse class in the InputManager library.

To move the mouse to a certain location (in screen coordinates), we use the Mouse.Move method:

C#
Mouse.Move(Point.X, Point.Y);
//Moves the mouse to a certain location on the screen

To perform a mouse click message, we use the Mouse.PressButton method:

C#
Mouse.PressButton(Mouse.MouseKeys.Left); //Performing a left click

To send mouse button down or up messages, we use the Mouse.ButtonDown/Mouse.ButtonUp methods:

C#
Mouse.ButtonDown(Mouse.MouseKeys.Left); //Holding the left mouse down
Mouse.ButtonUp(Mouse.MouseKeys.Left); //Releasing the left mouse button

To scroll up or down, we use the Mouse.Scroll method:

C#
Mouse.Scroll(Mouse.ScrollDirection.Down); //Scroling down
Mouse.Scroll(Mouse.ScrollDirection.Up); //Scrolling up

InputManagerSendingMessagesExample.jpg

Download this input simulating example at the top

Points of Interest

Writing this library (parts of it) was really fun and a bit challenging. There are other ways of sending mouse and keyboard messages with different APIs which don't support DirectX and are easier to use, but I wanted to simplify it and create my own library, so you don't even need to know what API means. With this library, you can send and receive input messages easily!

History

  • 13.10.2010: Added an explanation on how the library works (the concept behind it).

License

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


Written By
Software Developer Nemex Studios
Israel Israel
My name is Shay and I'm 21 years old.
At age 16 I created Nemex Studios (www.byshynet.com), an application development "company" (not official).
I'm the developer of "Mouse Recorder Pro", "Mouse Shaker" and many other applications, developed to help the community.

Comments and Discussions

 
AnswerRe: x86 version of Inputmanager Pin
shynet16-Aug-12 8:31
shynet16-Aug-12 8:31 
QuestionFinally sth. that works Pin
f4k0rg07722-Jul-12 7:17
f4k0rg07722-Jul-12 7:17 
Questionnice one Pin
mohamad yousef4-Jun-12 22:28
mohamad yousef4-Jun-12 22:28 
QuestionUpgrade to .net 4? Pin
zeppy1414-May-12 22:25
zeppy1414-May-12 22:25 
AnswerRe: Upgrade to .net 4? Pin
shynet17-May-12 6:08
shynet17-May-12 6:08 
QuestionKeys remapped after using Keyboard.KeyDown and Keyboard.KeyUp Pin
Larissa Schön4-Mar-12 4:28
professionalLarissa Schön4-Mar-12 4:28 
AnswerRe: Keys remapped after using Keyboard.KeyDown and Keyboard.KeyUp Pin
shynet8-Mar-12 3:50
shynet8-Mar-12 3:50 
QuestionMouse Resolution Pin
Larissa Schön24-Feb-12 9:27
professionalLarissa Schön24-Feb-12 9:27 
I was using it with the Darkness II demo and c#. I found that sending a mouse move from say

X = 1, Y = 4

to

X = 2, Y = 4

moves the mouse around 20° at once in the game. Using the real mouse it's smooth.

What to do about it?
AnswerRe: Mouse Resolution Pin
shynet24-Feb-12 23:12
shynet24-Feb-12 23:12 
GeneralRe: Mouse Resolution Pin
Larissa Schön24-Feb-12 23:32
professionalLarissa Schön24-Feb-12 23:32 
GeneralRe: Mouse Resolution Pin
shynet24-Feb-12 23:56
shynet24-Feb-12 23:56 
GeneralRe: Mouse Resolution Pin
Larissa Schön25-Feb-12 0:22
professionalLarissa Schön25-Feb-12 0:22 
GeneralIt worked Pin
futymax9-Feb-12 14:27
futymax9-Feb-12 14:27 
GeneralRe: It worked Pin
shynet10-Feb-12 3:17
shynet10-Feb-12 3:17 
QuestionHey bro הי אחי Pin
Member 849228115-Dec-11 8:07
Member 849228115-Dec-11 8:07 
AnswerRe: Hey bro הי אחי Pin
shynet15-Dec-11 8:20
shynet15-Dec-11 8:20 
QuestionDeclaration in VB.net Pin
Kakheng27-Oct-11 1:01
Kakheng27-Oct-11 1:01 
AnswerRe: Declaration in VB.net Pin
shynet27-Oct-11 2:47
shynet27-Oct-11 2:47 
GeneralRe: Declaration in VB.net Pin
Kakheng27-Oct-11 3:51
Kakheng27-Oct-11 3:51 
GeneralRe: Declaration in VB.net Pin
Kakheng27-Oct-11 4:04
Kakheng27-Oct-11 4:04 
GeneralRe: Declaration in VB.net Pin
shynet27-Oct-11 5:27
shynet27-Oct-11 5:27 
GeneralRe: Declaration in VB.net Pin
Kakheng27-Oct-11 14:22
Kakheng27-Oct-11 14:22 
GeneralRe: Declaration in VB.net Pin
shynet27-Oct-11 23:00
shynet27-Oct-11 23:00 
GeneralRe: Declaration in VB.net Pin
Kakheng28-Oct-11 3:26
Kakheng28-Oct-11 3:26 
QuestionRe: Declaration in VB.net Pin
crumbl3d11-Jul-13 4:53
crumbl3d11-Jul-13 4:53 

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.