自定义Uri启动其它APP与Notification

Home / Android MrLee 2015-4-1 3094

一、自定义Uri与外部启动

1、概述

上篇我们讲了Uri的结构,在这篇中,我们将看看如何利用自定义的URI来启动我的的应用。 有时,我们要通过外部Uri链接来启动我们的应用,主要是通过Uri隐式Intent匹配的方式:
Uri uri = Uri.parse("qijian://test.uri.activity?action=1");
Intent intent = new Intent("android.qijian.schemeurl.activity");
intent.setData(uri);
startActivity(intent);
这里通过隐式Intent匹配来启动应用,在这里我们自定义了一个Uri结构:qijian://test.uri.activity?action=1 我们的应用在隐式匹配Intent时,使用的语法为:

    
        
        
        
    

我们这里在匹配Intent时,使用指定scheme和host来精确匹配过来的Uri,以防止同名scheme就能启动我们的activity,即本来可能要启人家应用,确我们也横插一脚,用户体验很不好,一定要做到精确匹配,以防大家URI一样出现多个应用让用户选择的情况。 这样,第三方就能通过这个Uri来匿名启动我们的Activity了。

2、实例

(1)、新建用于外部启动的Activity 首先,我们先建一个应用,命名为:SchemeURL,在这个应用中我们新建一个Activity命名为:secondActivity,其XML代码如下: (这个Activity是为了在外部启动,为了标识这个Activity是这个应用的,把背景色改成了黄色,文字改上了“SchemeURL 的Activity”)

    
(2)、在AndroidManifest.xml中添加Intent-filter过滤代码:
在SecondAcitivity中添加上Intent-filter用于隐式启动Intent,由于我们定义的Uri格式为:qijian://test.uri.activity?action=1,所以我们固定schemeurl和host,通过query来传递参数即可; 除了Uri匹配,我这里还添加上了Action:“android.qijian.schemeurl.activity”,所以我们在第三方隐式匹配时要同时通过Uri和action来同时匹配才能通过这里的Intent-filter

    
        
        
        
    

(3)新建应用:UseSchemeURL,通过自定义的Uri来从外部调起SecondActivity 这个应用的外观是这个样子的:当点击按钮时,调起SchemeURL的SecondActivity:


代码为:
Button btn = (Button) findViewById(R.id.btn_try);
btn.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        Uri uri = Uri.parse("qijian://test.uri.activity?action=1");
        Intent intent = new Intent("android.qijian.schemeurl.activity");
        intent.setData(uri);
        startActivity(intent);
    }
});

(4)、进阶:通过Uri来传递参数,并处理 在上面,我们已经能够通过Uri来进入我们的应用,我们上面只是固定了Uri的scheme部分和host部分,对于其它部分并没有固定,所以我们可以通过其它部分来传递参数,进而完成指定的功能:比较进入指定的页面或做出指定的操作,等 比如,我们在SchemeURL中,对Uri进行接收,并将结果显示出来:
public class SecondActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        Intent intent = getIntent();
        if (null != intent) {
            Uri uri = intent.getData();
            if (uri == null) {
                return;
            }
            String acionData = uri.getQueryParameter("action");
            TextView tv = (TextView)findViewById(R.id.qijian_test_tv);
            tv.append("n传过来的action值为:" + acionData);
        }
    }
}

结果截图如下:


源码在文章底部给出;

二、特殊应用:Notification与应用启动

有关通过Uri启动APP的经典应用,应当数通过推送消息启动我们应用的指定页面或做出特定的操作了。这部分,我们就看看如何通过推送的通知栏消息来进入我们的应用。
效果如下:

  • 首先:SchemeURL工程代码都不变,我们依然通过隐式匹配Intent来启动SecondActivity.
  • 然后:在UseSchemeURL工程中新加一个按钮,当点击时,发送Notification通知,点击跳转到SchemeURL工程的SecondActivity
