android基础--WebView

章涵蓄
2023-12-01

[参考网页](http://blog.sina.com.cn/s/blog_8241e8510101btvk.html
http://www.2cto.com/kf/201411/354679.html
)
1.添加权限:”android.permission.INTERNET” 否则会出现webpage not available错误

2.在Activity中生成webView组件WebView webView=new WebView(this);

3.设置webView基本信息:
例如:
1).设置WebSettings类
WebSettings用来对WebView的配置进行配置和管理,比如是否可以进行文件操作、缓存的设置、页面是否支持放大和缩小、是否允许使用数据库api、字体及文字编码设置、是否允许js脚本运行、是否允许图片自动加载、是否允许数据及密码保存等等

2).触摸焦点起作用,如果设置,在点击网页文本输入框时,不能弹出软键盘及不响应其他的一些事件。
ebview.requestFoucus();
设置WebChromeClient子类
WebChromeClient会在一些影响浏览器ui交互动作发生时被调用,比如WebView关闭和隐藏、页面加载进展、js确认框和警告框、js加载前、js操作超时、webView获得焦点等等
mWebView.setWebChromeClient(new MyWebChromeClient());

3).取消滚动条:这个设置应该在其他场景也使用吧
this.setScrollBarStyle(SCROLLBARS_OUTSIDE_OVERLAY);

4)设置addJavascriptInterface方法
使Js调用Native本地Java对象,实现本地Java代码和HTML页面进行交互,
注意:因为安全问题的考虑 Google在使用Android API 17以上的版本的时候 需要通过@JavascriptInterface来注解的Java函数才能被识别可以被Js调用。

4.设置WevView要显示的网页:
互联网用:webView.loadUrl(“http://www.google.com“);
本地文件用:webView.loadUrl(“file:///android_asset/XX.html”); 本地文件存放在:assets文件中

5.如果希望点击链接由自己处理,而不是新开Android的系统browser(浏览器)中响应该链接。给WebView添加一个事件监听对象(WebViewClient)

webView.setWebViewClient(new WebViewClient() {  

    @Override 
    public boolean shouldOverrideUrlLoading(WebView view, String url) {  
        view.loadUrl(url);  
        return true;  
    }  
});

shouldOverrideUrlLoading表示当前webView中的一个新url需要加载时,给当前应用程序一个处理机会,如果没有重写此函数,webView请求ActivityManage选择合适的方式处理请求,就像弹出uc和互联网让用户选择浏览器一样。重写后return true表示让当前程序处理,return false表示让当前webView处理

6.listview,webview中滚动拖动到顶部或者底部时的阴影(滑动到项部或底部不固定)这个没用上不知道

           WebView.setOverScrollMode(View.OVER_SCROLL_NEVER);

7.localStorage本地储存缓存神马的(因为两个网页有不同的写法,我就都写上去了)
1)当使用WebView加载HTML网页时,会在我们data/应用package下生成database与cache两个文件夹:
我们请求的Url记录是保存在webviewCache.db里,而url的内容是保存在webviewCache文件夹下.

五种缓存模式的设置setCacheMode:
LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据。
LOAD_DEFAULT: 根据cache-control决定是否从网络上取数据。
LOAD_CACHE_NORMAL: API level 17中已经废弃, 从API level 11开始作用同LOAD_DEFAULT模式。
LOAD_NO_CACHE: 不使用缓存,只从网络获取数据。
LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。

WebSettings webSettings = mWebView.getSettings();
webSettings.setCacheMode(WebSettings.LOAD_DEFAULT);  //设置 缓存模式  
// 开启 DOM storage API 功能  
webSettings.setDomStorageEnabled(true);  
//开启 database storage API 功能  
webSettings.setDatabaseEnabled(true);

H5缓存
通过setAppCacheEnabled(boolean flag)设置H5的缓存是否打开,默认关闭。
根据setAppCachePath(String appCachePath)提供的路径,在H5使用缓存过程中生成的缓存文件。
通过setAppCacheMaxSize(long appCacheMaxSize)设置缓存最大容量。

String cacheDirPath = getCacheDir().getAbsolutePath()+ "/webViewCache ";
WebSettings webSettings = mWebView.getSettings();
//开启 database storage API 功能  
webSettings.setDatabaseEnabled(true);    
//设置数据库缓存路径  
webSettings.setDatabasePath(cacheDirPath); 
//开启Application H5 Caches 功能  
webSettings.setAppCacheEnabled(true); 
//设置Application Caches 缓存目录  
webSettings.setAppCachePath(cacheDirPath);

2)

