关于MFC项目中使用CDHtmlDialog禁止脚本错误的方法

本文探讨了在WebBrowser控件加载错误JS代码时如何屏蔽或控制错误提示,包括屏蔽错误信息和记录错误信息同时控制JS执行。通过自定义COleControlSite类并实现IOleCommandTarget接口,实现错误提示的定制化处理。

当WebBrowser控件(CDHtmlDialog自动创建了WebBrowser控件)加载的网页中含有错误Javascript代码时默认情况下控件会弹出错误信息提示对话框,相对于用户体验来说这样的提示完全不是开发人员想要的,针对这个问题有两个解决方案,一是完全屏蔽掉错误提示,二是控制错误的提示并且记录错误信息同时也可以控制出现错误后Javascript是否继续执行。

1、屏蔽错误信息提示

 

?
  1. m_pBrowserApp->put_Silent(VARIANT_TRUE);//禁止脚本错误提示  

在CDHtmlDialog::OnInitDialog()的代码中首先了创建WebBrowser控件,然后把控件的Browser对象赋值给m_pBrowserApp(这是CDHtmlDialog完成的不需要自己处理)。WebBrowser的put_Silent函数在官方给出的说明是禁用所有的对话框,但例外情况是它不会影响SSL安全认证需要的进示对话框。绝大多数情况下这就可以解决问题了,记得很久以前我遇到过一种情况就是虽然调用了put_Silent但是还是有极个别的js错误是无法屏蔽掉的依然会显示出来(在网页含有嵌套页面时会错误无法屏蔽,不知道是否还有其它情况),现在找不到这样的网页了,如果谁遇到这种情况了建议给我发上个URL让我也重温一下当年阳光灿烂的时刻。

2、控制错误提示并进行记录

  这要比第一种方法复杂上许多,简短的来说就是自定义COleControlSite类并实现IOleCommandTarget接口,IOleCommandTarget接口是错误控制的关健,错误发生时会触发此接口的Exec函数并为nCmdID参数赋值为OLECMDID_SHOWSCRIPTERROR,这样就可以得到错误信息了。

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
IOleCommandTarget : public IUnknown
     {
     public :
         virtual /* [input_sync] */ HRESULT STDMETHODCALLTYPE QueryStatus(
             /* [unique][in] */ __RPC__in_opt const GUID *pguidCmdGroup,
             /* [in] */ ULONG cCmds,
             /* [out][in][size_is] */ __RPC__inout_ecount_full(cCmds) OLECMD prgCmds[  ],
             /* [unique][out][in] */ __RPC__inout_opt OLECMDTEXT *pCmdText) = 0;
         
         virtual HRESULT STDMETHODCALLTYPE Exec(
             /* [unique][in] */ __RPC__in_opt const GUID *pguidCmdGroup,
             /* [in] */ DWORD nCmdID,
             /* [in] */ DWORD nCmdexecopt,
             /* [unique][in] */ __RPC__in_opt VARIANT *pvaIn,
             /* [unique][out][in] */ __RPC__inout_opt VARIANT *pvaOut) = 0;
         
     };
    

