MFC使用TRACKMOUSEEVENT觸發mouseHover和mouseLeave

來源:互聯網
上載者:User

為對話方塊添加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

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.