//android 中 webview 使用 localStorage
  WebSettings settings = mWebView.getSettings();

   // 设置可以使用localStorage
    settings.setDomStorageEnabled(true);

 // 应用可以有数据库
        settings.setDatabaseEnabled(true);  
        String dbPath = this.getApplicationContext().getDir("database",     Context.MODE_PRIVATE).getPath();
        settings.setDatabasePath(dbPath);

 // 应用可以有缓存
        settings.setAppCacheEnabled(true);            
        String appCaceDir = this.getApplicationContext().getDir("cache", Context.MODE_PRIVATE).getPath();
        settings.setAppCachePath(appCaceDir);

8.关于后退,用WebView点击链接查看网页后,如果不做任何处理。点击系统“Back”键,整个程序会finish()掉自己,所以想要回退的话,就要在Activity中处理并消费掉Back事件

 @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        //重写onKeyDown,当浏览网页,WebView可以后退时执行后退操作。
        if(keyCode==KeyEvent.KEYCODE_BACK && webView.canGoBack()){
            webView.goBack();
            return true;
        }
            return super.onKeyDown(keyCode, event);  
}

其中webView.canGoBack()在webView含有一个可后退的浏览记录时返回true
webView.goBack();表示返回至webView的上次访问页面

//or

webView.setWebChromeClient(new WebChromeClient(){

  @Override


  public void onProgressChanged(WebView view, int newProgress) {

    if(newProgress==100){  
    // 这里是设置activity的标题, 也可以根据自己的需求做一些其他的操作                      
       title.setText("加载完成");
     }else{
     title.setText("加载中.......");
    }
    }
});
 webView.setWebViewClient(new WebViewClient(){

 @Override

    public boolean shouldOverrideUrlLoading(WebView view, String url) {  //重写此方法表明点击网页里面的链接还是在当前的webview里跳转,不跳到浏览器那边

       view.loadUrl(url);

       return true;

  }

 @Override

public void onReceivedSslError(WebView view, SslErrorHandler handler, android.net.http.SslError error)
 { // 重写此方法可以让webview处理https请求

handler.proceed();

}



});

9.设置开始加载网页、加载完成、加载错误时处理
在WebViewClient子类中分别重写如下父类函数 代码如下:

webView.setWebViewClient(new WebViewClient() {  

    @Override
    public void onPageStarted(WebView view, String url, Bitmap favicon) {
        super.onPageStarted(view, url, favicon);
        // 开始加载网页时处理 如:显示"加载提示" 的加载对话框
        DialogManager.showLoadingDialog(this);
    }

    @Override
    public void onPageFinished(WebView view, String url) {
        super.onPageFinished(view, url);
        // 网页加载完成时处理  如:让 加载对话框 消失
        DialogManager.dismissLoadingDialog();
    }

    @Override
    public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
        super.onReceivedError(view, errorCode, description, failingUrl);
        // 加载网页失败时处理  如:
        view.loadDataWithBaseURL(null,
                "<span style="\"color:#FF0000\"">网页加载失败</span>",
                "text/html",
                "utf-8",
                null);
    }  
});

10.处理https请求,为WebView处理ssl证书设置
WebView默认是不处理https请求的,页面显示空白,需要进行如下设置
在WebViewClient子类中重写父类的onReceivedSslError函数 代码如下:

webView.setWebViewClient(new WebViewClient() {  

    @Override
    public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {
        handler.proceed();  // 接受信任所有网站的证书
        // handler.cancel();   // 默认操作 不处理
        // handler.handleMessage(null);  // 可做其他处理
    } 
});

我写下面的代码的时候也出现过空白的现象不过是因为setting的设置方面,这个我没有加上,也没有出问题

11.显示页面加载进度
在WebChromeClient子类中重写父类的onProgressChanged函数 代码如下:

webView.setWebChromeClient(new WebChromeClient() {

public void onProgressChanged(WebView view, int progress) {  
    setTitle("页面加载中,请稍候..." + progress + "%");  
    setProgress(progress * 100);  

    if (progress == 100) {  
        setTitle(R.string.app_name);  
    }  
}  

});
onProgressChanged通知应用程序当前页面加载的进度
progress表示当前页面加载的进度,为1至100的整数

12.
1) Js中调Native本地Java方法
设置webView的addJavascriptInterface方法,该方法有两个参数,第一个参数为被绑定到js中的类实例,第二个参数为在js中暴露的类别名,在js中引用java对象就是用这个名字
在Native Java代码如下:

mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.addJavascriptInterface(new JavaScriptInterface(this), "Android");

class JavaScriptInterface{

    Context mContext;

    /** Instantiate the interface and set the context */
    JavaScriptInterface(Context c) {
        mContext = c;
    }

    /** Show a toast from the web page 
      * 由Js调用执行Native本地Java方法
      */
    @JavascriptInterface
    public void showToast(String toast) {
        Log.d("TAG", "Js Invoker Native Function");
        Toast.makeText(mContext, toast, Toast.LENGTH_SHORT).show();
    }       

}

