Signal-iOS/SignalUI/Views/ContactsViewHelper.m

303 lines
12 KiB
Objective-C

//
// Copyright 2017 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//
#import "ContactsViewHelper.h"
#import <ContactsUI/ContactsUI.h>
#import <SignalCoreKit/NSString+OWS.h>
#import <SignalMessaging/OWSContactsManager.h>
#import <SignalMessaging/OWSProfileManager.h>
#import <SignalMessaging/SignalMessaging-Swift.h>
#import <SignalServiceKit/AppContext.h>
#import <SignalServiceKit/Contact.h>
#import <SignalServiceKit/PhoneNumber.h>
#import <SignalUI/SignalUI-Swift.h>
NS_ASSUME_NONNULL_BEGIN
#pragma mark -
@implementation ContactsViewHelper
- (instancetype)init
{
self = [super init];
if (!self) {
return self;
}
_observers = [NSHashTable weakObjectsHashTable];
AppReadinessRunNowOrWhenUIDidBecomeReadySync(^{
// setup() - especially updateContacts() - can
// be expensive, so we don't want to run that
// directly in runNowOrWhenAppDidBecomeReadySync().
// That could cause 0x8badf00d crashes.
//
// On the other hand, the user might quickly
// open a view (like the compose view) that uses
// this helper. If the helper hasn't completed
// setup, that view won't be able to display a
// list of users to pick from. Therefore, we
// can't use runNowOrWhenAppDidBecomeReadyAsync()
// which might not run for many seconds after
// the app becomes ready.
//
// Therefore we dispatch async to the main queue.
// We'll run very soon after app UI becomes ready,
// without introducing the risk of a 0x8badf00d
// crash.
dispatch_async(dispatch_get_main_queue(), ^{ [self setup]; });
});
return self;
}
- (void)setup
{
if (CurrentAppContext().isNSE) {
return;
}
[self updateContacts];
[self observeNotifications];
}
- (void)observeNotifications
{
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(signalAccountsDidChange:)
name:OWSContactsManagerSignalAccountsDidChangeNotification
object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(profileWhitelistDidChange:)
name:kNSNotificationNameProfileWhitelistDidChange
object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(blockListDidChange:)
name:BlockingManager.blockListDidChange
object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(hideListDidChange:)
name:RecipientHidingManagerObjcBridge.hideListDidChange
object:nil];
}
- (void)signalAccountsDidChange:(NSNotification *)notification
{
OWSAssertIsOnMainThread();
[self updateContacts];
}
- (void)profileWhitelistDidChange:(NSNotification *)notification
{
OWSAssertIsOnMainThread();
[self updateContacts];
}
- (void)addObserver:(id<ContactsViewHelperObserver>)observer
{
OWSAssertIsOnMainThread();
[self.observers addObject:observer];
}
- (void)blockListDidChange:(NSNotification *)notification
{
OWSAssertIsOnMainThread();
OWSAssertDebug(!CurrentAppContext().isNSE);
[self updateContacts];
}
- (void)hideListDidChange:(NSNotification *)notification
{
OWSAssertIsOnMainThread();
OWSAssertDebug(!CurrentAppContext().isNSE);
/// Hiding a recipient who is a system contact or is someone you've
/// chatted with 1:1 updates the profile whitelist, which already
/// triggers a call to `updateContacts`. However, recipients who
/// do not fit into these categories need this other mechanism to
/// trigger `updateContacts`.
[self updateContacts];
}
#pragma mark - Contacts
- (nullable SignalAccount *)fetchSignalAccountForAddress:(SignalServiceAddress *)address
{
OWSAssertIsOnMainThread();
OWSAssertDebug(address);
OWSAssertDebug(!CurrentAppContext().isNSE);
SignalAccount *_Nullable signalAccount;
if (address.serviceIdObjC) {
signalAccount = self.serviceIdSignalAccountMap[address.serviceIdObjC];
}
if (!signalAccount && address.phoneNumber) {
signalAccount = self.phoneNumberSignalAccountMap[address.phoneNumber];
}
return signalAccount;
}
- (NSArray<SignalAccount *> *)allSignalAccounts
{
OWSAssertDebug(!CurrentAppContext().isNSE);
return self.signalAccounts;
}
- (SignalServiceAddress *)localAddress
{
OWSAssertDebug(!CurrentAppContext().isNSE);
return [TSAccountManagerObjcBridge localAciAddressWithMaybeTransaction];
}
- (BOOL)hasUpdatedContactsAtLeastOnce
{
OWSAssertDebug(!CurrentAppContext().isNSE);
return self.contactsManagerImpl.hasLoadedSystemContacts;
}
- (NSArray<SignalAccount *> *)signalAccountsMatchingSearchString:(NSString *)searchText
transaction:(SDSAnyReadTransaction *)transaction
{
OWSAssertDebug(!CurrentAppContext().isNSE);
// Check for matches against "Note to Self".
NSMutableArray<SignalAccount *> *signalAccountsToSearch = [self.signalAccounts mutableCopy];
SignalAccount *selfAccount = [[SignalAccount alloc] initWithContact:nil address:self.localAddress];
[signalAccountsToSearch addObject:selfAccount];
return [self.fullTextSearcher filterSignalAccounts:signalAccountsToSearch
withSearchText:searchText
transaction:transaction];
}
#pragma mark - Editing
- (CNContactViewController *)contactViewControllerForAddress:(SignalServiceAddress *)address
editImmediately:(BOOL)shouldEditImmediately
{
OWSAssertDebug(!CurrentAppContext().isNSE);
return [self contactViewControllerForAddress:address
editImmediately:shouldEditImmediately
addToExistingCnContact:nil
updatedNameComponents:nil];
}
- (CNContactViewController *)contactViewControllerForAddress:(SignalServiceAddress *)address
editImmediately:(BOOL)shouldEditImmediately
addToExistingCnContact:(nullable CNContact *)existingContact
updatedNameComponents:(nullable NSPersonNameComponents *)updatedNameComponents
{
OWSAssertIsOnMainThread();
OWSAssertDebug(!CurrentAppContext().isNSE);
OWSAssertDebug(self.contactsManagerImpl.editingAuthorization == ContactAuthorizationForEditingAuthorized);
SignalAccount *signalAccount = [self fetchSignalAccountForAddress:address];
CNContactViewController *_Nullable contactViewController;
CNContact *_Nullable cnContact = nil;
if (existingContact) {
CNMutableContact *updatedContact = [existingContact mutableCopy];
NSMutableArray<CNLabeledValue *> *phoneNumbers
= (updatedContact.phoneNumbers ? [updatedContact.phoneNumbers mutableCopy] : [NSMutableArray new]);
// Only add recipientId as a phone number for the existing contact
// if its not already present.
BOOL hasPhoneNumber = NO;
if (address.phoneNumber) {
for (CNLabeledValue *existingPhoneNumber in phoneNumbers) {
CNPhoneNumber *phoneNumber = existingPhoneNumber.value;
if ([phoneNumber.stringValue isEqualToString:address.phoneNumber]) {
OWSFailDebug(
@"We currently only should the 'add to existing contact' UI for phone numbers that don't "
@"correspond to an existing user.");
hasPhoneNumber = YES;
break;
}
}
if (!hasPhoneNumber) {
CNPhoneNumber *phoneNumber = [CNPhoneNumber phoneNumberWithStringValue:address.phoneNumber];
CNLabeledValue<CNPhoneNumber *> *labeledPhoneNumber =
[CNLabeledValue labeledValueWithLabel:CNLabelPhoneNumberMain value:phoneNumber];
[phoneNumbers addObject:labeledPhoneNumber];
updatedContact.phoneNumbers = phoneNumbers;
// When adding a phone number to an existing contact, immediately enter
// "edit" mode.
shouldEditImmediately = YES;
}
cnContact = updatedContact;
}
}
if (signalAccount && !cnContact && signalAccount.contact.cnContactId != nil) {
cnContact = [self.contactsManager cnContactWithId:signalAccount.contact.cnContactId];
}
if (cnContact) {
if (updatedNameComponents) {
CNMutableContact *updatedContact = [cnContact mutableCopy];
updatedContact.givenName = updatedNameComponents.givenName;
updatedContact.familyName = updatedNameComponents.familyName;
cnContact = updatedContact;
}
if (shouldEditImmediately) {
// Not actually a "new" contact, but this brings up the edit form rather than the "Read" form
// saving our users a tap in some cases when we already know they want to edit.
contactViewController = [CNContactViewController viewControllerForNewContact:cnContact];
// Default title is "New Contact". We could give a more descriptive title, but anything
// seems redundant - the context is sufficiently clear.
contactViewController.title = @"";
} else {
contactViewController = [CNContactViewController viewControllerForContact:cnContact];
}
}
if (!contactViewController) {
CNMutableContact *newContact = [CNMutableContact new];
if (address.phoneNumber) {
CNPhoneNumber *phoneNumber = [CNPhoneNumber phoneNumberWithStringValue:address.phoneNumber];
CNLabeledValue<CNPhoneNumber *> *labeledPhoneNumber =
[CNLabeledValue labeledValueWithLabel:CNLabelPhoneNumberMain value:phoneNumber];
newContact.phoneNumbers = @[ labeledPhoneNumber ];
}
[self.databaseStorage readWithBlock:^(SDSAnyReadTransaction *transaction) {
newContact.givenName = [self.profileManagerImpl givenNameForAddress:address transaction:transaction];
newContact.familyName = [self.profileManagerImpl familyNameForAddress:address transaction:transaction];
newContact.imageData
= UIImagePNGRepresentation([self.profileManagerImpl profileAvatarForAddress:address
downloadIfMissing:YES
authedAccount:AuthedAccount.implicit
transaction:transaction]);
}];
if (updatedNameComponents) {
newContact.givenName = updatedNameComponents.givenName;
newContact.familyName = updatedNameComponents.familyName;
}
contactViewController = [CNContactViewController viewControllerForNewContact:newContact];
}
contactViewController.allowsActions = NO;
contactViewController.allowsEditing = YES;
contactViewController.edgesForExtendedLayout = UIRectEdgeNone;
contactViewController.view.backgroundColor = Theme.backgroundColor;
return contactViewController;
}
@end
NS_ASSUME_NONNULL_END