转自:vigoss331
首先,WM_MOUSELEAVE是鼠标离开窗口时发出的消息,但是这个消息与普通消息不同,要收到WM_MOUSELEAVE消息必须先调用 TrackMouseEvent 函数,并且每调用一次TrackMouseEvent窗口只能收到一次WM_MOUSELEAVE,也就是说如果要获得WM_MOUSELEAVE消息的话,当鼠标重新进入窗口时必须调用一次TrackMouseEvent。
函数原型:
BOOL TrackMouseEvent(LPTRACKMOUSEEVENT lpEventTrack);
typedef struct tagTRACKMOUSEEVENT {
DWORD cbSize;
DWORD dwFlags;
HWND hwndTrack;
DWORD dwHoverTimer;
}TRACKMOUSEEVENT, *LPTRACKMOUSEEVENT;
lpEventTrack 指向 TRACKMOUSEEVENT 结构体的指针
说明:在鼠标离开某一窗口或在某一窗口上停留超过某一特定时间长度时发送消息。
cbSize: 定义TRACKMOUSEEVENT结构体的大小;
dwFlags: 定义服务请求,可以是下列值的组合
hwndTrack: 待跟踪窗口的句柄
dwHoverTime:定义hover事件的耗尽时间,单位毫秒。可以使用HOVER_DEFAULT来使用系统默认的hover事件耗尽时间。
TME_CANCEL 取消前一次的跟踪请求,THE_CANCEL|THE_HOVER
TME_HOVER 悬停通知,发送WM_MOUSEHOVER消息
TME_LEAVE 离开通知,发送WM_MOUSELEAVE消息
TME_QUERY 这一项不是作为跟踪请求的。选中这一项时,当结构体被传送给TrackMouseEvent函数时,即产生当前跟踪。唯一不同的是返回的消耗时间,是真实的消耗时间而不是HOVER_DEFAULT,即使之前TrackMouseEvent函数所请求的HOVER_DEFAULT。
具体使用步骤:
- 设置一个状态量 m_bTracked 用于记录和标识鼠标是否在窗口内,初始化为 FALSE
- 添加 OnMouseMove() 、OnMouseLeave() 两个消息函数
- 在编辑函数如下:
void CWnd::OnMouseMove(UINTnFlags, CPoint point)
{
if (!m_bTracked)
{
TRACKMOUSEEVENTtme;
tme.cbSize = sizeof(tme);
tme.dwFlags = TME_LEAVE;
tme.dwHoverTime = 0;
tme.hwndTrack = m_hWnd;
TrackMouseEvent(&tme); // 调用函数
m_bTracked= TRUE; // 标记鼠标进入窗口
Invalidate(FALSE);
}
CBitmapButton::OnMouseMove(nFlags,point);
}
3.接收到WM_MOUSELEAVE,重置状态变量
void CWnd::OnMouseLeave() // 该函数的触发由 TrackMouseEvent 函数发出消息
{
m_bTracked = FALSE; // 标记鼠标离开窗口
Invalidate(FALSE);
CBitmapButton::OnMouseLeave();
}