Question

I want to make an application to intercept all UI events in all the forms of my application and to write them to a log. This data can than be used to see which controls are the most used, in what order, etc. The problem is that I want this to happen automatically, without modifying the existing classes.

I made a prototype that attaches a method to a click event for all controls in a form, but how can this be done for all forms? Reflection needs a target object when manipulating events, but only the startup form can be easily accessed.

Is there a way to hook the constructor of an object? Then I could "inject" my method in all the events of the new form. Or maybe there is another way to do this.

Thanks in advance!

Was it helpful?

Solution

You can install a message filter.

A message filter is an object that implements IMessageFilter. WinForms calls your PreFilterMessage method for every message that passes through your thread's message loop. This is enough to monitor user input across the application (and gives you the option of manipulating it).

OTHER TIPS

In Windows API this is done using local hooks (you can set local mouse hook using SetWindowsHookEx function). This is the proper way to do your task. In C# you need to use P/Invoke in order to get access to SetWindowsHookEx.

One more task would be to match the HWND (windows handle) to corresponding WinForms control. Read this article for how to do this (via WM_GETCONTROLNAME message).

Also see this question which is a duplicate of yours.

You will have to work with the Win32's API Messages, I guess.

Here's a little example under the form of tutorial.

You should be able to achieve what you want with message filters - no direct P/Invoke to Win32-APIs required!

See the help on the IMessageFilter interface for more info.

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top