WebView2控件应用详解系列博客

.NET桌面程序集成Web网页开发的十种解决方案

.NET混合开发解决方案1 WebView2简介

.NET混合开发解决方案2 WebView2与Edge浏览器的区别

.NET混合开发解决方案3 WebView2的进程模型

.NET混合开发解决方案4 WebView2的线程模型

.NET混合开发解决方案5 WebView2运行时与分发应用

.NET混合开发解决方案7 WinForm程序中通过NuGet管理器引用集成WebView2控件

.NET混合开发解决方案8 WinForm程序中通过设置固定版本运行时的BrowserExecutableFolder属性集成WebView2控件

  在我的博客《.NET混合开发解决方案7 WinForm程序中集成WebView2》中介绍了WinForm程序中集成WebView2组件的详细步骤以及注意事项。这只是最基本的应用,WebView2功能之所以强大,是因为它提供了很多开放的属性与事件供开发者调用以完成复杂的功能。具体可以参考我的博客《.NET混合开发解决方案2 WebView2与Edge浏览器的区别》

  本文介绍WebView2应用程序的导航事件。当WebView2实例中显示的内容发生特定的异步操作时,导航事件会运行。例如,当WebView2用户导航到新网站时,本机内容(WinForm、WPF、Win32、WinUI)通过侦听 NavigationStarting 事件来侦听更改。导航操作完成后,NavigationCompleted 事件将运行。

导航事件的正常顺序为:

  1. NavigationStarting
  2. SourceChanged
  3. ContentLoading
  4. HistoryChanged
  5. BasicAuthenticationRequested
  6. DOMContentLoaded
  7. NavigationCompleted

以下事件描述每次导航操作期间 WebView2 的状态:

上图显示了在各自的事件参数上具有相同NavigationId属性的导航事件。

  • 使用导航ID(在NavigationId事件中提供)跟踪每个新文档的导航事件。每次成功导航到新文档时,WebView2的NavigationId事件都会发生更改。
  • 具有不同NavigationId事件实例的导航事件可能会重叠。例如,启动导航事件时,必须等待相关的NavigationStarting事件。如果随后启动另一个导航,您将看到以下序列:
    1. 第一次导航的 NavigationStarting 事件。
    2. 第二次导航的 NavigationStarting 事件。
    3. 第一次导航的 NavigationCompleted 事件。
    4. 第二次导航的所有其他相应导航事件。
  • 在错误情况下,可能有或可能没有内容加载事件,这取决于导航是否继续导航到错误页面。
  • 如果发生HTTP重定向,则一行中有多个NavigationStarting事件,其中后面的事件参数设置了IsRedirect属性;但是,NavigationId事件保持不变。
  • 相同的文档导航事件(例如导航到同一文档中的片段)不会导致NavigationStarting事件,也不会增加NavigationId事件。
  • 要监视或取消WebView2实例中子框架内的导航事件,请使用FrameNavigationStarting和FrameNavigationCompleted事件。这些事件的行为类似于等效的非框架对应事件。

当在文本框中输入目标网址后,点击【导航】按钮,具体执行逻辑如下

发生错误时,会引发以下事件,这可能取决于对错误网页的导航:

  • SourceChanged
  • ContentLoading
  • HistoryChanged

如果发生HTTP重定向,则一行中有多个NavigationStarting事件。

示例演示

先看一个效果动画

逻辑代码

  1 public partial class Frm2Navigation : Form
