關於WM_NCHITTEST消息

以下爲轉載內容:

 

我爲了移動一個無標題欄的窗體,使用了WM_NCHITTEST消息,這個消息大概如下:

通常,我們拖動對話框窗口的標題欄來移動窗口,但有時候,我們想通過鼠標在客戶區上拖動來移動窗口。

一個容易想到的方案是,處理鼠標消息WM_LBUTTONDOWN和WM_LBUTTONUP。在OnLButtonUp函數中計算鼠標位置的變化,調用MoveWindow實現窗口的移動。

注意,拖動標題欄移動窗口的時候,會出現一個矩形框,它提示了窗口移動的當前位置。當鼠標左鍵放開的時候,窗口就移動到矩形框所在位置。而我們的實現方案中沒有這個功能。

要實現此功能,我們必須自己來畫這些矩形。

事實上,我們沒有必要自己來做這件事情,因爲Windows已經給我們做好了。

試想,如果我能夠欺騙Windows,告訴它現在鼠標正在拖動的是標題欄而不是客戶區,那麼窗口移動操作就由Windows來代勞了。

要欺騙Windows並不像想像中的困難,甚至非常簡單。

我們利用一個消息:WM_NCHITTEST。

MSDN對它的解釋是:

The WM_NCHITTEST message is sent to a window when the cursor moves, or when a mouse button is pressed or released. If the mouse is not captured, the message is sent to the window beneath the cursor. Otherwise, the message is sent to the window that has captured the mouse.

這個消息是當鼠標移動或者有鼠標鍵按下時候發出的。

Windows用這個消息來做什麼? “HITTEST”就是“命中測試”的意思,WM_NCHITTEST消息用來獲取鼠標當前命中的位置。

WM_NCHITTEST的消息響應函數會根據鼠標當前的座標來判斷鼠標命中了窗口的哪個部位,消息響應函數的返回值指出了部位,例如它可能會返回HTCAPTION,或者HTCLIENT等。(其返回值有很多,請查閱MSDN)。

爲了便於理解,我先描述一下Windows對鼠標鍵按下的響應流程:

1. 確定鼠標鍵點擊的是哪個窗口。Windows會用表記錄當前屏幕上各個窗口的區域座標,當鼠標驅動程序通知Windows鼠標鍵按下了,Windows根據鼠標的座標確定它點擊的是哪個窗口。

2. 確定鼠標鍵點擊的是窗口的哪個部位。Windows會向鼠標鍵點擊的窗口發送WM_NCHITTEST消息,來詢問鼠標鍵點擊的是窗口的哪個部位。(WM_NCHITTEST的消息響應函數的返回值會通知Windows)。通常來說,WM_NCHITTEST消息是系統來處理的,用戶一般不會主動去處理它(也就是說,WM_NCHITTEST的消息響應函數通常採用的是Windows默認的處理函數)。

3. 根據鼠標鍵點擊的部位給窗口發送相應的消息。例如:如果WM_NCHITTEST的消息響應函數的返回值是HTCLIENT,表示鼠標點擊的是客戶區,則Windows會向窗口發送WM_LBUTTONDOWN消息;如果WM_NCHITTEST的消息響應函數的返回值不是HTCLIENT(可能是HTCAPTION、HTCLOSE、HTMAXBUTTON等),即鼠標點擊的是非客戶區,Windows就會向窗口發送WM_NCLBUTTONDOWN消息。

我們有必要詳細討論一下:如果WM_NCHITTEST的消息響應函數的返回值是HTCAPTION,即指示了鼠標點擊了標題欄,接下去Windows的處理是怎樣的?

上面已經提到,接下來,Windows會向窗口發送WM_NCLBUTTONDOWN消息。

MSDN對WM_NCLBUTTONDOWN消息描述如下:

WM_NCLBUTTONDOWN

nHittest = (INT) wParam; // hit-test value

pts = MAKEPOINTS(lParam); // position of cursor

WM_NCLBUTTONDOWN的wParam指示了鼠標點擊的窗口部位,lParam指示了當前鼠標的座標。

如果應用程序沒有對該消息響應,則由系統默認處理。

系統默認處理又是怎樣的呢?系統發現wParam指示了鼠標點擊的是標題欄,就會標識當前窗口處於“拖拽狀態”(Windows內部記錄了每個窗口的狀態信息)。由於標識了“拖拽狀態”,則從此刻起到鼠標鍵放開之前,你的鼠標移動狀況完全由Windows跟蹤。它根據鼠標的移動,使得窗口作“同步”移動。

注意,這個過程中,窗口不會收到WM_NCMOUSEMOVE消息,因爲窗口和鼠標是“同步”移動的,你的鼠標相對於窗口是靜止的。

 

但問題同時也出現了, 我想在右鍵這個窗體的時候彈出一個菜單, 當我完成 MSG_WM_RBUTTONDOWN 這個消息的時候,發現窗體收不到這個消息, 將WM_NCHITTEST消息的實現去掉就可以了,看了一原因是:

因爲你在WM_NCHITTEST中處理了鼠標消息,把他定位成HTCAPTION,也就是鼠標在標題欄上,而標題欄屬於非客戶區(NC);
非客戶區的事件消息都是以WM_NC開頭的。也就是說,當你的WM_NCHITTEST返回HTCAPTION時,原來可以用WM_LBUTTONUP處理的消息,你只能用WM_NCLBUTTONUP來處理。

 

解決方法:

同時處理WM_NCHITTEST和WM_NCRBUTTONUP,而不處理WM_RBUTTONUP

 

 

 

原文地址:http://www.cnblogs.com/GnagWang/archive/2010/09/12/1824394.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章