Address

通讯录

  • 从iOS6开始,需要得到用户的授权才能访问通讯录,因此在使用之前,需要检查用户是否已经授权 获得通讯录的授权状态:ABAddressBookGetAuthorizationStatus()
  • 授权状态

    • kABAuthorizationStatusNotDetermined --> 用户还没有决定是否授权你的程序进行访问

    • kABAuthorizationStatusRestricted --> iOS设备上一些许可配置阻止程序与通讯录数据库进行交互

    • kABAuthorizationStatusDenied --> 用户明确的拒绝了你的程序对通讯录的访问

    • kABAuthorizationStatusAuthorized --> 用户已经授权给你的程序对通讯录进行访问

AddressBookUI.framework “iOS 9 已过期”

  • 提供了联系人列表界面、联系人详情界面、添加联系人界面等

    • 一般用于选择联系人
  • 创建联系人界面

      // 1.创建选择联系人的界面
      ABPeoplePickerNavigationController *ppnc = [[ABPeoplePickerNavigationController alloc] init];
    
      // 2.设置代理
      ppnc.peoplePickerDelegate = self;
    
      // 3.弹出选择联系人界面
      [self presentViewController:ppnc animated:YES completion:nil];
    
  • 遵守代理协议

    • ABPeoplePickerNavigationControllerDelegate
  • 实现代理方法

// 选中某一个联系人的时候,会执行该代理方法
// 如果实现了该方法,那么就不会进入联系人的详细界面
- (void)peoplePickerNavigationController:(ABPeoplePickerNavigationController *)peoplePicker didSelectPerson:(ABRecordRef)person
{
    /*
     __bridge NSString * : 将CoreFoundation框架的对象所有权交给Foundation框架来使用,但是Foundation框架中的对象并不能管理该对象内存
     __bridge_transfer NSString * : 将CoreFoundation框架的对象所有权交给Foundation来管理,如果Foundation中对象销毁,那么我们之前的对象(CoreFoundation)会一起销毁
     */
    // 1.获取选中联系人的姓名(姓lastname和名firstname)
    CFStringRef firstname = ABRecordCopyValue(person, kABPersonFirstNameProperty);
    CFStringRef lastname = ABRecordCopyValue(person, kABPersonLastNameProperty);
    NSString *firstName = (__bridge_transfer NSString *)(firstname);
    NSString *lastName = (__bridge_transfer NSString *)(lastname);
    NSLog(@"%@ %@", firstName, lastName);

    // 2.获取联系人的电话号码
    ABMultiValueRef phones = ABRecordCopyValue(person, kABPersonPhoneProperty);
    CFIndex count = ABMultiValueGetCount(phones);
    for (CFIndex i = 0; i < count; i++) {
        NSString *phoneLabel = (__bridge_transfer NSString *)ABMultiValueCopyLabelAtIndex(phones, i);
        NSString *phoneValue = (__bridge_transfer NSString *)ABMultiValueCopyValueAtIndex(phones, i);
        NSLog(@"label : %@ value : %@", phoneLabel, phoneValue);
    }

    // 3.释放不再使用的对象
    CFRelease(phones);
}

// 选择某一个联系人的某一个属性时,会执行该方法
// property选中的属性
// identifier : 每一个属性都由一个对应标示
// 如果实现了该方法,那么选中一个联系人的属性时,就会推出控制器.不会进入下一个页面
- (void)peoplePickerNavigationController:(ABPeoplePickerNavigationController *)peoplePicker didSelectPerson:(ABRecordRef)person property:(ABPropertyID)property identifier:(ABMultiValueIdentifier)identifier
{
    NSLog(@"选择了某一个联系人的某一个属性");
}

// 点击了取消按钮会执行该方法
- (void)peoplePickerNavigationControllerDidCancel:(ABPeoplePickerNavigationController *)peoplePicker
{

}

AddressBook.framework “iOS 9 已过期”

  • 纯C语言的API,仅仅是获得联系人数据
  • 没有提供UI界面展示,需要自己搭建联系人展示界面
  • 里面的数据类型大部分基于Core Foundation框架,使用起来极其蛋疼

    // 1.获取授权的状态
      ABAuthorizationStatus status = ABAddressBookGetAuthorizationStatus();
    
      // 2.判断授权状态,如果是未决定状态,才需要请求
      if (status == kABAuthorizationStatusNotDetermined) {
          // 2.1.创建通信录对象
          ABAddressBookRef addressBook = ABAddressBookCreateWithOptions(NULL, NULL);
    
          // 2.2.请求授权
          ABAddressBookRequestAccessWithCompletion(addressBook, ^(bool granted, CFErrorRef error) {
              if (granted) {
                  NSLog(@"授权成功");
              } else {
                  NSLog(@"授权失败");
              }
          });
          }
    
// 1.获取授权的状态
    ABAuthorizationStatus status = ABAddressBookGetAuthorizationStatus();

    // 2.如果用户已经授权
    if (status != kABAuthorizationStatusAuthorized) return;

    // 3.创建通信录对象
    ABAddressBookRef addressBook = ABAddressBookCreateWithOptions(NULL, NULL);

    // 4.从通信录对象中,将所有的联系人拷贝出来
    CFArrayRef peopleArray = ABAddressBookCopyArrayOfAllPeople(addressBook);

    // 5.遍历所有的联系人(每一个联系人都是一条记录)
    CFIndex peopleCount = CFArrayGetCount(peopleArray);
    for (CFIndex i = 0; i < peopleCount; i++) {

        // 6.获取到联系人
        ABRecordRef person = CFArrayGetValueAtIndex(peopleArray, i);

        // 7.获取姓名
        NSString *lastname = (__bridge_transfer NSString *)ABRecordCopyValue(person, kABPersonLastNameProperty);
        NSString *firstName = (__bridge_transfer NSString *)ABRecordCopyValue(person, kABPersonFirstNameProperty);
        NSLog(@"%@ %@", lastname, firstName);
    }

    // 8.释放不再使用的对象
    CFRelease(peopleArray);
    CFRelease(addressBook);

RHAddressBook 第三方框架

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {

    // 1.获取授权状态
    RHAuthorizationStatus status = [RHAddressBook authorizationStatus];

    // 2.判断授权状态,如果未决定状态,则请求授权
    if (status == RHAuthorizationStatusNotDetermined) {
        // 3.创建通信录对象
        RHAddressBook *addressBook = [[RHAddressBook alloc] init];

        // 4.请求授权
        [addressBook requestAuthorizationWithCompletion:^(bool granted, NSError *error) {
            if (granted) {
                NSLog(@"授权成功");
            } else {
                NSLog(@"授权失败");
            }
        }];
    }

    return YES;
}
 // 1.获取授权状态
    RHAuthorizationStatus status = [RHAddressBook authorizationStatus];

    // 2.判断如果是未授权,则直接返回
    if (status != RHAuthorizationStatusAuthorized) return;

    // 3.创建通信录
    RHAddressBook *addressBook = [[RHAddressBook alloc] init];

    // 4.获取所有的联系人
    NSArray *peopleArray = addressBook.people;

    // 5.遍历所有的联系人
    for (RHPerson *person in peopleArray) {

        // 6.获取联系人的姓名
        NSLog(@"%@ %@", person.firstName, person.lastName);

        // 7.获取电话号码
        RHMultiValue *phones = person.phoneNumbers;
        for (int i = 0; i < phones.count; i++) {
            // 8.获取电话号码和对应的Label
            NSString *phoneLabel = [phones labelAtIndex:i];
            NSString *phoneValue = [phones valueAtIndex:i];

            NSLog(@"%@ %@", phoneLabel, phoneValue);
        }
    }

ContactsUI ——> iOS 9 之后使用

  • 导入 Contacts/Contacts.h
// 1.创建联系人控制器
    CNContactPickerViewController *contactVc = [[CNContactPickerViewController alloc] init];

    // 2.设置代理
    contactVc.delegate = self;

    // 3.弹出控制器
    [self presentViewController:contactVc animated:YES completion:nil];
  • 遵守代理协议,实现代理方法
/** 点击了某个联系人的时候调用 */
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContact:(CNContact *)contact
{
    // 1.获取联系人的姓名
    NSString *firstname = contact.familyName;
    NSString *middlename = contact.middleName;
    NSString *lastname = contact.givenName;

    NSLog(@"%@ - %@ - %@",firstname,middlename,lastname);

    // 2.获取联系人的电话号码
    NSArray *phoneNumbers = contact.phoneNumbers;
    for (CNLabeledValue *value in phoneNumbers) {

        NSString *label = value.label;

        CNPhoneNumber *number = value.value;
        NSString *phonenumber = number.stringValue;

        NSLog(@"%@ - %@",label,phonenumber);
    }

}

/** 点击了某些联系人的时候调用 */
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContacts:(NSArray<CNContact *> *)contacts
{
}

/** 点击了联系人的属性调用 */
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContactProperty:(CNContactProperty *)contactProperty
{
}

/**
 * Both single contact and multiple contacts delegate methods are implemented, the single variants will be ignored
 * 实现了选择多个联系人回调的方法,就不会调用单个联系人的方法
 */
/** 点击了联系人的某些属性调用 */
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContactProperties:(NSArray<CNContactProperty *> *)contactProperties
{
}

/** 点击了取消 */
- (void)contactPickerDidCancel:(CNContactPickerViewController *)picker
{
}

contacts iOS 9 开始

  • iOS 10需要在info.plist配置NSContactsUsageDescription
<key>NSContactsUsageDescription</key>
<string>请求访问通讯录</string>
  • 程序启动时请求授权
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // 1.获取授权的状态
     CNAuthorizationStatus status = [CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts];

    // 2.判断授权状态,如果是未决定状态,才需要请求
    if (status == CNAuthorizationStatusNotDetermined)  {
        // 2.1.创建通信录对象
        CNContactStore *store = [[CNContactStore alloc] init];

        [store requestAccessForEntityType:CNEntityTypeContacts completionHandler:^(BOOL granted, NSError * _Nullable error) {

            if (granted) {
                NSLog(@"授权成功");
            }else
            {
                NSLog(@"授权失败");
            }

        }];
    }
    return YES;
}
  • 获取联系人
// 1.获取授权状态
    CNAuthorizationStatus status = [CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts];

    // 2.判断授权状态
    if (status != CNAuthorizationStatusAuthorized) {
        return;
    }

    // 3.创建通讯录对象
    CNContactStore *contactStore = [[CNContactStore alloc] init];

    // 4.创建获取通讯录的请求对象
    // 4.1 获取需要属性的对应key
    NSArray *phoneKeys = @[CNContactGivenNameKey,CNContactFamilyNameKey,CNContactPhoneNumbersKey];
    // 4.2 创建CNContactFetchRequest对象
    CNContactFetchRequest *request = [[CNContactFetchRequest alloc] initWithKeysToFetch:phoneKeys];

    // 5.遍历所有联系人
    [contactStore enumerateContactsWithFetchRequest:request error:nil usingBlock:^(CNContact * _Nonnull contact, BOOL * _Nonnull stop) {
       // 5.1 获取联系人姓名
        NSString *surName = contact.familyName;
        NSString *name = contact.givenName;
        NSLog(@"%@ %@",surName,name);

       // 5.2 获取联系人电话号码
        NSArray *phoneNumbers = contact.phoneNumbers;
        for (CNLabeledValue *labelValue in phoneNumbers) {
            // 5.2.1 获取电话号码的key
            NSString *label = labelValue.label;

            // 5.2.2 获取电话号码
            CNPhoneNumber *number = labelValue.value;
            NSString *phoneNumber = number.stringValue;

            NSLog(@"%@ %@",label,phoneNumber);
        }
    }];

results matching ""

    No results matching ""