当前位置: 代码迷 >> Android >> Android系列课程之十二:Intents and Intent Filters(三)
  详细解决方案

Android系列课程之十二:Intents and Intent Filters(三)

热度:375   发布时间:2016-05-01 16:40:53.0
Android系列教程之十二:Intents and Intent Filters(三)

接上节继续。。版权所有:飞雪无情,转载请注明出处:

http://flysnow.iteye.com/blog/978225

Android开发技术交流群86686524(已满)请加120059404

Android系列教程目录:

  1. Android教程之一:Window下搭建Android开发环境

  2. Android教程之二:Linux下搭建Android开发环境

  3. Android教程之三:第一个Android应用,HelloWorld

  4. Android系列教程之四:Android项目的目录结构

  5. Android系列教程之五:Activity的生命周期

  6. Android系列教程之六:TextView小组件的使用--附带超链接和跑马灯效果

  7. Android系列教程之七:EditText使用详解-包含很多教程上看不到的功能演示

  8. Android系列教程之八:ListView组件的使用

  9. Android系列教程之九:GridView组件的使用

  10. Android系列教程之十:Intents and Intent Filters(一)

  11. Android系列教程之十一:Intents and Intent Filters(二)

数据(data)检测

data标记也是在intent-filter中定义的,大致格式如下:

<intent-filter>        		<action android:name="android.intent.action.VIEW"></action>        		<category android:name="android.intent.category.DEFAULT"></category>        		<data android:mimeType="text/plain"></data>        	</intent-filter>

??? ?? 每个data定义一个URI和数据类型(MIME),URI由4个属性来定义,分别是android:scheme,android:host,android:port,android:path..这个四个属性构成如下格式的URI:scheme://host:port/path?? 如:content://com.flysnow.intent:8080/show/view。其中content就是scheme,com.flysnow.intent就是host,8080就是port,show/view就是path...如果有经常使用ContentProvider的应该熟悉。。我们经常定义的authority不就是host+port吗?还有这几个元素都是可选的,但是不是随便用就可以的,port要依赖于host,没有host,port就会被忽略,不起作用,同样,如果要使用host+port(authority)就必须指定scheme。而path则依赖于scheme和authority。。


?????? 还有一个很重要的类型就是mimeType,这个属性用于指定内容的类型,也就是这个组件可以处理哪些类型的内容。。如text/plain表示无格式文本类型,mimeType也支持通配符,使用text/*则表示所有文本类型。通过使用它,你可以很方便的开发出关联打开诸如txt文件,pdf文件的应用。后面的两个自理将会演示txt文件查看器,图片查看器的例子。。MIME可以参考http://www.w3school.com.cn/media/media_mimeref.asp。这里有所有的内容类型的定义。。

开发实例-拨打电话,text阅读器和图片查看器

下面通过一个例子来演示data的检测,项目名为Intents,应用名为Intents and Filters,运行在Android2.2版本上.主启动Activity为IntentsTestList。例子包括以下演示:

  1. 通过发送intent的方式“打开拨号界面并输入电话123456”。
  2. 创建一个Text文件阅读器
  3. 创建一个图片查看器

首先我们实现第一项,修改IntentsTestList类如下:

/** * Intents测试列表类 * @author 飞雪无情 * @since 2011-3-14 */public class IntentsTestList extends ListActivity {    /** Called when the activity is first created. */    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);      //定义ListAdapter		setListAdapter(new SimpleAdapter(this, getData(),				android.R.layout.simple_list_item_1, new String[] { "title" },new int[] {android.R.id.text1}));		getListView().setTextFilterEnabled(true);    }	@Override	protected void onListItemClick(ListView l, View v, int position, long id) {		Intent intent=(Intent)getData().get(position).get("intent");		Boolean isActivity=(Boolean)getData().get(position).get("isActivity");		if(isActivity){			startActivity(intent);		}else{			sendBroadcast(intent);		}	}	/**	 * 返回ListView需要的数据	 * @return ListView需要的数据	 */	private List<Map<String,Object>> getData() {		List<Map<String,Object>> data=new ArrayList<Map<String,Object>>();		addItem(data, "打开拨号界面并输入电话123456", new Intent(Intent.ACTION_DIAL, Uri.parse("tel://123456")), true);		return data;	}	/**	 * 给ListView添加数据	 * @param data 存储数据的List	 * @param name 要显示的Title	 * @param intent 单击某一项时要启动的Activity	 * @param isActivity 启动的是否是Activity,true是,false为广播	 */	private void addItem(List<Map<String,Object>> data, String name, Intent intent,boolean isActivity) {        Map<String, Object> temp = new HashMap<String, Object>();        temp.put("title", name);        temp.put("intent", intent);        temp.put("isActivity", isActivity);        data.add(temp);    }	}

?这时我们运行程序,单击“打开拨号界面并输入电话123456”就会打开系统的自带的拨号界面,并且默认已经录入了要拨打的号码“123456”。效果图如下:

?

? 然后我们实现第二功能-txt文件阅读器

新建TextWatcherActivity代码如下:

