http://www.cnblogs.com/GnagWang/archive/2010/09/12/1824394.htmlhtml
我爲了移動一個無標題欄的窗體,使用了WM_NCHITTEST消息,這個消息大概以下:函數
一般,咱們拖動對話框窗口的標題欄來移動窗口,但有時候,咱們想經過鼠標在客戶區上拖動來移動窗口。post
一個容易想到的方案是,處理鼠標消息WM_LBUTTONDOWN和WM_LBUTTONUP。在OnLButtonUp函數中計算鼠標位置的變化,調用MoveWindow實現窗口的移動。測試
注意,拖動標題欄移動窗口的時候,會出現一個矩形框,它提示了窗口移動的當前位置。當鼠標左鍵放開的時候,窗口就移動到矩形框所在位置。而咱們的實現方案中沒有這個功能。htm
要實現此功能,咱們必須本身來畫這些矩形。blog
事實上,咱們沒有必要本身來作這件事情,由於Windows已經給咱們作好了。事件
試想,若是我可以欺騙Windows,告訴它如今鼠標正在拖動的是標題欄而不是客戶區,那麼窗口移動操做就由Windows來代勞了。get
要欺騙Windows並不像想像中的困難,甚至很是簡單。同步
咱們利用一個消息:WM_NCHITTEST。it
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