cp网站建设,西安建设银行网站,wordpress开cdn好吗,凡科建站自助建站平台大家好#xff0c;我是 17。
已经有很多关于 Flutter WebView 的文章了#xff0c;为什么还要写一篇。两个原因#xff1a;
Flutter WebView 是 Flutter 开发的必备技能现有的文章都是关于老版本的#xff0c;新版本 4.x 有了重要变化#xff0c;基于 3.x 的代码很多要重…
大家好我是 17。
已经有很多关于 Flutter WebView 的文章了为什么还要写一篇。两个原因
Flutter WebView 是 Flutter 开发的必备技能现有的文章都是关于老版本的新版本 4.x 有了重要变化基于 3.x 的代码很多要重写。
WebView 的文章分两篇
在 Flutter 中使用 webview_flutter 4.0 | js 交互Flutter WebView 性能优化让 h5 像原生页面一样优秀
本篇讲 js 交互。首先了解下 4.0 有哪些重大变化。
最大的变化就是 WebView 类已被删除其功能已拆分为 WebViewController 和 WebViewWidget。让我们可以提前初始化 WebViewController。Android 的 PlatformView 的实现目前不再可配置。它在版本 23 上使用 Texture Layer Hybrid Compositiond在版本 19-23 回退到 Hybrid Composition。
第 2 条的变化让我们不需要再写判断 android 的代码了。
还有 api 的变化。总的来说让我们的编码更加容易了。
写本文的时候Flutter WebView 的版本是 4.0.2
环境准备
虽然文档上写的是支持 addroid SDK 19 or 20 但我们最好写 21 或更高不是说会影响 Flutter WebView 的使用而是太低了会影响其它插件的使用。如果能写 23 就更好了这样可以用 Texture Layer Hybrid Compositiond 了。
android {defaultConfig {minSdkVersion 21}
}iOS 支持 9.0 以上新版本的 flutter 默认配置是 ios 11.0 所以我们按 Flutter 默认的配置就好。
安装 webview_flutter
flutter pub add webview_flutter最简示例
一般举例都是先发一个 hello world咱们也发一个最简单的先跑起来。
完整代码贴到 main.dart 就能运行
引用 webview_flutter 插件创建 controller用 WebViewWidget 展示内容
import package:flutter/material.dart;
import package:webview_flutter/webview_flutter.dart;const htmlString
!DOCTYPE html
head
titlewebview demo | IAM17/title
meta nameviewport contentwidthdevice-width, initial-scale1.0, maximum-scale1.0, user-scalableno,viewport-fitcover /
style
*{margin:0;padding:0;
}
body{background:#BBDFFC; display:flex;justify-content:center;align-items:center;height:100px;color:#C45F84;font-size:20px;
}
/style
/head
html
body
div 大家好我是 17/div
/body
/html
;void main() {runApp(const MyApp());
}class MyApp extends StatelessWidget {const MyApp({super.key});overrideWidget build(BuildContext context) {return const MaterialApp(home: Scaffold(body: SafeArea(child: MyWebView()),));}
}class MyWebView extends StatefulWidget {const MyWebView({super.key});overrideStateMyWebView createState() _MyWebViewState();
}class _MyWebViewState extends StateMyWebView {late final WebViewController controller;double height 0;overridevoid initState() {controller WebViewController()..setJavaScriptMode(JavaScriptMode.unrestricted)..loadHtmlString(htmlString);super.initState();}overrideWidget build(BuildContext context) {return Column(children: [Expanded(child: WebViewWidget(controller: controller))],);}
}执行代码你将看到如下内容 WebView 内容的可以通过网址获取但这样不方便演示各种效果所以直接用 htmlString 替代了效果是一样的。
默认情况下 javascript 是被禁用的。必须手动开启 setJavaScriptMode(JavaScriptMode.unrestricted)否则对于绝大多数的网页都没法用了。
WebView 的小大
WebViewWidget 会尝试让自己获得最大高度和最大宽度所以 WebView 必须放在有限宽度和有限高度的 Widget 中。一般会用 SizedBox 这样的容器把 WebView 包起来。但是 WebView 内容的高度是未知的要如何设置 SizedBox 的 height 呢
一种方案是 height 采用固定高度如果 WebView 内容过多可以用上下滑动的方式来查看所有内容。如果 WebView 的内容高度是变化的用固定高度可能会产生大块空白这个时候应该把 height 设置成 WebView 内容的高度。
那么问题来了如何获得 WebView 内容的高度最理想的情况是网页是自己能控制的让网页自己报告高度。
网页自己报告高度
在 htmlString 中 增加 js
body
div classcontent大家好我是 17/div
scriptconst resizeObserver new ResizeObserver(entries Report.postMessage(document.scrollingElement.scrollHeight))resizeObserver.observe(document.body)
/script
/body 如果WebView 不支持 ResizeObserver 可以直接在合适的时机调用 Report.postMessage(document.scrollingElement.scrollHeight)) dart 代码中
增加一个变量 height 初始值为 0。增加 ScriptChannel注意名字和前面 script 中的名字必须一样本例中名字叫 Report用 SizedBox 替换 Expanded限定 WebViewWidget 的高度。
class _MyWebViewState extends StateMyWebView {late final WebViewController controller;double height 0;overridevoid initState() {controller WebViewController()..setJavaScriptMode(JavaScriptMode.unrestricted)..addJavaScriptChannel(Report, onMessageReceived: (message) {setState(() {height double.parse(message.message);});})..loadHtmlString(htmlString);super.initState();}overrideWidget build(BuildContext context) {return Column(children: [SizedBox(height: height, child: WebViewWidget(controller: controller)),],);}
}修改 html 代码中的 body 的样式 height:100px 为 height:200px;重新运行代码restarthot reload 不生效 发现 SizedBox 也变为 200px 高了。
无法修改页面
如果页面我们无权修改也没有办法协调修改那就只能通过注入 js 方式获取了。
如果页面的高度只由静态 css 决定可以简单的加一个小延时直接获取高度即可。
controller.setNavigationDelegate(NavigationDelegate(onPageFinished: (url) async {await Future.delayed(Duration(milliseconds: 50));var message await controller.runJavaScriptReturningResult(document.scrollingElement.scrollHeight);setState(() {height double.parse(message.toString());});},));如果页面加载完成后 js 又对页面进行了修改这个时间就很难预估了。js 可以随时修改页面导致高度改变所以要想时时跟踪页面高度只能靠监听。如果 webview 不支持 ResizeObserver还可以用 setInterval。 void initState() {controller WebViewController()..setJavaScriptMode(JavaScriptMode.unrestricted)..addJavaScriptChannel(Report, onMessageReceived: (message) {var msgHeight double.parse(message.message);setState(() {height msgHeight;});})..setNavigationDelegate(NavigationDelegate(onPageFinished: (url) async {// 注入 jscontroller.runJavaScript(const resizeObserver new ResizeObserver(entries Report.postMessage(document.scrollingElement.scrollHeight))resizeObserver.observe(document.body));},))..loadHtmlString(htmlString);super.initState();}必须等到页面加载完成后再注入 js否则页面文档还不存在往哪里注入啊。
因为代码都在 dart 这边免去了和页面开发沟通的成本。既使 WebView 加载的页面中可能还有链接跳到另一个地址js 注入的代码依然有效
页面的高度可能会在很短时间内连续变化我们可以只对最后一次的高度变化做更新用 Timer 可以做到。页面高度要限制一个最大值否则超出最大允许的高度就报错了。
可能你会觉得既然注入的方式这么多优点不需要页面报告那种方式了都用这种注入的方式就可以了。实际上每种方式都有它的利弊不然我就不会介绍了。页面报告的方式在于灵活想什么时候报告就什么时候报告页面高度变化了也可以不报告。在页面没有内容的时候可以先报告一个预估的高度会让页面避免从 0 开始突然变高。尽量把主动权交给页面因为页面是可以随时修改的app 不能
在网页中调用 Flutter 页面
拦截 url
url 以 /android 结尾时跳到对应的原生页面。否则继续原来的请求。
onNavigationRequest: (request) {if (request.url.endsWith(/android)) {// 跳到原生页面return NavigationDecision.prevent;} else {// 继续原来的请求return NavigationDecision.navigate;}},触发方式有两种
用 A 标签 a href/ios跳到 Flutter 页面/a用 js 跳转 window.location.href完整页面地址
用 js 跳转的地址一定是完整的页面地址。比如这样写都是可以的
https://juejin.cnaa:/bb
schema 可以自定义但不能没有。这样写是无效的 /android
js 调用 JavaScriptChannel 定义的方法
先定义跳转的通道对象为 Jump void initState() {controller WebViewController()..setJavaScriptMode(JavaScriptMode.unrestricted)..addJavaScriptChannel(Jump, onMessageReceived: (message) {//根据 message 信息跳转})..loadHtmlString(htmlString);super.initState();}在页面中执行 Jump.postMessage(video);
实际上flutter 拿到页面传过来的信息后除了可以跳转到 flutter 页面还可以执行其它功能比如调取相机。
总结
通过两个示例演示了页面与 flutter 通信的 3 种方式
flutter 拦截 urlflutter 设置 JavaScriptChannelflutter 向页面注入 js
向页面注入 js 需要等页面加载完成后再注入。注入 js 的能力非常强大的。几乎可以对页面做任意修改。比如
删除页面中不想要的部分修改页面的样式增加页面的功能比如给页面增加一个按钮点按钮跳到原生页面就好像原来的页面就有这个功能一样。
删除页面中不想要的部分这是有实际意义的。页面都会有页头这可能和 app 的头部冲突。有了注入 js 这个利器可以在不修改页面的情况下直接在 app 中不显示页头。
修改页面样式这个你懂的既然能注入 js 也就是能注入 css 了。相比于直接用 js 修改页面样式注入 css 的方式更加容易维护。
当然了凡事有利有弊不要滥用这个功能。在 app 单方面修改页面将来页面修改的时候可能会翻车即使做好沟通也会给页面开发造成限制或麻烦所以如何做一定要权衡各方面的得失。
app 不像页面那样可以随时修改所以要优先考虑让页面实现功能尽量把控制权交给页面说两遍了因为很重要。js 注入这种操作不是万不得已不要做把它做为最后的选项。
最后说一点示例中为了方便演示用 loadHtmlString实际应用中一般是用 loadRequest 加载网址。
loadHtmlString(htmlString) loadRequest(Uri.parse(https://juejin.cn))
本文到这里就结束了。谢谢观看 文章转载自: http://www.morning.lylkh.cn.gov.cn.lylkh.cn http://www.morning.hqrkq.cn.gov.cn.hqrkq.cn http://www.morning.qwpyf.cn.gov.cn.qwpyf.cn http://www.morning.bpmdh.cn.gov.cn.bpmdh.cn http://www.morning.lltdf.cn.gov.cn.lltdf.cn http://www.morning.gcysq.cn.gov.cn.gcysq.cn http://www.morning.jjzjn.cn.gov.cn.jjzjn.cn http://www.morning.dtzxf.cn.gov.cn.dtzxf.cn http://www.morning.hbdqf.cn.gov.cn.hbdqf.cn http://www.morning.qjzgj.cn.gov.cn.qjzgj.cn http://www.morning.tzjqm.cn.gov.cn.tzjqm.cn http://www.morning.wdpbq.cn.gov.cn.wdpbq.cn http://www.morning.qpsft.cn.gov.cn.qpsft.cn http://www.morning.xpmhs.cn.gov.cn.xpmhs.cn http://www.morning.kbynw.cn.gov.cn.kbynw.cn http://www.morning.ltqzq.cn.gov.cn.ltqzq.cn http://www.morning.zgnng.cn.gov.cn.zgnng.cn http://www.morning.zrkws.cn.gov.cn.zrkws.cn http://www.morning.gmgnp.cn.gov.cn.gmgnp.cn http://www.morning.pnntx.cn.gov.cn.pnntx.cn http://www.morning.zfhzx.cn.gov.cn.zfhzx.cn http://www.morning.wqfj.cn.gov.cn.wqfj.cn http://www.morning.zdnrb.cn.gov.cn.zdnrb.cn http://www.morning.npfkw.cn.gov.cn.npfkw.cn http://www.morning.rqhdt.cn.gov.cn.rqhdt.cn http://www.morning.bmmhs.cn.gov.cn.bmmhs.cn http://www.morning.lhgqc.cn.gov.cn.lhgqc.cn http://www.morning.chmkt.cn.gov.cn.chmkt.cn http://www.morning.rjjys.cn.gov.cn.rjjys.cn http://www.morning.yzdth.cn.gov.cn.yzdth.cn http://www.morning.dwxqf.cn.gov.cn.dwxqf.cn http://www.morning.rbxsk.cn.gov.cn.rbxsk.cn http://www.morning.rykmz.cn.gov.cn.rykmz.cn http://www.morning.hlrtzcj.cn.gov.cn.hlrtzcj.cn http://www.morning.ryznd.cn.gov.cn.ryznd.cn http://www.morning.mqmxg.cn.gov.cn.mqmxg.cn http://www.morning.fwcjy.cn.gov.cn.fwcjy.cn http://www.morning.rbkml.cn.gov.cn.rbkml.cn http://www.morning.zlrsy.cn.gov.cn.zlrsy.cn http://www.morning.wblpn.cn.gov.cn.wblpn.cn http://www.morning.bkgfp.cn.gov.cn.bkgfp.cn http://www.morning.jqswf.cn.gov.cn.jqswf.cn http://www.morning.qmzhy.cn.gov.cn.qmzhy.cn http://www.morning.nkkpp.cn.gov.cn.nkkpp.cn http://www.morning.kryr.cn.gov.cn.kryr.cn http://www.morning.sphft.cn.gov.cn.sphft.cn http://www.morning.nxrgl.cn.gov.cn.nxrgl.cn http://www.morning.yrjhr.cn.gov.cn.yrjhr.cn http://www.morning.bhgnj.cn.gov.cn.bhgnj.cn http://www.morning.dfdhx.cn.gov.cn.dfdhx.cn http://www.morning.wrbnh.cn.gov.cn.wrbnh.cn http://www.morning.yuanshenglan.com.gov.cn.yuanshenglan.com http://www.morning.mjtgt.cn.gov.cn.mjtgt.cn http://www.morning.fplwz.cn.gov.cn.fplwz.cn http://www.morning.xkzr.cn.gov.cn.xkzr.cn http://www.morning.qytby.cn.gov.cn.qytby.cn http://www.morning.bpmdh.cn.gov.cn.bpmdh.cn http://www.morning.zbtfz.cn.gov.cn.zbtfz.cn http://www.morning.prprz.cn.gov.cn.prprz.cn http://www.morning.dzfwb.cn.gov.cn.dzfwb.cn http://www.morning.grbp.cn.gov.cn.grbp.cn http://www.morning.slqgl.cn.gov.cn.slqgl.cn http://www.morning.xnqwk.cn.gov.cn.xnqwk.cn http://www.morning.qfrmy.cn.gov.cn.qfrmy.cn http://www.morning.kqzxk.cn.gov.cn.kqzxk.cn http://www.morning.tqhpt.cn.gov.cn.tqhpt.cn http://www.morning.ghjln.cn.gov.cn.ghjln.cn http://www.morning.gmdtk.cn.gov.cn.gmdtk.cn http://www.morning.lswgs.cn.gov.cn.lswgs.cn http://www.morning.ndxss.cn.gov.cn.ndxss.cn http://www.morning.yrccw.cn.gov.cn.yrccw.cn http://www.morning.kphsp.cn.gov.cn.kphsp.cn http://www.morning.rlfr.cn.gov.cn.rlfr.cn http://www.morning.rwhlf.cn.gov.cn.rwhlf.cn http://www.morning.wnhml.cn.gov.cn.wnhml.cn http://www.morning.rzysq.cn.gov.cn.rzysq.cn http://www.morning.spxk.cn.gov.cn.spxk.cn http://www.morning.rmfwh.cn.gov.cn.rmfwh.cn http://www.morning.tgfsr.cn.gov.cn.tgfsr.cn http://www.morning.jmllh.cn.gov.cn.jmllh.cn