当前位置: 首页 > 图文教程 > 网络编程 > ASP.NET > 将非模态对话框显示为模态对话框

ASP.NET
.Net中使用com组件后发生System.ArithmeticException异常的解决办法
SQL Server.net 和 OLE DB.net连接数据库的比较
后台更新DataTable行内容的方法
敏捷软件开发(原则,模式与实践)笔记1
确保文本框输入值为数值的代码
XML和数据库之间相互的映射
让你的.NET程序兼容不同版本的Dll文件。
.NET 的数据访问应用程序块(Data Access Application Block)
用控件仅一条指令实现界面换肤和多语言版本(YFSkins)
Microsoft User Interface Process Application Block 研究(3)
分享:处理Excel方法小结
基于ASP.NET实现全球化
.net 里面 protected private 的变量也可以访问(新发现)。
关于C#中{0}和{1}的问题初次在此发贴,问题对你易对我难,求救了
使用C#代码实现增加用户帐号
全世界都在关注-微软重大产品发布
教你做Rational Rose(UML Design)
OLE DB取得数据库的架构信息
VB 从零开始编外挂(三)
XPath序列之四

ASP.NET 中的 将非模态对话框显示为模态对话框


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-08-14   浏览: 227 ::
收藏到网摘: n/a

1、意图

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

2、简单实现

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


 
问题在于非模态窗口显示之后是立即返回的,那我们将父窗口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);

  }


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