vbAccelerator - Contents of code file: cMouseTrack.cls
VERSION 1.0 CLASS
BEGIN
MultiUse = -1 'True
END
Attribute VB_Name = "cMouseTrack"
Attribute VB_GlobalNameSpace = False
Attribute VB_Creatable = True
Attribute VB_PredeclaredId = False
Attribute VB_Exposed = False
Option Explicit
'
===============================================================================
=======
' Name: cMouseTrack.cls
' Author: Steve McMahon (steve@vbaccelerator.com)
' Date: 20 October 1999
'
' Requires: SSUBTMR.DLL
'
' Copyright 1999 Steve McMahon for vbAccelerator
'
-------------------------------------------------------------------------------
-------
' Visit vbAccelerator - advanced free source code for VB programmers
' http://vbaccelerator.com
'
-------------------------------------------------------------------------------
-------
'
' Implements a mouse leave event for any object with
' a hWnd.
'
' FREE SOURCE CODE - ENJOY!
' Do not sell this code. Credit vbAccelerator.
'
===============================================================================
=======
' 1. USER32 method:
Private Const WM_MOUSEHOVER = &H2A1&
Private Const WM_MOUSELEAVE = &H2A3&
Private Const TME_HOVER = &H1&
Private Const TME_LEAVE = &H2&
Private Const TME_QUERY = &H40000000
Private Const TME_CANCEL = &H80000000
Private Const HOVER_DEFAULT = &HFFFFFFFF
Private Type tagTRACKMOUSEEVENT
cbSize As Long
dwFlags As Long
hwndTrack As Long
dwHoverTime As Long
End Type
Private Declare Function TrackMouseEvent Lib "user32" _
(lpEventTrack As tagTRACKMOUSEEVENT) As Long
' 2. The COMCTL32.DLL Method:
'// Declare _TrackMouseEvent. This API tries to use the window manager's
'// implementation of TrackMouseEvent if it is present, otherwise it emulates.
Private Declare Function CCTrackMouseEvent Lib "COMCTL32.DLL" Alias
"_TrackMouseEvent" _
(lpEventTrack As tagTRACKMOUSEEVENT) As Long
Private Const MK_LBUTTON = &H1&
Private Const MK_RBUTTON = &H2&
Private Const MK_SHIFT = &H4&
Private Const MK_CONTROL = &H8&
Private Const MK_MBUTTON = &H10&
' 3 If ALL else fails, then use the work-around:
Private Declare Function SetCapture Lib "user32" (ByVal hwnd As Long) As Long
Private Declare Function ReleaseCapture Lib "user32" () As Long
Private Type RECT
Left As Long
Top As Long
Right As Long
Bottom As Long
End Type
Private Type POINTAPI
X As Long
Y As Long
End Type
Private Declare Function ClientToScreen Lib "user32" (ByVal hwnd As Long,
lpPoint As POINTAPI) As Long
Private Declare Function WindowFromPoint Lib "user32" (ByVal xPoint As Long,
ByVal yPoint As Long) As Long
Private Declare Function GetClientRect Lib "user32" (ByVal hwnd As Long, lpRect
As RECT) As Long
Private Declare Function PtInRect Lib "user32" (lpRect As RECT, ByVal X As
Long, ByVal Y As Long) As Long
Private Declare Function GetAsyncKeyState Lib "user32" (ByVal vKey As Long) As
Integer
Private Const WM_MOUSEMOVE = &H200
Private Const WM_MOUSEUP = &H200
Private Const WM_ACTIVATE = &H6
Private Const WM_LBUTTONUP = &H202
Private Const WM_MBUTTONUP = &H208
Private Const WM_RBUTTONUP = &H205
' Version detection:
' For OS:
Private Type OSVERSIONINFO
dwOSVersionInfoSize As Long
dwMajorVersion As Long
dwMinorVersion As Long
dwBuildNumber As Long
dwPlatformID As Long
szCSDVersion As String * 128 ' Maintenance string for PSS usage
End Type
Private Declare Function GetVersionEx Lib "kernel32" Alias "GetVersionExA"
(lpVersionInformation As OSVERSIONINFO) As Long
Private Const VER_PLATFORM_WIN32_NT = 2
Private Const VER_PLATFORM_WIN32_WINDOWS = 1
Private Const VER_PLATFORM_WIN32s = 0
' For COMCTL32.DLL
Private Const S_OK = &H0
Private Type DLLVERSIONINFO
cbSize As Long
dwMajor As Long
dwMinor As Long
dwBuildNumber As Long
dwPlatformID As Long
End Type
Private Declare Function LoadLibrary Lib "kernel32" Alias "LoadLibraryA" (ByVal
lpLibFileName As String) As Long
Private Declare Function FreeLibrary Lib "kernel32" (ByVal hLibModule As Long)
As Long
Private Declare Function GetProcAddress Lib "kernel32" (ByVal hModule As Long,
ByVal lpProcName As String) As Long
Private Declare Function DllGetVersion Lib "comctl32" (pdvi As DLLVERSIONINFO)
As Long
' Implementation:
Implements ISubclass
Public Event MouseHover(Button As MouseButtonConstants, Shift As
ShiftConstants, X As Single, Y As Single)
Public Event MouseLeave()
Private m_bTracking As Boolean
Private m_hWnd As Long
Private m_hWndParent As Long
Private m_bUseCC As Boolean
Private m_bUseCapture As Boolean
Public Enum EMouseTrackMethods
eMouseTrackDetect = -1
eMouseTrackUser32 = 0
eMouseTrackComCtl32 = 1
eMouseTrackWorkAround = 2
End Enum
Private m_eMethod As EMouseTrackMethods
Private Sub pDetectMethod()
Dim tVI As OSVERSIONINFO
' Default to use COMCTL32.DLL. (Requires IE4.0 or higher installed).
m_bUseCC = True
' Now we check for a window manager (user32.dll) implementation of
' TrackMouseEvent. We can rely on COMCTL32.DLL's version to use
' the window manager's version directly, except IE4 may not be installed.
tVI.dwOSVersionInfoSize = Len(tVI)
If (GetVersionEx(tVI) <> 0) Then
' NT4 or higher supports TrackMouseEvent in User32:
If (tVI.dwPlatformID = VER_PLATFORM_WIN32_NT) And (tVI.dwMajorVersion >
3) Then
' Using NT
m_bUseCC = False
' Win98 or higher supports TrackMouseEvent in User32:
ElseIf (tVI.dwMajorVersion >= 5) Then
' Using 98
m_bUseCC = False
End If
End If
If (m_bUseCC) Then
Dim hMod As Long
Dim lR As Long
Dim lptrDLLVersion As Long
Dim tDVI As DLLVERSIONINFO
Dim bCC As Boolean
hMod = LoadLibrary("comctl32.dll")
If (hMod <> 0) Then
lR = S_OK
'/*
' You must get this function explicitly because earlier versions of
the DLL
' don't implement this function. That makes the lack of implementation
of the
' function a version marker in itself. */
lptrDLLVersion = GetProcAddress(hMod, "DllGetVersion")
If (lptrDLLVersion <> 0) Then
tDVI.cbSize = Len(tDVI)
lR = DllGetVersion(tDVI)
If (lR = S_OK) Then
If (tDVI.dwMajor > 4) Then
bCC = True
ElseIf (tDVI.dwMajor = 4) And (tDVI.dwMinor > 70) Then
bCC = True
End If
End If
End If
FreeLibrary hMod
End If
If Not (bCC) Then
m_bUseCC = False
m_bUseCapture = True
End If
End If
If (m_bUseCC) Then
m_eMethod = eMouseTrackComCtl32
ElseIf (m_bUseCapture) Then
m_eMethod = eMouseTrackWorkAround
Else
m_eMethod = eMouseTrackUser32
End If
End Sub
Public Property Get Method() As EMouseTrackMethods
Method = m_eMethod
End Property
Public Sub AttachMouseTracking( _
objTo As Object, _
Optional ByVal eForceMethod As EMouseTrackMethods = eMouseTrackDetect _
)
m_bUseCapture = False
m_bUseCC = False
' Check for tracking type if not forced:
If (eForceMethod = eMouseTrackDetect) Then
pDetectMethod
Else
Select Case eForceMethod
Case eMouseTrackWorkAround
m_bUseCapture = True
Case eMouseTrackComCtl32
m_bUseCC = True
End Select
m_eMethod = eForceMethod
End If
' Start subclassing for WM_MOUSEHOVER and WM_MOUSELEAVE
' messages:
DetachMouseTracking
m_hWnd = objTo.hwnd
If (m_hWnd <> 0) Then
If (m_bUseCapture) Then
AttachMessage Me, m_hWnd, WM_MOUSEMOVE
AttachMessage Me, m_hWnd, WM_LBUTTONUP
AttachMessage Me, m_hWnd, WM_MBUTTONUP
AttachMessage Me, m_hWnd, WM_RBUTTONUP
m_hWndParent = objTo.Parent.hwnd
AttachMessage Me, m_hWndParent, WM_ACTIVATE
Else
AttachMessage Me, m_hWnd, WM_MOUSEHOVER
AttachMessage Me, m_hWnd, WM_MOUSELEAVE
End If
End If
End Sub
Public Sub StartMouseTracking()
Dim tET As tagTRACKMOUSEEVENT
Dim lR As Long
On Error GoTo ErrorHandler
' Tells Windows to start tracking the mouse over the specified
' hWnd:
If Not (m_bTracking) Then
' Tracking will stop whenever a WM_MOUSEHOVER or WM_MOUSELEAVE
' event occurs.
tET.cbSize = Len(tET)
tET.dwFlags = TME_HOVER Or TME_LEAVE
tET.dwHoverTime = HOVER_DEFAULT
tET.hwndTrack = m_hWnd
If (m_bUseCC) Then
lR = CCTrackMouseEvent(tET)
ElseIf (m_bUseCapture) Then
SetCapture m_hWnd
Else
lR = TrackMouseEvent(tET)
End If
m_bTracking = True
End If
Exit Sub
ErrorHandler:
' This occurs because the user has forced a method
' which is not supported. Raise error!
Err.Raise Err.Number, App.EXEName & ".cMouseTrack", Err.Description
' But don't allow this to get set...
m_bTracking = False
End Sub
Public Sub DetachMouseTracking()
' Stops subclassing for mouse tracking commands.
' Called automatically when the class terminates.
If (m_hWnd <> 0) Then
If (m_bUseCapture) Then
ReleaseCapture
DetachMessage Me, m_hWnd, WM_MOUSEMOVE
DetachMessage Me, m_hWnd, WM_LBUTTONUP
DetachMessage Me, m_hWnd, WM_MBUTTONUP
DetachMessage Me, m_hWnd, WM_RBUTTONUP
If (m_hWndParent <> 0) Then
DetachMessage Me, m_hWndParent, WM_ACTIVATE
End If
Else
DetachMessage Me, m_hWnd, WM_MOUSEHOVER
DetachMessage Me, m_hWnd, WM_MOUSELEAVE
End If
m_hWnd = 0
End If
End Sub
Public Property Get Tracking() As Boolean
' Returns whether windows is tracking or not (it stops
' everyime a WM_MOUSEHOVER or WM_MOUSELEAVE event is fired):
Tracking = m_bTracking
End Property
Private Sub Class_Initialize()
'
End Sub
Private Sub Class_Terminate()
' Clear up subclass:
DetachMouseTracking
End Sub
Private Property Let ISubclass_MsgResponse(ByVal RHS As SSubTimer.EMsgResponse)
'
End Property
Private Property Get ISubclass_MsgResponse() As SSubTimer.EMsgResponse
' Let Windows pre-process message:
ISubclass_MsgResponse = emrPreprocess
End Property
Private Function ISubclass_WindowProc(ByVal hwnd As Long, ByVal iMsg As Long,
ByVal wParam As Long, ByVal lParam As Long) As Long
Dim X As Single, Y As Single
' Respond to WM_MOUSEHOVER and WM_MOUSELEAVE messages:
Select Case iMsg
' ===============================================
' To effect the TrackMouseEvent User32 or
' Comctl32 methods:
Case WM_MOUSEHOVER
Dim Button As MouseButtonConstants
Dim Shift As ShiftConstants
m_bTracking = False
If (wParam And MK_LBUTTON) = MK_LBUTTON Then
Button = Button Or vbLeftButton
End If
If (wParam And MK_RBUTTON) = MK_RBUTTON Then
Button = Button Or vbRightButton
End If
If (wParam And MK_MBUTTON) = MK_MBUTTON Then
Button = Button Or vbMiddleButton
End If
If (wParam And MK_CONTROL) = MK_CONTROL Then
Shift = Shift Or vbCtrlMask
End If
If (wParam And MK_SHIFT) = MK_SHIFT Then
Shift = Shift Or vbShiftMask
End If
X = lParam And &HFFFF&
Y = lParam \ &H10000
RaiseEvent MouseHover(Button, Shift, X, Y)
Case WM_MOUSELEAVE
m_bTracking = False
RaiseEvent MouseLeave
' ===============================================
' ===============================================
' To effect the SetCapture/ReleaseCapture method:
Case WM_MOUSEMOVE, WM_LBUTTONUP, WM_RBUTTONUP, WM_MBUTTONUP
Dim tR As RECT, tP As POINTAPI
GetClientRect m_hWnd, tR
X = lParam And &HFFFF&
Y = lParam \ &H10000
tP.X = X
tP.Y = Y
ClientToScreen m_hWnd, tP
If (PtInRect(tR, X, Y) = 0) Or (WindowFromPoint(tP.X, tP.Y) <> m_hWnd)
Then
If (GetAsyncKeyState(vbKeyLButton) = 0) And
(GetAsyncKeyState(vbKeyMButton) = 0) And
(GetAsyncKeyState(vbKeyRButton) = 0) Then
m_bTracking = False
ReleaseCapture
RaiseEvent MouseLeave
End If
ElseIf (iMsg <> WM_MOUSEMOVE) Then
m_bTracking = False
StartMouseTracking
End If
Case WM_ACTIVATE
If (m_bTracking) Then
m_bTracking = False
ReleaseCapture
RaiseEvent MouseLeave
End If
End Select
' ===============================================
End Function
|
|