MFC message route (Part ONE)

王朝other·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

关于MFC中的消息流动

MFC中消息分为3类:

WM_COMMAND:所有的UI组件和加速键都会产生这种消息,所有派生于CCmdTarget的类都有能力处理该消息

标准消息:除WM_COMMAND之外的WM_xx消息都是标准消息,派生于CWnd的类都有能力处理该消息

控件通知消息:用于子窗口控件向父窗口发送的消息

在MFC的消息映射表的建立中,通过一组宏,你就可以让自己的类先于父类处理某些windows消息,这种行为很像虚函数,只是我们重载的内容不是虚函数,而是消息。

推动消息的泵

第一阶段 窗口过程

在产生一个窗口的时候,会调用CFrameWnd::Create,所有的故事也都从这里展开。

下面的代码为了简洁,去掉了不相关的代码

BOOL CFrameWnd::Create(…) {

// …

if (!CreateEx(…)) {

// …

}

// …

}

BOOL CWnd::CreateEx(…) {

// …

AfxHookWindowCreate(this);

HWND hWnd = ::CreateWindowEx(…);

// …

}

void AFXAPI AfxHookWindowCreate(CWnd* pWnd)

{

// …

if (pThreadState->m_hHookOldCbtFilter == NULL)

{

pThreadState->m_hHookOldCbtFilter = ::SetWindowsHookEx(WH_CBT,

_AfxCbtFilterHook, NULL, ::GetCurrentThreadId());

// …

}

// …

pThreadState->m_pWndInit = pWnd;

}

这样,通过AfxHookWindowCreate,在当前线程中安装了一个钩子,用来拦截和窗口相关的事件,每当:

另一个窗口成为active;

产生或摧毁一个窗口

Minimize或maximize一个窗口;

移动或缩放一个窗口;

完成一个来自系统菜单的命令;

从系统队列中取出一个消息;

时,都会先调用_AfxCbtFilterHook,接下来:

LRESULT CALLBACK

_AfxCbtFilterHook(int code, WPARAM wParam, LPARAM lParam)

{

// …

WNDPROC afxWndProc = AfxGetAfxWndProc();

oldWndProc = (WNDPROC)SetWindowLongPtr(hWnd, GWLP_WNDPROC,(DWORD_PTR)afxWndProc);

// …

}

WNDPROC AFXAPI AfxGetAfxWndProc()

{

// …

return &AfxWndProc;

}

这样,_AfxCbtFilterHook的工作总结起来就是通过窗口子类化,把新建的窗口的窗口过程设置成AfxWndProc。

到这里,我们终于找到了窗口过程,总结一下

CFrameWnd::Create创建窗口调用CWnd::CreateEx

CWnd::CreateEx调用AfxHookWindowCreate准备为窗口设置钩子

AfxHookWindowCreate调用::SetWindowHookEx为窗口设置了一个WH_CBT类型的钩子来过滤消息,并把过滤函数设置成_AfxCbtFilterHook

_AfxCbtFilterHook通过窗口子类化设置窗口的窗口过程为AfxWndProc

这样,通过::DispatchMessage发送给窗口的消息就会源源不断地送到AfxWndProc中来,可以想到,AfxWndProc利用MFC的消息映射表,分门别类的对消息进行分流。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航