2 {
3 public Frm2Navigation()
4 {
5 InitializeComponent();
6
7 InitializeAsync();
8
9 webView2.CoreWebView2InitializationCompleted += WebView2_CoreWebView2InitializationCompleted;
10
11 webView2.NavigationStarting += WebView2_NavigationStarting;
12 webView2.NavigationCompleted += WebView2_NavigationCompleted;
13 }
14
15 async void InitializeAsync()
16 {
17 await webView2.EnsureCoreWebView2Async(null);
18 }
19
20 private void WebView2_CoreWebView2InitializationCompleted(object? sender, CoreWebView2InitializationCompletedEventArgs e)
21 {
22 if (e.IsSuccess == false)
23 {
24 MessageBox.Show("WebView2_CoreWebView2InitializationCompleted 事件,发生异常。"
25 + Environment.NewLine + e.InitializationException.Message,
26 "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
27 }
28
29 webView2.SourceChanged += WebView2_SourceChanged;
30 webView2.ContentLoading += WebView2_ContentLoading;
31 webView2.CoreWebView2.HistoryChanged += CoreWebView2_HistoryChanged;
32 webView2.CoreWebView2.DOMContentLoaded += CoreWebView2_DOMContentLoaded;
33
34 webView2.CoreWebView2.BasicAuthenticationRequested += CoreWebView2_BasicAuthenticationRequested;
35 webView2.CoreWebView2.ProcessFailed += CoreWebView2_ProcessFailed;
36 }
37
38 private void WebView2_NavigationStarting(object? sender, CoreWebView2NavigationStartingEventArgs e)
39 {
40 string uri = e.Uri;
41 if (!uri.StartsWith("https://"))
42 {
43 //webView2.CoreWebView2.ExecuteScriptAsync($"alert('{uri} 不安全, 请尝试https链接。')");
44
45 DialogResult dr = MessageBox.Show($"{uri} 不安全, 请尝试https链接。\r\n\r\n 确定要访问吗?", "提示", MessageBoxButtons.OKCancel, MessageBoxIcon.Warning);
46 if (dr == DialogResult.Cancel)
47 {
48 e.Cancel = true;
49 }
50 }
51 }
52
53 private void WebView2_SourceChanged(object? sender, CoreWebView2SourceChangedEventArgs e)
54 {
55 //if(e.IsNewDocument)
56 //{
57 // webView2.CoreWebView2.ExecuteScriptAsync("alert('WebView2_SourceChanged事件。= e.IsNewDocument = true')");
58 //}
59
60 MessageBox.Show("WebView2_SourceChanged 事件。"
61 + Environment.NewLine + "e.IsNewDocument = " + e.IsNewDocument,
62 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
63 }
64
65 private void WebView2_ContentLoading(object? sender, CoreWebView2ContentLoadingEventArgs e)
66 {
67 //webView2.CoreWebView2.ExecuteScriptAsync("alert('WebView2_SourceChanged事件。= e.IsNewDocument = true')");
68
69 MessageBox.Show("WebView2_ContentLoading 事件。"
70 + Environment.NewLine + "e.IsErrorPage = " + e.IsErrorPage,
71 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
72 }
73
74 private void CoreWebView2_HistoryChanged(object? sender, object e)
75 {
76 MessageBox.Show("CoreWebView2_HistoryChanged 事件。",
77 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
78 }
79
80 private void CoreWebView2_DOMContentLoaded(object? sender, CoreWebView2DOMContentLoadedEventArgs e)
81 {
82 MessageBox.Show("CoreWebView2_DOMContentLoaded 事件。"
83 + Environment.NewLine + "e.NavigationId = " + e.NavigationId,
84 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
85 }
86
87 private void WebView2_NavigationCompleted(object? sender, CoreWebView2NavigationCompletedEventArgs e)
88 {
89 MessageBox.Show("WebView2_NavigationCompleted 事件。"
90 + Environment.NewLine + "e.NavigationId = " + e.NavigationId
91 + Environment.NewLine + "e.IsSuccess = " + e.IsSuccess
92 + Environment.NewLine + "e.WebErrorStatus = " + e.WebErrorStatus,
93 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
94 }
95
96
97 private void CoreWebView2_ProcessFailed(object? sender, CoreWebView2ProcessFailedEventArgs e)
98 {
99 MessageBox.Show("CoreWebView2_ProcessFailed 事件。"
100 + Environment.NewLine + "e.ExitCode = " + e.ExitCode
101 + Environment.NewLine + "e.FrameInfosForFailedProcess = " + e.FrameInfosForFailedProcess
102 + Environment.NewLine + "e.ProcessDescription = " + e.ProcessDescription
103 + Environment.NewLine + "e.ProcessFailedKind = " + e.ProcessFailedKind
104 + Environment.NewLine + "e.Reason = " + e.Reason,
105 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
106 }
107
108
109 private void CoreWebView2_BasicAuthenticationRequested(object? sender, CoreWebView2BasicAuthenticationRequestedEventArgs e)
110 {
111 MessageBox.Show("CoreWebView2_BasicAuthenticationRequested 事件。"
112 + Environment.NewLine + "e.Uri = " + e.Uri
113 + Environment.NewLine + "e.Cancel = " + e.Cancel
114 + Environment.NewLine + "e.Challenge = " + e.Challenge
115 + Environment.NewLine + "e.Response = " + e.Response,
116 "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
117
118 }
119
120
121 // 访问url
122 private void btnGo_Click(object sender, EventArgs e)
123 {
124 string rawUrl = txtUrl.Text;
125 if (string.IsNullOrWhiteSpace(rawUrl))
126 {
127 MessageBox.Show("请输入网址。", "提示", MessageBoxButtons.OK, MessageBoxIcon.Warning);
128
129 txtUrl.Focus();
130
131 return;
132 }
133
134 Uri uri;
135
136 if (Uri.IsWellFormedUriString(rawUrl, UriKind.Absolute))
137 {
138 uri = new Uri(rawUrl);
139 }
140 else if (rawUrl.Contains(" ") == false && rawUrl.Contains(".") == false)
141 {
142 // 无效的URI包含一个点且没有空格,请尝试在前面添加http://标记
143 uri = new Uri("http://" + rawUrl);
144 }
145 else if (rawUrl.StartsWith("http://") == false)
146 {
147 uri = new Uri("http://" + rawUrl);
148 }
149 else
150 {
151 // 其他情况将其视为网络搜索
152 uri = new Uri("https://bing.com/search?q=" + string.Join("+", Uri.EscapeDataString(rawUrl).Split(new string[] { "%20" }, StringSplitOptions.RemoveEmptyEntries)));
153 }
154
155 //webView2.CoreWebView2.Navigate(url.Trim());
156 webView2.Source = uri;
157 }
158 }

注意事项

CoreWebview2的相关事件必须在它的 CoreWebView2InitializationCompleted 事件(CoreWebView2对象初始化完成后事件)里面注册

  • webView2.CoreWebView2.HistoryChanged
  • webView2.CoreWebView2.DOMContentLoaded
  • webView2.CoreWebView2.BasicAuthenticationRequested
  • webView2.CoreWebView2.ProcessFailed

如上第31、32、34、35行。

.NET混合开发解决方案9 WebView2控件的导航事件的更多相关文章

  1. .NET混合开发解决方案10 WebView2控件调用网页JS方法

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  2. .NET混合开发解决方案15 WebView2控件集成到WinForm程序编译后的文件及结构说明

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  3. .NET混合开发解决方案11 WebView2加载的网页中JS调用C#方法

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  4. .NET混合开发解决方案14 WebView2的基本身份验证

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  5. .NET混合开发解决方案24 WebView2对比CefSharp的超强优势

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  6. 混合开发 webview 中file 控件 点击后无反应解决方法

    最近在做个项目 ,需要 使用 file 控件上传 图片到服务器 ,在手机浏览器中 可以正常选择照片,但是放到 android 应用中的webview中,file 控件点击后就没有反应. 百度了一番后, ...

  7. .NET混合开发解决方案5 WebView2运行时与分发应用

    系列目录     [已更新最新开发文章,点击查看详细] 发布使用Microsoft Edge WebView2的应用程序时,客户端计算机上需要安装WebView2运行时,可以安装自动更新的Evergr ...

  8. .NET混合开发解决方案13 自定义WebView2中的上下文菜单

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  9. .NET混合开发解决方案16 管理WebView2的用户数据

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

随机推荐

  1. 什么是编织(Weaving)?

    为了创建一个 advice 对象而链接一个 aspect 和其它应用类型或对象,称为编 织(Weaving).在 Spring AOP 中,编织在运行时执行.

  2. 学习Puppet(二)

    puppet的工作流程 1.简介 puppet是一种采用C/S星状结构的linux.Unix平台的集中配置管理系统.puppet拥有自己的语言,可管理配置文件.用户.cron任务.软件包.系统服务等. ...

  3. React 可视化开发工具 Shadow Widget 非正经入门(之六:markdown)

    本系列博文从 Shadow Widget 作者的视角,解释该框架的设计要点.本篇讲解 Markdown 在 Shadow Widget 中的应用. Markdown 在 Shadow Widget 中 ...

  4. vue和mint-ui loadMore 实现上拉加载和下拉刷新

    首先安装mint-ui组件库 npm install mint-ui 在main.js中引入mint-ui和样式 import 'mint-ui/lib/style.css' import MintU ...

  5. .NET程序设计实验三

    实验三  Windows 应用程序开发 一.实验目的 1. 掌握窗口控件的使用方法: 2. 掌握Windows 的编程基础. 二.实验要求 根据要求,编写 C#程序,并将程序代码和运行结果写入实验报告 ...

  6. PAT B1015德才论

    题目描述: 宋代史学家司马光在<资治通鉴>中有一段著名的"德才论":"是故才德全尽谓之圣人,才德兼亡谓之愚人,德胜才谓之君子,才胜德谓之小人.凡取人之术,苟不 ...

  7. 升级DLL plugin 到AutoDllPlugin

    为了使打包构建速度加快使用的DLLPlugin,但是我们还是需要手动把dll文件引入文件, HTMLwebpackplugin 结合autoDLLplugin可以自动引入打包文件, 十份地方便

  8. HDMS(Headend Device Management System)软件下载

    进入官网http://www.pbi-china.com/CHS/index.aspx点击右下角的下载通道.

  9. docker下将容器按照端口号分配

    问题情境:现在有一个服务器主机,安装了docker,想给成员分配各自的容器,但不想成员通过宿主机进入容器.那么成员如何直接访问容器呢? 成员可以通过ip加端口号访问 因此,需要生成一个容器,将容器的2 ...

  10. xtrabackup备份和恢复数据脚本

    该脚本用于备份和恢复MySQL数据库. 总结xtrabackup备份的两个坑: 1.在恢复数据的过程中,如果中途出错,则数据将会被破坏,后续很难再恢复. 2.在恢复过程中,如果版本过低,在准备全量数据 ...