Introduction
(Rev 2.0)
I often use Spy++ to track window messages but have rarely used it to look at the attributes of a window.
When i have used it, I've found using the finder tool to be a really awkward device because it presents a static interface to what i find to be a much more dynamic activity.
I decided therefore to see if I could devise an alternative interface which would allow me to look at a window's attributes with the minimum of fuss and bother.
The result is MiniSpy.
There's nothing very special about it, but it does provide the following features:
- All attributes are visible in one pane
- Dynamically updates as you drag it around
- Displays control specific styles for all of the standard windows and common controls separately to the standard window styles.
- Provides a filter drop-down so that if you have registered you own window class and it is based on a standard control class you can manually select the base window class to see the control styles
- 'Refresh' button for refreshing the current window's attributes
- Provides tooltips to view long rows
History
New Features (2.0)
- 'Lock Updates' button/shortcut to prevent updating of the picked window as you move MiniSpy.
- Buttons/shortcuts to navigate to parent window, first child windows, next/previous windows (note: these buttons will automatically lock updates to prevent the new window being immediately overridden).
- Vertical toolbar so that all buttons are visible.
- Manual implementation of
RealWindowFromPoint()
for win95 (see bug fixes (1.2) below).
- Addition of control styles for tooltips_class32.
Bug Fixes (2.0)
- Missing Menu (#32768), Desktop (#32769), Icon Title (#32772), tooltips_class32, Task Switcher (#32771) classnames added to filter combobox.
New Features (1.2)
- Show the 'hittest' result returned by the window in response to a
WM_NCHITTEST
message
Bug Fixes (1.2)
- Correctly pick windows contained by a group box. Thanks to Philippe Lhoste (PhiLho@GMX.net)
Previously, MiniSpy used the method ::WindowFromPoint()
to pick the window. However, as Philippe pointed out this does not always work with controls within group boxes, specifically when the controls are behind the group box in the z-order.
To solve this I have extended the window picking to also use ::RealChildWindowFromPoint()
once the initial pick is done. The documentation for this function effectively states that it ignores windows which return HTTRANSPARENT
to a WM_NCHITTEST
message, which group boxes do.
This solves the problem except for nested group boxes which are ignored by ::RealChildWindowFromPoint
because they too return HTTRANSPARENT
.
The docs also indicated that ::RealChildWindowFromPoint()
is not supported on windows 95 which is why I have linked to it dynamically.
New Features (1.1)
- Can use any one of its corners to pick the windows
- Shows window properties (via
EnumPropsEx
)
- Includes richedit
ES_
styles
- Includes richedit
ENM_
styles
- Includes common control CCS_ styles for toolbars and status bars
- Provides a 'Restore Size' button to return to the startup size
Bug Fixes (1.11)
- Crash under XP when enumerating properties using EnumPropsEx(). Thanks to limax (limax@hot.ee).
Some Comments on the Source Code
- All the class, window, windowEx and control styles that are used are located in styles.h. If I've missed any, feel free to add them to your version and mail them to me for addition
- I decided to make MiniSpy 'always on top' as i find it most useful in that mode, but when i turned on tooltips in the list control these appeared behind the MiniSpy Window! Since the tooltips are a very useful feature for the style rows, I came up with an interesting workaround: I handle
WM_ACTIVATEAPP
and make MiniSpy 'topmost' when its not the active window and 'not topmost' when it is active.
- To retrieve the window caption for the window and its parent I've used
WM_GETTEXT
instead of GetWindowText
. This is simply because, for one reason or another, GetWindowText
did not seem to work for some windows in this situation.