iOS 通訊錄獲取使用

通訊錄簡介

image

通訊錄使用場景:

  1. 電商類的 App,設(shè)置收貨人電話號碼。
  2. 即時通訊類 App,添加手機(jī)聯(lián)系人好友。

通訊錄獲取方案:

一、iOS 9 以前的通訊錄框架

  1. AddressBookUI.framework 框架

    1. 提供了聯(lián)系人列表界面、聯(lián)系人詳情界面、添加聯(lián)系人界面等。
    2. 一般用于選擇聯(lián)系人。
  2. AddressBook.framework 框架

    1. 純 C 語言的 API,僅僅是獲得聯(lián)系人數(shù)據(jù)。
    2. 沒有提供 UI 界面展示,需要自己搭建聯(lián)系人展示界面。
    3. 里面的數(shù)據(jù)類型大部分基于 Core Foundation 框架,使用起來炒雞復(fù)雜。

二、 iOS 9 以后最新通訊錄框架

  1. ContactsUI.framework 框架。

    • 擁有 AddressBookUI.framework 框架的所有功能,使用起來更加的面向?qū)ο蟆?/li>
  2. Contacts.framework 框架。

    • 擁有 AddressBook.framework 框架的所有功能,不再是 C 語言的 API,使用起來非常簡單。

iOS 9 以前的通訊錄框架

AddressBookUI

實現(xiàn)步驟

一、創(chuàng)建選擇聯(lián)系人的控制器

// 創(chuàng)建聯(lián)系人選擇控制器    
ABPeoplePickerNavigationController *pvc = [[ABPeoplePickerNavigationController alloc] init];

二、設(shè)置代理(用來接收用戶選擇的聯(lián)系人信息)

// 設(shè)置代理
pvc.peoplePickerDelegate = self;

三、彈出聯(lián)系人控制器

if (ABAddressBookGetAuthorizationStatus() == kABAuthorizationStatusNotDetermined)
{
   ABAddressBookRef bookRef = ABAddressBookCreate();
   ABAddressBookRequestAccessWithCompletion(bookRef, ^(bool granted, CFErrorRef error) {
       if (granted)
       {
           NSLog(@"授權(quán)成功!");
           [self presentViewController:pvc animated:YES completion:nil];
       }
       else
       {
           NSLog(@"授權(quán)失敗!");
       }
   });
}
else if (ABAddressBookGetAuthorizationStatus() == kABAuthorizationStatusAuthorized)
{
   [self presentViewController:pvc animated:YES completion:nil];
}

四、實現(xiàn)代理方法

// 選擇某個聯(lián)系人時調(diào)用
- (void)peoplePickerNavigationController:(ABPeoplePickerNavigationController*)peoplePicker didSelectPerson:(ABRecordRef)person
{
    NSLog(@"選中聯(lián)系人");
    CFStringRef firstName = ABRecordCopyValue(person, kABPersonFirstNameProperty);
    CFStringRef lastName = ABRecordCopyValue(person, kABPersonLastNameProperty);

    NSString *fir = CFBridgingRelease(firstName);
    NSString *las = CFBridgingRelease(lastName);

    NSLog(@"%@---%@", fir, las);

    ABMultiValueRef multi = ABRecordCopyValue(person, kABPersonPhoneProperty);
    CFIndex count = ABMultiValueGetCount(multi);
    for (int i = 0; i  < count; i++) 
    {
        NSString *label = (__bridge_transfer NSString *)ABMultiValueCopyLabelAtIndex(multi, i);
        NSString *phone =(__bridge_transfer NSString *)  ABMultiValueCopyValueAtIndex(multi, i);
        NSLog(@"%@---%@", label, phone);
    }
}

Core Foundation 對象手動管理內(nèi)存,如果是 Create、Copy、Retain 等字樣創(chuàng)建的對象,需要手動 CFRelease。類似 Objective-C 的 MRC。

