专栏首页分享达人秀Intent 属性详解(下)

Intent 属性详解(下)

上一期学习了Intent的前三个属性,本期接着学习其余四个属性,以及Android系统常用内置组件的启动。

四、Data和Type属性

Data属性通常用于向Action属性提供操作的数据。Data属性接受一个Uri对象,一个Uri 对象通常通过如下形式的字符串来表示:

content://com.android.contacts/contacts/1tel:123

Uri字符串总满足如下格式:

scheme://host:port/path

例如上面给出的 content://com.android.contacts/contacts/l,其中 content 是 scheme 部分, com.android.contacts 是 host 部分,port 部分被省略了,/contacts/1 是 path 部分。

Type属性用于指定该Data属性所指定Uri对应的MIME类型,这种MIME类型可以是任何自定义的MIME类型,只要符合abc/xyz格式的字符串即可。

Data属性与Type属性的关系比较微妙,这两个属性会相互覆盖,例如:

  • 如果为Intent先设置Data属性,后设置Type属性,那么Type属性将会覆盖Data属性。
  • 如果为Intent先设置Type属性,后设置Data属性,那么Data属性将会覆盖Type属性。
  • 如果希望Intent既有Data属性,也有Type属性,则应该调用Intent的setDataAndType() 方法。

下面的示例演示了 Intent的Data属性与Type属性互相覆盖的情形,该示例的界面布局文件很简单,只是定义了三个按钮,并为三个按钮绑定了事件处理。 Activity的代码如下:

public class MainActivity extends AppCompatActivity {   
 @Override    
 protected void onCreate(Bundle savedInstanceState) {        
   super.onCreate(savedInstanceState);        
   setContentView(R.layout.activity_main);        
   Button overrideTypeBtn = (Button) findViewById(R.id.override_type_btn);        
   Button overrideDataBtn = (Button) findViewById(R.id.override_data_btn);        
   Button dataAndTypeBtn = (Button) findViewById(R.id.data_type_btn);        
   overrideTypeBtn.setOnClickListener(new View.OnClickListener() {          
     @Override            
     public void onClick(View view) {           
          Intent intent = new Intent();                // 先为Intent设置Type属性                
          intent.setType("abc/xyz");                // 再为Intent设置Data属性,覆盖Type属性                
          intent.setData(Uri.parse("scheme://host:port/path"));                
          Toast.makeText(MainActivity.this, intent.toString(), 
          Toast.LENGTH_LONG).show();            }        });        
          overrideDataBtn.setOnClickListener(new View.OnClickListener() {           
           @Override            
           public void onClick(View view) {          
                 Intent intent = new Intent();                // 先为Intent设置Data属性               
                 intent.setData(Uri.parse("scheme://host:port/path"));                // 再为Intent设置Type属性,覆盖Data属性                
                 intent.setType("abc/xyz");                
                 Toast.makeText(MainActivity.this, intent.toString(), Toast.LENGTH_LONG).show();
                 }       
            });        
     dataAndTypeBtn.setOnClickListener(new View.OnClickListener() {          
       @Override            
       public void onClick(View view) {              
         Intent intent = new Intent();                // 同时设置Intent的Data、Type属性                
         intent.setDataAndType(Uri.parse("scheme://host:port/path"), "abc/xyz");                
         Toast.makeText(MainActivity.this, intent.toString(), Toast.LENGTH_LONG).show();        
             }      
               });  
 }}

上面的三个事件监听方法分别为Intent设置了 Data、Type属性,第一个事件监听方法先设置了 Type属性,再设置了 Data属性,这将导致Data属性覆盖Type属性。单击按钮激发该事件监听方法,将可以看到下图所示提示信息。

从上图可以看出,此时的Intent只有Data属性,Type属性被覆盖了。

第二个事件监听方法先设置了 Data属性,再设置了 Type属性,这将导致Type属性覆盖 Data属性。单击按钮激发该事件监听方法,将可以看到下图左侧所示提示信息。

第三个事件监听方法同时设置了 Data、Type属性,这样该Intent中才会同时具有Data、Type属性。单击按钮激发该事件监听方法,将可以看到上图右侧图所示提示信息。

在 AndroidManifest.xml 文件中为组件声明 Data、Type 属性都通过<data.../>元素,<data.../> 元素的格式如下:

<data android:mimeType=""    
 android:scheme=""    
 android:host=""    
 android:port=""    
 android:path=""    
 android:pathPrefix=""    
 android:pathPattern=""/>