在HTML中Js调用Native方法 代码如下

<input value="Say hello" onclick="showAndroidToast('Hello Android!')" type="button">

<script type="text/javascript">
    function showAndroidToast(toast) {
        Android.showToast(toast);
    }
</script>

2) Java调Js方法
比如在HTML中有如下Js函数

<script type="text/javascript">
      function showAlert() {
        alert("Be executed by Native");
    }
</script>

在Native调Js方法如下:

mWebView.loadUrl("javascript:showAlert()");

13.加快HTML网页加载完成速度
默认情况html代码下载到WebView后,webkit开始解析网页各个节点,发现有外部样式文件或者外部脚本文件时,会异步发起网络请求下载文件,但如果在这之前也有解析到image节点,那势必也会发起网络请求下载相应的图片。在网络情况较差的情况下,过多的网络请求就会造成带宽紧张,影响到css或js文件加载完成的时间,造成页面空白loading过久。解决的方法就是告诉WebView先不要自动加载图片,等页面finish后再发起图片加载。
故在WebView初始化时设置如下代码:

public void int () {
    if(Build.VERSION.SDK_INT >= 19) {
        webView.getSettings().setLoadsImagesAutomatically(true);
    } else {
        webView.getSettings().setLoadsImagesAutomatically(false);
    }
}

同时在WebView的WebViewClient子类中重写onPageFinished()方法添加如下代码:

@Override
public void onPageFinished(WebView view, String url) {
    if(!webView.getSettings().getLoadsImagesAutomatically()) {
        webView.getSettings().setLoadsImagesAutomatically(true);
    }
}

从上面的代码,可以看出我们对系统API在19以上的版本作了兼容。因为4.4以上系统在onPageFinished时再恢复图片加载时,如果存在多张图片引用的是相同的src时,会只有一个image标签得到加载,因而对于这样的系统我们就先直接加载。

14.WebView硬件加速导致页面渲染闪烁问题解决方法
关于Android硬件加速 开始于Android 3.0 (API level 11),在四个级别上开启/关闭硬件加速
1)、Application级别:为整个应用程序开启硬件加速,在AndroidManifest中加入如下配置

</application>是断片了么。。。

2)Activity级别:控制每个activity是否开启硬件加速,只需在activity元素中添加android:hardwareAccelerated属性即可
3)Window级别:注:目前还不支持在Window级别上关闭硬件加速

getWindow().setFlags(
    WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED,
    WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED);

4)View级别:运行时单个view硬件加速,目前Android还不支持在View级别开启硬件加速 代码如下:

mView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);

我们开启硬件加速后,WebView渲染页面更加快速,拖动也更加顺滑。但有个副作用就是容易会出现页面加载白块同时界面闪烁现象。解决这个问题的方法是设置WebView暂时关闭硬件加速 代码如下:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
    webview.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}

15.其他
1> 从网络上下载html页面的过程应放在工作线程(后台线程)中
2> html下载成功后渲染出html的步骤应放在UI主线程,不然WebView加载网页过程会容易报错

以下是简单运用时的代码,其实也没用上面多少

package com.example.myapplication01.education.com.example.activity;

import android.app.Activity;
import android.os.Bundle;
import android.view.KeyEvent;
import android.webkit.WebSettings;
import android.webkit.WebView;
import android.webkit.WebViewClient;

import com.example.myapplication01.education.R;

/**
 * Created by zhihui on 2016/2/18.
 */
public class webActivity extends Activity {
    WebView webView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.webview);
        webView=(WebView)findViewById(R.id.wv_webview);
        loadWeb();
    }
    public void loadWeb(){
        String url="http://www.baidu.com";
        //此方法可以在webview中打开链接而不会跳转到外部浏览器
        webView.setWebViewClient(new WebViewClient());
        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true);
        webSettings.setCacheMode(WebSettings.LOAD_DEFAULT);
        webSettings.setDomStorageEnabled(true);
        webSettings.setDatabaseEnabled(true);
        webSettings.setAppCacheEnabled(true);
        webSettings.setAllowFileAccess(true);
        webSettings.setSupportZoom(true);
        webSettings.setBuiltInZoomControls(true);
        webView.loadUrl(url);
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        //重写onKeyDown,当浏览网页,WebView可以后退时执行后退操作。
        if(keyCode==KeyEvent.KEYCODE_BACK && webView.canGoBack()){
            webView.goBack();
            return true;
        }
        return super.onKeyDown(keyCode, event);
    }
}

R.layout.webview

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:orientation="vertical"
    android:layout_width="match_parent" android:layout_height="match_parent">
    <WebView
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:id="@+id/wv_webview"/>

</LinearLayout>
 类似资料: