这就是我在做什么:
- (void)doCreateGroup {
[[self contentView] endEditing:true];
NSString * newString = [[[[self contentView] groupNameField] text] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
NSString * firstError = nil;
if ([newString length] == 0) {
firstError = @"Missing group name";
}
NSError * groupsError = nil;
NSArray * groups = [self.contactStore groupsMatchingPredicate:nil error:&groupsError];
for (CNGroup * group in groups) {
if ([group.name isEqualToString:newString]) {
firstError = @"Group already exists";
}
}
if (firstError) {
[self presentViewController:[WLGCommonUtilities doProcessErrorWithOkay:@"Error" errorMessage:firstError] animated:YES completion:nil];
return;
}
CNMutableGroup * newGroup = [CNMutableGroup new];
[newGroup setName:newString];
CNSaveRequest *saveRequest = [CNSaveRequest new];
[saveRequest addGroup:newGroup toContainerWithIdentifier:nil];
NSError * error = nil;
[self.contactStore executeSaveRequest:saveRequest error:&error];
if (error) {
[self presentViewController:[WLGCommonUtilities doProcessErrorWithOkay:@"Error" errorMessage:[error localizedDescription]] animated:YES completion:nil];
} else {
CNSaveRequest *saveRequest2 = [CNSaveRequest new];
NSArray * groupsAgain = [self.contactStore groupsMatchingPredicate:nil error:&groupsError];
CNGroup * gotGroup;
for (CNGroup * group in groupsAgain) {
if ([group.name isEqualToString:newString]) {
gotGroup = group;
}
}
for (CNContact * contact in self.selectedContactsArray) {
[saveRequest2 addMember:contact toGroup:gotGroup];
}
NSError * error1 = nil;
[self.contactStore executeSaveRequest:saveRequest2 error:&error1];
if (error) {
[self presentViewController:[WLGCommonUtilities doProcessErrorWithOkay:@"Error" errorMessage:[error1 localizedDescription]] animated:YES completion:nil];
} else {
[[self navigationController] dismissViewControllerAnimated:true completion:nil];
}
}
}
这用于创建CNGroup,然后将联系人添加到所述CNGroup。适用于所有联系人(统一联系人除外)。我已经尽一切可能使这项工作成功了,但事实并非如此。它可能与统一的CNContact标识符有关,因为该标识符仅存储在临时内存中,因此由于它实际上没有REAL CNContact标识符,因此无法将其添加到CNGroup。联系人框架是一团糟!任何帮助,将不胜感激。我还向Apple提出了技术支持请求。
编辑: 解决此问题的一种方法是使用现在不建议使用的地址框架。这样可以将尽可能多的统一联系人添加到地址组。
ABRecordRef group = ABGroupCreate();
ABAddressBookAddRecord(addressBook, group, nil);
ABRecordSetValue(group, kABGroupNameProperty,@"My Groups", nil);
for (int i=0;i < nPeople;i++) {
ABRecordRef ref = CFArrayGetValueAtIndex(allPeople,i);
ABGroupAddMember(group, ref, nil);
ABAddressBookSave(addressBook, nil);
}
这确实将联系人簿中的所有内容保存到一个组中,即所有可见的联系人。因此它将确实将统一联系人存储到组中。如果您在组中同时取消联系的链接,则两个联系人都将保留在组中。所以旧的框架可以解决这个问题。似乎无法使用新的Contacts框架解决该问题。同样,新的Contacts框架可能会丢失一些内容,因此,如果iOS中的当前Contacts框架可以实现此功能,请告诉我。
答案 0 :(得分:2)
我想通了。这是一团糟
第一步:
NSMutableArray * finalArray = [NSMutableArray array];
NSMutableArray * unifiedContacts = [NSMutableArray array];
NSMutableArray * fullContacts = [NSMutableArray array];
CNContactFetchRequest * request = [[CNContactFetchRequest alloc] initWithKeysToFetch:keys];
[request setSortOrder:CNContactSortOrderGivenName];
[self.contactStore enumerateContactsWithFetchRequest:request error:&error usingBlock:^(CNContact * _Nonnull contact, BOOL * _Nonnull stop) {
[unifiedContacts addObject:contact];
}];
CNContactFetchRequest * request2 = [[CNContactFetchRequest alloc] initWithKeysToFetch:keys];
[request2 setUnifyResults:false];
[request2 setSortOrder:CNContactSortOrderGivenName];
[self.contactStore enumerateContactsWithFetchRequest:request2 error:nil usingBlock:^(CNContact * _Nonnull contact, BOOL * _Nonnull stop) {
[fullContacts addObject:contact];
}];
for (CNContact * contctUn in unifiedContacts) {
NSMutableArray * nestedContacts = [NSMutableArray array];
for (CNContact * contct in fullContacts) {
if ([contctUn isUnifiedWithContactWithIdentifier:contct.identifier]) {
[nestedContacts addObject:contct];
}
}
if (nestedContacts.count) {
[finalArray addObject:@{@"contact" : contctUn, @"linked" : nestedContacts}];
} else {
[finalArray addObject:@{@"contact" : contctUn}];
}
}
self.mainArray = [finalArray mutableCopy];
此操作从统一联系人中拉出所有联系人,然后拉出所有未统一的联系人,将组拼接在一起,并将它们保存为词典,如果“确实”链接到联系人中,则“链接”是链接联系人的数组问题。
第2步:创建一个群组...这非常简单,因为这非常容易,因此无需显示代码
第3步:
for (id obj in self.filteredSearchArray) {
if ([obj valueForKey:@"linked"]) {
for (id obj2 in [obj valueForKey:@"linked"]) {
[self.selectedContactsArray addObject:obj2];
}
}
}
CNSaveRequest *saveRequest2 = [CNSaveRequest new];
for (CNContact * contact in self.selectedContactsArray) {
[saveRequest2 addMember:contact toGroup:[newGroup copy]];
}
NSError * error1 = nil;
[self.contactStore executeSaveRequest:saveRequest2 error:&error1];
self.selectedContactsArray是包含您要在组中的联系人的数组。它包含您要在组中的所有联系人,另外,如果您要在组中的联系人已链接到用户,则它还包含子链接的联系人。
此保存请求执行时,该组现在包含统一联系人。
这是一团糟。 iOS中的Contacts Framework一片混乱,但这是可行的。没有为联系人创建群组的应用可以解决此问题,因此这是一百万美元的解决方案。
答案 1 :(得分:0)
这确实很奇怪。至少有一种解决方法,您是否尝试过将CNContactFetchRequest
设置为unifyResults
的{{1}}来获取选定的联系人?
我的意思是,我不知道您从何处获得false
,我假设您可以修改现有的请求以相应地向您提供该数据,或者您必须以某种方式再次重新获取联系人。这可能真的非常丑陋,因为您必须使用谓词或键集构造一个获取请求,以确保匹配相同的联系人(并且仅匹配那些联系人)以及设置为{{1}的说selectedContactsArray
成员}。
我会想像这样的东西(对不起,使用swift,现在对我来说有点紧凑,我希望可以):
unifyResults
我承认我对Contacts框架不是很熟悉,所以我不能说这是否真的可行。特别是,如果您没有保证可以包含在您想要的所有联系人中的键,那么必须向false
方法提供的键集可能会很棘手。
我为这样一个半生半熟的答案表示歉意,但也许它至少可以给您带来新的动力。