為對話方塊添加WM_MOUSEHOVER或WM_MOUSELEAVE訊息並不會響應。
MFC需要特殊處理,其中一法就是使用TRACKMOUSEEVENT
void CmfcDlgDlg::OnMouseMove(UINT nFlags, CPoint point){ // TODO: 在此添加訊息處理常式代碼和/或調用預設值 //GetDlgItem(IDC_BUTTON)->SetWindowText("Move"); if (!m_bTracking) { TRACKMOUSEEVENT tme; tme.cbSize = sizeof(TRACKMOUSEEVENT); tme.dwFlags = TME_LEAVE | TME_HOVER;//要觸發的訊息 tme.hwndTrack = this->m_hWnd; tme.dwHoverTime = 10;// 若不設此參數,則無法觸發mouseHover if (::_TrackMouseEvent(&tme)) //MOUSELEAVE|MOUSEHOVER訊息由此函數觸發 { m_bTracking = true; } } CDialogEx::OnMouseMove(nFlags, point);}void CmfcDlgDlg::OnMouseHover(UINT nFlags, CPoint point){ // TODO: 在此添加訊息處理常式代碼和/或調用預設值 GetDlgItem(IDC_BUTTON)->SetWindowText("Hover"); m_bTracking = false; CDialogEx::OnMouseHover(nFlags, point);}void CmfcDlgDlg::OnMouseLeave(){ // TODO: 在此添加訊息處理常式代碼和/或調用預設值 GetDlgItem(IDC_BUTTON)->SetWindowText("Leave"); m_bTracking = false; CDialogEx::OnMouseLeave();}
本來打算自繪CButton,時間不足,總結一下這個知識點。
隨後研究一下MFC的自繪,可以製作非常牛的介面。
url:http://greatverve.cnblogs.com/archive/2013/02/06/TRACKMOUSEEVENT.html
預設情況下,視窗是不響應 WM_MOUSELEAVE 和 WM_MOUSEHOVER 訊息的,所以要使用 _TrackMouseEvent 函數來啟用這兩個訊息。調用這個函數後,當滑鼠在指定視窗上停留超過一定時間或離開視窗後,該函數會 Post 這兩個訊息到指定視窗。
使用方法:
1. 在對話方塊類中定義一個變數來標識是否追蹤當前滑鼠狀態,之所以要這樣定義是要避免滑鼠已經在表單之上時,一移動滑鼠就不斷重複產生 WM_MOUSEHOVER 。
BOOL _bMouseTrack=TRUE ;
2. 在 OnMouseMove 中調用 _TrackMouseEvent 函數
if (_bMouseTrack) // 若允許 追蹤,則。
{
TRACKMOUSEEVENT csTME;
csTME.cbSize = sizeof (csTME);
csTME.dwFlags = TME_LEAVE|TME_HOVER;
csTME.hwndTrack = m_hWnd ;// 指定要 追蹤 的視窗
csTME.dwHoverTime = 10; // 滑鼠在按鈕上停留超過 10ms ,才認為狀態為 HOVER
::_TrackMouseEvent (&csTME); // 開啟 Windows 的 WM_MOUSELEAVE , WM_MOUSEHOVER 事件支援
_bMouseTrack=FALSE ; // 若已經 追蹤 ,則停止 追蹤
}
(#add 摘自MSDN:The _TrackMouseEvent function posts messages when the mouse pointer leaves a window or hovers over a window for a specified amount of time. This function calls TrackMouseEvent if it exists, otherwise it emulates it.)
3. 在 OnMouseLeave 中再次允許追蹤滑鼠狀態
_bMouseTrack=TRUE ;
4. 備忘:這兩個訊息的映射要自己寫
ON_MESSAGE(WM_MOUSEHOVER,OnMouseHover)
ON_MESSAGE(WM_MOUSELEAVE,OnMouseLeave)
注意:也可以用下面代碼在PreTranslateMessage函數中接收,不需要自己寫WM_MOUSELEASE和WM_MOUSEHOVER訊息的響應函數(當然你要自己寫也行):
if(pMsg-> message==WM_MOUSELEAVE)
AfxMessageBox( "mouse leave ");
else if(pMsg->message == WM_MOUSEHOVER)
AfxMessageBos("mouse leave");
return CDialog::PreTranslateMessage(pMsg);
附一個例子:
.h檔案加入:
afx_msg LRESULT OnMouseLeave(WPARAM ,LPARAM );
afx_msg LRESULT OnMouseHover(WPARAM ,LPARAM );
.cpp檔案加入:
ON_MESSAGE(WM_MOUSELEAVE,OnMouseLeave)
ON_MESSAGE(WM_MOUSEHOVER,OnMouseHover)
LRESULT CIconButton::OnMouseLeave(WPARAM wParam,LPARAM lParam)
{
InvalidateRect(NULL);
return 0;
}
LRESULT CIconButton::OnMouseHover(WPARAM wParam,LPARAM lParam)
{
//擷取滑鼠座標
POINT point;
::GetCursorPos(&point);
ScreenToClient(&point) ;
//亦用如下方法,推薦.OnMouseMove 也可用.
CPoint pt;
pt.x = LOWORD(lParam); // horizontal position of cursor
pt.y = HIWORD(lParam); // vertical position of cursor
//還有如下,和上邊其實是一樣
POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) };
CClientDC dc(this);
CRect rt;
GetClientRect(&rt);
dc.Draw3dRect(0,0,
m_rect.right-m_rect.left,m_rect.bottom-m_rect.top,
RGB(0,0,0),RGB(10,10,10));
return 0;
}
void CIconButton::OnMouseMove(UINT nFlags, CPoint point)
{
// TODO: Add your message handler code here and/or call default
CButton::OnMouseMove(nFlags, point);
TRACKMOUSEEVENT tme;
tme.cbSize=sizeof(TRACKMOUSEEVENT);
tme.dwFlags=TME_HOVER | TME_LEAVE;
tme.dwHoverTime=HOVER_DEFAULT;
tme.hwndTrack=m_hWnd;
_TrackMouseEvent(&tme);
}
可用如下自訂宏:
#ifndef ON_WM_MOUSELEAVE
#define ON_WM_MOUSELEAVE() \
{ WM_MOUSELEAVE, 0, 0, 0, AfxSig_vv, \
(AFX_PMSG)(AFX_PMSGW) \
(static_cast< void (AFX_MSG_CALL CWnd::*)(void) > ( /*&ThisClass :: */OnMouseLeave)) },
#endif
#ifndef ON_WM_MOUSEHOVER
#define ON_WM_MOUSEHOVER() \
{ WM_MOUSEHOVER, 0, 0, 0, AfxSig_vwp, \
(AFX_PMSG)(AFX_PMSGW)(void (AFX_MSG_CALL CWnd::*)(UINT, CPoint))&OnMouseHover },
#endif