MFC漫談(四)——消息的路由
有了消息映射表,如何利用呢?消息如何被分門別類的派發出去的?用最基本的一句話概述,鉤子函數起了很大作用。故事是這樣的,有些漫長,也需要些耐心。
MFC中消息分為3類:
?1. WM_COMMAND:所有的UI組件和加速鍵都會產生這種消息,所有派生于CCmdTarget的類都有能力處理該消息
?2. 標準消息:除WM_COMMAND之外的WM_xx消息都是標準消息,派生于CWnd的類都有能力處理該消息
?3. 控件通知消息:用于子窗口控件向父窗口發送的消息
在MFC的消息映射表的建立中,通過一組宏,你就可以讓自己的類先于父類處理某些Windows消息,這種行為很像虛函數,只是我們重載的內容不是虛函數,而是消息。
推動消息的泵
第一階段 窗口過程
在產生一個窗口的時候,會調用CFrameWnd::Create,所有的故事也都從這里展開。下面的代碼為了簡潔,去掉了不相關的代碼



































這樣,通過AfxHookWindowCreate,在當前線程中安裝了一個鉤子,用來攔截和窗口相關的事件,每當:
1. 另一個窗口成為active;
2. 產生或摧毀一個窗口
3. Minimize或maximize一個窗口;
4. 移動或縮放一個窗口;
5. 完成一個來自系統菜單的命令;
6. 從系統隊列中取出一個消息;
時,都會先調用_AfxCbtFilterHook,接下來看看鉤子函數作了什么:
















這樣,_AfxCbtFilterHook的工作總結起來就是通過窗口子類化,把新建的窗口的窗口過程設置成AfxWndProc。
到這里,我們終于找到了窗口過程。
結論
CFrameWnd::Create創建窗口調用CWnd::CreateEx
CWnd::CreateEx調用AfxHookWindowCreate準備為窗口設置鉤子
AfxHookWindowCreate調用::SetWindowHookEx為窗口設置了一個WH_CBT類型的鉤子來過濾消息,并把過濾函數設置成_AfxCbtFilterHook
_AfxCbtFilterHook通過窗口子類化設置窗口的窗口過程為AfxWndProc
這樣,通過::DispatchMessage發送給窗口的消息就會源源不斷地送到AfxWndProc中來,可以想到,AfxWndProc利用MFC的消息映射表,分門別類的對消息進行分流。
(待續……)
posted on 2006-05-18 17:29 nacci 閱讀(3864) 評論(2) 編輯 收藏 引用 所屬分類: C++漫談