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

来源:互联网 发布:办公软件自动化证书 编辑:程序博客网 时间:2024/05/16 17:49
当WebBrowser控件(CDHtmlDialog自动创建了WebBrowser控件)加载的网页中含有错误Javascript代码时默认情况下控件会弹出错误信息提示对话框,相对于用户体验来说这样的提示完全不是开发人员想要的,针对这个问题有两个解决方案,一是完全屏蔽掉错误提示,二是控制错误的提示并且记录错误信息同时也可以控制出现错误后Javascript是否继续执行。

1、屏蔽错误信息提示

?
view plain
  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 : publicIUnknown
   {
   public:
       virtual/* [input_sync] */ HRESULT STDMETHODCALLTYPE QueryStatus(
           /* [unique][in] */ __RPC__in_opt constGUID *pguidCmdGroup,
           /* [in] */ ULONG cCmds,
           /* [out][in][size_is] */ __RPC__inout_ecount_full(cCmds) OLECMD prgCmds[  ],
           /* [unique][out][in] */ __RPC__inout_opt OLECMDTEXT *pCmdText) = 0;
       
       virtualHRESULTSTDMETHODCALLTYPE Exec(
           /* [unique][in] */ __RPC__in_opt constGUID *pguidCmdGroup,
           /* [in] */ DWORD nCmdID,
           /* [in] */ DWORD nCmdexecopt,
           /* [unique][in] */ __RPC__in_opt VARIANT *pvaIn,
           /* [unique][out][in] */ __RPC__inout_opt VARIANT *pvaOut) = 0;
       
   };
  

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

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


对话框头文件加入声明:

view plain
  1. virtualBOOL CreateControlSite(COleControlContainer* pContainer,  
  2.         COleControlSite** ppSite,UINT  nID , REFCLSID  clsid ); 

对应源文件:

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

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

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

http://blog.csdn.net/cnfixit/article/details/6780583

原创粉丝点击