其中发送Notification代码如下:
private void pushNotify() {
    NotificationManager notifyManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    NotificationCompat.Builder builder;
    builder = new NotificationCompat.Builder(this);
    builder.setSmallIcon(R.drawable.ic_launcher)
            .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher))
            .setContentTitle("harvic")
            .setContentText("test schemeURL")
            .setTicker("有新消息")
            .setOngoing(false)
            .setWhen(System.currentTimeMillis())
            .setPriority(Notification.PRIORITY_DEFAULT)
            .setAutoCancel(true);
    Uri uri = Uri.parse("qijian://test.uri.activity?action=1");
    Intent intent = new Intent("android.qijian.schemeurl.activity");
    intent.setData(uri);
    PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
    builder.setContentIntent(pendingIntent);
    Notification notification = builder.build();
    notifyManager.notify(1111, notification);
}

有关Notification的知识,我就不再讲了,最重要的是PendingIntent的封装!
即如下代码:
Uri uri = Uri.parse("qijian://test.uri.activity?action=1");
Intent intent = new Intent("android.qijian.schemeurl.activity");
intent.setData(uri);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);

在这里大家也可以看到,这是利用隐式Intent匹配的方式来启动我们的Activity的!所有的APP在通知栏进入到自己的应用,都是通过这个方式来的!
源码在文章底部给出

三、有关path、pathPrefix、pathPattern 之间的区别

细心的同学可能在AndroidManifest.xml中已经发现,Intent-filter的data域除了scheme、host、port这些已知的还有三个参数我们没用过——path、pathPrefix、pathPattern;它们也是用来隐式匹配Intent的,即:

    
        
        
        
    

这里主要说的区别是 path、pathPrefix、pathPattern 之间的区别
  • path: 用来匹配完整的路径,如:http://example.com/blog/abc.html,这里将 path 设置为 /blog/abc.html 才能够进行匹配;
  • pathPrefix: 用来匹配路径的开头部分,拿上面的 Uri 来说,这里将 pathPrefix 设置为 /blog 就能进行匹配了;
  • pathPattern: 用表达式来匹配整个路径,这里需要说下匹配符号与转义。
匹配符号: 1、“” 用来匹配0次或更多,如:“a” 可以匹配“a”、“aa”、“aaa”… 2、“.” 用来匹配任意字符,如:“.” 可以匹配“a”、“b”,“c”… 3、因此 “.*” 就是用来匹配任意字符0次或更多,如:“.*html” 可以匹配 “abchtml”、“chtml”,“html”,“sdf.html”… 转义: 因为当读取 Xml 的时候,“ ” 是被当作转义字符的(当它被用作 pathPattern 转义之前),因此这里需要两次转义,读取 Xml 是一次,在 pathPattern 中使用又是一次。如:“” 这个字符就应该写成 “ ”,“ ” 这个字符就应该写成 “ ”。

样例:匹配 http 以 “.pdf” 结尾的路径

如果我们想要匹配 http 以 “.pdf” 结尾的路径,使得别的程序想要打开网络 pdf 时,用户能够可以选择我们的程序进行下载查看。
我们可以将 scheme 设置为 “http”,pathPattern 设置为 “.* .pdf”,整个 intent-filter 设置为:

     
     
     

如果你只想处理某个站点的 pdf,那么在 data 标签里增加 android:host="yoursite.com" 则只会匹配 http://yoursite.com/xxx/xxx.pdf,但这不会匹配 www.yoursite.com,如果你也想匹配这个站点的话,你就需要再添加一个 data 标签,除了 android:host 改为 “www.yoursite.com” 其他都一样。

四、特殊:如何从网页中通过Uri启动我们的应用

如果想要从网页中点击一个链接跳转到我们的应用,那除了Intent-filter中的各种匹配工作,还应该加上一个属性:

即,以我们的SecondActivity为例,它完整的AndroidManifest.xml的配置方式为:

    
        
        
        
        
    

参考文章:《intent-filter 之 data 「scheme, host, port, mimeType, path, pathPrefix, pathPattern」》

源码内容:(两个文件夹分别是)
1、Part1:自定义Uri与外部启动 2、PART2:特殊应用:Notification与应用启动
每个文件夹里面包含两个工程:
1、SchemeURL:这里提供第三方调用的SecondActivity
2、UseScheme:在这个APP里通过隐式匹配调用SchemeURL的SecondActivity

如果本文有帮到你,记得关注哦
源码下载地址:http://download.csdn.net/detail/harvic880925/8551159
转于:http://blog.csdn.net/harvic880925

本文链接:https://www.it72.com/1732.htm

推荐阅读
最新回复 (0)
返回