MFC創建一個可通過鼠標點擊客戶區拖動窗口位置的方法

通過添加消息響應函數
afx_msg UINT OnNcHitTest(CPoint point);
添加宏
ON_WM_NCHITTEST()

然后, 定義OnNcHitTest的行為即可, OnNcHitTest的代碼如下:
UINT CTimerDlg::OnNcHitTest(CPoint point)
{
    CRect rect;
    GetClientRect(rect);
    ClientToScreen(rect);

    
if(rect.PtInRect(point))
        
return HTCAPTION;
    
return CDialog::OnNcHitTest(point);
}

首先, 發生WM_NCHITTEST消息的時候, 程序先取得客戶區的窗口矩形, 這里用到GetClientRect方法得到, 由于GetClientRect取得的是相對于窗口坐標系統, 但OnNcHitTest的參數point卻是相對于屏幕的坐標系統, 所以這里用到了ClientToScreen把取到的客戶區矩形轉成相對于屏幕的坐標, 然后用CRect類的PtInRect方法判斷point是否落在這個矩形里面, 如果是, 則返回HTCAPTION(就相當于這次點擊是點擊標題欄), 所以, 就可以這樣來拖動窗口了. 如果, OnNcHitTest直接返回HTCAPTION的話, 那么, 但用戶點擊非客戶區的時候, 有些功能會失效, 比如點擊系統的關閉按鈕, 試一下就會發現這個按鈕無效了; ok, 如果點擊的是非客戶端, 者直接返回CDialog::OnNcHitTest(point), 這就是按默認處理了.

以上的這個方法比較麻煩, 其實可以更簡單一點, 只要先調用父類的OnNcHitTest(point), 取得返回值, 在判斷是不是HTCLIENT, 如果是, 則返回HTCAPTION, 否則直接返回父類OnNcHitTest(point)的返回值就ok了, 修改后如下:
UINT CTimerDlg::OnNcHitTest(CPoint point)
{
    UINT uRet 
= CDialog::OnNcHitTest(point);
    
if(HTCLIENT == uRet)
        
return HTCAPTION;
    
return uRet;
}

再簡化一下, 就成這樣子:
UINT CTimerDlg::OnNcHitTest(CPoint point)
{
    UINT uRet 
= CDialog::OnNcHitTest(point);
    
return (HTCLIENT == uRet) ? HTCAPTION : uRet;
}