无法将统一的CNContact添加到iOS中的CNGroup

时间:2018-10-05 17:53:52

标签: ios objective-c

这就是我在做什么:

- (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框架可以实现此功能,请告诉我。

2 个答案:

答案 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方法提供的键集可能会很棘手。

我为这样一个半生半熟的答案表示歉意,但也许它至少可以给您带来新的动力。