Flutter webview 使用和交互

flutter.png

在Flutter中使用webview,在pub.dev中查看,有几个组件都可以使用webview

flutter_webview.png

本文使用的是webview_flutter,它是Flutter团队开发的,目前还不是正式版,但已经可以使用。在iOS中底层调用的是WKWebView,在Android中底层调用的是WebView

webview_flutter 使用
  1. 添加组件 添加webview_flutter
  2. 主要代码
WebView(
   initialUrl: "https://flutterchina.club/",
   //JS执行模式 是否允许JS执行
   javascriptMode: JavascriptMode.unrestricted,
 )
  1. 针对iOS,需要在ios-Runner-info.plist中添加
<key>io.flutter.embedded_views_preview</key>
<string>YES</string>

第三步不可少,否则会报错。

webview.gif
Flutter调用JS

在调用之前 先看一下WebView的其他参数:

WebView创建完成时调用
onWebViewCreated,

要显示的url
initialUrl

JS执行模式 默认是不调用
javascriptMode = JavascriptMode.disabled

使用javascriptChannel JS可以调用Flutter
javascriptChannels

拦截请求
navigationDelegate

手势
gestureRecognizers

页面加载完成
onPageFinished

通过WebViewController调用evaluateJavascript方法,即可调用JS。

了解了每个参数的作用之后,交互就看似简单了。

Flutter调用JS的流程:

  1. WebView创建完成时在onWebViewCreated 中,获取到WebViewController实例。
  2. 当页面加载完成之后,即onPageFinished之后,通过WebViewControllerevaluateJavascript方法调用JS。
  3. evaluateJavascript返回的是Future<String>,通过Future可以获取到JS的返回结果。

示例代码: 在页面加载完成之后,获取网页标题,并显示在导航栏上。

WebView(
    initialUrl: "https://flutterchina.club/",
    //JS执行模式 是否允许JS执行
    javascriptMode: JavascriptMode.unrestricted,
    onWebViewCreated: (controller) {
      _controller = controller;
    },
    onPageFinished: (url) {
        _controller.evaluateJavascript("document.title").then((result){
          setState(() {
            _title = result;
          });
        }
      );
    },
  )

效果:

webview_title.gif
JS调用Flutter

JS调用Flutter主要看剩下的两个参数:

javascriptChannels

navigationDelegate

这两个参数都可以从JS调用Flutter,可以单独使用,也可以混合使用。

javascriptChannels使用:

javascriptChannels类似于往JS中注册方法,这些方法名要和web端约定好。

javascriptChannels参数接受Set<JavascriptChannel>一个成员类型为JavascriptChannel的Set集合。

来看一下JavascriptChannel的用法:

JavascriptChannel(
  name: "share",
  onMessageReceived: (JavascriptMessage message)    {
        print("参数为: ${message.message}");
  }
)

在JS端就可以调用share方法,同时可以传递参数,在Flutter中通过JavascriptMessage可以获取到参数。

navigationDelegate使用:
navigationDelegate: (NavigationRequest request) {
  //对于需要拦截的操作 做判断
  if(request.url.startsWith("myapp://")) {
    print("即将打开 ${request.url}");
    //做拦截处理
    //pushto.... 
    
    Application.push(context, request.url);
    return NavigationDecision.prevent;
  }
  
  //不需要拦截的操作
  return NavigationDecision.navigate;
} ,

例如想要通过webview打开app的原生页面,通过约定好的链接,拦截到指定链接后可跳转到原生页面。

完整示例代码:

import 'package:flutter/cupertino.dart';
import 'package:webview_flutter/webview_flutter.dart';

class WebViewPage extends StatefulWidget {
  @override
  _WebViewPageState createState() => _WebViewPageState();
}

class _WebViewPageState extends State<WebViewPage> {
  WebViewController _controller;
  String _title = "webview";

  @override
  Widget build(BuildContext context) {
    return CupertinoPageScaffold(
        navigationBar: CupertinoNavigationBar(
          middle: Text("$_title"),
        ),
        child: SafeArea(
          child: WebView(
            initialUrl: "https://flutterchina.club/",
            //JS执行模式 是否允许JS执行
            javascriptMode: JavascriptMode.unrestricted,
            onWebViewCreated: (controller) {
              _controller = controller;
            },
            onPageFinished: (url) {
               _controller.evaluateJavascript("document.title").then((result){
                  setState(() {
                    _title = result;
                  });
                }
              );
            },
            navigationDelegate: (NavigationRequest request) {
              if(request.url.startsWith("myapp://")) {
                print("即将打开 ${request.url}");
                
                return NavigationDecision.prevent;
              }
              return NavigationDecision.navigate;
            } ,
            javascriptChannels: <JavascriptChannel>[
              JavascriptChannel(
                name: "share",
                onMessageReceived: (JavascriptMessage message) {
                  print("参数: ${message.message}");
                }
              ),
            ].toSet(),

          ),
        ),
    );
  }
}

可能已经有同学看出来了,JS调用Flutter时,JS获取不到Flutter的返回值,目前还没有找到直接的解决办法,但可以通过迂回的方法解决。

下面提供两种思路:

  1. js调用Flutter时,除了传递业务需要的参数之外,再添加一个callbackname参数
    通过callbackname参数把JS调用Flutter和Flutter返回参数绑定起来。
JavascriptChannel(
       name: "share",
       onMessageReceived: (JavascriptMessage message) {
       print("参数: ${message.message}");
                
        String callbackname = message.message; //实际应用中要通过map通过key获取
                  
        String data = "收到消息调用了";
        String script = "$callbackname($data)";
         _controller.evaluateJavascript(script);
       }
  )
  1. 在网页加载成功后,把JS需要获取的Flutter的值,注入到widow上, 在JS中就可以通过widow获取到相应的值。
String script = "window.isLogin=是否登录";
 _controller.evaluateJavascript(script).then((result){

   }
);
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,311评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,339评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,671评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,252评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,253评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,031评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,340评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,973评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,466评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,937评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,039评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,701评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,254评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,259评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,497评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,786评论 2 345

推荐阅读更多精彩内容