关键字:非模态、模态、hook、wh_cbt、cbtproc、
1、意图
有时候我们希望将非模态窗口显示为模态窗口。比如在ie的“文件”菜单下选择“打印”,弹出的“打印”对话框就是非模态的(也许我们不太清楚microsoft的设计意图,一般来说这里的“打印”对话框应该是模态的)。这种情况下如何将“打印”对话框显示为模态的呢(这个对话框对我们来说是black box)?
2、简单实现
简单地说,模态窗口显示时,其父窗口是被disable的,所以模态窗口才呈现“模态”,所以只要在显示我们非模态窗口前将父窗口disable即可实现,如下:
……
afxgetmainwnd()->enablewindow(false);//将主窗口disable,显示出的非模态窗口就变成模态的了
showmodelesswindow();
……
问题在于非模态窗口显示之后是立即返回的,那我们将父窗口enable的代码放在哪里呢?笨办法是用时钟,不断地检测显示出来的非模态窗口是否已经关闭,若关闭则将父窗口enable。
当然,还要更好的办法。
3、wh_cbt hook
wh_cbt钩子的详细说明请参阅msdn,我们仅仅需要知道的是在窗口创建、销毁之前系统都会调用挂上了wh_cbt的钩子函数,这正是我们需要的。具体就是在显示非模态窗口之前挂上我们的wh_cbt钩子处理函数,之后非模态窗口创建的句柄就可以在钩子函数的ncode为hcbt_createwnd(创建窗口)时从wparam参数获得,将其保存下来,并在钩子函数的ncode为hcbt_destroywnd(销毁窗口)时与wparam参数进行比较,如果匹配则恢复主窗口的enable状态。
2、实现
1)首先定义两个变量,此处为全局静态变量。
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);
g_hwnddialog = 0; //可能多次调用,需要重置保存窗口句柄的变量
g_hhook = setwindowshookex(wh_cbt, cbtproc, null, getcurrentthreadid());
if (!g_hhook)
{
afxgetmainwnd()->enablewindow(true);
return;
}
调用“打印”对话框
}
lresult callback cmyhtmlview::cbtproc(int ncode, wparam wparam, lparam lparam)
{
switch (ncode)
{
case hcbt_createwnd:
{
hwnd hwnd = (hwnd)wparam;
lpcbt_createwnd pcbt = (lpcbt_createwnd)lparam;
lpcreatestruct pcs = pcbt->lpcs;
if ((dword)pcs->lpszclass == 0x00008002)//#32770,“打印”对话框类名
{
if ( g_hwnddialog == 0 )
g_hwnddialog = hwnd; // 只保存一次保存“打印”窗口的句柄
}
break;
}
case hcbt_destroywnd:
{
hwnd hwnd = (hwnd)wparam;
if (hwnd == g_hwnddialog)
{
afxgetmainwnd()->enablewindow(true);//恢复窗口状态
unhookwindowshookex(g_hhook);//去除挂钩
}
break;
}
}
return callnexthookex(g_hhook, ncode, wparam, lparam);
}
很简单吧,更重要的是这种方法确实有效。