拓展:__bridge,__bridge_retained__bridge_transfer 三個轉(zhuǎn)換關(guān)鍵字的區(qū)別。

  1. __bridge 只做類型轉(zhuǎn)換,但是不修改對象(內(nèi)存)管理權(quán);
  2. __bridge_retained(也可以使用CFBridgingRetain)將 Objective-C 的對象轉(zhuǎn)換為 Core Foundation 的對象,同時將對象(內(nèi)存)的管理權(quán)交給我們,后續(xù)需要使用 CFRelease 或者相關(guān)方法來釋放對象;
  3. __bridge_transfer(也可以使用 CFBridgingRelease )將 Core Foundation 的對象轉(zhuǎn)換為 Objective-C 的對象,同時將對象(內(nèi)存)的管理權(quán)交給 ARC。

五、在對應(yīng)的代理方法中獲取聯(lián)系人信息

// 1.選擇聯(lián)系人時使用(不展開詳情)
- (void)peoplePickerNavigationController:(ABPeoplePickerNavigationController *)peoplePicker didSelectPerson:(ABRecordRef)person;

// 2.選擇聯(lián)系人某個屬性時調(diào)用(展開詳情)
- (void)peoplePickerNavigationController:(ABPeoplePickerNavigationController*)peoplePicker didSelectPerson:(ABRecordRef)person property:(ABPropertyID)property identifier:(ABMultiValueIdentifier)identifier;

// 3.取消選中聯(lián)系人時調(diào)用
- (void)peoplePickerNavigationControllerDidCancel:(ABPeoplePickerNavigationController *)peoplePicker;

注意:選擇聯(lián)系人的不展開詳情(代理方法1)和展開詳情(代理方法2)的代理方法都寫了的時候,展開詳情的代理方法就不執(zhí)行。

AddressBook

一、請求授權(quán)

從 iOS 6 開始,需要得到用戶的授權(quán)才能訪問通訊錄,因此在使用之前,需要檢查用戶是否已經(jīng)授權(quán)。

// 獲得通訊錄的授權(quán)狀態(tài)
ABAddressBookGetAuthorizationStatus()

授權(quán)狀態(tài)

  1. 用戶還沒有決定是否授權(quán)你的程序進(jìn)行訪問:kABAuthorizationStatusNotDetermined

  2. iOS 設(shè)備上一些許可配置阻止程序與通訊錄數(shù)據(jù)庫進(jìn)行交互:kABAuthorizationStatusRestricted

  3. 用戶明確的拒絕了你的程序?qū)νㄓ嶄浀脑L問:kABAuthorizationStatusDenied

  4. 用戶已經(jīng)授權(quán)給你的程序?qū)νㄓ嶄涍M(jìn)行訪問:kABAuthorizationStatusAuthorized

// 判斷當(dāng)前的授權(quán)狀態(tài)是否是用戶還未選擇的狀態(tài)
if (ABAddressBookGetAuthorizationStatus() == kABAuthorizationStatusNotDetermined) 
{
   ABAddressBookRef bookRef = ABAddressBookCreate();
   ABAddressBookRequestAccessWithCompletion(bookRef, ^(bool granted, CFErrorRef error) {
       if (granted) 
       {
           NSLog(@"授權(quán)成功!");
       }
       else
       {
           NSLog(@"授權(quán)失敗!");
       }
   });
}

二、判斷授權(quán)狀態(tài)

如果已授權(quán),則繼續(xù);未授權(quán),則提示用戶,并返回。

// 判斷當(dāng)前的授權(quán)狀態(tài)
if (ABAddressBookGetAuthorizationStatus() != kABAuthorizationStatusAuthorized) 
{
    NSLog(@"您的通訊錄暫未允許訪問,請去設(shè)置->隱私里面授權(quán)!");
    return;
}

三、創(chuàng)建通訊錄對象

// 創(chuàng)建通訊錄對象
ABAddressBookRef bookRef = ABAddressBookCreate();

四、從通信錄對象中, 獲取所有的聯(lián)系人

// 獲取通訊錄中所有的聯(lián)系人
CFArrayRef arrayRef = ABAddressBookCopyArrayOfAllPeople(bookRef);

