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