上面的<data.../>元素支持如下属性。

  • mimeType:用于声明该组件所能匹配的Intent的Type属性。
  • scheme:用于声明该组件所能匹配的Intent的Data属性的scheme部分。
  • host:用于声明该组件所能匹配的Intent的Data属性的host部分。
  • port:用于声明该组件所能匹配的Intent的Data属性的port部分。
  • path:用于声明该组件所能匹配的Intent的Data属性的path部分。
  • pathPrefix:用于声明该组件所能匹配的Intent的Data属性的path前缀。
  • pathPattern:用于声明该组件所能匹配的Intent的Data属性的path字符串模板。

Intent的Type属性也用于指定该Intent的要求,对应组件中<intent-filter.../>元素的<data.../>子元素的mimeType属性必须与此相同,才能启动该组件。

Intent的Data属性则略有差异,程序员为Intent指定Data属性时,Data属性的Uri对象实际上可分为scheme、host、port和path部分,此时并不要求被启动组件的<intent-filter.../>中<data.../>之元素android:scheme、android:host、android:port、android:path 完全满足。

Data属性的“匹配”过程则有些差别,它会先检查<intent-filter.../>里的<data.../>子元素, 然后:

  1. 如果目标组件的<data.../>子元素只指定了 android:scheme属性,那么只要Intent的Data 属性的scheme部分与android:scheme属性值相同,即可启动该组件。
  2. 如果目标组件的<data.../>子元素只指定了 android:scheme、android:host属性,那么只 要 Intent 的 Data 属性的 scheme、host 部分与 android:scheme、android:host 属性值相同,即可启动该组件。
  3. 如果目标组件的<data.../>子元素指定了 android:scheme、android:host、android:port 属性,那么只要 Intent 的 Data 属性的 scheme、host、port 部分与 android:scheme、android:host、 android:host属性值相同,即可启动该组件。
  4. 如果目标组件的<data.../>子元素只指定了 android:scheme、android:host、android:path 属性,那么只要 Intent 的 Data 属性的 scheme、host、path 部分与 android:scheme、 android:host、android:path属性值相同,即可启动该组件。
  5. 如果目标组件的<data…/>子元素指定了android:scheme、android:host、android:port、 android:path 属性,那么就要求 Intent 的 Data 属性的 scheme、host、port、path 部分依次与 android:scheme、android:host、android:port、android:path 属性值相同,才可启动该组件。

这里有两点需要注意:

  1. 如果<data…/>子元素只有android:port属性,没有指定android:host属性,那么android.port属性将不会起作用。
  2. 如果<data.../>子元素只有android:path属性,没有指定android:host属性,那么android:path属性将不会起作用。

一旦为Intent同时指定了 Action、Data属性,Android就可根据指定的数据类型来启动特定的应用程序,并对指定数据执行相应的操作。

下面几个是常见的Data属性:

  • tel://:号码数据格式,后跟电话号码。
  • mailto://:邮件数据格式,后跟邮件收件人地址。
  • smsto://:短信数据格式,后跟短信接收号码。
  • content://:内容数据格式,后跟需要读取的内容。
  • file://:文件数据格式,后跟文件路径。
  • market://search?q=pname:pkgname:市场数据格式,在Google Market里搜索包名为pkgname的应用。
  • geo://latitude, longitude:经纬数据格式,在地图上显示经纬度所指定的位置。

五、Extra属性

Intent的Extra属性通常用于在多个Action之间进行数据交换,Intent的Extra属性值应该是一个Bundle对象,Bundle对象就像一个Map对象,它可以存入多个key-value对,这样就可以通过Intent在不同Activity之间进行数据交换了。

接下来学习一下系统内置的几个Extra常量:

  • EXTRA_BCC:存放邮件密送人地址的字符串数组。
  • EXTRA_CC:存放邮件抄送人地址的字符串数组。
  • EXTRA_EMAIL:存放邮件地址的字符串数组。
  • EXTRA_SUBJECT:存放邮件主题字符串。
  • EXTRA_TEXT:存放邮件内容。
  • EXTRA_KEY_EVENT:以KeyEvent对象方式存放触发Intent的按键。
  • EXTRA_PHONE_NUMBER:存放调用ACTION_CALL时的电话号码。

关于Extra属性的使用,会在学习Activity数据传递的时候详细讲解,此处不做过多介绍。

