如何妥善处理WebBrowser对Javascript的错误问题,阻止JS弹出框,提高用户体验

时间:2023-03-09 02:57:14
如何妥善处理WebBrowser对Javascript的错误问题,阻止JS弹出框,提高用户体验
由于项目需求,最近转战客户端,开始搞浏览器开发。众所周知,现在在微软平台上开发浏览器,最常用的方法就是扩展Webbrowser,但是首先要清楚的是,WebBrowser控件仅仅是对WebBrowser ActiveX 控件提供了托管包装而已(详细了解http://msdn.microsoft.com/zh-cn/library/w290k23d(VS.80).aspx),要写一个像点样子的浏览器,很多方面还是要自己去扩展的,否则开发出来的也只能是个样子,没多少实际功能。
本篇随笔重点将介绍如何处理WebBrowser对JS的错误处理问题。 可能有些朋友看到上句话会觉得,这问题不是很简单嘛,设置一下 ScriptErrorsSuppressed属性不就完了嘛,殊不知这样做,像其他的譬如网页安全验证的弹出框,甚至很多其他网页内部弹出窗口也都被禁止掉了,如此当然是行不通的。可能还会有一部分朋友会说,那可以通过WebBrowser向网页中注入JS错误处理脚本,捕获JS错误,不是也可以嘛。那我会告诉你,这种方法在面对使用很多iframe的网站是也玩完。 那么接下来,我将介绍,如何更好的避免WebBrowser中的JS错误。 避免错误,首先要想到办法去捕获错误,查遍了网上的很多资料,发现是要去实现IOleCommandTarget接口,调用接口中的Exec方法来捕获异常,然后解决,当然如果需要去调用IE的一些功能,也得需实现IObjectWithSite接口。 定义IOleCommandTarget接口、OLECMD类: public static class NativeMethods
{
public enum OLECMDF
{
// Fields
OLECMDF_DEFHIDEONCTXTMENU = 0x20,
OLECMDF_ENABLED = ,
OLECMDF_INVISIBLE = 0x10,
OLECMDF_LATCHED = ,
OLECMDF_NINCHED = ,
OLECMDF_SUPPORTED =
} public enum OLECMDID
{
// Fields
OLECMDID_PAGESETUP = ,
OLECMDID_PRINT = ,
OLECMDID_PRINTPREVIEW = ,
OLECMDID_PROPERTIES = ,
OLECMDID_SAVEAS = ,
OLECMDID_SHOWSCRIPTERROR =
}
public enum OLECMDEXECOPT
{
// Fields
OLECMDEXECOPT_DODEFAULT = ,
OLECMDEXECOPT_DONTPROMPTUSER = ,
OLECMDEXECOPT_PROMPTUSER = ,
OLECMDEXECOPT_SHOWHELP =
} [ComImport, InterfaceType(ComInterfaceType.InterfaceIsIUnknown), Guid("B722BCCB-4E68-101B-A2BC-00AA00404770"), ComVisible(true)]
public interface IOleCommandTarget
{
[return: MarshalAs(UnmanagedType.I4)]
[PreserveSig]
int QueryStatus(ref Guid pguidCmdGroup, int cCmds, [In, Out] NativeMethods.OLECMD prgCmds, [In, Out] IntPtr pCmdText);
[return: MarshalAs(UnmanagedType.I4)]
[PreserveSig]
int Exec(ref Guid pguidCmdGroup, int nCmdID, int nCmdexecopt, [In, MarshalAs(UnmanagedType.LPArray)] object[] pvaIn, ref int pvaOut);
} [StructLayout(LayoutKind.Sequential)]
public class OLECMD
{
[MarshalAs(UnmanagedType.U4)]
public int cmdID;
[MarshalAs(UnmanagedType.U4)]
public int cmdf;
public OLECMD()
{
}
} public const int S_FALSE = ;
public const int S_OK = ; public static readonly Guid CGID_DocHostCommandHandler = new Guid("f38bc242-b950-11d1-8918-00c04fc2c836"); public const int VARIANT_TRUE = -;
public const int VARIANT_FALSE = ;
public const int OLECMDERR_E_NOTSUPPORTED = -; } 然后定义一个类ExtendedWebBrowserSite,去实现接口: class ExtendedWebBrowserSite : WebBrowserSite, NativeMethods.IOleCommandTarget
{
/// <summary>
/// Creates a new instance of the <see cref="ExtendedWebBrowserSite"/> class
/// </summary>
/// <param name="host">The <see cref="ExtendedWebBrowser"/> hosting the browser</param>
public ExtendedWebBrowserSite(ExtendedWebBrowser host)
: base(host)
{
_host = host;
} private ExtendedWebBrowser _host;
private ExtendedWebBrowser Host
{
get
{
return _host;
}
} #region IOleCommandTarget Members int NativeMethods.IOleCommandTarget.QueryStatus(ref Guid pguidCmdGroup, int cCmds, NativeMethods.OLECMD prgCmds, IntPtr pCmdText)
{
return NativeMethods.S_FALSE;
} int NativeMethods.IOleCommandTarget.Exec(ref Guid pguidCmdGroup, int nCmdID, int nCmdexecopt, object[] pvaIn, ref int pvaOut)
{
int hResult = NativeMethods.S_OK;
if (pguidCmdGroup == null)
return hResult;
// Check for invalid pointers (or get a NullReferenceException on a value type???)
//if (NativeMethods.CGID_DocHostCommandHandler.Equals(pguidCmdGroup))
{
switch (nCmdID)
{
case (int)NativeMethods.OLECMDID.OLECMDID_SHOWSCRIPTERROR:
// Hide the dialog
pvaOut = NativeMethods.VARIANT_TRUE; break;
default:
hResult = NativeMethods.OLECMDERR_E_NOTSUPPORTED;
break;
}
}
return hResult;
} #endregion }
} 当然这里主要是通过实现IOleCommandTarget.Exec方法去捕获OLECMDID.OLECMDID_SHOWSCRIPTERROR消息,然后避免异常。 最后在你的扩展的WebBrowser里面重载一下CreateWebBrowserSiteBase方法,完成调用。 protected override WebBrowserSiteBase CreateWebBrowserSiteBase()
{
return new ExtendedWebBrowserSite(this);
}

原文:http://blog.csdn.net/tangyanzhi1111/article/details/8990459