修复flutter_webview_plugin在页面滑出时web图层残留的问题

前言

目前pub上关于webview有两个点赞最多的插件,

webview_flutter 和 flutter_webview_plugin

经过一番比较选择了后者:flutter_webview_plugin,这里将记录写出来,希望对你有所帮助

两者区别

webview_flutter :

flutter官方开发维护,采用的platformView显示。

受flutter端控制(在树内),对于页面过渡动画是可协调,受控制的。

flutter_webview_plugin :

flutter 社区开发维护,采用的是原生端添加渲染的方式。

因为是原生端绘制,不在flutter 树内,不受其控制,显示和隐藏是需要methodChannel进行通知的。

看起来前者要比后者灵活方便,但是唯一也是最严重的扣分项就是性能问题 :

webview_flutter 的性能要明显弱于 flutter_webview_plugin,其所造成的卡顿是肉眼可见,不需要看什么fps、dumpsy啥的...尤其是稍微复杂一些的页面。

基于此我选择了flutter_webview_plugin,当然它也有不足。

flutter_webview_plugin

遇到的问题

由于其本身是采用原生端渲染(以安卓为例,是通过addContentView(webview)),因此其不在flutter 的widget树内,也就无从谈起flutter对其的控制了。

那么当我们的页面采用了过渡动画,如滑动进入/退出,由于flutter 页面在没有走完过渡动画时,是不会真正退出的(走dispose),而插件的显隐和释放是在页面的dispose中才进行的,这就导致了,背景虽然滑出去了(或者漏出了上层页面),但是webview的内容依然残留了一会才消失。

问题演示

image

问题分析

查看了flutter_webview_plugin的源码,它的ui结构和运行流程如下图

image

代码大致结构

    class _WebviewScaffoldState extends state{
        widget build(){
            return Scaffold(
                body:_WebviewPlaceholder(
                    onRectChanged:(Rect rect){
                        webviewReference.launch(
                            rect:rect
                            ...
                        );
                    }
                )
            );
        }
    }

在创建的renderBox的paint方法调用后,就会回调onRectChanged 这个方法并携带显示区域rect,然后通过

webviewReference.launch 启动原生端的view添加绘制,绘制区域基于所传的rect。

webviewReference extends FlutterWebviewPlugin 这个类是一个通信类,

这个类还对外暴露了一个resize方法用于在rect改变时进行相应的调整
  /// resize webview
  Future<Null> resize(Rect rect) async {
    final args = {};
    args['rect'] = {
      'left': rect.left,
      'top': rect.top,
      'width': rect.width,
      'height': rect.height,
    };
    await _channel.invokeMethod('resize', args);
  }

经过上面的分析,只要我们改动这个rect就可以改变webview的显示位置和大小。

首先我想到的是对页面做动画的PageRouteBuilder;

初版解决方案

经过对PageRouteBuilder这类的源码一层一层分析后

PageRouteBuilder 嵌套极多,同时我还捎带了看了一下push方法,所得的大致的流程图我放在文章结尾,有兴趣的可以看一下

发现通过builder.animation可以对过渡动画进行监听

              SlideRightRouteBuilder builder = SlideRightRouteBuilder(ComplexPage());
              Navigator.of(context).push(builder);
              ///要放在push后面,不然报错,原因见文章末尾的流程图
              builder.animation.addListener(() {

              });

那么我给ComplexPage传入一个 key,通过这个获取context,进而取到它的offset,然后在回调函数中执行以下操作

final RenderBox box = context.findRenderObject();
final Offset offset = box.localToGlobal(Offset.zero);

这个offset就是包裹webview的那个父widget,它是在widget树上,受动画控制的,换言之随着动画的进行,这个offset也会变化。

之后我们只需要调用

webviewReference.resize(_rect.shift(offset));

在这个过程中,因为builder和resize分别在不同的widget(页面),只能通过各种接口传输/调用,这样就发生了严重的耦合,在考虑需要兼容 滑动/缩放动画,并pr到插件仓库后,便直接放弃了这个方法。

终版解决方案-兼容滑动/缩放

重新思考,发现对于builder的依赖,只是对animation的监听,并触发重绘(resize),对于进度值,完全可以通过其他方法解决。所以便有了下面的方案。

首先我在插件的通信类FlutterWebviewPlugin,定义了支持的过渡到动画类型

/// the transition animation type of page on/off screen
enum TransitionType{
  Non,
  Slide,
  Scale
}

之后在插件WebviewScaffold的构造函数中增加了对应的参数

final TransitionType transitionType;

在state的initState()中调用我创建的方法:

perceptionPageTransition();
  /// coordinate the webview rect whit page's transition
 void perceptionPageTransition(){
   if(widget.transitionType != TransitionType.Non){
     WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
       //avoid to concurrent modification exception
       WidgetsBinding.instance.addPersistentFrameCallback((timeStamp) {
         if(context != null){
           driveWebView();
         }
       });
     });
   }
 }

通过上面这个方法,我就可以模拟出builder.animation的监听了。再看driveWebView()方法


  void driveWebView(){
   final RenderBox box = context.findRenderObject();
   final Offset offset = box.localToGlobal(Offset.zero);
   //获得可绘制的大小
   final Size size = box.size;
   //获得可绘制的区域
   final Rect rect = box.paintBounds;
   
   //当变动位置等于绘制区域的位置时,说明动画已经执行完毕,直接退出,避免过度绘制
   if(offset.dx == rect.left)return;
   //这个值用于缩放动画
   //根据当前位置的dx值/除以size的宽度,就可以计算出动画进度value
   final double value = offset.dx/size.width;
   //根据传入的动画类型,对rect进行位移或者缩放
   switch(widget.transitionType){
     case TransitionType.Slide:
       webviewReference.resize(_rect.shift(offset));
       break;
     case TransitionType.Scale:
       final double www = box.size.width*(value*2);
       final double hhh = box.size.height*(value*2);
       webviewReference.resize(Rect.fromLTWH(offset.dx,offset.dy,size.width-www , size.height-hhh));
       break;
     case TransitionType.Non:
       // TODO: Handle this case.
       break;
   }
 }

这样我们就完成了初版的功能,同时使插件和项目进行了解耦。

效果图

debug模式下,第一次过度会有错位之后正常
性能模式和release 则完全正常
image
image

分析时记录的一些流程图

.push()

[图片上传失败...(image-8f4dbc-1598493593844)]

pageRouteBuilder

image

结语

希望以上对你有所帮助,如果不足之处欢迎指出,喜欢的点个赞撒 ;)

该项目已经PR,大佬正在帮忙审核,不知道什么时候合并。

Fork的仓库地址

我的其它文章

Bedrock——基于MVVM+Provider的Flutter快速开发框架

Flutter自定义View——仿高德三级联动Drawer

Flutter 自定义View——仿同花顺自选股列表

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