六、Flag属性

Intent的Flag属性用于为该Intent添加一些额外的控制旗标,Intent可调用addFlags()方法来添加控制旗标。其实在之前的Activity任务栈和启动模式章节里面有简单提到过。

Intent包含了如下几个常用的Flag属性:

  • FLAG_ACTIVITY_BROUGHT_TO_FRONT:如果通过该 Flag 启动的 Activity 己经存在,下次再次启动时,将只是把该Activity带到前台。例如,现在Activity栈中有Activity A,此时以该旗标启动 Activity B (即 Activity B 是以 FLAG_ACTIVITY_BROUGHT_ TO_FRONT旗标启动的),然后在Activity B中启动Activity C、D,如果这时在Activity D中再启动Activity B,将直接把Activity栈中的Activity B带到前台。此时Activity 栈中情形是 Activity A、C、D、B。
  • FLAG_ACTIVITY_CLEAR_TOP:该Flag相当于加载模式中的singleTask,通过这种 Flag启动的Activity将会把要启动的Activity之上的Activity全部弹出Activity栈。例如,Activity栈中包含A、B、C、D四个Activity,如果采用该Flag从Activity D跳转到Activity B,那么此时Activity栈中只包含A、B两个Activity。
  • FLAG_ACTIVITY_NEW_TASK:默认的启动旗标,该旗标控制重新创建一个新的 Activity。
  • FLAG_ACTIVITY_NO_ANIMATION:该旗标控制启动Activity时不使用过渡动画。
  • FLAG_ACTIVITY_NO_HISTORY:该旗标控制被启动的Activity将不会保留在Activity 栈中。例如,Activity栈中原来有A、B、C三个Activity,此时在Activity C中以该Flag 启动 Activity D,Activity D 再启动 Activity E,此时 Activity 栈中只有 A、B、C、E 四个 Activity,Activity D 不会保留在 Actvity 栈中。
  • FLAG_ACTIVITY_REORDER_TO_FRONT:该 Flag 控制如果当前己有 Activity,则直接将该Activity带到前台。例如,现在Activity栈中有A、B、C、D四个Activity,如果使用FLAG_ACTIVITY_REORDER_TO_FRONT旗标来启动Activity B,那么启动后的Activity栈中情形为A、C、D、B。
  • FLAG_ACTMTY_SINGLE_TOP:该Flag相当于加载模式中的singleTop模式。例如,原来 Activity 栈中有 A、B、C、D.四个 Activity,在 Activity D 中再次启动 Activity D,Activity桟中依然还是A、B、C、D四个Activity。

Android为Intent提供了大量的Flag,每个Flag都有其特定的功能,具体请参考关于Intent 的API文档,这里也不做过多介绍。

七、启动Android系统常用内置组件

在实际开发中,很多时候不只是调用自定义的Activity,还会调用Android系统定义的Activity。接下来介绍一些系统常见Activity的调用方法。

1. 拨打电话

如给移动客服10086拨打电话,其核心代码如下:

Uri uri = Uri.parse("tel:10086");
Intent intent = new Intent(Intent.ACTION_DIAL, uri);
startActivity(intent);

2. 发送短信

如给10086发送内容为“Hello”的短信,其核心代码如下:

Uri uri = Uri.parse("smsto:10086");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
intent.putExtra("sms_body", "Hello");
startActivity(intent);

3. 发送彩信

相当于发送带附件的短信,其核心代码如下:

Intent intent = new Intent(Intent.ACTION_SEND);
intent.putExtra("sms_body", "Hello");
Uri uri = Uri.parse("content://media/external/images/media/23");
intent.putExtra(Intent.EXTRA_STREAM, uri);
intent.setType("image/png");startActivity(intent);

4. 打开浏览器

如打开百度主页,其核心代码如下:

Uri uri = Uri.parse("http://www.baidu.com");
Intent intent  = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

5. 发送电子邮件

如给cqkxzsxy@163.com发邮件,其核心代码如下:

Uri uri = Uri.parse("mailto:cqkxzsxy@163.com");
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
startActivity(intent);

当然一般发送邮件都会带有邮件内容,如给cqkxzsxy@163.com发邮件发送内容为“Hello”的邮件,其核心代码如下:

Intent intent = new Intent(Intent.ACTION_SEND);
intent.putExtra(Intent.EXTRA_EMAIL, "cqkxzsxy@163.com");
intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
intent.putExtra(Intent.EXTRA_TEXT, "Hello");
intent.setType("text/plain");
startActivity(intent);

