本文记录 Android
中 WebView
控件的相关使用,不断完善中…
主要包括:
WebView
缓存相关内容 Java
与 Js
的交互 WebView
打开本地应用(支付宝等) sd
卡路径, assert
目录路径的方法 WebView
支持下载等其它一些内容
汇总的记录一下 WebView
的配置方法,重要的属性后面会展开说明。
@SuppressLint("SetJavaScriptEnabled") public static void initWebViewSettings(WebView mWebView) { //支持获取手势焦点 mWebView.requestFocusFromTouch(); // 触觉反馈,暂时没发现用处在哪里 mWebView.setHapticFeedbackEnabled(false); WebSettings settings = mWebView.getSettings(); // 支持插件 settings.setPluginState(WebSettings.PluginState.ON); // 允许js交互 settings.setJavaScriptEnabled(true); // 设置WebView是否可以由 JavaScript 自动打开窗口,默认为 false // 通常与 JavaScript 的 window.open() 配合使用。 settings.setJavaScriptCanOpenWindowsAutomatically(true); // 允许中文编码 settings.setDefaultTextEncodingName("UTF-8"); // 使用大视图,设置适应屏幕 settings.setUseWideViewPort(true); settings.setLoadWithOverviewMode(true); // 支持多窗口 settings.setSupportMultipleWindows(true); // 隐藏自带缩放按钮 settings.setBuiltInZoomControls(false); // 支持缩放 settings.setSupportZoom(true); // 设置可访问文件 settings.setAllowFileAccess(true); // 当WebView调用requestFocus时为WebView设置节点 settings.setNeedInitialFocus(true); //支持自动加载图片 settings.setLoadsImagesAutomatically(true); // 指定WebView的页面布局显示形式,调用该方法会引起页面重绘。 // NORMAL,SINGLE_COLUMN 过时, NARROW_COLUMNS 过时 ,TEXT_AUTOSIZING settings.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN); // 从Lollipop(5.0)开始WebView默认不允许混合模式,https当中不能加载http资源,需要设置开启 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { settings.setMixedContentMode(WebSettings.MIXED_CONTENT_ALWAYS_ALLOW); } }
清除缓存, true
表示清除磁盘缓存,这个方法是全局的,也就是说会清除掉整个应用所有的 网页缓存。
mWebView.clearCache(true);
清除历史记录
mWebView.clearHistory();
网上有介绍说,设置缓存的目录,清除缓存时删除掉缓存文件,但是使用下面的方法设置缓存路径后,发现文件并没有缓存到指定的目录,不知道是怎么回事?求指教
缓存模式 | 描述 |
---|---|
LOAD_DEFAULT | 默认的缓存使用模式。在进行页面前进或后退的操作时,如果缓存可用并未过期就优先加载缓存,否则从网络上加载数据。这样可以减少页面的网络请求次数。 |
LOAD_CACHE_ELSE_NETWORK | 只要缓存可用就加载缓存,哪怕它们已经过期失效。如果缓存不可用就从网络上加载数据。 |
LOAD_NO_CACHE | 不加载缓存,只从网络加载数据。 |
LOAD_CACHE_ONLY | 不从网络加载数据,只从缓存加载数据。 |
<pre>private static void initWebViewCache(WebView mWebView) { String cachePath = new File(Environment.getExternalStorageDirectory() , "webCache").getAbsolutePath(); WebSettings settings = mWebView.getSettings(); settings.setAppCacheEnabled(true); settings.setAppCachePath(cachePath); settings.setDatabaseEnabled(true); // 过时 settings.setDatabasePath(cachePath); // 开启dom缓存 settings.setDomStorageEnabled(true); // 加载模式 settings.setCacheMode(WebSettings.LOAD_NO_CACHE); // 缓存最大值,过时 settings.setAppCacheMaxSize(1000 * 1024); }</pre>
加载网络路径, headers
不是必选的
<div>Map<String,String> headers = new HashMap<>();</div> <div>headers.put("auth","110");</div> <div>mWebView.loadUrl("https://www.baidu.com/",headers);</div>
加载 assert
路径
<div>// WebViewUtil.java</div> <div>public static String getAssertUrl(String fileUrl) {</div> <div> return "file:///android_asset/" + fileUrl;</div> <div>}</div> <div></div> <div>mWebView.loadUrl(WebViewUtil.getAssertUrl("index.html"));</div>
加载 sd
卡路径
<div>// WebViewUtil.java</div> <div>public static String getSdUrl(String fileUrl) {</div> <div> return "file://" + Environment.getExternalStorageDirectory() + "/" + fileUrl;</div> <div>}</div> <div></div> <div>mWebView.loadUrl(WebViewUtil.getSdUrl("index.html"));</div>
Java
与 Js
交互需要定义一个带有 @JavascriptInterface
注解方法的对象,如下:
<pre>public class JsBridge { @JavascriptInterface public void toast(String msg) { ToastUtils.show(msg); } @JavascriptInterface public void log(String msg) { Log.e(TAG, msg); } }</pre>
使用 JsBridge
连接 Java
和 Js
,使用 mWebView.addJavascriptInterface(obj, name);
方法。
JavascriptInterface
mWebView.getSettings().setJavaScriptEnabled(true); mWebView.addJavascriptInterface(new JsBridge(), "android");
在添加调用接口时,我们添加了一个标记,如上面代码中添加的是 android
,它相当于调用接口对象的一个别名,因此在 js
中调用 java
方法时,只需要使用 window
对象,如下:
window.android.log('test js invoke java method');
使用 java
调用 js
方法相对简单,假如在 js
中应该声明如下 function
,
<div><script></div> <div>// 有参无返回值</div> <div>function funcParam(param){</div> <div> alert(param+"");</div> <div>}</div> <div></div> <div>// 有参有返回值</div> <div>function newFunc(param1,param2,param3){</div> <div> alert((param1 + param2) + " " + param3);</div> <div> return param3;</div> <div>}</div> <div></script></div>
在 java
层调用时只需 loadUrl("Javascript:js方法名())
即可,但是 Android 4.4 之后在调用 js
方法时可以获取返回值,写一个方法兼容一下
<pre>public void invokeJs(String jsFunc, final ValueCallback<String> callback) { if (!jsFunc.startsWith(JS_FUNC_PREFIX)) { jsFunc = JS_FUNC_PREFIX + jsFunc; } // api 19 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { mWebView.evaluateJavascript(jsFunc, callback); } else { mWebView.loadUrl(jsFunc); } }</pre>
在与 js
进行参数传递时,基本数据类型可以直接传递,字符串类型要使用引号包含,而且直接拼接字符串不太好,因此写一个简化构建 js
方法,自动按顺序进行参数的拼接。
<pre>// 创建一个 js 方法 public static String generateJsFunc(String funcName, Object... params) { StringBuilder sb = new StringBuilder(funcName).append("("); for (int i = 0; i < params.length; i++) { if (params[i] != null) { if (params[i] instanceof String) { sb.append("'").append(params[i]).append("'"); } else { sb.append(params[i]); } if (i != params.length - 1) { sb.append(","); } } } sb.append(")"); return sb.toString(); }</pre>
简单调用
<div>// 调用有参有返回值的方法</div> <div>String jsFunc = JsBridge.generateJsFunc("newFunc", "str", 100, 100);</div> <div>mJsBridge.invokeJs(jsFunc, new ValueCallback<String>() {</div> <div> @Override</div> <div> public void onReceiveValue(String value) {</div> <div> Log.e(TAG, "value = " + value);</div> <div> }</div> <div>});</div> <div></div> <div>// 调用有参无返回值方法</div> <div>String funcParam = JsBridge.generateJsFunc("funcParam", 100);</div> <div>mJsBridge.invokeJs(funcParam, null);</div>
主要原理是由于某些应用会暴露一些页面出来供别的 app 唤起,因此我们只需要拦截这种 Url
,然后使用 intent
打开即可,下面以支付宝为例说明:
前端使用支付宝进行支付时,需要打开手机支付宝客户端,断点看到支付宝会加载一个 scheme
为 alipays
的url,链接中配置了支付的相关信息,客户端要做的就是拦截该 Url
,使用 intent
打开支付宝。
当 html
加载网页之前会先走 shouldOverrideUrlLoading()
方法,此时截断不使用 webView
加载,而是使用 intent
打开,此方法不仅适用支付宝,也适用打开其他应用。
<pre>mWebView.setWebViewClient(new WebViewClient() { @Override public boolean shouldOverrideUrlLoading(WebView view, String url) { // 如果不使用 intent 覆盖加载这个链接,就走原来 if (!shouldOverrideIntentUrl(getContext(), url)) { view.loadUrl(url, Api.makeHttpHeaders(getContext())); } return true; } }); public static final String ALIPAY_SCHEME = "alipays"; public static boolean shouldOverrideIntentUrl(Context context, String url) { Uri uri = Uri.parse(url); if (uri != null && uri.getScheme() != null && uri.getScheme().equals(ALIPAY_SCHEME)) { try { Intent intent = Intent.parseUri(url, Intent.URI_INTENT_SCHEME); intent.addCategory(Intent.CATEGORY_BROWSABLE); intent.setComponent(null); intent.setSelector(null); context.startActivity(intent); } catch (Exception e) { e.printStackTrace(); if (e instanceof ActivityNotFoundException) { ToastUtil.show("未检测到支付宝客户端"); } return true; } return true; } return false; }</pre>
当网页中加载一个下载的链接,如 http://xxx.apk
这种链接,会走 shouldOverrideUrlLoading()
方法,但是,它是无法加载一个网页的,结果就是没有任何反应,此时需要设置 DownloadListener
,需要下载的链接会进入监听,你可以在监听中自己进行网络下载保存到文件,下面我使用直接打开浏览器的方式,更简单一些。
<pre>public static void setDefDownloadListener(final WebView webView) { webView.setDownloadListener(new DownloadListener() { @Override public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype, long contentLength) { //去下载 Intent intent = new Intent(Intent.ACTION_VIEW); String downLoadUrl = url; if (!downLoadUrl.contains("http://")) { downLoadUrl = "http://" + downLoadUrl; } intent.setData(Uri.parse(downLoadUrl)); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); webView.getContext().startActivity(intent); } }); }</pre>
WebView
默认是无法弹出 alert()
的,需要设置
WebChromeClient
mWebView.setWebChromeClient(new WebChromeClient());
更多内容补充中 …