1、意图
有时候我们希望将非模态窗口显示为模态窗口。比如在IE的“文件”菜单下选择“打印”,弹出的“打印”对话框就是非模态的(也许我们不太清楚Microsoft的设计意图,一般来说这里的“打印”对话框应该是模态的)。这种情况下如何将“打印”对话框显示为模态的呢(这个对话框对我们来说是Black Box)?
2、简单实现
简单地说,模态窗口显示时,其父窗口是被Disable的,所以模态窗口才呈现“模态”,所以只要在显示我们非模态窗口前将父窗口Disable即可实现,如下:
| 以下为引用的内容: …… 站长.站 AfxGetMainWnd()->EnableWindow(FALSE);//将主窗口Disable,显示出的非模态窗口就变成模态的了 ShowModelessWindow(); …… 中国.站.长站 |
问题在于非模态窗口显示之后是立即返回的,那我们将父窗口Enable的代码放在哪里呢?笨办法是用时钟,不断地检测显示出来的非模态窗口是否已经关闭,若关闭则将父窗口Enable。
Www.zhuye123.com
当然,还要更好的办法。
3、WH_CBT Hook
WH_CBT钩子的详细说明请参阅MSDN,我们仅仅需要知道的是在窗口创建、销毁之前系统都会调用挂上了WH_CBT的钩子函数,这正是我们需要的。具体就是在显示非模态窗口之前挂上我们的WH_CBT钩子处理函数,之后非模态窗口创建的句柄就可以在钩子函数的nCode为HCBT_CREATEWND(创建窗口)时从wParam参数获得,将其保存下来,并在钩子函数的nCode为HCBT_DESTROYWND(销毁窗口)时与wParam参数进行比较,如果匹配则恢复主窗口的Enable状态。 中国
2、实现
中国站长.站
1)首先定义两个变量,此处为全局静态变量。 Www~Chinaz~com
| 以下为引用的内容: static HHOOK g_hHook = NULL; static HWND g_hWndDialog = NULL;//用以保存窗口句柄 |
2)再添加一个函数CbtProc,由于是回调函数,注意要声明为static。 站.长站
static LRESULT CALLBACK CbtProc(int nCode, WPARAM wParam, LPARAM lParam); 中国站.长站
3)挂钩
假设下面是我们的某个浏览器中调用“打印”对话框的函数 中国
| 以下为引用的内容: void CMyHtmlView::OnFilePrint() { 中.国.站长站 AfxGetMainWnd()->EnableWindow(FALSE); Www.zhuye123.com g_hWndDialog = 0; //可能多次调用,需要重置保存窗口句柄的变量 站长.站 g_hHook = SetWindowsHookEx(WH_CBT, CbtProc, NULL, GetCurrentThreadId()); 站.长站 if (!g_hHook) 中国.站.长站 { Www.zhuye123.com AfxGetMainWnd()->EnableWindow(TRUE); return; 站长.站 } |
调用“打印”对话框 中国
| 以下为引用的内容: } Chinaz_com LRESULT CALLBACK CMyHtmlView::CbtProc(int nCode, WPARAM wParam, LPARAM lParam) 中国.站.长站 { 站.长站 switch (nCode) Chinaz_com { case HCBT_CREATEWND: 站.长站 { Chinaz~com HWND hWnd = (HWND)wParam; LPCBT_CREATEWND pcbt = (LPCBT_CREATEWND)lParam; 中国站长_站,为中文网站提供动力 LPCREATESTRUCT pcs = pcbt->lpcs; 中国站长.站 Chinaz~com if ((DWORD)pcs->lpszClass == 0x00008002)//#32770,“打印”对话框类名 Www.zhuye123.com { Chinaz if ( g_hWndDialog == 0 ) Www~Chinaz~com g_hWndDialog = hWnd; // 只保存一次保存“打印”窗口的句柄 } 中国站长_站,为中文网站提供动力 break; } 中.国. case HCBT_DESTROYWND: 中.国站长站 { Www.zhuye123.com HWND hwnd = (HWND)wParam; 中.国.站长站 if (hwnd == g_hWndDialog) 中国站长_站,为中文网站提供动力 { 中国.站长站 AfxGetMainWnd()->EnableWindow(TRUE);//恢复窗口状态 中.国. UnhookWindowsHookEx(g_hHook);//去除挂钩 中国站长.站 } break; 中国.站.长站 } 站长.站 } return CallNextHookEx(g_hHook, nCode, wParam, lParam); Www~Chinaz~com } |
很简单吧,更重要的是这种方法确实有效。 中国.站长站