本文记录 Android
中 WebView
控件的相关使用,不断完善中…
主要包括:
- 基本属性的配置
WebView
缓存相关内容Java
与Js
的交互- 打开本地应用(支付宝等)
- 加载远程网页,本地网页,
assert
目录下的网页 - 支持自定义下载
- 自定义
WebChromeClient
本地化弹窗、文件选择
更详细的扩展实现都可以在 GitHub-WebKit 中找到。
基本配置汇总
汇总的记录一下 WebView
的配置方法,重要的属性后面会展开说明。
1 | "SetJavaScriptEnabled") ( |
WebView 缓存
清除缓存,true
表示清除磁盘缓存,这个方法是全局的,也就是说会清除掉整个应用所有的 网页缓存。
1 | mWebView.clearCache(true); |
清除历史记录
1 | mWebView.clearHistory(); |
网上有介绍说,设置缓存的目录,清除缓存时删除掉缓存文件,但是使用下面的方法设置缓存路径后,发现文件并没有缓存到指定的目录,不知道是怎么回事?求指教
缓存模式 | 描述 |
---|---|
LOAD_DEFAULT | 默认的缓存使用模式。在进行页面前进或后退的操作时,如果缓存可用并未过期就优先加载缓存,否则从网络上加载数据。这样可以减少页面的网络请求次数。 |
LOAD_CACHE_ELSE_NETWORK | 只要缓存可用就加载缓存,哪怕它们已经过期失效。如果缓存不可用就从网络上加载数据。 |
LOAD_NO_CACHE | 不加载缓存,只从网络加载数据。 |
LOAD_CACHE_ONLY | 不从网络加载数据,只从缓存加载数据。 |
1 | private static void initWebViewCache(WebView mWebView) { |
加载 Url
加载网络路径,headers
不是必选的
1 | Map<String,String> headers = new HashMap<>(); |
加载 assert
路径
1 | // WebViewUtil.java |
加载 sd
卡路径
1 | // WebViewUtil.java |
Js 调用 Java
Java
与 Js
交互需要定义一个带有 @JavascriptInterface
注解方法的对象,如下:
1 | public class JsBridge { |
使用 JsBridge
连接 Java
和 Js
,使用 mWebView.addJavascriptInterface(obj, name);
方法。
- obj:带有
JavascriptInterface
注解方法的对象实例。 - name:一个标志符,js 将会使用该标志来调用 java 层的方法。
1 | mWebView.getSettings().setJavaScriptEnabled(true); |
在添加调用接口时,我们添加了一个标记,如上面代码中添加的是 android
,它相当于调用接口对象的一个别名,因此在 js
中调用 java
方法时,只需要使用 window
对象,如下:
1 | window.android.log('test js invoke java method'); |
Java 调用 Js
使用 java
调用 js
方法相对简单,假如在 js
中应该声明如下 function
,
1 | <script> |
在 java
层调用时只需 loadUrl("Javascript:js方法名())
即可,但是 Android 4.4 之后在调用 js
方法时可以获取返回值,写一个方法兼容一下
1 | public static final String JS_FUNC_PREFIX = "javascript:"; |
在与 js
进行参数传递时,基本数据类型可以直接传递,字符串类型要使用引号包含,而且直接拼接字符串不太好,因此写一个简化构建 js
方法,自动按顺序进行参数的拼接。
1 | // 创建一个 js 方法 |
简单调用
1 | // 调用有参有返回值的方法 |
拦截 Url 打开应用(支付宝)
主要原理是由于某些应用会暴露一些页面出来供别的 app 唤起,因此我们只需要拦截这种 Url
,然后使用 intent
打开即可,下面以支付宝为例说明:
前端使用支付宝进行支付时,需要打开手机支付宝客户端,断点看到支付宝会加载一个 scheme
为 alipays
的url,链接中配置了支付的相关信息,客户端要做的就是拦截该 Url
,使用 intent
打开支付宝。
当 html
加载网页之前会先走 shouldOverrideUrlLoading()
方法,此时截断不使用 webView
加载,而是使用 intent
打开,此方法不仅适用支付宝,也适用打开其他应用。
1 | mWebView.setWebViewClient(new WebViewClient() { |
支持下载链接
当网页中加载一个下载的链接,如 http://xxx.apk
这种链接,会走 shouldOverrideUrlLoading()
方法,但是,它是无法加载一个网页的,结果就是没有任何反应,此时需要设置 DownloadListener
,需要下载的链接会进入监听,你可以在监听中自己进行网络下载保存到文件,下面我使用直接打开浏览器的方式,更简单一些。
1 | public static void setDefDownloadListener(final WebView webView) { |
WebChromeClient
本地化弹窗
WebView
默认是无法弹出 alert()
的,需要设置 WebChromeClient
1 | mWebView.setWebChromeClient(new WebChromeClient()); |
从网页上弹起的弹窗与设备整体风格不是很搭,需要重载 onJsAlert()
、onJsConfirm()
和 onJsPrompt()
来将弹窗本地化。JavaScript
与本地方法对应如下:
1 | // 提示 返回值为 undefind - onJsAlert() |
以 onJsConfirm()
为例实现响应用户输入的方法,使用 JsResult
向 JavaScript
返回值,否则网页端会一直阻塞,不能操作。
1 |
|
文件选择
需要重载 onShowFileChooser()
方法,使用 Intent
打开文件选择,然后返回文件,在 api21
以上提供了一些特有的方法,在这个版本以下要自己适配。
1 |
|
在 onActivityResult()
中接受返回的数据.
1 | public void onActivityResult(int requestCode, int resultCode, Intent data) { |
todo
更多内容补充中 …