利用WH_CBT Hook将非模态对话框显示为模态对话框

原创 2004年09月15日 23:48:00

关键字:非模态、模态、HookWH_CBTCBTProc

 

1、意图

有时候我们希望将非模态窗口显示为模态窗口。比如在IE的“文件”菜单下选择“打印”,弹出的“打印”对话框就是非模态的(也许我们不太清楚Microsoft的设计意图,一般来说这里的“打印”对话框应该是模态的)。这种情况下如何将“打印”对话框显示为模态的呢(这个对话框对我们来说是Black Box)?

 

2、简单实现

简单地说,模态窗口显示时,其父窗口是被Disable的,所以模态窗口才呈现“模态”,所以只要在显示我们非模态窗口前将父窗口Disable即可实现,如下:

 

……

AfxGetMainWnd()->EnableWindow(FALSE);//将主窗口Disable,显示出的非模态窗口就变成模态的了

ShowModelessWindow();

……

 

问题在于非模态窗口显示之后是立即返回的,那我们将父窗口Enable的代码放在哪里呢?笨办法是用时钟,不断地检测显示出来的非模态窗口是否已经关闭,若关闭则将父窗口Enable

当然,还要更好的办法。

 

3WH_CBT Hook

WH_CBT钩子的详细说明请参阅MSDN,我们仅仅需要知道的是在窗口创建、销毁之前系统都会调用挂上了WH_CBT的钩子函数,这正是我们需要的。具体就是在显示非模态窗口之前挂上我们的WH_CBT钩子处理函数,之后非模态窗口创建的句柄就可以在钩子函数的nCodeHCBT_CREATEWND(创建窗口)时从wParam参数获得,将其保存下来,并在钩子函数的nCodeHCBT_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);

}

 

很简单吧,更重要的是这种方法确实有效。

 

参考资料

MSDNCBTProc

 

引用地址利用WH_CBT Hook将非模态对话框显示为模态对话框

MFC非模态对话框的创建及显示

我们需要在原对话框CMFC_Kinect_TrackingDlg设计一个按钮,点击这个按钮后弹出一个非模态对话框Color_Based_SettingDlg, 并且同时两个对话框都是激活状态,都可以...
  • denyz
  • denyz
  • 2016年03月05日 11:05
  • 878

Duilib 模态对话框和非模态对话框

[cpp] view plaincopy void CMainDlg::DoTask()   {       CTaskDlg * pDlg = NULL;    ...
  • u012519333
  • u012519333
  • 2014年12月25日 14:54
  • 2684

模态对话框和非模态对话框区别

MFC中对话框有两种形式,一个是模态对话框(model dialog box),一个是非模态对话框(modeless dialog box)。  一、模态对话框(model dialog box) ...
  • u013776188
  • u013776188
  • 2017年07月12日 11:47
  • 174

c#中的模态对话框和非模态对话框

模态对话框 弹出窗口阻止调用窗口的所有消息响应。 只有在弹出窗口结束后调用窗口才能继续。 在模态窗口“关闭”后,可以读取模态窗口中信息,包括窗口的返回状态,窗口子控件的值。 非模态对话框 可...
  • c914620529
  • c914620529
  • 2015年08月05日 21:50
  • 5157

MFC模态对话框与非模态对话框的区别

对话框主要分为两个类型:模式对话框和非模式对话框。 1、模式对话框的特点是: a)在关闭模式对话框前,无法操作背景窗口,除非自身就是主窗口; b)使用CDialog::DoModal函数弹出模式...
  • djb100316878
  • djb100316878
  • 2014年12月30日 16:40
  • 2039

模态对话框与非模态对话的几种销毁方法与区别

模态对话框与非模态对话的几种销毁方法与区别  先着一个注意点,自己误识了好久: CloseWindow是一个函数,可以用来最小化指定的窗口,但并不销毁该窗口。 先说下模式对话框调用的原...
  • rankun1
  • rankun1
  • 2016年05月26日 16:00
  • 2134

Qt4中的模态与非模态对话框用法总结

模态对话框(Modal Dialog)与非模态对话框(Modeless Dialog)的概念不是Qt所独有的,在各种不同的平台下都存在。又有叫法是称为模式对话框,无模式对话框等。 所谓模态对话框就是...
  • liuligui5200
  • liuligui5200
  • 2015年08月28日 14:12
  • 1039

MFC调用对话框dll弹出非模态对话框

思路: 1、新建基于MFC的DLL,在DLL中做一个导出函数,在该导出函数中new出对话框类,导致函数返回CWnd*也就是new出来的窗口实例指针。如下 CWnd *  InitData( CW...
  • q357010621
  • q357010621
  • 2016年03月01日 00:01
  • 609

JS高级应用:操作模态与非模态对话框

我们知道,对话框一般分为两种类型:模态类型(modal)与非模态类型(modeless)。所谓模态对话框,就是指除非采取有效的关闭手段,用户的鼠标焦点或者输入光标将一直停留在其上的对话框。非模态对话框...
  • duxikuan
  • duxikuan
  • 2016年01月28日 10:27
  • 1546

<MFC>模态对话框和非模态对话框的创建弹出

一、模态对话框的创建及显示 1、Windows对话框分为两类:模态对话框和非模态对话框。     模态对话框:当它弹出后,本应用程序其他窗口将不再接受用户输入,只有该对话框响应用户输入,在对它进...
  • xieziwei2008
  • xieziwei2008
  • 2013年10月16日 16:10
  • 1450
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:利用WH_CBT Hook将非模态对话框显示为模态对话框
举报原因:
原因补充:

(最多只允许输入30个字)