WebView 5.0+闪烁以及白屏问题完美解决

时间:2022-04-19 04:00:24

Android webView 在5.0+上启动硬件加速,造成部分手机出现闪烁、白屏等现象

必须写下这篇博客,遇到的问题搞了很久,百度谷歌就是没有完整的答案,记录下来,方便博友们;

需求:一个简单的WebView,底部是评论列表;

实现:ListView+头布局,webview放在头布局中

问题所在:在Android5.0以下系统,一切正常,非常完美;但是5.0+的系统有GIF的动态图片页面会闪烁不停,评论列表出现、webView滑动到底部快要消息的时候也会闪烁;

bug解决之前代码:加载webView核心代码

……
final String url = "http://www.chinichi.cn/news/index/app_detail.html?id=638";
mWebView = (MyWebView) findViewById(R.id.……);
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.loadUrl(url);
mWebView.setWebViewClient(new WebViewClient() {
@Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
mWebView.loadUrl(url);
return true;
}
……
……
}

一、造成闪烁的原因是WebView5.0开启了硬件加速,所以首要任务是关闭硬件加速,有三种

  1、 AndroidManifest.xml中的Activity配置:android:hardwareAccelerated="false"

  2、xml中:android:layerType="software"

  3、Java代码设置:

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
CookieManager cookieManager = CookieManager.getInstance();
cookieManager.setAcceptThirdPartyCookies(mWebView, true);
mWebView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}

二、但是关闭硬件加速后会造成:View too large to fit into drawing cache, needs 183047040 bytes, only 8294400 available。有人说是设置android:hardwareAccelerated="false"即可解决,但本人项目中并无卵用,所以,

  1、设置:mWebView.setDrawingCacheEnabled(false);  mWebView.getSettings().setLoadWithOverviewMode(true);

  2、wenView重写:

  

public class MyWebView extends android.webkit.WebView {
public MyWebView(Context context) {
super(context);
} public MyWebView(Context context, AttributeSet attrs) {
super(context, attrs);
} public MyWebView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
invalidate();
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
}

然后拿公司华为手机5.0+测,完美解决。可是,,,,魅族5.0+直接白屏。。。

不知道啥原理,一通乱按:

1、android:layerType="software"在ListView中也给设置(如果你用的是ScrollView的话给他也要设置)

2、 

mWebView.setWebViewClient(new WebViewClient() {
      //再增加这个方法
@Override
public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {
super.onReceivedSslError(view, handler, error);
handler.proceed();
} @Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
mWebView.loadUrl(url);
return true;
}
}

最后总结:

  1、在activity配置中增加 android:hardwareAccelerated="false"

  2、WebView xml中:android:layerType="software",ListView(或者外层嵌套ScrollView)android:layerType="software”

  3、重写WebView,代码看上面

  4、Java代码中:

  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
CookieManager cookieManager = CookieManager.getInstance();
cookieManager.setAcceptThirdPartyCookies(mWebView, true);
mWebView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}
 mWebView.setDrawingCacheEnabled(false);
mWebView.getSettings().setLoadWithOverviewMode(true);
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.loadUrl(url);
mWebView.setWebViewClient(new WebViewClient() {
@Override
public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {
super.onReceivedSslError(view, handler, error);
handler.proceed();
} @Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
mWebView.loadUrl(url);
return true;
} @Override
public void onPageFinished(WebView view, String url) {
super.onPageFinished(view, url);
}
});
mWebView.setWebChromeClient(new WebChromeClient() {
@Override
public void onProgressChanged(WebView view, int newProgress) {
super.onProgressChanged(view, newProgress);
}
});