前言:相信好多小伙伴跟我一样,项目赶得急,用到混合开发的时候自然选择webview网上随便找一个用法就可以了,已加载看出来了东西,OK过了,可是当你真正的要拦截URL做一些事情的时候,你发现网上说的并不正确。
一:写代码
WebSettings webSettings =webView.getSettings();
//设置支持js方法
webSettings.setJavaScriptEnabled(true);
//支持自动加载图片
webSettings.setLoadsImagesAutomatically(true);
webSettings.setDomStorageEnabled(true);
webSettings.setAllowFileAccess(true);
webSettings.setAppCacheEnabled(true);
//设置 缓存模式
webSettings.setCacheMode(WebSettings.LOAD_DEFAULT);
// 开启 DOM storage API 功能
webSettings.setDomStorageEnabled(true);
webView.loadUrl(url);
//覆盖WebView默认使用第三方或系统默认浏览器打开网页的行为,使网页用WebView打开
webView.setWebViewClient(new WebViewClient() {
@Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
//返回值是true的时候控制去WebView打开,为false调用系统浏览器或第三方浏览器
Share.d("shouldOverrideUrlLoading=" + url);
if (url.contains(Path.WEBVIEW +"showMemberInfo?memberId=")) {
//对应班级详情
String first = url.substring(url.indexOf("="));
String second = first.replace("=", "");
Share.d("classidfirst" + first);
Share.d("classidsecond" + second);
if (!ClassPathResource.isEmptyOrNull(second)) {
if (second.equals("spouse")) {
//配偶自动生成的时候没有id,没有详细信息跳转添加页面添加信息
Bundle bundle =new Bundle();
bundle.putString("type", "spouse");
openActivity(AddFamilyActivity.class, bundle);
}else {
//有id直接跳转编辑页面
Bundle bundle =new Bundle();
bundle.putString("id", second);
openActivity(FamilyDetailActivity.class, bundle);
}
}
return true;
}else if (url.matches(Path.WEBVIEW +"login.jsp")) {
Share.d("fase=" + url);
return true;
}else {
return false;
//return super.shouldOverrideUrlLoading(view, url);
}
}
@Override
public void onPageStarted(WebView view, String url, Bitmap favicon) {
super.onPageStarted(view, url, favicon);
Share.d("onPageStarted=" + url);
showLoading();
}
@Override
public void onPageFinished(WebView view, String url) {
super.onPageFinished(view, url);
dismissLoading();
}
});
}
/**
* Back键控制网页后退
*
* @param
* @param
*/
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if ((keyCode ==KEYCODE_BACK) &&webView.canGoBack()) {
//webView.goBack();
finish();
return true;
}
return super.onKeyDown(keyCode, event);
}
/**
* 在 Activity 销毁( WebView )的时候,先让 WebView 加载null内容,然后移除 WebView,再销毁 WebView,最后置空。
*
* @param
* @param
*/
@Override
protected void onDestroy() {
if (webView !=null) {
webView.loadDataWithBaseURL(null, "", "text/html", "utf-8", null);
webView.clearHistory();
((ViewGroup)webView.getParent()).removeView(webView);
webView.destroy();
webView =null;
}
super.onDestroy();
UMShareAPI.get(this).release();
}
说明:这是我的一个页面使用的,关键点在于 webView.setWebViewClient(new WebViewClient() {重写他的方法。今天重新认识一下
onPageStarted:页面最先加载的URL,然后就是在第一个URL上面点击的时候先加载shouldOverrideUrlLoading,马上进去onPageStarted,onPageStarted是资源准备好,要显示了,shouldOverrideUrlLoading是正准备加载的二级链接,还有一点就是按返回键的时候不会走shouldOverrideUrlLoading方法,而是onPageStarted方法,所以在重写回退键webview.goback。
shouldOverrideUrlLoading:重点,网上大多数是view.webview(url) return turn,要这样写不拦截无所谓,但是你拦截做逻辑就完蛋了,到底return true,fase是什么。
源码:
仔细阅读注释部分,可以得知在一个新的链接即将被加载时,会出现以下几种情况:
如果没有提供 WebViewClient 对象,则 WebView 会请求 Activity 管理者选择合适的 URL 处理方式,一般情况就是启动浏览器来加载URL;
如果提供了 WebViewClient 对象且shouldOverrideUrlLoading 方法返回 true,则宿主应用(意思应该是自己的Android应用)处理URL;
如果提供了 WebViewClient 对象且shouldOverrideUrlLoading 方法返回 false,则当前 WebView 处理URL;
再结合 shouldOverrideUrlLoading 方法源码就可以得出上面图片中的结论,其实我们没必要自定义 WebViewClient 并重写其 shouldOverrideUrlLoading 方法,因为 WebViewClient 源码中 shouldOverrideUrlLoading 方法已经返回 false,也就是说只要你设置了 WebViewClient 就可以实现在WebView 中加载新的链接而不去调用浏览器加载。
如果重写了;记住fase是当前webview处理了,ture就是不处理在当前webview.
1、 默认返回:return super.shouldOverrideUrlLoading(view, url); 这个返回的方法会调用父类方法,也就是跳转至手机浏览器,平时写webview一般都在方法里面写 webView.loadUrl(url); 然后把这个返回值改成下面的false。
2、返回: return true; webview处理url是根据程序来执行的。
3、返回: return false; webview处理url是在webview内部执行。
总结:知道啥意思了吗,就是说你重写的那个方法如果写了webView.loadUrl(url);不管你return啥都是在webview内部处理了,如果不写的话,fase在webview处理; true; webview处理url是根据程序来执行的。