Android WebView在Android平台上是一个特殊的View, 基于webkit引擎、展示web页面的控件,这个类能够被用来在你的app中仅仅显示一张在线的网页,还能够用来开发浏览器。WebView内部实现是采用渲染引擎来展现view的内容,提供网页前进后退,网页放大,缩小,搜索。Android的Webview在低版本和高版本采用了不一样的webkit版本内核,4.4后直接使用了Chrome。javascript
如今不少APP都内置了Web网页,好比说不少电商平台,淘宝、京东、聚划算等等。WebView比较灵活,不须要升级客户端,只须要修改网页代码便可。一些常常变化的页面能够用WebView这种方式去加载网页。例如中秋节跟国庆节打开的页面不同,若是是用WebView显示的话,只修改修改html页面就行,而不须要升级客户端。html
Webview功能强大,能够直接使用html文件(本地sdcard/assets目录),还能够直接加载url,使用JavaScript能够html跟原生APP互调。java
webView.loadUrl("http://139.196.35.30:8080/OkHttpTest/apppackage/test.html");//加载url webView.loadUrl("file:///android_asset/test.html");//加载asset文件夹下html //方式3:加载手机sdcard上的html页面 webView.loadUrl("content://com.ansen.webview/sdcard/test.html"); //方式4 使用webview显示html代码 webView.loadDataWithBaseURL(null,"<html><head><title> 欢迎您 </title></head>" + "<body><h2>使用webview显示 html代码</h2></body></html>", "text/html" , "utf-8", null);
使用WebView基本都会使用这两个类,那他们有哪些区别呢?android
WebViewClient主要帮助WebView处理各类通知、请求事件的,有如下经常使用方法:
- onPageFinished 页面请求完成
- onPageStarted 页面开始加载
- shouldOverrideUrlLoading 拦截url
- onReceivedError 访问错误时回调,例如访问网页时报错404,在这个方法回调的时候能够加载错误页面。git
WebChromeClient主要辅助WebView处理Javascript的对话框、网站图标、网站title、加载进度等,有如下经常使用方法。
- onJsAlert webview不支持js的alert弹窗,须要本身监听而后经过dialog弹窗
- onReceivedTitle 获取网页标题
- onReceivedIcon 获取网页icon
- onProgressChanged 加载进度回调github
由于须要加载网页url,因此须要在AndroidManifest.xml中添加访问网络权限。web
<uses-permission android:name="android.permission.INTERNET" />
布局文件:activity_main.xml浏览器
<?xml version="1.0" encoding="utf-8"?> <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <WebView android:id="@+id/webview" android:layout_width="match_parent" android:layout_height="match_parent"/> <ProgressBar android:id="@+id/progressbar" style="@android:style/Widget.ProgressBar.Horizontal" android:layout_width="match_parent" android:layout_height="3dip" android:max="100" android:progress="0" android:visibility="gone"/> </FrameLayout>
外层FrameLayout,里面有WebView跟ProgressBar,WebView的宽高匹配父类,ProgressBar横向进度条,高度3dip,按照FrameLayout布局规则,ProgressBar会覆盖在WebView之上,默认是隐藏不显示。缓存
MainActivity.java网络
public class MainActivity extends AppCompatActivity { private WebView webView; private ProgressBar progressBar; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); progressBar= (ProgressBar)findViewById(R.id.progressbar);//进度条 webView = (WebView) findViewById(R.id.webview); // webView.loadUrl("file:///android_asset/test.html");//加载asset文件夹下html webView.loadUrl("http://139.196.35.30:8080/OkHttpTest/apppackage/test.html");//加载url //使用webview显示html代码 // webView.loadDataWithBaseURL(null,"<html><head><title> 欢迎您 </title></head>" + // "<body><h2>使用webview显示 html代码</h2></body></html>", "text/html" , "utf-8", null); webView.addJavascriptInterface(this,"android");//添加js监听 这样html就能调用客户端 webView.setWebChromeClient(webChromeClient); webView.setWebViewClient(webViewClient); WebSettings webSettings=webView.getSettings(); webSettings.setJavaScriptEnabled(true);//容许使用js /** * LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据 * LOAD_DEFAULT: (默认)根据cache-control决定是否从网络上取数据。 * LOAD_NO_CACHE: 不使用缓存,只从网络获取数据. * LOAD_CACHE_ELSE_NETWORK,只要本地有,不管是否过时,或者no-cache,都使用缓存中的数据。 */ webSettings.setCacheMode(WebSettings.LOAD_NO_CACHE);//不使用缓存,只从网络获取数据. //支持屏幕缩放 webSettings.setSupportZoom(true); webSettings.setBuiltInZoomControls(true); //不显示webview缩放按钮 // webSettings.setDisplayZoomControls(false); } //WebViewClient主要帮助WebView处理各类通知、请求事件 private WebViewClient webViewClient=new WebViewClient(){ @Override public void onPageFinished(WebView view, String url) {//页面加载完成 progressBar.setVisibility(View.GONE); } @Override public void onPageStarted(WebView view, String url, Bitmap favicon) {//页面开始加载 progressBar.setVisibility(View.VISIBLE); } @Override public boolean shouldOverrideUrlLoading(WebView view, String url) { Log.i("ansen","拦截url:"+url); if(url.equals("http://www.google.com/")){ Toast.makeText(MainActivity.this,"国内不能访问google,拦截该url",Toast.LENGTH_LONG).show(); return true;//表示我已经处理过了 } return super.shouldOverrideUrlLoading(view, url); } }; //WebChromeClient主要辅助WebView处理Javascript的对话框、网站图标、网站title、加载进度等 private WebChromeClient webChromeClient=new WebChromeClient(){ //不支持js的alert弹窗,须要本身监听而后经过dialog弹窗 @Override public boolean onJsAlert(WebView webView, String url, String message, JsResult result) { AlertDialog.Builder localBuilder = new AlertDialog.Builder(webView.getContext()); localBuilder.setMessage(message).setPositiveButton("肯定",null); localBuilder.setCancelable(false); localBuilder.create().show(); //注意: //必需要这一句代码:result.confirm()表示: //处理结果为肯定状态同时唤醒WebCore线程 //不然不能继续点击按钮 result.confirm(); return true; } //获取网页标题 @Override public void onReceivedTitle(WebView view, String title) { super.onReceivedTitle(view, title); Log.i("ansen","网页标题:"+title); } //加载进度回调 @Override public void onProgressChanged(WebView view, int newProgress) { progressBar.setProgress(newProgress); } }; @Override public boolean onKeyDown(int keyCode, KeyEvent event) { Log.i("ansen","是否有上一个页面:"+webView.canGoBack()); if (webView.canGoBack() && keyCode == KeyEvent.KEYCODE_BACK){//点击返回按钮的时候判断有没有上一页 webView.goBack(); // goBack()表示返回webView的上一页面 return true; } return super.onKeyDown(keyCode,event); } /** * JS调用android的方法 * @param str * @return */ @JavascriptInterface //仍然必不可少 public void getClient(String str){ Log.i("ansen","html调用客户端:"+str); } @Override protected void onDestroy() { super.onDestroy(); //释放资源 webView.destroy(); webView=null; } }
运行代码,效果图以下: