编写:spencer198711 - 原文:http://developer.android.com/training/contacts-provider/modify-data.html
这一课向你展示了如何使用Intent去插入一个新的联系人或者修改联系人的数据。我们不是直接访问Contacts Provider,而是通过Intent去启动Contacts应用的适当的Activity。对于这一课中描述的数据修改行为,如果你向Intent发送扩展的数据,它会自动填充进启动的Activity页面中。
使用Intent去插入或者更新一个联系人是比较推荐的修改Contacts Provider的做法。原因如下:
- 节省了你去开发自己的UI和代码的时间和精力
- 避免了由于不符合Contacts Provider规则的修改,从而引入错误
- 减少你的应用所需要申请的权限数量。你的应用不需要去申请写Contacts Provider的权限,因为它把修改行为委托给了已经拥有这个权限的Contacts应用。
当你的应用接收到新的数据时,你通常会允许用户去插入一个新的联系人。例如,一个餐馆评论应用可以允许用户在评论餐馆的时候,把这个餐馆添加为一个联系人。可以使用Intent去做这个任务,使用你所拥有的尽可能多的数据去创建对应的Intent,然后发送这个Intent到Contacts应用。
使用Contacts应用去插入一个联系人将会向Contacts Provider中的ContactsContract.RawContact表中插入一个原始联系人。必要的情况下,在创建原始联系人的时候,Contacts应用将会提示用户选择账户类型和要使用的账户。如果联系人已经存在,Contacts应用也会告知用户。用户将会有取消插入的选项,在这种情况下不会有联系人创建。想要知道更多关于原始联系人的信息,请参阅Contacts Provider的API指导。
开始需要创建一个拥有Intents.Insert.ACTION的Intent对象,并设置其MIME类型为RawContacts.CONTENT_TYPE。例如:
...
// Creates a new Intent to insert a contact
Intent intent = new Intent(Intents.Insert.ACTION);
// Sets the MIME type to match the Contacts Provider
intent.setType(ContactsContract.RawContacts.CONTENT_TYPE);
如果你已经获得了此联系人的详细信息,比如说电话号码或者email地址,你可以把它们作为扩展数据添加到Intent中。对于键值,需要使用Intents.Insert中对应的常量。Contacts应用将会在插入界面显示这些数据,以便用户作进一步的数据编辑和数据添加。
/* Assumes EditText fields in your UI contain an email address
* and a phone number.
*
*/
private EditText mEmailAddress = (EditText) findViewById(R.id.email);
private EditText mPhoneNumber = (EditText) findViewById(R.id.phone);
...
/*
* Inserts new data into the Intent. This data is passed to the
* contacts app's Insert screen
*/
// Inserts an email address
intent.putExtra(Intents.Insert.EMAIL, mEmailAddress.getText())
/*
* In this example, sets the email type to be a work email.
* You can set other email types as necessary.
*/
.putExtra(Intents.Insert.EMAIL_TYPE, CommonDataKinds.Email.TYPE_WORK)
// Inserts a phone number
.putExtra(Intents.Insert.PHONE, mPhoneNumber.getText())
/*
* In this example, sets the phone type to be a work phone.
* You can set other phone types as necessary.
*/
.putExtra(Intents.Insert.PHONE_TYPE, Phone.TYPE_WORK);
一旦你已经创建好Intent,调用startActivity()发送到Contacts应用。
/* Sends the Intent
*/
startActivity(intent);
这个调用将会打开Contacts应用的界面,并允许用户进入一个新的联系人。这个联系人可用的账户类型和账户名字列在屏幕的上方。一旦用户输入数据并点击确定。Contacts应用的联系人列表则会显示出来。用户可以点击Back键返回。
如果用户已经选择了一个感兴趣的联系人,使用Intent去编辑这个已存在的联系人会很有用。例如,一个用来查找拥有邮政地址但是缺少邮政编码的联系人的应用,可以给用户查找邮政编码的选项,然后把它添加到这个联系人中。
使用Intent编辑已经存在的联系人,同插入一个联系人的步骤类似。创建一个在使用Intent插入新的联系人那一节课描述的Intent,但是需要给这个Intent添加对应联系人的Contacts.CONTENT_LOOKUP_URI和MIME类型Contacts.CONTENT_ITEM_TYPE。如果你想要使用已经拥有的详情信息编辑这个联系人,你需要把这些数据放到Intent的扩展数据中。同时注意有些列是不能使用Intent编辑的,这些不可编辑的列在ContactsContract.Contacts的API参考文档中的摘要部分的“Update”标题下有列出。
最后发送这个Intent。作为回应,Contacts应用会显示一个编辑界面。当用户编辑完成并保存,Contacts应用会显示一个联系人列表。当用户点击Back,就回显示你自己的应用。
为了能够编辑一个联系人,需要调用Intent(action)去创建一个拥有ACTION_EDIT行为的Intent,调用setDataAndType()去设置这个Intent要编辑的联系人的Contacts.CONTENT_LOOKUP_URI和MIME类型Contacts.CONTENT_ITEM_TYPE。因为对setType()的调用会重写Intent的当前data数据,你必须同时设置data数据和MIME类型。
为了得到联系人的Contacts.CONTENT_LOOKUP_URI,需要调用Contacts.getLookupUri(id, lookupkey)方法,并用这个联系人的Contacts._ID和Contacts.LOOKUP_KEY作为参数。
以下的代码片段展示了如何创建这个Intent:
// The Cursor that contains the Contact row
public Cursor mCursor;
// The index of the lookup key column in the cursor
public int mLookupKeyIndex;
// The index of the contact's _ID value
public int mIdIndex;
// The lookup key from the Cursor
public String mCurrentLookupKey;
// The _ID value from the Cursor
public long mCurrentId;
// A content URI pointing to the contact
Uri mSelectedContactUri;
...
/*
* Once the user has selected a contact to edit,
* this gets the contact's lookup key and _ID values from the
* cursor and creates the necessary URI.
*/
// Gets the lookup key column index
mLookupKeyIndex = mCursor.getColumnIndex(Contacts.LOOKUP_KEY);
// Gets the lookup key value
mCurrentLookupKey = mCursor.getString(mLookupKeyIndex);
// Gets the _ID column index
mIdIndex = mCursor.getColumnIndex(Contacts._ID);
mCurrentId = mCursor.getLong(mIdIndex);
mSelectedContactUri =
Contacts.getLookupUri(mCurrentId, mCurrentLookupKey);
...
// Creates a new Intent to edit a contact
Intent editIntent = new Intent(Intent.ACTION_EDIT);
/*
* Sets the contact URI to edit, and the data type that the
* Intent must match
*/
editIntent.setDataAndType(mSelectedContactUri,Contacts.CONTENT_ITEM_TYPE);
在android 4.0(API版本14)之后,Contacts应用中的一个问题会导致错误的页面导航。当你的应用发送一个编辑联系人的Intent到Contacts应用,然后用户编辑并保存这个联系人,当用户点击Back键的时候会看到联系人列表页面,为了能够导航回到你自己的应用,用户不得不点击最近使用的应用,然后选择你的应用,才能回到之前的页面。
要在android 4.0.3(API版本15)及以后的版本解决此问题,需要添加finishActivityOnSaveCompleted扩展数据参数到这个Intent,并将它的值设置为true。Android 4.0之前的版本也能够接受这个参数,但是不起作用。为了设置扩展数据,请按照以下方式去做:
// Sets the special extended data for navigation
editIntent.putExtra("finishActivityOnSaveCompleted", true);
对Intent添加额外的扩展数据,需要调用putExtra()。可以为常见的联系人数据字段添加扩展数据,这些常见字段的key值可以从Intents.InsertAPI参考文档中查到。记住ContactsContract.Contacts表中的有些列是不能编辑的,这列在ContactsContract.Contacts的API参考文档中的摘要部分的“Update”标题下有列出。
最后,发送你已经构建好的Intent。例如:
// Sends the Intent
startActivity(editIntent);
你可以通过发送ACTION_INSERT_OR_EDIT
行为的Intent,让用户去选择是插入联系人还是编辑已有的联系人。例如,一个email客户端应用会允许用户添加一个收件地址到新的联系人,或者仅仅作为额外的邮件地址添加到已有的联系人。需要为这个Intent设置MIME类型Contacts.CONTENT_ITEM_TYPE,但是不需要设置数据URI。
当你发送这个Intent后,Contacts应用会展示一个联系人列表,用户可以选择是插入一个新的联系人还是挑选一个存在的联系人去编辑。任何你添加到Intent中得扩展数据字段都会填充在界面上。你可以使用任何在Intents.InsertAPI参考文档中制定的的key值。以下的代码片段展示了如何构建和发送这个Intent:
// Creates a new Intent to insert or edit a contact
Intent intentInsertEdit = new Intent(Intent.ACTION_INSERT_OR_EDIT);
// Sets the MIME type
intentInsertEdit.setType(Contacts.CONTENT_ITEM_TYPE);
// Add code here to insert extended data, if desired
...
// Sends the Intent with an request ID
startActivity(intentInsertEdit);