DBMNG数据库管理与应用

书籍是全世界的营养品。生活里没有书籍,就好像没有阳光;智慧里没有书籍,就好像鸟儿没有翅膀。
当前位置:首页 > 移动应用 > Android

android通过自定义Uri Scheme从外部启动APP与Notification

一、自定义Uri与外部启动

1、概述

上篇我们讲了Uri的结构,在这篇中,我们将看看如何利用自定义的URI来启动我的的应用。 有时,我们要通过外部Uri链接来启动我们的应用,主要是通过Uri隐式Intent匹配的方式:

  1. Uri uri = Uri.parse("qijian://test.uri.activity?action=1");  
  2. Intent intent = new Intent("android.qijian.schemeurl.activity");  
  3. intent.setData(uri);  
  4. startActivity(intent);  
这里通过隐式Intent匹配来启动应用,在这里我们自定义了一个Uri结构:qijian://test.uri.activity?action=1
我们的应用在隐式匹配Intent时,使用的语法为:

  1. <activity  
  2.     android:name=".SecondActivity"  
  3.     android:label="@string/title_activity_second">  
  4.     <intent-filter>  
  5.         <action android:name="android.qijian.schemeurl.activity" />  
  6.         <category android:name="android.intent.category.DEFAULT" />  
  7.         <data  
  8.             android:scheme="qijian"  
  9.             android:host="test.uri.activity" />  
  10.     </intent-filter>  
  11. </activity>  
我们这里在匹配Intent时,使用指定scheme和host来精确匹配过来的Uri,以防止同名scheme就能启动我们的activity,即本来可能要启人家应用,确我们也横插一脚,用户体验很不好,一定要做到精确匹配,以防大家URI一样出现多个应用让用户选择的情况。
这样,第三方就能通过这个Uri来匿名启动我们的Activity了。

2、实例


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



  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:background="#ffff00"  
  6.     tools:context="com.harvic.com.schemeurl.SecondActivity">  
  7.   
  8.     <TextView  
  9.         android:text="SchemeURL 的Activity"  
  10.         android:layout_width="wrap_content"  
  11.         android:layout_height="wrap_content" />  
  12. </RelativeLayout>  
(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

  1. <activity  
  2.     android:name=".SecondActivity"  
  3.     android:label="@string/title_activity_second">  
  4.     <intent-filter>  
  5.         <action android:name="android.qijian.schemeurl.activity" />  
  6.         <category android:name="android.intent.category.DEFAULT" />  
  7.         <data  
  8.             android:scheme="qijian"  
  9.             android:host="test.uri.activity" />  
  10.     </intent-filter>  
  11. </activity>  
(3)新建应用:UseSchemeURL,通过自定义的Uri来从外部调起SecondActivity
这个应用的外观是这个样子的:当点击按钮时,调起SchemeURL的SecondActivity:


代码为:



  1. Button btn = (Button) findViewById(R.id.btn_try);  
  2. btn.setOnClickListener(new View.OnClickListener() {  
  3.     @Override  
  4.     public void onClick(View v) {  
  5.         Uri uri = Uri.parse("qijian://test.uri.activity?action=1");  
  6.         Intent intent = new Intent("android.qijian.schemeurl.activity");  
  7.         intent.setData(uri);  
  8.         startActivity(intent);  
  9.     }  
  10. });  
(4)、进阶:通过Uri来传递参数,并处理
在上面,我们已经能够通过Uri来进入我们的应用,我们上面只是固定了Uri的scheme部分和host部分,对于其它部分并没有固定,所以我们可以通过其它部分来传递参数,进而完成指定的功能:比较进入指定的页面或做出指定的操作,等
比如,我们在SchemeURL中,对Uri进行接收,并将结果显示出来:

  1. public class SecondActivity extends Activity {  
  2.   
  3.     @Override  
  4.     protected void onCreate(Bundle savedInstanceState) {  
  5.         super.onCreate(savedInstanceState);  
  6.         setContentView(R.layout.activity_second);  
  7.         Intent intent = getIntent();  
  8.         if (null != intent) {  
  9.             Uri uri = intent.getData();  
  10.             if (uri == null) {  
  11.                 return;  
  12.             }  
  13.             String acionData = uri.getQueryParameter("action");  
  14.   
  15.             TextView tv = (TextView)findViewById(R.id.qijian_test_tv);  
  16.             tv.append("\n传过来的action值为:" + acionData);  
  17.         }  
  18.     }  
  19. }  
结果截图如下:


源码在文章底部给出;

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

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

效果如下:


  • 首先:SchemeURL工程代码都不变,我们依然通过隐式匹配Intent来启动SecondActivity.
  • 然后:在UseSchemeURL工程中新加一个按钮,当点击时,发送Notification通知,点击跳转到SchemeURL工程的SecondActivity 
其中发送Notification代码如下:
  1. private void pushNotify() {  
  2.     NotificationManager notifyManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);  
  3.     NotificationCompat.Builder builder;  
  4.     builder = new NotificationCompat.Builder(this);  
  5.     builder.setSmallIcon(R.drawable.ic_launcher)  
  6.             .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher))  
  7.             .setContentTitle("harvic")  
  8.             .setContentText("test schemeURL")  
  9.             .setTicker("有新消息")  
  10.             .setOngoing(false)  
  11.             .setWhen(System.currentTimeMillis())  
  12.             .setPriority(Notification.PRIORITY_DEFAULT)  
  13.             .setAutoCancel(true);  
  14.   
  15.     Uri uri = Uri.parse("qijian://test.uri.activity?action=1");  
  16.     Intent intent = new Intent("android.qijian.schemeurl.activity");  
  17.     intent.setData(uri);  
  18.     PendingIntent pendingIntent = PendingIntent.getActivity(this0, intent, PendingIntent.FLAG_UPDATE_CURRENT);  
  19.     builder.setContentIntent(pendingIntent);  
  20.     Notification notification = builder.build();  
  21.     notifyManager.notify(1111, notification);  
  22. }  
