热线电话:0574--87304500 / 0574--87329560 / 0574--87329570 / 0574--87324970
天威诚信证书服务
服务中心
工作时间:08:30 -- 17:00
热线电话:0574--87304500
热线电话:0574--87329570
公司地址:宁波市海曙区华楼街14号世纪广场A座407室(315000)
跳过导航链接首页 > 关于我们 > 内刊在线
WebBrowser插件使用分享

  互联网的普及使用户非常熟悉基于浏览器的操作界面,对传统GUI模式应用的界面提出了更高的要求,希望能够具有类似于网页程序的表达力、易用性。对于Windows平台的GUI应用程序,WebBrowser插件是一个很好的解决方案,使用该插件相当在程序中嵌入了一个浏览器窗口,通过该窗口中可以充分利用IE内核的强大功能。
  程序嵌入WebBrowser时,逻辑上分为主窗口、WebBrowser以及ATLSimpleObject,其中ATLSimpleObject供WebBrowser中的JavaScript脚本调用。逻辑结构如下:

\

  图1 WebBrowser嵌入逻辑结构

  关于如何嵌入WebBrowser?如何在GUI程序中添加ATLSimpleObject对象?可以通过MSDN文档查询到。本文仅就在WebBrowser使用过程中的一些经验进行分享。
  我们最常用Dialog开发程序,亦即使用函数doModal()显示的对话框,嵌入WebBrowser浏览器时会有一些奇怪的现象。
  1、如果在WebBrowser页面中敲击回车键盘,该事件直接由Dialog处理,网页无法处理该事件。我们希望敲击回车键时由网页处理该事件,而Dialog的缺省处理是关闭窗口。
  2、在网页中,敲击ESC键时,Dialog会直接退出。
  3、在网页中无法使用TAB键在网页元素间进行导航。

  经过对Win32程序处理过程以及MFC代码跟踪分析,发现是因为在doModal()函数内部的处理过程中,会将所有获取到的消息首先交由Dialog窗口进行处理,如果Dialog没有处理该消息,再交由该消息真正的目标窗口处理。基于这样的过程,当用户在网页中输入ENTER, ESC,TAB键时,Dialog认为是对话框事件并进行消息处理,导致WebBrowser不会接受到该消息以进行后续处理。

  首先想到的方案Subclass Dialog 窗口,在Dialog处理键盘消息之前,将该消息交由WebBrowser处理。实际情况是,我们Subclass后根本没有接收到上述键盘消息,方案不可行。是因为Windows的Dialog对前述键盘消息的处理是在我们Subclass之前,导致我们无法处理该消息。经过网上大量查询,证明在doModal模式下,没有办法实现上述键盘消息的处理。

  之后决定采用Dialog的CreateWidnow, ShowWindow的模式进行处理。在这种模式下,需要自己编写Message Loop代码。关键点在于,在Message Loop中将键盘、鼠标消息首先交由WebBrowser处理,对于WebBrowser未处理的消息,再交由后续的TranslateMessage, DispatchMessage函数处理。主要步骤如下:
  1、WebBrowser的创建,在Dialog中处理

public:
 CAxWindow m_axwin;

// OnInitDialog
 m_axwin.Create(m_hWnd, rect, NULL,  WS_TABSTOP | WS_CHILD | WS_VISIBLE | WS_BORDER, 0, 1);
 m_axwin.CreateControl(OLESTR("http://www.itrus.com.cn"));

  2、WebBrowser的窗口大小调整
// OnSize
 if(m_axwin.IsWindow())
  m_axwin.MoveWindow(0,0,cx,cy);

  3、Dialog的WM_CLOSE消息需要特殊处理
  因为在ShowWindow模式下,EndDialog函数无法关闭窗口,需要对WM_CLOSE进行处理,以中止外部的Message Loop。
// OnClose
 PostQuitMessage(0);

  4、Dialog的MessageLoop
 CAtlWinDemo1Dlg m_dlg;
 m_dlg.Create(IDD_ATLWINDEMO1_DIALOG);
 m_dlg.SetWindowPos(NULL,100,100,800,600,NULL);
 m_dlg.CenterWindow();
 m_dlg.ShowWindow(SW_SHOW);

 MSG msg;
 BOOL bRet;
while( (bRet = GetMessage( &msg, NULL, 0, 0 )) != 0)
    {
if (bRet == -1)
        {
//break;
        }
else
        {
   LRESULT lTranslated = 0;
   if((msg.message >= WM_KEYFIRST && msg.message <= WM_KEYLAST) ||
      (msg.message >= WM_MOUSEFIRST && msg.message <= WM_MOUSELAST))
   {

 Label