在PhoneGap应用中拦截CordovaWebView的url

最近在用PhoneGap做混合应用,遇到了一个比较奇葩的需求,客户公司原先做了一个触屏版的webapp,之后又开始做的手机app,公司决定用PhoneGap开发,可以利用一些webapp页面,为了解决不让app变成一个浏览器页面随便跳,所以出来了这个需求:在PhoneGap中对web页面上的链接进行拦截。

思路:

通常在android的webView中可以采用设置webViewClient的方法拦截

/**
 * 返回true意味着宿主应用程序处理URL,则返回false意味着目前的WebView处理URL
 */
@Override
public boolean shouldOverrideUrlLoading(WebView view, String url) {
	//返回值是true的时候控制去WebView打开,为false调用系统浏览器或第三方浏览器
        url="www.baidu.com";
        view.loadUrl(url);
 Log.i("URL", "URL改变了");return true;}
 具体的操作网上有很多讲解,这里就不多说了,主要说明一点,我用了这个方法结果项目出问题了,总是提示连接失败。 
  

于是我开始看源码,看看cordova到底是如何工作的,我的版本是cordova-2.9.0。

下面是相关源码:

package org.apache.cordova;

public class DroidGap extends CordovaActivity
{
}
MainActivity继承的DroidGap只是一个空壳,进入CordovaActivity看看究竟,源码很多,稍微贴点

package org.apache.cordova;

import...

public class CordovaActivity extends Activity
  implements CordovaInterface
{
  public static String TAG = "CordovaActivity";
  protected CordovaWebView appView;
  protected CordovaWebViewClient webViewClient;
上面的代码是CordovaActivity类的开头部分,引入的包被我略去了,下面是进入关键代码

  public void loadUrl(String url)
  {
    if (this.appView == null) {
      init();
    }

    this.backgroundColor = getIntegerProperty("backgroundColor", -16777216);
    this.root.setBackgroundColor(this.backgroundColor);

    this.keepRunning = getBooleanProperty("keepRunning", true);

    loadSpinner();

    this.appView.loadUrl(url);
  }
上面的函数就是android在MainActivity中引入html页面调用的loadUrl方法,下面是MainActivity中的oncreate()方法

@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		super.init();
		super.loadUrl("file:///android_asset/www/html/index.html", 2500);
	}
从源码的loadUrl方法中执行了一个init()方法,那我们去看看到底都干了些什么

  public void init()
  {
    CordovaWebView webView = new CordovaWebView(this);
    CordovaWebViewClient webViewClient;
    CordovaWebViewClient webViewClient;
    if (Build.VERSION.SDK_INT < 11)
    {
      webViewClient = new CordovaWebViewClient(this, webView);
    }
    else
    {
      webViewClient = new IceCreamCordovaWebViewClient(this, webView);
    }
    init(webView, webViewClient, new CordovaChromeClient(this, webView));
  }

  @SuppressLint({"NewApi"})
  public void init(CordovaWebView webView, CordovaWebViewClient webViewClient, CordovaChromeClient webChromeClient)
  {
    LOG.d(TAG, "CordovaActivity.init()");

    this.appView = webView;
    this.appView.setId(100);

    this.appView.setWebViewClient(webViewClient);
    this.appView.setWebChromeClient(webChromeClient);
    webViewClient.setWebView(this.appView);
    webChromeClient.setWebView(this.appView);

    this.appView.setLayoutParams(new LinearLayout.LayoutParams(-1, -1, 1.0F));

    if ((getBooleanProperty("disallowOverscroll", false)) && 
      (Build.VERSION.SDK_INT >= 9)) {
      this.appView.setOverScrollMode(2);
    }

    this.appView.setVisibility(4);
    this.root.addView(this.appView);
    setContentView(this.root);

    this.cancelLoadUrl = false;
  }
在init()的刚开始出现的 CordovaWebView就是我们显示html的控件,它继承自webView。

从之后的代码可以看出在初始化的时候给 CordovaWebView 设置了 CordovaWebViewClient,它继承自WebViewClient,这个在webView中对应的就是在文章刚开始提到的webViewClient(上面已蓝色显示),于是去CordovaWebViewClient里面看看有什么吧。源代码又很多,但是在里面发现了下面这个方法

public boolean shouldOverrideUrlLoading(WebView view, String url)
是不是很眼熟?这个方法和文章刚开始提到的webView中拦截url的方法一样,这是不是就意味着我可以用这个方法去拦截呢?下面开始尝试拦截。

由于CordovaWebViewClient里面的shouldOverrideUrlLoading不是我想要的功能,于是我重写了CordovaWebViewClient

package com.example.opalbb.tools;

import org.apache.cordova.CordovaWebView;
import org.apache.cordova.CordovaWebViewClient;
import org.apache.cordova.api.CordovaInterface;

import android.util.Log;
import android.webkit.WebView;

public class MyCordovaWebViewClient extends CordovaWebViewClient {

	public MyCordovaWebViewClient(CordovaInterface cordova) {
		super(cordova);
		// TODO Auto-generated constructor stub
	}

	public MyCordovaWebViewClient(CordovaInterface cordova, CordovaWebView view) {
		super(cordova, view);
		// TODO Auto-generated constructor stub
	}

	@Override
	public boolean shouldOverrideUrlLoading(WebView view, String url) {
		Log.i("在这里可以修改url", url);
		view.loadUrl(url);
		} catch (Exception e) {
			e.printStackTrace();
		}
		return true;
	}
}
到这里似乎可以试一试拦截的效果了,先别着急,我们如何让 CordovaWebView 使用我们自己写的MyCordovaWebViewClient类呢?下面的代码是我修改了CordovaWebView的初始化函数,其实就是把上面MainActivity中的super.init()替换成下面的代码

@Override
	public void init() {
		CordovaWebView webView = new CordovaWebView(this);
		MyCordovaWebViewClient webViewClient;
		webViewClient = new MyCordovaWebViewClient(this, webView);
		init(webView, webViewClient, new CordovaChromeClient(this, webView));
	}
@Override
	public void init(CordovaWebView webView, CordovaWebViewClient webViewClient, CordovaChromeClient webChromeClient) {
		LOG.d(TAG, "CordovaActivity.init()");

		this.appView = webView;
		this.appView.setId(100);

		this.appView.setWebViewClient(webViewClient);
		this.appView.setWebChromeClient(webChromeClient);
		webViewClient.setWebView(this.appView);
		webChromeClient.setWebView(this.appView);

		this.appView.setLayoutParams(new LinearLayout.LayoutParams(-1, -1, 1.0F));

		if ((getBooleanProperty("disallowOverscroll", false)) && (Build.VERSION.SDK_INT >= 9)) {
			this.appView.setOverScrollMode(2);
		}

		this.appView.setVisibility(4);
		this.root.addView(this.appView);
		setContentView(this.root);

		this.cancelLoadUrl = false;
	}
其他代码都不用改,开始运行,就可以看出我们刚才代码中的Log了,我的代码已经完成了,就不还原回去运行了,但是保证这样就可以拦截CordovaWebView的url了,今天就写到这里了,下班回家吃饭

你可能感兴趣的:(PhoneGap,android)