有时候也需要给多个人发送邮件,其核心代码如下:

Intent intent = new Intent(Intent.ACTION_SEND);
String[] tos = {"cqkxzsxy@163.com", "cqkxzsxy@163.com"}; // 收件人
String[] ccs = {"852100303@qq.com", "852100303@qq.com"}; // 抄送
String[] bccs = {"test@abc.com", "test@abc.com"}; // 密送
intent.putExtra(Intent.EXTRA_EMAIL, tos);
intent.putExtra(Intent.EXTRA_CC, ccs);
intent.putExtra(Intent.EXTRA_BCC, bccs);
intent.putExtra(Intent.EXTRA_SUBJECT, "Subject");
intent.putExtra(Intent.EXTRA_TEXT, "Hello");
intent.setType("message/rfc822");
startActivity(intent);

6. 显示地图

如打开Google地图中国北京位置(北纬39.9,东经116.3),其核心代码如下:

Uri uri = Uri.parse("geo:39.9,116.3");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

7. 路径规划

如从北京某地(北纬39.9,东经116.3)到上海某地(北纬31.2,东经121.4),其核心代码如下:

Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=39.9 116.3&daddr=31.2 121.4");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

8. 多媒体播放

如播放SD卡里面的foo.mp3文件,其核心代码如下:

Intent intent = new Intent(Intent.ACTION_VIEW);
Uri uri = Uri.parse("file:///sdcard/foo.mp3");
intent.setDataAndType(uri, "audio/mp3");
startActivity(intent);

如获取SD卡下所有音频文件,然后播放第一首,其核心代码如下:

Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

9. 打开摄像头拍照

如打开拍照程序,其核心代码如下:

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);startActivityForResult(intent, 0);

如取出照片数据,其核心代码如下:

Bundle extras = intent.getExtras();Bitmap bitmap = (Bitmap) extras.get("data");

如调用系统相机应用程序并存储拍下来的照片,其核心代码如下:

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
long time = Calendar.getInstance().getTimeInMillis();
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment  
.getExternalStorageDirectory().getAbsolutePath()+"/tucue", time + ".jpg")));
startActivityForResult(intent, ACTIVITY_GET_CAMERA_IMAGE);

10. 获取并剪切图片

如获取并剪切图片,其核心代码如下:

Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");intent.putExtra("crop", "true"); // 开启剪切
intent.putExtra("aspectX", 1); // 剪切的宽高比为1:2
intent.putExtra("aspectY", 2);
intent.putExtra("outputX", 20); // 保存图片的宽和高
intent.putExtra("outputY", 40);
intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp"))); // 保存路径
intent.putExtra("outputFormat", "JPEG");// 返回格式
startActivityForResult(intent, 0);

如剪切特定图片,其核心代码如下:

Intent intent = new Intent("com.android.camera.action.CROP");
intent.setClassName("com.android.camera", "com.android.camera.CropImage");
intent.setData(Uri.fromFile(new File("/mnt/sdcard/temp")));
intent.putExtra("outputX", 1); // 剪切的宽高比为1:2
intent.putExtra("outputY", 2);
intent.putExtra("aspectX", 20); // 保存图片的宽和高
intent.putExtra("aspectY", 40);
intent.putExtra("scale", true);
intent.putExtra("noFaceDetection", true);
intent.putExtra("output", Uri.parse("file:///mnt/sdcard/temp"));
startActivityForResult(intent, 0);

11. 打开Google Market

如打开Google Market直接进入该程序的详细页面,其核心代码如下:

Uri uri = Uri.parse("market://details?id=" + "com.demo.app");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

12. 进入手机设置界面

如进入无线网络设置界面,其核心代码如下:

Intent intent = new Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS);
startActivityForResult(intent, 0);

13. 安装apk

安装apk其核心代码如下:

Uri installUri = Uri.fromParts("package", "xxx", null);
Intent intent = new Intent(Intent.ACTION_PACKAGE_ADDED, installUri);
startActivity(intent);

14. 卸载apk

卸载apk其核心代码如下:

Uri uri = Uri.fromParts("package", strPackageName, null);
Intent it = new Intent(Intent.ACTION_DELETE, uri);
startActivity(it);

15. 发送附件

