android ScollView 嵌套 WebView 底部空白,高度无法自适应解决

2018-10-21 22:29:24


毕竟webview要自适应的话的场景并不是很多,除非和原生一起使用,所以我选择了检测新的请求跳转新的webview。这样解决的。那个里面只有一个webview,网页自己滚动。


最近要做一个页面,需要 ScrollView 嵌套 WebView,怎么嵌套,怎么解决焦点和 touch 事件冲突,网上一大堆,这里就不赘述了,但是发现 WebView 从一个高度很高的网页加载一个高度很低的网页的时候,高度无法自适应了,造成底部会有一大片的空白,解决方案找到了挺多,描述一下:


Google 官方建议

  Google 建议不要在 ScrollView 中使用 WebView,但是毕竟无法避免,产品的需求;


移除重新添加新的 WebView

  每次加载新的 url 的时候 remove 掉旧的 WebView,重新加入一个新的 WebView 去加载这个新的 url,这种方案是可行的,但是毕竟不太好,效率不高,而且会造成体验上不太好;


JS 注入,获取页面内容高度,设置成 WebView 高度

  JS 注入,通过前端 JS 调用 native 方法,把高度值传递过来,然后通过 layoutparams 去设置 webview 的高度:


mWebView.setWebViewClient(new WebViewClient() {

    @Override

    public void onPageFinished(WebView view, String url) {

        mWebView.loadUrl("javascript:App.resize(document.body.getBoundingClientRect().height)");

        super.onPageFinished(view, url);

    }

});

mWebView.addJavascriptInterface(this, "App");

1

@JavascriptInterface

public void resize(final float height) {

    getActivity().runOnUiThread(new Runnable() {

        @Override

        public void run() {

            //Toast.makeText(getActivity(), height + "", Toast.LENGTH_LONG).show();

            //此处的 layoutParmas 需要根据父控件类型进行区分,这里为了简单就不这么做了

            mWebView.setLayoutParams(new LinearLayout.LayoutParams(getResources().getDisplayMetrics().widthPixels, (int) (height * getResources().getDisplayMetrics().density)));

        }

    });

}

1

2

3

4

5

6

7

8

9

10

11

需要注意的是上面的 resize 方法一定要上面那么写,写成:


mWebView.getLayoutParams().height = (int) (height * getResources().getDisplayMetrics().density);

1

这样是无法成功的,看来宽度也是有影响的


感谢 @imesong 同学,这段代码有亮点可以完善的地方:


mWebVew.setLayoutParam() 时,应该根据 WebView 父控件的类型,创建 LayoutParam,或者直接用 mWebView.getLayoutParams() 获取 LayoutParams;

调用 JAVA 方法的 JS ,可以在网页加载完成时自己加载下面这段 JS 代码(区别在于一个是终端处理,一个是前段处理),比如:

(function(w){

    window.App.resize(document.body.getBoundingClientRect().height);

})(window); 

1

2

3

加载这段 JS 就可以了,“APP” 和上面一样是自己注册是自己注册的接口名字; 

思路是一样的,处理方式有差异。 

试验不太成功的方式

  网上也找到了一些其他的方式,试了一下,不太成功,不知道是不是使用姿势的问题,感兴趣的也可以试试,反馈结果给我。


先加载一个空网页

  在从高网页加载低网页的时候,先加载一个空内容,接着立马做相关的处理操作(比如加载新的网页,更改字体等),


// load empty data to shrink the WebView instance

mArticleWebView.loadUrl(Constants.ASSETS_EMPTY);

// load real data

mArticleWebView.loadDataWithBaseURL(null,data,"text/html","utf-8",null);

1

2

3

4

Constants.ASSETS_EMPTY 仅仅只是一个 String 子串,标识该空网页的位置,所以该变量可以为 “file:///android_asset/Empty.html”. 

试了一下,也是不好用的,不知道是不是我的操作问题, 

