VC定制個性化的MessageBox解決方法
相信學(xué)過VC的人都知道MessageBox()函數(shù)的用法:
int MessageBox( HWND hWnd, // handle to owner window LPCTSTR lpText, // text in message box LPCTSTR lpCaption, // message box title UINT uType // message box style );
雖然在參數(shù)uType中可以指定一些樣式,但你在程序中能夠?qū)essageBox的外觀所做的定義卻不多。原因是當調(diào)用MessageBox()函數(shù)后,它在內(nèi)部有自己的消息循環(huán)(所有的模式對話框都有自己的消息循環(huán)),返回時MessageBox對話框窗口已經(jīng)被Destroy,所以你沒有辦法得到MessageBox對話框的窗口句柄。但你可以根據(jù)自己的不同需求用下面的方法中去定制你的MessageBox:
如果你只是想用自己的icon去代替系統(tǒng)MessageBox提供的那幾種有限的icon,用MessageBoxIndirect()函數(shù)就可以了:
int MessageBoxIndirect( CONST LPMSGBOXPARAMS lpMsgBoxParams // message box parameters ); typedef struct { UINT cbSize; HWND hwndOwner; HINSTANCE hInstance; LPCTSTR lpszText; LPCTSTR lpszCaption; DWORD dwStyle; LPCTSTR lpszIcon; DWORD_PTR dwContextHelpId; MSGBOXCALLBACK lpfnMsgBoxCallback; DWORD dwLanguageId; } MSGBOXPARAMS, *PMSGBOXPARAMS;
看到MSGBOXPARAMS結(jié)構(gòu)中的lpszIcon吧,在使用過程中你應(yīng)當準備一個合適的MSGBOXPARAMS結(jié)構(gòu),如果你要用自己的icon,你一定要用MB_USERICON這個flag。
上面我們也講到不能定制MessageBox對話框的原因是沒有辦法得到它的窗口句柄,但我們真的沒有辦法了嗎?當然有辦法,那就是使用HOOK(鉤子)機制。在windows系統(tǒng)中有多種HOOK,但在這里我們只用到HK_CBT類型的鉤子。HK_CBT鉤子過程在WM_MOVE、WM_SIZE、WM_ACTIVE、WM_CREATE、WM_DESTROY時被系統(tǒng)調(diào)用,所以HK_CBT鉤子可以在這里用。下面讓我們看如何實現(xiàn)MessageBox的定制過程。
1.安裝HK_CBT鉤子;
2.調(diào)用MessageBox()函數(shù);
3.移除HK_CBT鉤子。
整個過程很簡單吧?我們在這里介紹第一步和第三步。
安裝HK_CBT鉤子:
HHOOK hMsgBoxHook = SetWindowsHookEx( WH_CBT, // Type of hook CBTProc, // Hook procedure (see below) NULL, // Module handle. Must be NULL (see docs) GetCurrentThreadId() // Only install for THIS thread!!! );
重要的是SetWindowHookEx()函數(shù)的后邊兩個參數(shù),用它可以區(qū)別安裝是一個全局鉤子還是一個線程鉤子,在這里我們只要安裝一個線程鉤子。所以我們將Module handle設(shè)置為NULL,同時將thread ID設(shè)為本線程的ID。
在SetWindowHookEx()函數(shù)中有一個hook procedure,這是window調(diào)用的一個回調(diào)函數(shù),在windows系統(tǒng)中有一個HOOK鏈,我們在編寫hook procedure時要注意保證此鏈的完整,所以我們的hook procedure要調(diào)用CallNextHookEx()函數(shù)。下面就是我們的hook procedure:
LRESULT CALLBACK CBTProc(int nCode, WPARAM wParam, LPARAM lParam) { HWND hwnd; if(nCode < 0) return CallNextHookEx(hMsgBoxHook, nCode, wParam, lParam); switch(nCode) { case HCBT_ACTIVATE: // 現(xiàn)在wParam中就是message box的句柄 hwnd = (HWND)wParam; // 我們已經(jīng)有了message box的句柄,在這里我們就可以定制message box了! return 0; } // Call the next hook, if there is one return CallNextHookEx(hMsgBoxHook, nCode, wParam, lParam); }
移除HK_CBT鉤子:
只要調(diào)用UnhookWindowsHookEx()函數(shù)就可以了
好了,我們將在上面的三步寫成一個函數(shù),如下:
int MsgBoxEx(HWND hwnd, TCHAR *szText, TCHAR *szCaption, UINT uType) { int ret; // Install a thread hook, so we can customize it hMsgBoxHook = SetWindowsHookEx( WH_CBT, CBTProc, NULL, GetCurrentThreadId() ); // Display a standard message box ret = MessageBox(hwnd, szText, szCaption, uType); // remove the window hook UnhookWindowsHookEx(hMsgBoxHook); return ret; }
其實你也可以鉤住WM_CREATE消息,不過那樣處理要復(fù)雜一些。在早期的windows platform SDK中就有這樣一個例子。
你可能說,定制一個MessageBox有什么用處,我想有下面的用途:
1.你可以用CreateWindowEx()給MessageBox添加一個check box控件,并子類化MessageBox來處理check box的消息
2.通過子類化改變messagebox、button或icon,以便和你整個程序的界面風(fēng)格相一致
只要有了MessageBox對話框的句柄,你能做的很多,很多...
另外,如果你對模式對話框的機理很了解,你可以自己寫出一個"MessageBox"來代替系統(tǒng)MessageBox用在你的程序中。你可以參考Jeffrey Richter的《Windows 95程式設(shè)計指南》,在書中給出了模式對話框的偽碼。這本書的繁體電子版可以在候捷的個人網(wǎng)站上下載。這種方法也比較簡單(添加一個消息循環(huán),Enable/Disable Owner窗口),示例代碼這里就不實現(xiàn)了。讀者可以參考相關(guān)資料加以完善。
相關(guān)文章
C++中memcpy和memmove的區(qū)別總結(jié)
這篇文章主要介紹了C++中memcpy和memmove的區(qū)別總結(jié),這個問題經(jīng)常出現(xiàn)在C++的面試題目中,需要的朋友可以參考下2014-10-10VScode搭建C/C++開發(fā)環(huán)境的詳細過程
最近迷上了vscode,小巧美觀,最主要的是全平臺,但是vscode并不是ide,必須得自己配置環(huán)境,下面這篇文章主要給大家介紹了關(guān)于VScode搭建C/C++開發(fā)環(huán)境的詳細過程,需要的朋友可以參考下2023-06-06Windows安裝配置C/C++(VS2017)OpenSSL開發(fā)環(huán)境配置教程
這篇文章主要為大家詳細介紹了Windows安裝配置C/C++,OpenSSL開發(fā)環(huán)境配置教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07