一直以来都不习惯使用MFC做程序,尽管写了这么多年的VC代码,直接用MFC的项目也就2个。直接使用WIN32 API让人感觉简洁和高效,当然代价是得花上更多时间去写。
回到这篇文章的话题来,还是从提出问题开始吧,大家都知道默认情况下在对话框的情况下,按一下esc键对话框就会退出,要想拦截这个情况很简单把WM_KEYDOWN消息拦下,判断wParam的值是否为VK_ESCAPE 。 如果写基于对话框程序的都知道其消息过程是没有WM_KEYDOWN消息,这就是为什么MFC的 CDialog 的消息映射里并没有响应 WM_KEYDOWN 的函数对应,那怎样拦截?很多人第一反应就是重截CDialog::PreTranslateMessage() ,确实,正是这样做,因为WM_KEYDOWN是在对话框消息循环时被 TranslateMessage() 搞掉的,所以要得到它就必须在这个函数调用之用截下来,MFC也正是这样做。
那又为什么CDialog::PreTranslateMessage()能收到而对话框消息过程又收不到的消息? MSDN的说法是 “Used by class CWinApp to translate window messages before they are dispatched to the TranslateMessage and DispatchMessage Windows functions.” 但DoModal()显然是模式对话框,模式对话框的话程序自身是不用管理消息队列的,怎么又能在TranslateMessage()之前做动作? 这就得进入DoModal()内部看一下了。下面代码来自MFC,先看红色字体部分。
INT_PTR CDialog::DoModal()
{
if (hWndParent && hWndParent != ::GetDesktopWindow() && ::IsWindowEnabled(hWndParent))
{
::EnableWindow(hWndParent, FALSE);
bEnableParent = TRUE;
#ifndef _AFX_NO_OLE_SUPPORT
pMainWnd = AfxGetMainWnd();
if (pMainWnd && pMainWnd->IsFrameWnd() && pMainWnd->IsWindowEnabled())
{
//
// We are hosted by non-MFC container
//
pMainWnd->EnableWindow(FALSE);
bEnableMainWnd = TRUE;
}
#endif
}