五、遍歷所有的聯(lián)系人

// 遍歷所有聯(lián)系人
CFIndex count = CFArrayGetCount(arrayRef);
for (int i = 0; i < count; i++) 
{
   ABRecordRef record = CFArrayGetValueAtIndex(arrayRef, i);

   // 獲取姓名
   NSString *firstName = (__bridge_transfer NSString *)ABRecordCopyValue(record, kABPersonFirstNameProperty);
   NSString *lastName = (__bridge_transfer NSString *)ABRecordCopyValue(record, kABPersonLastNameProperty);
   NSLog(@"firstName = %@, lastName = %@", firstName, lastName);

   // 獲取電話號碼
   ABMultiValueRef multiValue = ABRecordCopyValue(record, kABPersonPhoneProperty);
   CFIndex count = ABMultiValueGetCount(multiValue);
   for (int i = 0; i < count; i ++) 
   {
       NSString *label = (__bridge_transfer NSString *)ABMultiValueCopyLabelAtIndex(multiValue, i);
       NSString *phone = (__bridge_transfer NSString *)ABMultiValueCopyValueAtIndex(multiValue, i);
       NSLog(@"label = %@, phone = %@", label, phone);
   }

   CFRelease(multiValue);
}

六、釋放不再使用的對象

CFRelease(bookRef);
CFRelease(arrayRef);

聯(lián)系人屬性定義

所有的屬性常量值都定義在了 ABPerson.h 頭文件中。

聯(lián)系人屬性包括以下類型:

  1. 簡單屬性:姓、名等
  2. 多重屬性:電話號碼、電子郵件等
  3. 組合屬性:地址等

注意:使用 ABRecordCopyValue 可以從一條 Person 記錄中獲取到對應(yīng)的記錄,但是后續(xù)處理則需要根據(jù)記錄的具體類型加以區(qū)分。

簡單屬性

一個聯(lián)系人就是一個 ABRecordRef,每個聯(lián)系人都有自己的屬性,比如名字、電話、郵件等。
使用 ABRecordCopyValue 函數(shù)可以從 ABRecordRef 中獲得聯(lián)系人的簡單屬性(例如:一個字符串)。
ABRecordCopyValue 函數(shù)接收 2 個參數(shù)。
第 1 個參數(shù)是 ABRecordRef 實例。
第 2 個參數(shù)是屬性關(guān)鍵字,定義在 ABPerson.h 中。
ABPersonCopyLocalizedPropertyName 函數(shù)可以根據(jù)指定的關(guān)鍵字獲取對應(yīng)的標(biāo)簽文本。

獲得所有的聯(lián)系人數(shù)據(jù)

// 獲取所有聯(lián)系人記錄
CFArrayRef array = ABAddressBookCopyArrayOfAllPeople(addressBook);
NSInteger count = CFArrayGetCount(array);

for (NSInteger i = 0; i < count; ++i) {
    // 取出一條記錄
    ABRecordRef person = CFArrayGetValueAtIndex(array, i);

    // 取出個人記錄中的詳細(xì)信息
    // 名
    CFStringRef firstNameLabel = ABPersonCopyLocalizedPropertyName(kABPersonFirstNameProperty);
    CFStringRef firstName = ABRecordCopyValue(person, kABPersonFirstNameProperty);
    CFStringRef lastNameLabel = ABPersonCopyLocalizedPropertyName(kABPersonLastNameProperty);
    // 姓
    CFStringRef lastName = ABRecordCopyValue(person, kABPersonLastNameProperty);

    NSLog(@"%@ %@ - %@ %@", lastNameLabel, lastName, firstNameLabel, firstName);
}

CoreFoundation 與 Foundation之間的橋接

// 1\. 獲取通訊錄引用
ABAddressBookRef addressBook = ABAddressBookCreateWithOptions(NULL, nil);
// 2\. 獲取所有聯(lián)系人記錄
NSArray *array = (__bridge NSArray *)(ABAddressBookCopyArrayOfAllPeople(addressBook));
for (NSInteger i = 0; i < array.count; i++) {
    // 取出一條記錄
    ABRecordRef person = (__bridge ABRecordRef)(array[i]);
    // 取出個人記錄中的詳細(xì)信息
    NSString *firstNameLabel = (__bridge NSString *)(ABPersonCopyLocalizedPropertyName(kABPersonFirstNameProperty));
    NSString *firstName = (__bridge NSString *)(ABRecordCopyValue(person, kABPersonFirstNameProperty));
    NSString *lastNameLabel = (__bridge NSString *)(ABPersonCopyLocalizedPropertyName(kABPersonLastNameProperty));
    NSString *lastName = (__bridge NSString *)(ABRecordCopyValue(person, kABPersonLastNameProperty));
    NSLog(@"%@ %@ - %@ %@", lastNameLabel, lastName, firstNameLabel, firstName);
}
CFRelease(addressBook);

多重屬性

聯(lián)系人的有些屬性值就沒這么簡單,一個屬性可能會包含多個值
比如郵箱,分為工作郵箱、住宅郵箱、其他郵箱等
比如電話,分為工作電話、住宅電話、其他電話等
如果是復(fù)雜屬性,那么 ABRecordCopyValue 函數(shù)返回的就是 ABMultiValueRef 類型的數(shù)據(jù),例如郵箱或者電話

// 取電話號碼
ABMultiValueRef phones = ABRecordCopyValue(person, kABPersonPhoneProperty);
// 取記錄數(shù)量
NSInteger phoneCount = ABMultiValueGetCount(phones);
// 遍歷所有的電話號碼
for (NSInteger i = 0; i < phoneCount; i++) 
{

}

獲取復(fù)雜屬性的方法

// 電話標(biāo)簽
CFStringRef phoneLabel = ABMultiValueCopyLabelAtIndex(phones, i);
// 本地化電話標(biāo)簽
CFStringRef phoneLocalLabel = ABAddressBookCopyLocalizedLabel(phoneLabel);
// 電話號碼
CFStringRef phoneNumber = ABMultiValueCopyValueAtIndex(phones, i);

添加聯(lián)系人的步驟

添加聯(lián)系人的步驟:

  1. 通過 ABPersonCreate 函數(shù)創(chuàng)建一個新的聯(lián)系人(返回 ABRecordRef)。
  2. 通過 ABRecordSetValue 函數(shù)設(shè)置聯(lián)系人的屬性。
  3. 通過 ABAddressBookAddRecord 函數(shù)將聯(lián)系人添加到通訊錄數(shù)據(jù)庫中。
  4. 通過 ABAddressBookSave 函數(shù)保存剛才所作的修改。

可以通過 ABAddressBookHasUnsavedChanges 函數(shù)判斷是否有未保存的修改
當(dāng)決定是否更改通訊錄數(shù)據(jù)庫后,你可以分別使用 AbAddressBookSaveABAddressBookRevert 方式來保存或放棄更改 。

添加群組的步驟

添加群組的步驟大體和添加聯(lián)系人一致:

  1. 通過 ABPersonCreate 函數(shù)創(chuàng)建一個新的組。(返回 ABRecordRef
  2. 通過 ABRecordSetValue 函數(shù)設(shè)置組名。
  3. 通過 ABAddressBookAddRecord 函數(shù)將組添加到通訊錄數(shù)據(jù)庫中。
  4. 通過 ABAddressBookSave 函數(shù)保存剛才所作的修改。

操作聯(lián)系人的頭像

想操作聯(lián)系人的頭像,有以下函數(shù)
BPersonHasImageData
判斷通訊錄中的聯(lián)系人是否有圖片

ABPersonCopyImageData
取得圖片數(shù)據(jù)(假如有的話)

ABPersonSetImageData
設(shè)置聯(lián)系人的圖片數(shù)據(jù)

通訊錄的修改回調(diào)

// 創(chuàng)建通訊錄
self.addressBook = ABAddressBookCreate();  
// 注冊通知  
ABAddressBookRegisterExternalChangeCallback(self.addressBook, _addressBookChange, nil);            

// 處理收到通知的 Action
void _addressBookChange(ABAddressBookRef addressBook, CFDictionaryRef info, void *context)
{
}

- (void)dealloc
{
    // 注銷通知
    ABAddressBookUnregisterExternalChangeCallback(self.addressBook, _addressBookChange, nil);
    // 釋放對象
    CFRelease(self.addressBook);
}

iOS 9 以后的通訊錄新框架

iOS 9 之前操作通訊錄還是比較麻煩的,iOS 9 以后蘋果推出了全新的通訊錄框架,使用起來更加的面向?qū)ο蟆?/p>

CNContactUI

實現(xiàn)步驟

一、創(chuàng)建選擇聯(lián)系人的控制器

// 創(chuàng)建聯(lián)系人選擇控制器    
CNMutableContact *contact = [[CNMutableContact alloc] init];
CNLabeledValue *labelValue = [CNLabeledValue labeledValueWithLabel:CNLabelPhoneNumberMobile
ontact.phoneNumbers = @[labelValue];                                                                     value:[CNPhoneNumber phoneNumberWithStringValue:phoneNum]];
CNContactViewController *contactController = [CNContactViewController viewControllerForNewContact:contact];
UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:contactController];

二、設(shè)置代理(用來接收用戶選擇的聯(lián)系人信息)

// 設(shè)置代理
contactController.delegate = self;

三、彈出聯(lián)系人控制器

[controller presentViewController:nav animated:YES completion:nil];

四、實現(xiàn)代理方法

// 選擇某個聯(lián)系人時調(diào)用
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContactProperty:(CNContactProperty *)contactProperty
{
    CNContact *contact = contactProperty.contact;
    NSString *name = [CNContactFormatter stringFromContact:contact style:CNContactFormatterStyleFullName];
    CNPhoneNumber *phoneValue= contactProperty.value;
    NSString *phoneNumber = phoneValue.stringValue;
    NSLog(@"%@--%@",name, phoneNumber);
}

五、在對應(yīng)的代理方法中獲取聯(lián)系人信息

// 1.選擇聯(lián)系人時使用(不展開詳情)
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContact:(CNContact *)contact;

// 2.選擇聯(lián)系人某個屬性時調(diào)用(展開詳情)
- (void)contactPicker:(CNContactPickerViewController *)picker didSelectContactProperty:(CNContactProperty *)contactProperty;

// 3.取消選中聯(lián)系人時調(diào)用
- (void)contactPickerDidCancel:(CNContactPickerViewController *)picker;

注意:與 AddressBookUI 一樣,選擇聯(lián)系人的不展開詳情(代理方法1)和展開詳情(代理方法2)的代理方法都寫了的時候,展開詳情的代理方法就不執(zhí)行。

CNContact

實現(xiàn)步驟

一、請求授權(quán)

// 獲得通訊錄的授權(quán)狀態(tài)
CNAuthorizationStatus status = [CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts]

授權(quán)狀態(tài)

  1. 用戶還沒有決定是否授權(quán)你的程序進(jìn)行訪問:CNAuthorizationStatusNotDetermined

  2. iOS 設(shè)備上一些許可配置阻止程序與通訊錄數(shù)據(jù)庫進(jìn)行交互:CNAuthorizationStatusRestricted

  3. 用戶明確的拒絕了你的程序?qū)νㄓ嶄浀脑L問:CNAuthorizationStatusDenied

  4. 用戶已經(jīng)授權(quán)給你的程序?qū)νㄓ嶄涍M(jìn)行訪問:CNAuthorizationStatusAuthorized

