熙熙-WebBrowser判断登录成功-WebBrowser-404错误-500错误-屏蔽消息窗口-Webbrowser判断是否加载成功
2011-06-24 21:11
218 查看
在这里,我来讲讲怎么应付WebBrowser里面的那点破事:
文档真的完成了吗?
也许你和我一样,打开一个页面,听到了不少次的哒哒哒哒的声音,但是单个页面怎么会造成多次的触发DocumentCompleted函数?不信自己试试,判断文档是否真的完成,需要在这个处理函数内判断ReadyState如下:
[code],WebBrowserDocumentCompletedEventArgse)
[/code]
404错误怎么办?500错误怎么办?
正如你所猜测的,即便是页面浏览错了,在页面中还是会显示内容,还是会无耻的触发DocumentCompleted事件,那么怎么判断到底是404页面还是500页面呢?看招:
[code]sw.NavigateError+=sw_NavigateError;
[/code]
上面的代码中存在一个很丑陋的转换Enum的操作,我懒得修改了,这个枚举是俺自己定义的,定义的就是错误的名字信息,如下:
[code]{
[/code]
Winform的WebBrowser居然不自己定义错误值,真是一个半成品啊。
让内部的Html调用的JS可以使用外部提供的C#类的函数
做起来简单。首先生成一个类(第一行不能少)
[code]publicclassA
[/code]
构造WebBrowser的时候设置属性ObjectForScripting=newA();
页面中使用类似如下的代码(注意函数名为window.external):
赶紧试试吧
事件调用转换成过程调用
需求是这样的,我需要做一个模拟的操作界面,包括:登录,打开特定页面,填写内容,提交几步。在事件模型之下,我需要先调用Navigate函数,
然后在DocumentCompleted的处理函数中处理每个的返回内容,这样很麻烦,能不能使用一个函数将事件处理直接屏蔽掉
(SynchronizedNavigate),我的操作转换成:
SynchronizedNavigate(登录页面)
填写登录信息
模拟点击登录按钮
SynchronizedNavigate(数据填充页面)
填写页面内容
提交到服务器
取得返回页面校验实际输入的值
是不是很玄妙,其实很简单(注意不能缺少函数Application.DoEvents())
[code]{
[/code]
怎么屏蔽内部的消息窗口?
直接看代码,很无耻滴注册了函数进去
[code]{
[/code]
怎么屏蔽网页内部弹出内容到新窗口?
也许你可以使用Navigate的变种,里面有voidNavigate(stringurlString,boolnewWindow)和Navigate(Uriurl,boolnewWindow)这两个函数,俺没有仔细试过,我使用的是COM接口:
[code]SHDocVw.WebBrowsersw=this.ActiveXInstanceasSHDocVw.WebBrowser;
[/code]
自定义协议?
你是不是想玩玩在<a
href=”personal://username=1235”>用户信息</a>的点击的时候弹出一个内部的页面,而不是默认的
Http访问?实现类似的协议很简单,在_Navigating(objectsender,
WebBrowserNavigatingEventArgs
e)这个事件处理中处理即可:当碰到您的协议的时候,e.Cancel=true;然后生成HTML,设置DocumentText就完成了。
取得当前的选择的文本
高亮指定的文本
[code]{
[/code]
文档真的完成了吗?
也许你和我一样,打开一个页面,听到了不少次的哒哒哒哒的声音,但是单个页面怎么会造成多次的触发DocumentCompleted函数?不信自己试试,判断文档是否真的完成,需要在这个处理函数内判断ReadyState如下:
privatevoidWebBrowserEx_DocumentCompleted(objectsender
[code],WebBrowserDocumentCompletedEventArgse)
{
if(this.ReadyState==WebBrowserReadyState.Complete)
{
if(null!=this.Document&&null!=this.OnDocumentCompleted)
{
this.OnDocumentCompleted(sender,e);
}
}
}
[/code]
404错误怎么办?500错误怎么办?
正如你所猜测的,即便是页面浏览错了,在页面中还是会显示内容,还是会无耻的触发DocumentCompleted事件,那么怎么判断到底是404页面还是500页面呢?看招:
SHDocVw.WebBrowsersw=this.ActiveXInstanceasSHDocVw.WebBrowser;
[code]sw.NavigateError+=sw_NavigateError;
//-----------------------------------------
voidsw_NavigateError(objectpDisp,refobjectURL,refobjectFrame
,refobjectStatusCode,refboolCancel)
{
ErrorCodeserrorcode=ErrorCodes.HTTP_STATUS_BAD_REQUEST;
foreach(ErrorCodesecodeinEnum.GetValues(typeof(ErrorCodes)))
{
if(((long)ecode)==(Int32)StatusCode)
{
errorcode=ecode;
break;
}
}
System.Console.WriteLine("Error"+URL+">>"+errorcode);
}
[/code]
上面的代码中存在一个很丑陋的转换Enum的操作,我懒得修改了,这个枚举是俺自己定义的,定义的就是错误的名字信息,如下:
publicenumErrorCodes:long
[code]{
HTTP_STATUS_BAD_REQUEST=400,
HTTP_STATUS_DENIED=401,
HTTP_STATUS_PAYMENT_REQ=402,
HTTP_STATUS_FORBIDDEN=403,
HTTP_STATUS_NOT_FOUND=404,
HTTP_STATUS_BAD_METHOD=405,
HTTP_STATUS_NONE_ACCEPTABLE=406,
HTTP_STATUS_PROXY_AUTH_REQ=407,
HTTP_STATUS_REQUEST_TIMEOUT=408,
HTTP_STATUS_CONFLICT=409,
HTTP_STATUS_GONE=410,
HTTP_STATUS_LENGTH_REQUIRED=411,
HTTP_STATUS_PRECOND_FAILED=412,
HTTP_STATUS_REQUEST_TOO_LARGE=413,
HTTP_STATUS_URI_TOO_LONG=414,
HTTP_STATUS_UNSUPPORTED_MEDIA=415,
HTTP_STATUS_RETRY_WITH=449,
HTTP_STATUS_SERVER_ERROR=500,
HTTP_STATUS_NOT_SUPPORTED=501,
HTTP_STATUS_BAD_GATEWAY=502,
HTTP_STATUS_SERVICE_UNAVAIL=503,
HTTP_STATUS_GATEWAY_TIMEOUT=504,
HTTP_STATUS_VERSION_NOT_SUP=505,
INET_E_INVALID_URL=0x800C0002L,
INET_E_NO_SESSION=0x800C0003L,
INET_E_CANNOT_CONNECT=0x800C0004L,
INET_E_RESOURCE_NOT_FOUND=0x800C0005L,
INET_E_OBJECT_NOT_FOUND=0x800C0006L,
INET_E_DATA_NOT_AVAILABLE=0x800C0007L,
INET_E_DOWNLOAD_FAILURE=0x800C0008L,
INET_E_AUTHENTICATION_REQUIRED=0x800C0009L,
INET_E_NO_VALID_MEDIA=0x800C000AL,
INET_E_CONNECTION_TIMEOUT=0x800C000BL,
INET_E_INVALID_REQUEST=0x800C000CL,
INET_E_UNKNOWN_PROTOCOL=0x800C000DL,
INET_E_SECURITY_PROBLEM=0x800C000EL,
INET_E_CANNOT_LOAD_DATA=0x800C000FL,
INET_E_CANNOT_INSTANTIATE_OBJECT=0x800C0010L,
INET_E_REDIRECT_FAILED=0x800C0014L,
INET_E_REDIRECT_TO_DIR=0x800C0015L,
INET_E_CANNOT_LOCK_REQUEST=0x800C0016L,
INET_E_USE_EXTEND_BINDING=0x800C0017L,
INET_E_TERMINATED_BIND=0x800C0018L,
INET_E_INVALID_CERTIFICATE=0x800C0019L,
INET_E_CODE_DOWNLOAD_DECLINED=0x800C0100L,
INET_E_RESULT_DISPATCHED=0x800C0200L,
INET_E_CANNOT_REPLACE_SFP_FILE=0x800C0300L,
INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY=0x800C0500L,
INET_E_CODE_INSTALL_SUPPRESSED=0x800C0400L,
}
[/code]
Winform的WebBrowser居然不自己定义错误值,真是一个半成品啊。
让内部的Html调用的JS可以使用外部提供的C#类的函数
做起来简单。首先生成一个类(第一行不能少)
[System.Runtime.InteropServices.ComVisibleAttribute(true)]
[code]publicclassA
{
publicStringFun()
{
returnGuid.NewGuid().ToString();
}
}
[/code]
构造WebBrowser的时候设置属性ObjectForScripting=newA();
页面中使用类似如下的代码(注意函数名为window.external):
<ahref="javascript:{alert(window.external.Fun());}">测试按钮</a>
赶紧试试吧
事件调用转换成过程调用
需求是这样的,我需要做一个模拟的操作界面,包括:登录,打开特定页面,填写内容,提交几步。在事件模型之下,我需要先调用Navigate函数,
然后在DocumentCompleted的处理函数中处理每个的返回内容,这样很麻烦,能不能使用一个函数将事件处理直接屏蔽掉
(SynchronizedNavigate),我的操作转换成:
SynchronizedNavigate(登录页面)
填写登录信息
模拟点击登录按钮
SynchronizedNavigate(数据填充页面)
填写页面内容
提交到服务器
取得返回页面校验实际输入的值
是不是很玄妙,其实很简单(注意不能缺少函数Application.DoEvents())
publicvoidSynchronizedNavigate(StringstrUrl)
[code]{
this.Navigate(strUrl);
WaitNavigatingDone();
}
publicvoidWaitNavigatingDone()
{
while(m_isDocumentationCompleted==false)
{
Application.DoEvents();
Thread.Sleep(50);
}
}
[/code]
怎么屏蔽内部的消息窗口?
直接看代码,很无耻滴注册了函数进去
privatevoidWebBrowserEx_Navigated(objectsender,WebBrowserNavigatedEventArgse)
[code]{
if(this.Document==null||this.Document.DomDocument==null)
{
return;
}
mshtml.IHTMLDocument2doc2=this.Document.DomDocumentasmshtml.IHTMLDocument2;
if(this.BlockClientMessage)
{
if(null!=doc2&&null!=doc2.parentWindow)
{
//blockalertandconfirm
doc2.parentWindow.execScript(@"functionalert(){}functionconfirm(){returntrue;}"
,"javaScript");
}
}
}
[/code]
怎么屏蔽网页内部弹出内容到新窗口?
也许你可以使用Navigate的变种,里面有voidNavigate(stringurlString,boolnewWindow)和Navigate(Uriurl,boolnewWindow)这两个函数,俺没有仔细试过,我使用的是COM接口:
this.Navigate("about:blank");
[code]SHDocVw.WebBrowsersw=this.ActiveXInstanceasSHDocVw.WebBrowser;
if(null!=sw)
{
sw.NewWindow3+=newSHDocVw.DWebBrowserEvents2_NewWindow3EventHandler(sw_NewWindow3);
}
----------------------
voidsw_NewWindow3(refobjectppDisp,refboolCancel,uintdwFlags,stringbstrUrlContext,stringbstrUrl)
{
if(null==OnNewWindow&&this.BlockPopWindow)
{
Cancel=true;
this.Navigate(bstrUrl);
}
elseif(null!=OnNewWindow)
{
OnNewWindow(this,bstrUrl,refCancel);
}
}
[/code]
自定义协议?
你是不是想玩玩在<a
href=”personal://username=1235”>用户信息</a>的点击的时候弹出一个内部的页面,而不是默认的
Http访问?实现类似的协议很简单,在_Navigating(objectsender,
WebBrowserNavigatingEventArgs
e)这个事件处理中处理即可:当碰到您的协议的时候,e.Cancel=true;然后生成HTML,设置DocumentText就完成了。
取得当前的选择的文本
publicstringSelectedText
{
get
{
IHTMLDocument2doc=(IHTMLDocument2)this.Document.DomDocument;
IHTMLTxtRangetxt=(IHTMLTxtRange)doc.selection.createRange();
returntxt.htmlText;
}
}
高亮指定的文本
publicvoidHilightText(stringkeyword,intnindexK)
[code]{
if(null==keyword||
keyword.Trim().Length<1||
null==this.Document||
this.Document.DomDocument==null||
this.IsBusy||
this.IsDisposed
)
{
return;
}
HTMLDocumentdocument=(HTMLDocument)this.Document.DomDocument;
IHTMLDOMNodebodyNode=(IHTMLDOMNode)this.Document.Body.DomElement;
HilightText(document,bodyNode,keyword.Trim(),nindexK);
}
privatevoidHilightText(HTMLDocumentdocument,IHTMLDOMNodenode,stringkeyword,intnindexK)
{
//nodeType=3:text节点
if(node.nodeType==3)
{
stringnodeText=node.nodeValue.ToString();
//如果找到了关键字
if(nodeText.Contains(keyword))
{
IHTMLDOMNodeparentNode=node.parentNode;
//将关键字作为分隔符,将文本分离,并逐个添加到原text节点的父节点
string[]result=nodeText.Split(newstring[]{keyword},StringSplitOptions.None);
for(inti=0;i<result.Length-1;i++)
{
if(result[i]!="")
{
IHTMLDOMNodetxtNode=document.createTextNode(result[i]);
parentNode.insertBefore(txtNode,node);
}
IHTMLDOMNodeorgNode=document.createTextNode(keyword);
IHTMLDOMNodehilightedNode=(IHTMLDOMNode)document.createElement("SPAN");
IHTMLStylestyle=((IHTMLElement)hilightedNode).style;
style.color="black";
style.backgroundColor=colorTables[nindexK%colorTables.Length];
hilightedNode.appendChild(orgNode);
parentNode.insertBefore(hilightedNode,node);
}
if(result[result.Length-1]!="")
{
IHTMLDOMNodepostNode=document.createTextNode(result[result.Length-1]);
parentNode.insertBefore(postNode,node);
}
parentNode.removeChild(node);
}//EndofnodeText.Contains(keyword)
}
else
{
//如果不是text节点,则递归搜索其子节点
IHTMLDOMChildrenCollectionchildNodes=node.childNodesasIHTMLDOMChildrenCollection;
foreach(IHTMLDOMNodeninchildNodes)
{
HilightText(document,n,keyword,nindexK);
}
}
}
[/code]
相关文章推荐
- 熙熙-WebBrowser判断登录成功-WebBrowser-404错误-500错误-屏蔽消息窗口-Webbrowser判断是否加载成功
- WebBrowser判断登录成功-WebBrowser-404错误-500错误-屏蔽消息窗口-Webbrowser判断是否加载成功
- 判断webbrowser页面是否完全加载完毕的方法
- Delphi中WebBrowser判断页面及JS是否加载完成
- c# sqlserver 判断登录是否成功
- 判断ftp是否登录成功
- 页面状态javascript 判断 iframe是否加载成功
- 基于JS判断iframe是否加载成功的方法(多种浏览器)
- Delphi中WebBrowser判断页面及JS是否加载完成
- 从控制台输入用户名和密码, 然后 判断输入的用户名是否是@“Frank”, 密码 是否是 @“lanou”, 如果用户名和密码都正确,则输出登录成功, 否则输出登录失败.
- 在主函数中提示用户输入用户名和密码。另写一方法来判断用户输入是否正确。该方法分别返回一个bool类型的登录结果和和一个string类型的登录信息。如登录成功,返回true及“登录成功”,若登录失败则返回false及“用户名错误”或“密码错误”(使用out参数)
- Delphi中WebBrowser判断页面及JS是否加载完成
- 从控制台输入用户名和密码, 然后 判断输入的用户名是否是@“Frank”, 密码 是否是 @“lanou”, 如果用户名和密码都正确,则输出登录成功, 否则输出登录失败. 提示:
- 判断一个窗口是否被挂起(发WM_NULL消息,或者调用IsHungAppWindow API进行测试)
- js判断iframe是否加载成功
- C#使用Webbrowser来判断网页是否加载完毕
- VB 判断 WebBrowser是否已经加载网页完毕
- ARM Linux判断SD卡是否加载成功
- web 开发,个人中心每个请求,判断用户是否登录,若没有登录,则跳转到登录页面,登录成功后返回之前页面
- 判断Iframe中的网页是否加载成功