发送附件其核心代码如下:

Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/eoe.mp3");
it.setType("audio/mp3");
startActivity(Intent.createChooser(it, "Choose Email Client"));

16. 进入联系人页面

进入联系人页面,其核心代码如下:

Intent intent = new Intent();intent.setAction(Intent.ACTION_VIEW);
intent.setData(Contacts.People.CONTENT_URI);
startActivity(intent);

17. 查看指定联系人

查看指定联系人,其核心代码如下:

Uri personUri = ContentUris.withAppendedId(People.CONTENT_URI, info.id);// info.id联系人ID
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setData(personUri);
startActivity(intent);

18. 调用系统编辑添加联系人

调用系统编辑添加联系人,其核心代码如下:

Intent intent = newIntent(Intent.ACTION_INSERT_OR_EDIT);
intent.setType(People.CONTENT_ITEM_TYPE);
intent.putExtra(Contacts.Intents.Insert.NAME, "My Name");
intent.putExtra(Contacts.Intents.Insert.PHONE, "+1234567890");
intent.putExtra(Contacts.Intents.Insert.PHONE_TYPE,Contacts.PhonesColumns.TYPE_MOBILE);
intent.putExtra(Contacts.Intents.Insert.EMAIL, "cqkxzsxy@163.com");
intent.putExtra(Contacts.Intents.Insert.EMAIL_TYPE, Contacts.ContactMethodsColumns.TYPE_WORK);
startActivity(intent);

19. 打开另一程序

打开另一程序,其核心代码如下:

Intent intent  = new Intent();
ComponentName cn = new ComponentName("com.ibeifeng.test","com.jinyu.cqkxzsxy.android.test.MainActivity");
intent .setComponent(cn);intent .setAction("android.intent.action.MAIN");
startActivityForResult(intent, RESULT_OK);

20. 打开录音机

打开录音机,其核心代码如下:

Intent mi = new Intent(Media.RECORD_SOUND_ACTION);
startActivity(mi);

这里将常见启用Android系统内置功能的核心代码总结了出来,关于具体的使用建议自己多加练习。

本文分享自微信公众号 - 分享达人秀(ShareExpert),作者:鑫鱻

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2017-10-20

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Intent 属性详解(上)

    Android应用将会根据Intent来启动指定组件,至于到底启动哪个组件,则取决于Intent的各属性。本期将详细介绍Intent的各属性值,以及 A...

    分享达人秀
  • Activity中启动和关闭其他Activity

    上一期我们学习了Activity的创建和配置,当时留了一个悬念,如何才能在默认启动的Activity中打开其他新建的Activity呢?那么本期一起来学...

    分享达人秀
  • 四大组件的纽带——Intent

    前面学习Activity时己经多次使用了 Intent,当一个Activity需要启动另一个Activity时, 程序并没有直接告诉系统要启动哪个Act...

    分享达人秀
  • Android实现开机自动启动APP

    我们在开发的过程中,可能都会遇到要求开机自动启动一个APP这样的需求,于是就有了本文,看下文↓

    IT大飞说
  • Intent 属性详解(上)

    Android应用将会根据Intent来启动指定组件,至于到底启动哪个组件,则取决于Intent的各属性。本期将详细介绍Intent的各属性值,以及 A...

    分享达人秀
  • 我所理解的Intent 和Intent-filter

    1.Intent Intent 是一个消息传递对象,可以使用它从其他应用组件请求操作。尽管 Intent 可以通过多种方式促进组件之间的通信,但其基本用例主要包...

    非著名程序员
  • 英国政府要征收2%的“数字税”,目标是谷歌、亚马逊

    美国科技巨头们最近的日子可能有点不好过,就在本周,英国政府对外宣布,将针对跨国科技公司的收入征收2%的“数字服务税”,该税率预计会于2020年生效,征税范围主要...

    镁客网
  • Android Bundle简介

    Bundle经常使用在Activity之间或者线程间传递数据,传递的数据可以是boolean、byte、int、long、float、double、string...

    李小白是一只喵
  • “一带一路”和丝博会告诉我们:硬科技才是大国崛起的核心助推力

    镁客网
  • 投融资汇总|本周(11.25-12.01),传地平线机器人获得10亿美元的B轮融资

    本周,由前百度研究院院长林元庆创立的Aibee获得A轮融资,后者曾在天使轮刷新当时中国AI初创企业天使轮融资额新纪录。

    镁客网

扫码关注云+社区

领取腾讯云代金券