---- C++ Builer的VCL提供了對大多數Windows消息的處理機制,這對於一般應用程序是足夠了,但VCL也不是無所不包的,對於那些VCL沒有處理的Windows消息,在需要時如何進行捕獲呢?C++ Builder采用了消息映像表機制,通過消息映像表將特定的Windows消息與代碼中的函數聯系起來,當窗口捕獲到消息時就會調用這個函數,這其實和事件句柄非常相似。
---- C++ Builder消息映像表定義形式如下:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
……
MESSAGE_HANDLER(< message >,
< message structure >,< message handler >)
END_MESSAGE_MAP(ClassName)
---- ClassName是基礎類名,message參數是要捕獲的Windows消息,message structure參數用於傳遞VCL剖析消息後用於傳遞參數的消息剖析結構名,裡面包含處理消息時所需要的全部參數,不同的消息其消息剖析結構是不同的,一般性的消息剖析結構定義如下:
struct Tmessage
{
unsigned int Msg;//Windows消息
long Wparam;
long Lparam;
long Result;
}
---- 這裡關鍵是Result數據成員,它用於設置處理消息後的返回值,程序可以根據不同的返回值進行相應的處理。message handler參數是對消息進行處理的函數名稱。
---- 俗話說:千言不如一圖。下面就用一個例子具體說明一下,讓我們來拖動一個無標題的窗口,實現的原理是通過捕獲WM_NCHITTEST消息,在鼠標單擊窗體的客戶區時發送消息,讓Windows認為是個窗體的標題欄上。
---- 創建一個新工程,將單元文件保存為mainform.cpp,工程文件保存為nocaption.bpr。首先得創建一個無標題欄的窗體,將Form1的BorderIcons集合設為空,這就刪除了窗口的系統菜單、最大化和最小化按鈕,將BorderStyle屬性設為bsDialog。注意在設計時創建的所有C++ Builder窗體都會有一個標題欄,你並不能象VB中那樣簡單地通過將Form1的Caption屬性設為空來得到一個無標題欄窗體,你必須在TForm的派生類中重載CreateParams函數,去除相應的窗口標識位,在運行時刪去窗體的標題欄。打開mainform.h文件,在private段增加CreateParams的原型:
void __fastcall CreateParams(TCreateParams& Params);
TCreateParams結構包含了創建窗體的各種參數,
在這裡我們要屏蔽掉WS_CAPTION位。
在mainform.cpp中增加如下代碼:
void __fastcall TForm1::CreateParams(TCreateParams& Params)
{
TForm::CreateParams(Params);//先調用基礎類中的成員函數
Params.Style &=~WS_CAPTION;
}
---- 下面創建消息映像表,對WM_NCHITTEST消息進行捕獲,在mainform.h的public段輸入代碼:
BEGIN_MESSAGE_MAP
MESSAGE_HANDLER
(WM_NCHITTEST,Tmessage,WMNchitTest)
END_MESSAGE_MAP(TForm)
---- WM_NCHITTEST消息名稱為"非客戶區命中測試",它優先於所有其它的客戶區和非客戶區鼠標消息,Windows應用程序通常把這個消息傳送給DefWindowProc,然後Windows用這個消息產生基於鼠標位置的所有其它鼠標消息。
---- 在private段加入消息處理函數的聲明:
void __fastcall WMNchitTest(Tmessage &message);
---- 在mainform.cpp中輸入此函數的定義:
void __fastcall TForm1::WMNchitTest(Tmessage &message)
{
if(GetAsyncKeyState(VK_LBUTTON)< 0)
---- message.Result=HTCAPTION; //如果鼠標左鍵按下,就通知Windows鼠標所在區域是標題欄區,Windows就會按要求完成拖動操作。
else
message.Result=HTCLIENT;
}
---- 這裡用Windows API函數GetAsyncKeyState來實時檢查鼠標按鈕的狀態,通過檢測虛擬鍵碼VK_LBUTTON的高位是否被置位來探測鼠標動作,如果按下鼠標左鍵,此鍵碼的高位被置1,此時GetAsyncKeyState函數將返回一個負數。
---- 按F9編譯並運行程序,單擊窗體任一位置就可以象單擊在標題欄上一樣拖動窗體了,這裡沒有給出結束程序運行的機制,可以通過增加PopMenu來實現。
---- 以上代碼在C++ Builder3、Pwin98環境下編譯、運行通過。