现在我们开始实现自定义的COleControlSite,代码如下:

  1. CMyControlSite.h  
  1. #pragma once  
  2. #include "afxocc.h"  
  3. #include "Mshtml.h"//应该加入这个头文件  
  4. #include "Mshtmhst.h"//这个也是  
  5. class CMyControlSite :public COleControlSite  
  6. {  
  7. public:  
  8.     CMyControlSite(COleControlContainer *pCntr):COleControlSite(pCntr) {}  
  9.     ~CMyControlSite(void);  
  10. protected:  
  11.     DECLARE_INTERFACE_MAP()    
  12.     BEGIN_INTERFACE_PART(OleCommandTarget, IOleCommandTarget)    
  13.         STDMETHOD(QueryStatus)(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText);    
  14.         STDMETHOD(Exec)(const GUID* pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANTARG* pvaIn, VARIANTARG* pvaOut);    
  15.     END_INTERFACE_PART(OleCommandTarget)    
  16. };  
  1. CMyControlSite.cpp  
  1. #include "StdAfx.h"  
  2. #include "MyControlSite.h"  
  3.   
  4.   
  5. BEGIN_INTERFACE_MAP(CMyControlSite, COleControlSite)    
  6.     INTERFACE_PART(CMyControlSite, IID_IOleCommandTarget, OleCommandTarget)  
  7. END_INTERFACE_MAP()    
  8.   
  9.   
  10.   
  11. CMyControlSite::~CMyControlSite(void)  
  12. {  
  13. }  
  14.   
  15. HRESULT CMyControlSite::XOleCommandTarget::Exec    
  16. (const GUID* pguidCmdGroup, DWORD nCmdID,    
  17.  DWORD nCmdexecopt, VARIANTARG* pvaIn, VARIANTARG* pvaOut )    
  18. {    
  19.     HRESULT hr = OLECMDERR_E_NOTSUPPORTED;    
  20.     //return S_OK;    
  21.     if (pguidCmdGroup && IsEqualGUID(*pguidCmdGroup, CGID_DocHostCommandHandler))    
  22.     {    
  23.   
  24.         switch (nCmdID)     
  25.         {    
  26.   
  27.         case OLECMDID_SHOWSCRIPTERROR:    
  28.             {    
  29.                 IHTMLDocument2*             pDoc = NULL;    
  30.                 IHTMLWindow2*               pWindow = NULL;    
  31.                 IHTMLEventObj*              pEventObj = NULL;    
  32.                 BSTR                        rgwszNames[5] =     
  33.                 {     
  34.                     SysAllocString(L"errLine"),    
  35.                     SysAllocString(L"errCharacter"),    
  36.                     SysAllocString(L"errCode"),    
  37.                     SysAllocString(L"errMsg"),    
  38.                     SysAllocString(L"errUrl")    
  39.                 };    
  40.                 DISPID                      rgDispIDs[5];    
  41.                 VARIANT                     rgvaEventInfo[5];    
  42.                 DISPPARAMS                  params;    
  43.                 BOOL                        fContinueRunningScripts = false;  //修改此处为false禁止脚本错误提示  
  44.   
  45.                 params.cArgs = 0;    
  46.                 params.cNamedArgs = 0;    
  47.                   
  48.                 hr = pvaIn->punkVal->QueryInterface(IID_IHTMLDocument2, (void **) &pDoc);        
  49.                    
  50.                 hr = pDoc->get_parentWindow(&pWindow);    
  51.                 pDoc->Release();    
  52.                   
  53.                 hr = pWindow->get_event(&pEventObj);    
  54.                   
  55.                 for (int i = 0; i < 5; i++)     
  56.                 {      
  57.                       
  58.                     hr = pEventObj->GetIDsOfNames(IID_NULL, &rgwszNames[i], 1,     
  59.                         LOCALE_SYSTEM_DEFAULT, &rgDispIDs[i]);    
  60.                   
  61.                     hr = pEventObj->Invoke(rgDispIDs[i], IID_NULL,    
  62.                         LOCALE_SYSTEM_DEFAULT,    
  63.                         DISPATCH_PROPERTYGET, &params, &rgvaEventInfo[i],    
  64.                         NULL, NULL);    
  65.                     //可以在此记录错误信息                    //必须使用SysFreeString来释放SysAllocString分配的内存,SysAllocString在分配的内存中记录了字符的长度  
  66.                     SysFreeString(rgwszNames[i]);    
  67.                 }    
  68.   
  69.                 // At this point, you would normally alert the user with     
  70.                 // the information about the error, which is now contained    
  71.                 // in rgvaEventInfo[]. Or, you could just exit silently.    
  72.   
  73.                 (*pvaOut).vt = VT_BOOL;    
  74.                 if (fContinueRunningScripts)    
  75.                 {    
  76.                     // 在页面中继续执行脚本   
  77.                     (*pvaOut).boolVal = VARIANT_TRUE;    
  78.                 }    
  79.                 else   
  80.                 {    
  81.                     // 停止在页面中执行脚本    
  82.                     (*pvaOut).boolVal = VARIANT_FALSE;       
  83.                 }     
  84.                 break;    
  85.             }    
  86.         default:    
  87.             hr =OLECMDERR_E_NOTSUPPORTED;   
  88.             break;    
  89.         }    
  90.     }    
  91.     else   
  92.     {    
  93.         hr = OLECMDERR_E_UNKNOWNGROUP;  
  94.     }    
  95.     return (hr);    
  96. }    
  97.   
  98.   
  99. ULONG FAR EXPORT CMyControlSite::XOleCommandTarget::AddRef()     
  100. {     
  101.     METHOD_PROLOGUE(CMyControlSite, OleCommandTarget)     
  102.         return pThis->ExternalAddRef();     
  103. }     
  104.   
  105.   
  106. ULONG FAR EXPORT CMyControlSite::XOleCommandTarget::Release()     
  107. {     
  108.     METHOD_PROLOGUE(CMyControlSite, OleCommandTarget)     
  109.         return pThis->ExternalRelease();     
  110. }     
  111.   
  112. HRESULT FAR EXPORT CMyControlSite::XOleCommandTarget::QueryInterface(REFIID riid, void **ppvObj)     
  113. {     
  114.     METHOD_PROLOGUE(CMyControlSite, OleCommandTarget)     
  115.         HRESULT hr = (HRESULT)pThis->ExternalQueryInterface(&riid, ppvObj);     
  116.     return hr;     
  117. }    
  118.   
  119. STDMETHODIMP CMyControlSite::XOleCommandTarget::QueryStatus(     
  120.     /* [unique][in] */ const GUID __RPC_FAR *pguidCmdGroup,     
  121.     /* [in] */ ULONG cCmds,     
  122.     /* [out][in][size_is] */ OLECMD __RPC_FAR prgCmds[ ],     
  123.     /* [unique][out][in] */ OLECMDTEXT __RPC_FAR *pCmdText     
  124.     )     
  125. {     
  126.     METHOD_PROLOGUE(CMyControlSite, OleCommandTarget)     
  127.         return OLECMDERR_E_NOTSUPPORTED;     
  128. }     