有关Notification的知识,我就不再讲了,最重要的是PendingIntent的封装!
即如下代码:
  1. Uri uri = Uri.parse("qijian://test.uri.activity?action=1");  
  2. Intent intent = new Intent("android.qijian.schemeurl.activity");  
  3. intent.setData(uri);  
  4. PendingIntent pendingIntent = PendingIntent.getActivity(this0, intent, PendingIntent.FLAG_UPDATE_CURRENT);  
在这里大家也可以看到,这是利用隐式Intent匹配的方式来启动我们的Activity的!所有的APP在通知栏进入到自己的应用,都是通过这个方式来的!
源码在文章底部给出


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

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


  1. <activity  
  2.     android:name=".SecondActivity"  
  3.     android:label="@string/title_activity_second">  
  4.     <intent-filter>  
  5.         <action android:name="android.qijian.schemeurl.activity" />  
  6.         <category android:name="android.intent.category.DEFAULT" />  
  7.         <data  
  8.             android:scheme="qijian"  
  9.             android:host="test.uri.activity"  
  10.             android:path=""  
  11.             android:pathPrefix=""  
  12.             android:pathPattern=""/>  
  13.     </intent-filter>  
  14. </activity>  
这里主要说的区别是 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 设置为:

  1. <intent-filter>  
  2.      <action android:name="android.intent.action.VIEW"></action>  
  3.      <category android:name="android.intent.category.DEFAULT"></category>  
  4.      <data android:scheme="http" android:pathPattern=".*\\.pdf"></data>  
  5. </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中的各种匹配工作,还应该加上一个属性:


  1. <category android:name="android.intent.category.BROWSABLE"/>  
即,以我们的SecondActivity为例,它完整的AndroidManifest.xml的配置方式为:

  1. <activity  
  2.     android:name=".SecondActivity"  
  3.     android:label="@string/title_activity_second">  
  4.     <intent-filter>  
  5.         <action android:name="android.qijian.schemeurl.activity" />  
  6.         <category android:name="android.intent.category.DEFAULT" />  
  7.         <category android:name="android.intent.category.BROWSABLE"/>  
  8.         <data  
  9.             android:scheme="qijian"  
  10.             android:host="test.uri.activity"/>  
  11.     </intent-filter>  
  12. </activity>  


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


源码内容:(两个文件夹分别是)

1、Part1:自定义Uri与外部启动
2、PART2:特殊应用:Notification与应用启动

每个文件夹里面包含两个工程:

1、SchemeURL:这里提供第三方调用的SecondActivity

2、UseScheme:在这个APP里通过隐式匹配调用SchemeURL的SecondActivity

from:http://blog.csdn.net/harvic880925/article/details/44781557

本站文章内容,部分来自于互联网,若侵犯了您的权益,请致邮件chuanghui423#sohu.com(请将#换为@)联系,我们会尽快核实后删除。
Copyright © 2006-2023 DBMNG.COM All Rights Reserved. Powered by DEVSOARTECH            豫ICP备11002312号-2

豫公网安备 41010502002439号