访问地址簿和单个联系人数据的接口是基于C语言的函数,接口传递对地址簿各种对象的引用作为参数。管理地址簿中条目的基类对象是 ABRecord。一个 ABRecord 可以表示一个人 或者一个群体 ABGroup 。无论何时,在界面上选中一条记录,或者使用框架进行查询返回一条结果,系统都会返回一个指向 ABRecord 的指针,标示为 ABRecordRef。与地址簿 API 的大部分交互都涉及使用 ABRecordRef 这个引用类型。
常用访问记录的函数如下:
ABRecordID ABRecordGetRecord(ABRecordRef record);
返回 ABRecordID ,代表了 记录在底层数据库中的ID号。具有唯一性。
ABRecordType ABRecordGetRecordType(ABRecordRef record);
返回记录类型。可以是 kABPersonType 和 kABGroupType
CFStringRef ABRecordCopyCompositeName(ABRecordRef record);
返回个人或群体完整名称。例:NSString* name = (NSString*)ABRecordCopyCompositeName(record);
一、高层地址簿函数
1.获得地址簿句柄
你必须先初始化地址簿,然后才能对其进读写操作。要获得一个地址簿句柄,可以使用 ABAddressBookCreate函数:
[java] view plain copy print ?
- #import <AddressBook/AddressBook.h>
- AddressBookRef ab = AddressBookCreate();
2. 保存地址簿
获得了地址簿的引用就可以对其进行操作了,操作完毕要记得保存:
[java] view plain copy print ?
- CFErrorRef err;
- BOOL success = ABAddressBookSave(ab, &err);
如果不确定是否需要保存,则可以用:
[java] view plain copy print ?
- BOOL hasUnsavedChanges = ABAddressBookHasUnsavedChanges(ab);
3.添加/删除 记录
[java] view plain copy print ?
- CFErrorRef err;
- BOOL success =ABAddressBookAddRecord(ab, record, &err);
[java] view plain copy print ?
- CFErrorRef err;
- BOOL success =ABAddressBookRemoveRecord(ab, record, &err);
二、查询地址簿
地址簿框架仅仅提供了基本的查询功能。可以用函数根据名字来查询多个记录,或者是根据特定记录 ID 查询单个记录。
1.获取地址簿中记录总个数:
[java] view plain copy print ?
- CFIndex count = ABAddressBookGetPersonCount(ab);
- printf("%ld total entries in the address book\n",count);
2. 获取所有联系人:
[java] view plain copy print ?
- NSArray* array = (NSArray*)ABAddressBookCopyArrayOfAllPeople(ab);
- printf("Retrieved %d contacts\n",[array count]);
3. 在联系人列表中查询一个特定的名字:
[java] view plain copy print ?
- NSArray* arrayByName = (NSArray*)ABAddressBookCopyPeopleWithName(ab, CFSTR("Liu Wei"));
函数如其名,这个函数返回的并不是地址簿中实际的对象,而是拷贝。要访问这个数组的单个记录,就用NSArray的方法啦:
[java] view plain copy print ?
- ABRecordRef myRecord = [arrayByName objectAtIndex:0];
除了通过名字查询,你也可以直接通过ID查询(假如你知道ID的话):
[java] view plain copy print ?
- ABRecordRef myRecord = ABAddressBookGetPersonWithRecordID(ab, recordID);
三、创建记录
创建新的联系人可以使用 ABPersonCreate 函数。这样可以得到一个空记录,然后就可以向其中填充信息:
[java] view plain copy print ?
- ABRecordRef record = ABPersonCreate();
四、操纵记录
一旦获得了 ABRecordRef,就可以确定是属于个人还是群体,也就可以访问更进一步的信息了。姓名以及其他各项信息都可以通过相应属性来操纵。只有ABPerson 类型实体记录才会有属性,而每个记录都具有各种各样的信息。
查询给定记录的信息,可以使用 ABRecordCopyValue 函数。这个函数原型如下:
[java] view plain copy print ?
- CFTypeRef ABRecordCopyValue(ABRecordRef record,kABPersonFirstNameProperty);
调用这个函数时,会将你指定 的属性拷贝一份,并返回引用:
[java] view plain copy print ?
- CFStringRef firstName = ABRecordCopyValue(record, kABPersonFirstNameProperty);
由于kABPersonFirstNameProperty 属性是一个 CFString,你可以将其转换为 NSString* :
[java] view plain copy print ?
- NSString* firstName =NSString* ABRecordCopyValue(record, kABPersonFirstNameProperty);
就像CFStringRef 可以转换为 NSString* 一样,如果有属性返回类型是CFDateRef,你也可以转换为 NSDate* :
[java] view plain copy print ?
- NSDate* birthday = (NSDate*) ABRecordCopyValue(record, kABPersonBirthdayProperty);
上面所指定的 ABPropertyID, 是一个与在记录中查找信息相对应的值。根据这个值返回ABPerson 对象属性。由于 ABRecordCopyValue 函数返回的数据类型是通用的 CFTypeRef ,所以结果可疑被转换为一个与属性相对应的更细化的数据类型,如下表:
[java] view plain copy print ?
-
- extern const ABPropertyID kABPersonFirstNameProperty;
- extern const ABPropertyID kABPersonLastNameProperty;
- extern const ABPropertyID kABPersonMiddleNameProperty;
- extern const ABPropertyID kABPersonPrefixProperty;
- extern const ABPropertyID kABPersonSuffixProperty;
- extern const ABPropertyID kABPersonNicknameProperty;
- extern const ABPropertyID kABPersonFirstNamePhoneticProperty;
- extern const ABPropertyID kABPersonLastNamePhoneticProperty;
- extern const ABPropertyID kABPersonMiddleNamePhoneticProperty;
- extern const ABPropertyID kABPersonOrganizationProperty;
- extern const ABPropertyID kABPersonJobTitleProperty;
- extern const ABPropertyID kABPersonDepartmentProperty;
- extern const ABPropertyID kABPersonEmailProperty;
- extern const ABPropertyID kABPersonBirthdayProperty;
- extern const ABPropertyID kABPersonNoteProperty;
- extern const ABPropertyID kABPersonCreationDateProperty;
- extern const ABPropertyID kABPersonModificationDateProperty;
五、写入属性
向记录写入属性,使用ABRecordSetValue 函数:
[java] view plain copy print ?
- CFStringRef name = CFSTR("IUKEY");
- BOOL suc = ABRecordSetValue(record, kABPersonNicknameProperty, name, &err);
- if (suc) {
- NSLog(@"setValue succeed");
- }
删除属性:
[java] view plain copy print ?
- BOOL succeed = ABRecordRemoveValue(record, kABPersonFirstNameProperty, &err);
当修改完以后不要忘记保存地址簿。
六、多值属性
除了前面列出的属性之外,一个记录还可能会有一些属性,其中包含多个值。多只属性可以用一种索引机制来处理,使用时首先查询值的总数,然后通过特定的索引得到一个条目。指向多值数据的指针,可以首先通过前面提到的 ABRecordCopyValue 方法得到,然后转换成 MultiValueRef :
[java] view plain copy print ?
- ABMultiValueRef phoneNumbers = ABRecordCopyValue(record, kABPersonPhoneProperty);
然后你可以通过这个引用来确定值的个数,并按照索引获取其中的单个值。函数ABMultiGetCount 可以返回条目个数,按照索引拷贝指定条目,可以用ABMultiValueCopyValueAtIndex 函数。
下面列出了多值属性中的条目:
[java] view plain copy print ?
- extern const ABPropertyID kABPersonEmailProperty;
- extern const ABPropertyID kABPersonAddressProperty;
- extern const ABPropertyID kABPersonDateProperty;
- extern const ABPropertyID kABPersonPhoneProperty;
- extern const ABPropertyID kABPersonInstantMessageProperty;
- extern const ABPropertyID kABPersonURLProperty;
除了多值属性中真正的值,每个条目还有一个标签。标签描述了返回的条目的类型。例如,一个电话号码标签有可能指明了这个号码是家庭电话还是手机号。地址的标签则可一描述是家庭地址还是工作地址。要查询特定条目标签,可以使用 ABMultValueCopyLabelAtIndex 函数:
[java] view plain copy print ?
- CFStringRef label = ABMultiValueCopyLabelAtIndex(phoneNumbers, i);
默写属性具有一组预定义的标签。下面这些 CFStringRef 类型标签的原型是在 ABPerson.h 中指定的:
[java] view plain copy print ?
- extern const ABPropertyID kABPersonDateProperty;
- extern const CFStringRef kABPersonAnniversaryLabel;
[java] view plain copy print ?
-
- extern const ABPropertyID kABPersonPhoneProperty;
- extern const CFStringRef kABPersonPhoneMobileLabel;
- extern const CFStringRef kABPersonPhoneIPhoneLabel __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_3_0);
- extern const CFStringRef kABPersonPhoneMainLabel;
- extern const CFStringRef kABPersonPhoneHomeFAXLabel;
- extern const CFStringRef kABPersonPhoneWorkFAXLabel;
- extern const CFStringRef kABPersonPhoneOtherFAXLabel __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_5_0);
- extern const CFStringRef kABPersonPhonePagerLabel;
[java] view plain copy print ?
-
- extern const ABPropertyID kABPersonInstantMessageProperty;
- extern const CFStringRef kABPersonInstantMessageServiceKey;
- extern const CFStringRef kABPersonInstantMessageServiceYahoo;
- extern const CFStringRef kABPersonInstantMessageServiceJabber;
- extern const CFStringRef kABPersonInstantMessageServiceMSN;
- extern const CFStringRef kABPersonInstantMessageServiceICQ;
- extern const CFStringRef kABPersonInstantMessageServiceAIM;
[java] view plain copy print ?
-
- extern const ABPropertyID kABPersonURLProperty;
- extern const CFStringRef kABPersonHomePageLabel;
[java] view plain copy print ?
-
- extern const ABPropertyID kABPersonRelatedNamesProperty;
- extern const CFStringRef kABPersonFatherLabel;
- extern const CFStringRef kABPersonMotherLabel;
- extern const CFStringRef kABPersonParentLabel;
- extern const CFStringRef kABPersonBrotherLabel;
- extern const CFStringRef kABPersonSisterLabel;
- extern const CFStringRef kABPersonChildLabel;
- extern const CFStringRef kABPersonFriendLabel;
- extern const CFStringRef kABPersonSpouseLabel;
- extern const CFStringRef kABPersonPartnerLabel;
- extern const CFStringRef kABPersonAssistantLabel;
- extern const CFStringRef kABPersonManagerLabel;
许多属性使用一组通用标签,标识工作、家庭以及其它地点。这些通用的标签如下:
kABWorkLabel
kABHomeLabel
kABOtherLabel
写入多只属性条目:
为了在现有的属性中加入一个值,你必须首先从记录中复制出多值字典。然后 ABMultiValueAddValueAndLabel 函数操作拷贝,将新的值-标签加入到字典中。最后用函数 ABRecordSetValue 将字典条目写回到地址簿记录中,完全替换掉整个多值属性。
七、使用字典
地址簿记录用字典来 表示地址和即时通讯账号。这些字典是内嵌在多值属性条目之中的。要访问这些字典,需要将值复制出来,并转换成 NSDictionary* 。然后你就可以用一组预定于的键值来访问字典了。
八、图像数据
某些联系人可能会有与之相关联的图像。可以使用 ABPersonCopyImageData 函数获取这些图像数据,返回将是一个 CFDataRef。可以把它转换成 NSData*,然后用来初始化一个UIImage对象。
[java] view plain copy print ?
- if (ABPersonHasImageData(record)) {
- UIImage *addressVookImage = [UIImage imageWithData:(NSData*)ABPersonCopyImageData(record)];
- }
九、地址簿界面
地址簿界面框架提供了两种关键的用户界面:一个“找人”导航控件,来选择联系人;以及一个视图控件,用于显示单个联系人。
1. 联系人视图
ABPersonViewController 提供了一个简单的界面,可以向用户显示一个联系人。联系人视图需要一个 CFRecordRef 。
[java] view plain copy print ?
- ABPersonViewController *viewController = [[ABPersonViewController alloc]init];
可以将希望显示的记录赋值给 displayedPerson 属性:
[java] view plain copy print ?
- viewController.displayedPerson=record;
然后,你可以创建一个数组,其中包含你想要显示给用户的属性。只有指定的属性才会被显示出来,不过如果联系人被修改了,所有属性都将显示出来。可用的属性值与前面提到的枚举值相同。每个都作为一个 NSNumber 对象加入到数组中。
[java] view plain copy print ?
- NSMutableArray* properties = [[NSMutableArray alloc]init];
- [properties addObject:[NSNumber numberWithInt:kABPersonFirstNameProperty ]];
- [properties addObject:[NSNumber numberWithInt:kABPersonLastNameProperty ]];
- [properties addObject:[NSNumber numberWithInt:kABPersonOrganizationProperty ]];
- viewController.allowsEditing =YES;
十、联系人选取器
如果应用程序要选取一系列联系人,ABPeoplePickerNavigationController 类正好适合你。这个导航控件 可以显示联系人,让用户从中选择其一。选好后,你可以选择向用户显示该联系人,也可以通过一个委托方法实现你自己的行为。
[java] view plain copy print ?
- ABPeoplePickerNavigationController* peoplePicker = [[ABPeoplePickerNavigationController alloc]init];
如果希望允许用户查看单个联系人,你可以赋予其一组希望用户看到的属性。默认情况下会向用户显示所有的项目。 可用的属性值就是本章前面讲授的那些枚举值。每个作为一个NSNumber对象加入到数组中:
[java] view plain copy print ?
- NSMutableArray* properties = [[NSMutableArray alloc]init];
- [properties addObject:[NSNumber numberWithInt:kABPersonFirstNameProperty ]];
- [properties addObject:[NSNumber numberWithInt:kABPersonLastNameProperty ]];
- [properties addObject:[NSNumber numberWithInt:kABPersonOrganizationProperty ]];
- peoplePicker.displayedProperties = properties;
可以指定一个委托,在用户选定联系人时接收通知:
[java] view plain copy print ?
- peoplePicker.peoplePickerDelegate =self;
[java] view plain copy print ?
- [self.view addSubview:peoplePicker.view];
委托方发:
[java] view plain copy print ?
- -(void)peoplePickerNavigationControllerDidCancel:(ABPeoplePickerNavigationController *)peoplePicker{
-
- }
-
- -(BOOL)peoplePickerNavigationController:(ABPeoplePickerNavigationController *)peoplePicker shouldContinueAfterSelectingPerson:(ABRecordRef)person{
-
-
- }
- -(BOOL)peoplePickerNavigationController:(ABPeoplePickerNavigationController *)peoplePicker shouldContinueAfterSelectingPerson:(ABRecordRef)person property:(ABPropertyID)property identifier:(ABMultiValueIdentifier)identifier{
-
-
- }