对话框头文件加入声明:

  1. virtual BOOL CreateControlSite(COleControlContainer* pContainer,   
  2.         COleControlSite** ppSite, UINT  nID , REFCLSID  clsid );  

 

对应源文件:

  1. BOOL CXDlg::CreateControlSite(COleControlContainer* pContainer,   
  2.         COleControlSite** ppSite, UINT  nID , REFCLSID  clsid )  
  3. {  
  4.         if(ppSite == NULL)  
  5.     {  
  6.         ASSERT(FALSE);  
  7.         return FALSE;  
  8.     }  
  9.   
  10.     CMyControlSite *pBrowserSite =   
  11.         new CMyControlSite (pContainer);//  
  12.     if (!pBrowserSite)  
  13.         return FALSE;  
  14.   
  15.     *ppSite = pBrowserSite;  
  16.     return TRUE;  
  17. }  

hinkingfor原文中为CDHtmlDialog,同样适用于CDialog.

参考文章  CDHtmlDialog探索----WebBrowser扩展和网页Javascript错误处理 作者:thinkingfor

http://blog.youkuaiyun.com/cnfixit/article/details/6780583

ATL发展到现在,从未提供像MFC中的CDHtmlDialog一样的HTML页面布局且方便开发者使用的窗口基类,虽然也有DHtml Control可供使用,但并不能像MFC那样通过宏映射来方便地让开发者与页面元素之间进行任意的双向交互,尤其是需要响应页面元素事件的时候,ATL/WTL爱好者必须自己编写相应的代码来完成这些工作。基于这个原因,通过理解分析MFCCDHtmlDialog类的功能和实现行为,这里完全使用ATL一样的实现机制来模仿MFC中实现的功能编写了一个头文件,使ATL爱好者在无需MFC庞大的支持库的情形下实现跟CDHtmlDialog一样的功能,并且扩展了其能力。下载的压缩包中只有一个头文件 atldhtmldlg.h,头文件的开始部分是一段开发者使用示例的注释,以方便开发者容易地上手。这个文件提供了若干个类和模板类,开发者只需关注其中两个模板类:CDHtmlDialogImpl和CMultiPageDHtmlDialogImpl,第一个模板类实现了WEB页面布局的对话框,第二个模板类以第一个类为基础,扩展成了在一个对话框中支持多个页面。使用方式非常简单,从上述两个模板类之一继承实现一个对话框类,然后添加相应的映射宏,实现宏映射中的方法即可构造一个完美、表现能力强、控制方便的对话框窗口,例子代码像下面这样:class CMainDlg : public CDHtmlDialogImpl{ ...... // 实现页面元素事件的处理 // 请注意响应函数原型定义:HRESULT Foo(IHTMLElement*) BEGIN_DHTML_EVENT_MAP(CMainDlg) DHTML_EVENT_ONCLICK(_T("elementid1"), OnClick) DHTML_EVENT_ONMOUSEMOVE(_T("elementid2"), OnMouseMove) DHTML_EVENT_ELEMENT(DISPID_HTMLELEMENTEVENTS_ONMOUSEOVER, _T("elementid"), OnMouseOver) // id为elementid的元素事件响应 DHTML_EVENT_CLASS(DISPID_HTMLELEMENTEVENTS_ONMOUSEOVER, _T("myclass"), OnMouseOver) // class为myclass的元素事件响应 DHTML_EVENT_TAG(DISPID_HTMLELEMENTEVENTS_ONMOUSEOVER, _T("div"), OnMouseOver) // 所有DIV元素的onmouseover事件响应 DHTML_EVENT_AXCONTROL(controlMethodDISPID, _T("objectid"), OnControlMethod) // ActiveX控件事件响应 END_DHTML_EVENT_MAP() // 实现外部DISPATCH方法,能在这里添加任意的方法供页面中的脚本调用 // 调用方式如下:window.external.about(123, "abc") // 请注意响应函数的原型定义:void Foo(VARIANT*,VARIANT*,VARIANT*) // 目前仅仅定义了三个参数,一般也足够使用了。如果脚本实际调用中只提供了2个参数,则第三个指针参数是NULL。以此类推 BEGIN_EXTERNAL_METHOD_MAP(CMainDlg) EXTERNAL_METHOD(_T("about"), OnAbout) END_EXTERNAL_METHOD_MAP() void OnAbout(VARIANT* para1, VARIANT* para2, VARIANT* para3) { // your code is here. } HRESULT OnClick(IHTMLElement *pElement) { // your code is here. return S_OK; } HRESULT OnMouseMove(IHTMLElement *pElement) { // your code is here. return S_OK; } ......}除了上面的例子,还有CMultiPageDHtmlDialogImpl中的一些映射宏,以及还有一些虚函数可供重载。对话框设计好后,调用方法也很简单,例子如下: CMainDlg dlg; dlg.m_nHtmlResID = IDR_YOUR_HTML_RESOURCE_ID; // open your resource or dlg.m_szHtmlResID = _T("C:\\yourname.html"); // open your resource or dlg.m_strCurrentUrl = _T("www.microsoft.com"); // open local html file or external URL dlg.Create(NULL); // or dlg.DoModal(NULL);好了,介绍就到这里,如果有更多疑问,请下载头文件查看源代码以及注释,本次下载的文件版本是 1.02。当前版本的一个缺憾是尚未支持DDX/DDV,原因是ATL并未提供DDX机制,下个版本将完全仿照WTL的方式实现该类的DDX/DDV。另外为了通用性以及减少依赖,代码中完全未使用CString或CAtlString,主要是MFC、ATL、WTL各自提供了自己的CString实现。版权特别声明:本软件源码完全属作者James(胡柏华)自创,作者拥有修改和变更代码特性的权利,任何团体或个人均可以自由下载并免费使用。如用于商业用途,请在所属商业软件的版权声明中加注本声明。使用使用本源码的过程中产生的任何错误,作者并无义务提供技术支持,由此导致的任何损失,作者概不负责。2007-11-7
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值