// 判斷當(dāng)前的授權(quán)狀態(tài)是否是用戶還未選擇的狀態(tài)
if (status == CNAuthorizationStatusNotDetermined)
{
    CNContactStore *store = [CNContactStore new];
    [store requestAccessForEntityType:CNEntityTypeContacts completionHandler:^(BOOL granted, NSError * _Nullable error) {
        if (granted) 
        {
            NSLog(@"授權(quán)成功!");
        }
        else
        {
            NSLog(@"授權(quán)失敗!");
        }
    }];
}

二、判斷授權(quán)狀態(tài)

如果已授權(quán),則繼續(xù);未授權(quán),則提示用戶,并返回。

// 判斷當(dāng)前的授權(quán)狀態(tài)
if (status != CNAuthorizationStatusAuthorized) 
{
    NSLog(@"您的通訊錄暫未允許訪問,請去設(shè)置->隱私里面授權(quán)!");
    return;
}

三、創(chuàng)建通訊錄對象

// 創(chuàng)建通訊錄對象
CNContactStore *contactStore = [CNContactStore new];

四、設(shè)置訪問的屬性 Key,每個 Key 對應(yīng)一個屬性,iOS 9 新增,如果沒有設(shè)置,訪問該屬性就會崩潰。

// 姓名前綴
CNContactNamePrefixKey     
// 名                 
CNContactGivenNameKey                       
// 中間名
CNContactMiddleNameKey  
// 姓                   
CNContactFamilyNameKey            
// 婚前姓         
CNContactPreviousFamilyNameKey
// 姓名后綴
CNContactNameSuffixKey   
// 昵稱                   
CNContactNicknameKey                        
// 公司
CNContactOrganizationNameKey                
// 部門
CNContactDepartmentNameKey                  
// 職位
CNContactJobTitleKey                        
// 名字拼音或音標(biāo)
CNContactPhoneticGivenNameKey
// 中間名拼音或音標(biāo)              
CNContactPhoneticMiddleNameKey
// 姓拼音或音標(biāo)
CNContactPhoneticFamilyNameKey  
// 公司拼音或音標(biāo)            
CNContactPhoneticOrganizationNameKey      
// 生日  
CNContactBirthdayKey   
// 農(nóng)歷                    
CNContactNonGregorianBirthdayKey    
// 備注        
CNContactNoteKey                            
// 圖片
CNContactImageDataKey                       
// 縮略圖
CNContactThumbnailImageDataKey              
// 圖片是否允許訪問
CNContactImageDataAvailableKey              
// 類型
CNContactTypeKey                            
// 號碼
CNContactPhoneNumbersKey                    
// 電子郵件
CNContactEmailAddressesKey                  
// 地址
CNContactPostalAddressesKey                 
// 日期
CNContactDatesKey   
// URL                        
CNContactUrlAddressesKey                    
// 關(guān)聯(lián)人
CNContactRelationsKey                       
// 社交
CNContactSocialProfilesKey                  
// 即時通訊
CNContactInstantMessageAddressesKey         

NSArray *keys = @[CNContactPhoneNumbersKey,CNContactGivenNameKey];

五、從通信錄對象中, 獲取所有的聯(lián)系人,并遍歷

// 獲取通訊錄中所有的聯(lián)系人
CNContactFetchRequest *request = [[CNContactFetchRequest alloc] initWithKeysToFetch:keys];

[contactStore enumerateContactsWithFetchRequest:request error:nil usingBlock:^(CNContact * _Nonnull contact, BOOL * _Nonnull stop) {
    // 獲取姓名
    NSString *firstName = contact.familyName;
    NSString *lastName = contact.givenName;

    NSLog(@"%@--%@",firstName,lastName);

    // 獲取電話號碼

    for (CNLabeledValue *labeledValue in contact.phoneNumbers)
    {
         CNPhoneNumber *phoneValue = labeledValue.value;
         NSString *phoneNumber = phoneValue.stringValue;
         NSString *label = [CNLabeledValue localizedStringForLabel:labeledValue.label];
         NSLog(@"%@--%@",label,phoneNumber);
    }

}];

通訊錄的修改回調(diào)

// 注冊通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_contactStoreDidChange) name:CNContactStoreDidChangeNotification object:nil]            

