Mouse manipulation

By | 2002-06-01

At some point, you may find it useful to manipulate the location of the mouse cursor. Perhaps you are
designing an interactive tutorial, a walkthrough, or maybe you plan on controlling another application
through mouse events. Regardless, you will quickly find a number of hurdles to overcome and it is the
goal of this example to help you over, under, or around these hurdle

Original Author: Tims VB 5 tips and tricks

Assumptions

First, a little bit of background about the mouse in general. Since the days of DOS, mouse drivers have
reported their location in graphic applications by returning X/Y coordinates based up on a resolution
independent coordinate system. This coordinate system neatly breaks the screen down into 65535
units on each axis. A unit of measurement in this system is known as a Mickey. This system was
devised to insure that the specification for mouse drivers would be a lasting one, and that screen
resolution would never overtake the resolution of the mouse driver.
Why mention this? Well, the Win32 API function call which allows you to specify the location for the
mouse wants the location provided in mickeys. And the first hurdle to overcome is converting screen
coordinates to mouse coordinates.
In order to make the coversion, we first need to get the screen’s height and width with
GetSystemMetrics. The GetScreenRes subroutine illustrates how this is done.
Once the resolution of the display is known, we can convert the pixels returned by GetScreenRes into
mickeys. There are four conversion routines included with this example, two to handle pixel
conversions to mickeys (PixelXToMickey, PixelYToMickey), and two to handle mickey to pixel
conversions (MickeyXToPixel, MickeyYToPixel).
Now that we have conversion routines, we can actually do some work. Included with this example is
CenterMouseOn, a function that will center the mouse cursor on anything that has an hWnd. An
example of using this function to put the cursor over a commandbutton appears as:
CenterMouseOn (command1.hWnd)
If you need to move the mouse but don’t have an hWnd to reference, the MouseMove function will
allow you to specify an X/Y coordinate for the mouse cursor. And once it is moved, you can use the
MouseFullClick function to simulate a mouseclick.

Side Effects

There are a series of mouse coordinate to screen coordinate routines included in this example. Due to
rounding problems, it is quite likely that the calculations may be off by a pixel. If your application
requires extremely precise pointer placement, you may want to develop or look for a more precise
method.
One of the uglier portions of this code are the mickey to pixel routines. They use a series of
temporary singles to store values prior to being converted. This was done to improve the accuracy of
the conversion, but even so, rounding errors continue to creep in. If you know of a better, more
accurate way to accomplish the same task, I would appreciate hearing about it.

API Declarations

‘ ———————————————-
‘ * MouseEvent Related Declares *
‘ ———————————————-
Private Const MOUSEEVENTF_ABSOLUTE = &H8000
Private Const MOUSEEVENTF_LEFTDOWN = &H2
Private Const MOUSEEVENTF_LEFTUP = &H4
Private Const MOUSEEVENTF_MIDDLEDOWN = &H20
Private Const MOUSEEVENTF_MIDDLEUP = &H40
Private Const MOUSEEVENTF_MOVE = &H1
Private Const MOUSEEVENTF_RIGHTDOWN = &H8
Private Const MOUSEEVENTF_RIGHTUP = &H10
Private Declare Sub mouse_event Lib “user32” (ByVal dwFlags As Long, _
ByVal dx As Long, ByVal dy As Long, ByVal cbuttons As Long, _
ByVal dwExtraInfo As Long)
‘ ———————————————-
‘ * GetSystemMetrics Related Declares *
‘ ———————————————-
Private Const SM_CXSCREEN = 0
Private Const SM_CYSCREEN = 1
Private Const TWIPS_PER_INCH = 1440
Private Const POINTS_PER_INCH = 72
Private Declare Function GetSystemMetrics Lib “user32” (ByVal nIndex _
As Long) As Long
‘ ———————————————-
‘ * GetWindowRect Related Declares *
‘ ———————————————-
Private Type RECT
Left As Long
Top As Long
Right As Long
Bottom As Long
End Type
Private Declare Function GetWindowRect Lib “user32” (ByVal hwnd As Long, _
lpRect As RECT) As Long

‘ ———————————————-
‘ * Internal Constants and Types *
‘ ———————————————-
Private Const MOUSE_MICKEYS = 65535
Public Enum enReportStyle
rsPixels
rsTwips
rsInches
rsPoints
End Enum
Public Enum enButtonToClick
btcLeft
btcRight
btcMiddle
End Enum