原文链接 http://vision-apps.blogspot.hk/2012/08/android-webview-tips-tricks.html


修改 Setting 属性

wv.getSettings().setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);

wv.setVerticalScrollBarEnabled(false);

wv.setVerticalScrollbarOverlay(false);

wv.setHorizontalScrollBarEnabled(false);

wv.setHorizontalScrollbarOverlay(false);

1

2

3

4

5

一样不好用,原文链接http://blog.csdn.net/wht3xr/article/details/21004053


另外一些人的讨论

  一些人的讨论,比较有参考价值:http://stackoverflow.com/questions/15546416/how-to-shrink-webview-size-dynamically-according-to-its-content


总结

  综合考虑来看,第三种方式是比较合适的,我当时也是使用的第三种方式解决的这个问题。


  • 2019-05-21 12:43:26

    (重要)RecycleView的缓存机制

    RecycleView的四级缓存是由三个类共同作用完成的,Recycler、RecycledViewPool和ViewCacheExtension。Recycler用于管理已经废弃或者与RecyclerView分离的ViewHolder,这里面有两个重要的成员,为可以看见的屏幕的内部缓存成员mAttachedScrap、mChangedScrap和滑出屏幕外的外部缓存成员mCachedViews二者共同完成ViewHolder的缓存;RecycledViewPool类是用来缓存整体所有的ViewHolder,是对mCachedViews缓存的补充;ViewCacheExtension是扩展内的缓存对象,默认不加载,需实现方法getViewForPositionAndType(Recycler recycler, int position, int type)来实现自己的缓存。接下来对四级缓存一步步介绍。

  • 2019-05-21 12:44:31

    对嵌套RecyclerView的优化

    RecyclerView 是一个更高级的 ListView ,它可以重用view避免额外创建太多的view从而带来流畅的滚动性能。RecyclerView通过叫做 View pool 的东西持有不再可见的 view ,让它可被回收

  • 2019-05-25 14:54:50

    commit your changes or stash them before you can merge

    Your local changes to the following files would be overwritten by merge:         protected/config/main.php Please, commit your changes or stash them before you can merge. --------------------- 作者:陈小峰_iefreer 来源:CSDN 原文:https://blog.csdn.net/iefreer/article/details/7679631 版权声明:本文为博主原创文章,转载请附上博文链接!

  • 2019-05-27 10:43:34

    IntelliJ IDEA中C盘文件过大怎么办

    当我在D:\ 安装完IDEA9.0之后,建立了一个工程,发现C:\Users\Administrator\.IntelliJIdea90 竟然增大到了500+M,并且随着使用在逐渐增大,这样占用系统盘资源是非常让人不爽的,那么如何将其修改到其他路径呢?

  • 2019-05-28 13:33:20

    BRVAH+MTRVA,复杂?不存在的

    言归正传,这样的一个首页,我们需要做怎么样的基础工作呢?或者说,碰到以后更复杂的页面我们应该怎么做?这里小提示下,不要再用什么类似ScrollView的这种东西了,诶,好像说的有点绝对,尽量不要用,这不是谷歌想要看到的,5.0谷歌推出了RecyclerView,从它的整个设计架构来看,简直就是为这而生的。而RecyclerView的视图是通过Adapter来渲染的。原始的Adapter,让人很蛋疼,重复工作太多,我们应该要有封装的思想,把最需要的部分提供出来,其它不用管。

  • 2019-05-29 14:19:19

    解决Git中fatal: refusing to merge unrelated histories

    Git的报错 在使用Git的过程中有时会出现一些问题,那么在解决了每个问题的时候,都需要去总结记录下来,下次不再犯。 一、fatal: refusing to merge unrelated histories 今天在使用Git创建项目的时候,在两个分支合并的时候,出现了下面的这个错误。

  • 2019-05-29 15:47:51

    撤销commit

    在git push的时候,有时候我们会想办法撤销git commit的内容