/** * 显示文本的Activity * @author 飞雪无情 * @since 2011-3-24 */public class TextWatcherActivity extends Activity {	private TextView mTextView;	@Override	protected void onCreate(Bundle savedInstanceState) {		super.onCreate(savedInstanceState);		mTextView=new TextView(this);		setContentView(mTextView);	}	@Override	protected void onStart() {		super.onStart();		Intent txtIntent=getIntent();		Uri data=txtIntent.getData();		String txt;		try {			txt = readTxt(data);		} catch (IOException e) {			txt="打开txt文件异常";		}		mTextView.setText(txt);	}	/**	 * 读取txt文本	 * @param txtUri	 * @return	 * @throws IOException	 */	private String readTxt(Uri txtUri) throws IOException{		BufferedReader bufferedReader=new BufferedReader(new InputStreamReader(getContentResolver().openInputStream(txtUri),Charset.forName("GBK")));		StringBuilder txt=new StringBuilder();		String buf="";		while((buf=bufferedReader.readLine())!=null){			txt.append(buf).append("\n");		}		return txt.toString();	}		}

?然后在AndroidManifest.xml中加入如下定义:

<activity android:name=".TextWatcherActivity"        	android:label="查看TXT文件">        	<intent-filter>        		<action android:name="android.intent.action.VIEW"></action>        		<category android:name="android.intent.category.DEFAULT"></category>        		<data android:mimeType="text/plain"></data>        	</intent-filter>        </activity>

?这样在单击txt文件的时候就可以选择我们的这个Activity对txt文件处理,显示其内容.我们新建一个1.txt文件,写上一些内容,放在我们的sd卡中,使用文件管理工具查看这个txt文件,会弹出如下图的提示,看到我们刚刚做的《TXT阅读器》了吧。

选择“查看TXT文件”,就可以看到我们的txt内容:

最后实现第三个功能--图片查看器

新建ImageWatcherActivity,代码如下:

/** * 显示文本的Activity * @author 飞雪无情 * @since 2011-3-24 */public class ImageWatcherActivity extends Activity {	private final String IMAGE_URI_KEY="imageUriKey";	private Uri image;	private ImageView mImageView;	@Override	protected void onCreate(Bundle savedInstanceState) {		super.onCreate(savedInstanceState);		mImageView=new ImageView(this);		setContentView(mImageView);	}	@Override	protected void onStart() {		super.onStart();		Intent txtIntent=getIntent();		image=txtIntent.getData();		//对于大图片未做优化处理		mImageView.setImageURI(image);	}	@Override	protected void onRestoreInstanceState(Bundle savedInstanceState) {		super.onRestoreInstanceState(savedInstanceState);		image=savedInstanceState.getParcelable(IMAGE_URI_KEY);		mImageView.setImageURI(image);	}	@Override	protected void onSaveInstanceState(Bundle outState) {		super.onSaveInstanceState(outState);		outState.putParcelable(IMAGE_URI_KEY,image);	}		}

?然后在AndroidManifest.xml中加入如下定义:

<activity android:name=".ImageWatcherActivity"        	android:label="查看图片文件">        	<intent-filter>        		<action android:name="android.intent.action.VIEW"></action>        		<category android:name="android.intent.category.DEFAULT"></category>        		<data android:mimeType="image/*"></data>        	</intent-filter>        </activity>

?这样在单击图片文件的时候就可以选择我们的这个Activity对txt文件处理并且显示。

?

数据(data)检测小结

对于data的匹配,如果说怎么怎么匹配,在什么情况下通过可能会比较难以理解,这里以一种简单的方式来解说。

假定我们定义的Intent Filter 的data标签为集合A,传递的Intent中包含的data为集合B,当B是A的子集时就通过了(Action和Category也得检测通过)。如果B为空(不配置data),那么A也得为空(不配置data)才能通过。更详细(繁琐)的介绍请参考doc

?

Intents and Intent Filters总结

Android提供了以Intent的方式调用Android设备的内置Google应用,比如打电话,调用Google浏览器打开网页,搜索等。关于这方便的介绍可以参考Android开发文档《Intents List: Invoking Google Applications on Android Devices》这一节的介绍,很详细。docs/guide/appendix/g-app-intents.html。


Intent是一个很好的设计,它提供了一种在各个组建之间通信的方式,也为我们使用其他的应用的功能提供了可能,这样如果我们想在自己的应用打开一个网页,我们就不用特意迁入一个webview,我们直接调用Android内的浏览器打开即可。。


最后值得一提的是PackageManager这个类中为我们提供了一系列的query...()方法,可以让我们根据我们定义的Intent查询特定的匹配Intent Filter标记的所有组件。。有兴趣的可以研究一下。。

1 楼 深夜未眠 2011-03-30  
鄙人初学Android,向博主学习。
2 楼 飞雪无情 2011-03-30  
深夜未眠 写道
鄙人初学Android,向博主学习。

共同学习
3 楼 fortianwei 2011-05-15  
楼主在吗,请问你有没有弄android付费软件?需要LVL授权验证那个?我现在遇到的问题就是:不管是谁的手机测试,返回的都是LICENSED,都自动回调allow()函数,付费软件默认的不应该是NOT_LICENSED的吗?如果楼主看到了还请您帮帮我啊。
4 楼 飞雪无情 2011-05-15  
fortianwei 写道
楼主在吗,请问你有没有弄android付费软件?需要LVL授权验证那个?我现在遇到的问题就是:不管是谁的手机测试,返回的都是LICENSED,都自动回调allow()函数,付费软件默认的不应该是NOT_LICENSED的吗?如果楼主看到了还请您帮帮我啊。
不好意思,美做过付费的,这个没有研究过,不过应该需要Google Market的License
  相关解决方案