Code


     ' Returns the screen size in pixels or, optionally,
     ' in others scalemode styles
     Public Sub GetScreenRes(ByRef X As Long, ByRef Y As Long, Optional ByVal _
       ReportStyle As enReportStyle)
       X = GetSystemMetrics(SM_CXSCREEN)
       Y = GetSystemMetrics(SM_CYSCREEN)
       If Not IsMissing(ReportStyle) Then
         If ReportStyle <> rsPixels Then
           X = X * Screen.TwipsPerPixelX
           Y = Y * Screen.TwipsPerPixelY
           If ReportStyle = rsInches Or ReportStyle = rsPoints Then
             X = X TWIPS_PER_INCH
             Y = Y TWIPS_PER_INCH
             If ReportStyle = rsPoints Then
               X = X * POINTS_PER_INCH
               Y = Y * POINTS_PER_INCH
             End If
           End If
         End If
       End If
     End Sub

     ' Convert's the mouses coordinate system to
     ' a pixel position.
     Public Function MickeyXToPixel(ByVal mouseX As Long) As Long
       Dim X As Long
       Dim Y As Long
       Dim tX As Single
       Dim tmouseX As Single
       Dim tMickeys As Single
      
       GetScreenRes X, Y
       tX = X
       tMickeys = MOUSE_MICKEYS
       tmouseX = mouseX
      
       MickeyXToPixel = CLng(tmouseX / (tMickeys / tX))
      
     End Function

     ' Converts mouse Y coordinates to pixels
     Public Function MickeyYToPixel(ByVal mouseY As Long) As Long
       Dim X As Long
       Dim Y As Long
       Dim tY As Single
       Dim tmouseY As Single
       Dim tMickeys As Single
      
       GetScreenRes X, Y
       tY = Y
       tMickeys = MOUSE_MICKEYS
       tmouseY = mouseY
      
       MickeyYToPixel = CLng(tmouseY / (tMickeys / tY))
      
     End Function

     ' Converts pixel X coordinates to mickeys
     Public Function PixelXToMickey(ByVal pixX As Long) As Long
       Dim X As Long
       Dim Y As Long
       Dim tX As Single
       Dim tpixX As Single
       Dim tMickeys As Single
      
       GetScreenRes X, Y
       tMickeys = MOUSE_MICKEYS
       tX = X
       tpixX = pixX
      
       PixelXToMickey = CLng((tMickeys / tX) * tpixX)
     End Function

     ' Converts pixel Y coordinates to mickeys
     Public Function PixelYToMickey(ByVal pixY As Long) As Long
       Dim X As Long
       Dim Y As Long
       Dim tY As Single
       Dim tpixY As Single
       Dim tMickeys As Single
      
       GetScreenRes X, Y
       tMickeys = MOUSE_MICKEYS
       tY = Y
       tpixY = pixY
      
       PixelYToMickey = CLng((tMickeys / tY) * tpixY)
     End Function

     ' The function will center the mouse on a window
     ' or control with an hWnd property. No checking
     ' is done to ensure that the window is not obscured
     ' or not minimized, however it does make sure that
     ' the target is within the boundaries of the
     ' screen.
     Public Function CenterMouseOn(ByVal hwnd As Long) As Boolean
       Dim X As Long
       Dim Y As Long
       Dim maxX As Long
       Dim maxY As Long
       Dim crect As RECT
       Dim rc As Long
       GetScreenRes maxX, maxY
       rc = GetWindowRect(hwnd, crect)
      
       If rc Then
         X = crect.Left + ((crect.Right - crect.Left) / 2)
         Y = crect.Top + ((crect.Bottom - crect.Top) / 2)
         If (X >= 0 And X <= maxX) And (Y >= 0 And Y <= maxY) Then
           MouseMove X, Y
           CenterMouseOn = True
         Else
           CenterMouseOn = False
         End If
       Else
         CenterMouseOn = False
       End If
     End Function

     ' Simulates a mouse click
     Public Function MouseFullClick(ByVal MBClick As enButtonToClick) As Boolean
       Dim cbuttons As Long
       Dim dwExtraInfo As Long
       Dim mevent As Long
      
       Select Case MBClick
         Case btcLeft
           mevent = MOUSEEVENTF_LEFTDOWN Or MOUSEEVENTF_LEFTUP
         Case btcRight
           mevent = MOUSEEVENTF_RIGHTDOWN Or MOUSEEVENTF_RIGHTUP
         Case btcMiddle
           mevent = MOUSEEVENTF_MIDDLEDOWN Or MOUSEEVENTF_MIDDLEUP
         Case Else
           MouseFullClick = False
           Exit Function
       End Select
       mouse_event mevent, 0&, 0&, cbuttons, dwExtraInfo
       MouseFullClick = True
      
     End Function

     Public Sub MouseMove(ByRef xPixel As Long, ByRef yPixel As Long)
       Dim cbuttons As Long
       Dim dwExtraInfo As Long
      
       mouse_event MOUSEEVENTF_ABSOLUTE Or MOUSEEVENTF_MOVE, _
         PixelXToMickey(xPixel), PixelYToMickey(yPixel), cbuttons, dwExtraInfo
     End Sub

Author: dwirch

Derek Wirch is a seasoned IT professional with an impressive career dating back to 1986. He brings a wealth of knowledge and hands-on experience that is invaluable to those embarking on their journey in the tech industry.

Leave a Reply

Your email address will not be published. Required fields are marked *

This site uses Akismet to reduce spam. Learn how your comment data is processed.