// 處理收到通知的 Action
- (void)_contactStoreDidChange
{
}

- (void)dealloc
{   
    // 注銷通知
    [[NSNotificationCenter defaultCenter] removeObserver:self name:CNContactStoreDidChangeNotification object:nil];
}

當(dāng) App 活躍(前臺+后臺活動期間)的時候,當(dāng)通訊錄修改的時候,會收到通知
當(dāng) App 不活躍的時候(掛起的時候),App 收不到通知;而是,當(dāng) App 到前臺的時候收到延遲的通知。

LJContactManager

介紹

LJContanctManager 是我寫的一款操作通訊錄的類庫,iOS 9 之前使用的是 AddressBook 和 AddressBookUI 系統(tǒng)庫,iOS 9 之后使用蘋果新推出的 Contacts 和 ContactsUI 框架。

安裝

CocoaPods

  1. 在 Podfile 中添加 pod 'LJContactManager'。
  2. 執(zhí)行 pod installpod update。
  3. 導(dǎo)入 <LJContactManager.h>。

手動安裝

  1. 下載 LJContactManager 文件夾內(nèi)的所有內(nèi)容。
  2. 將 LJContactManager 內(nèi)的源文件添加(拖放)到你的工程。
  3. 導(dǎo)入 LJContactManager.h

使用

主要提供以下的方法:

  • 選擇聯(lián)系人
/**
 選擇聯(lián)系人

 @param controller 控制器
 @param completcion 回調(diào)
 */
- (void)selectContactAtController:(UIViewController *)controller
                      complection:(void (^)(NSString *name, NSString *phone))completcion;

image
  • 創(chuàng)建新聯(lián)系人
/**
 創(chuàng)建新聯(lián)系人

 @param phoneNum 手機(jī)號
 @param controller 當(dāng)前 Controller
 */
- (void)createNewContactWithPhoneNum:(NSString *)phoneNum controller:(UIViewController *)controller;

image
  • 添加到現(xiàn)有聯(lián)系人
/**
 添加到現(xiàn)有聯(lián)系人

 @param phoneNum 手機(jī)號
 @param controller 當(dāng)前 Controller
 */
- (void)addToExistingContactsWithPhoneNum:(NSString *)phoneNum controller:(UIViewController *)controller;

image
  • 獲取聯(lián)系人列表(未分組的通訊錄)
/**
 獲取聯(lián)系人列表(未分組的通訊錄)

 @param completcion 回調(diào)
 */
- (void)accessContactsComplection:(void (^)(BOOL succeed, NSArray <LJPerson *> *contacts))completcion;

image
  • 獲取聯(lián)系人列表(已分組的通訊錄)
/**
 獲取聯(lián)系人列表(已分組的通訊錄)

 @param completcion 回調(diào)
 */
- (void)accessSectionContactsComplection:(void (^)(BOOL succeed, NSArray <LJSectionPerson *> *contacts, NSArray <NSString *> *keys))completcion;

image
  • 通訊錄變更回調(diào)(未分組的通訊錄)
/**
 通訊錄變更回調(diào)(未分組的通訊錄)
 */
@property (nonatomic, copy) void (^contactsChangeHanlder) (BOOL succeed, NSArray <LJPerson *> *newContacts);

  • 通訊錄變更回調(diào)(已分組的通訊錄)
/**
 通訊錄變更回調(diào)(已分組的通訊錄)
 */
@property (nonatomic, copy) void (^sectionContactsHanlder) (BOOL succeed, NSArray <LJSectionPerson *> *newSectionContacts, NSArray <NSString *> *keys);

最后

由于筆者水平有限,文中如果有錯誤的地方,或者有更好的方法,還望大神指出。
附上本文的所有 demo 下載鏈接,【GitHub】。
如果你看完后覺得對你有所幫助,還望在 GitHub 上點個 star。贈人玫瑰,手有余香。

作者:LeeJay
鏈接:http://www.itdecent.cn/p/55d1c90f62c8

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容