美文网首页半栈工程师
Android业务组件化之URL Scheme使用

Android业务组件化之URL Scheme使用

作者: 火星局 | 来源:发表于2018-04-19 17:09 被阅读18次

Android业务组件化之URL Scheme使用
H5打包的apk使用(Android平台通过UrlSchemes与第三方应用相互调用)

什么是 URL Scheme?

android中的scheme是一种页面内跳转协议,是一种非常好的实现机制,通过定义自己的scheme协议,可以非常方便跳转app中的各个页面;通过scheme协议,服务器可以定制化告诉App跳转那个页面,可以通过通知栏消息定制化跳转页面,可以通过H5页面跳转页面等。

URL Scheme应用场景:

客户端应用可以向操作系统注册一个 URL scheme,该 scheme 用于从浏览器或其他应用中启动本应用。通过指定的 URL 字段,可以让应用在被调起后直接打开某些特定页面,比如商品详情页、活动详情页等等。也可以执行某些指定动作,如完成支付等。也可以在应用内通过 html 页来直接调用显示 app 内的某个页面。综上URL Scheme使用场景大致分以下几种:
● 服务器下发跳转路径,客户端根据服务器下发跳转路径跳转相应的页面
● H5页面点击锚点,根据锚点具体跳转路径APP端跳转具体的页面
● APP端收到服务器端下发的PUSH通知栏消息,根据消息的点击跳转路径跳转相关页面
● APP根据URL跳转到另外一个APP指定页面

URL Scheme协议格式:
先来个完整的URL Scheme协议格式:
xl://goods:8888/goodsDetail?goodsId=10011002
通过上面的路径 Scheme、Host、port、path、query全部包含,基本上平时使用路径就是这样子的。
● xl代表该Scheme 协议名称
● goods代表Scheme作用于哪个地址域
● goodsDetail代表Scheme指定的页面
● goodsId代表传递的参数
● 8888代表该路径的端口号

URL Scheme如何使用:
1.)在AndroidManifest.xml中对<activity />标签增加<intent-filter />设置Scheme

<activity
            android:name=".GoodsDetailActivity"
            android:theme="@style/AppTheme">
            <!--要想在别的App上能成功调起App,必须添加intent过滤器-->
            <intent-filter>
                <!--协议部分,随便设置-->
                <data android:scheme="xl" android:host="goods" android:path="/goodsDetail" android:port="8888"/>
                <!--下面这几行也必须得设置-->
                <category android:name="android.intent.category.DEFAULT"/>
                <action android:name="android.intent.action.VIEW"/>
                <category android:name="android.intent.category.BROWSABLE"/>
            </intent-filter>
        </activity>

2.)获取Scheme跳转的参数

Uri uri = getIntent().getData();
if (uri != null) {
    // 完整的url信息
    String url = uri.toString();
    Log.e(TAG, "url: " + uri);
    // scheme部分
    String scheme = uri.getScheme();
    Log.e(TAG, "scheme: " + scheme);
    // host部分
    String host = uri.getHost();
    Log.e(TAG, "host: " + host);
    //port部分
    int port = uri.getPort();
    Log.e(TAG, "host: " + port);
    // 访问路劲
    String path = uri.getPath();
    Log.e(TAG, "path: " + path);
    List<String> pathSegments = uri.getPathSegments();
    // Query部分
    String query = uri.getQuery();
    Log.e(TAG, "query: " + query);
    //获取指定参数值
    String goodsId = uri.getQueryParameter("goodsId");
    Log.e(TAG, "goodsId: " + goodsId);
}

3.)调用方式

网页
<a href="xl://goods:8888/goodsDetail?goodsId=10011002">打开商品详情</a>

原生调用

  Intent intent = new Intent(Intent.ACTION_VIEW,Uri.parse("xl://goods:8888/goodsDetail?goodsId=10011002"));
  startActivity(intent);

4.)如何判断一个Scheme是否有效

PackageManager packageManager = getPackageManager();
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse("xl://goods:8888/goodsDetail?goodsId=10011002"));
List<ResolveInfo> activities = packageManager.queryIntentActivities(intent, 0);
boolean isValid = !activities.isEmpty();
if (isValid) {
    startActivity(intent);
}

说明:如果遇到特殊情况参考

实际使用中的一些小细节
当自定义的URL配置在LAUNCHER对应的Activity上时,上述配置就足够了。
但是当自定义的URL配置在非LAUNCHER对应的Activity时,还需要增加额外几步操作。

问题一:使用自定义的URL启动Activity时,默认是已FLAG_ACTIVITY_NEW_TASK的方式启动的,所以可能存在URL启动的Activity跟应用已启动的Activity不再同一个堆栈的现象。
解决方式:这种情况下,需要在manifest中将Activity多配置一个taskAffinity属性,约束URL启动的Activity与应用自身的启动的Activity在同一个堆栈中。

问题二:应用A使用url的方式唤起应用B的Activity时,可能存在应用B的Activity启动了,但是堆栈仍然在后台的现象,即应用B的Activity没有聚焦的问题。
解决方式:这种情况下,应用B的Activity收到启动的请求后,可以主动将Activity对应的堆栈移动到最前端。

  1. ActivityManager activityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);  
  2. activityManager.moveTaskToFront(getTaskId(), ActivityManager.MOVE_TASK_WITH_HOME);

使用这种方式需要注意的是该api是在Android 3.0(api 11)以后才提供的,现在基本上手机rom版本都是Android4.4以上了,就不太需要关注3.0一下怎么处理了,且使用这个需要在manifest中申请android.permission.REORDER_TASKS权限。

H5部分

Android平台可以通过Url Scheme来调用第三方应用,设置步骤如下:

5+应用配置支持urlscheme
双击应用的manifest.json文件,切换到“代码视图”,在根节点下添加plus->distribute->google下添加schemes节点数据如下:


QQ截图20180419170602.jpg

其值为字符串数组,每个字符串为一个urlscheme,使用小写字母,可设置多个。
保存后提交App云端打包生效
浏览器中通过href启动应用
安装应用后,我们可以在html页面中,通过href直接调用应用:

<a href="test://abc">test:<a><br/>

5+应用中处理urlscheme启动传递的参数
在其它应用中通过href调用Url Scheme传递过来的值,可以通过plus.runtime.arguments获取
其值为完整的urlscheme字符串,如上面href的值启动应用后获取的plus.runtime.arguments值为“test://abc

代码示例如下:
document.addEventListener('plusready',function(){
    checkArguments();
},false);
// 判断启动方式
function checkArguments(){
    console.log("plus.runtime.launcher: "+plus.runtime.launcher);
    var args= plus.runtime.arguments;
    if(args){
        // 处理args参数,如打开新页面等
    }
}
// 处理从后台恢复
document.addEventListener('newintent',function(){
    console.log("addEventListener: newintent");
    checkArguments();
},false);

android实例
参考 :https://www.jianshu.com/writer#/notebooks/24554617/notes/26872651

iOS平台请参考:

iOS平台通过UrlSchemes与第三方应用相互调用

相关文章

网友评论

    本文标题:Android业务组件化之URL Scheme使用

    本文链接:https://www.haomeiwen.com/subject/jmowkftx.html