|
|
|
//
|
|
|
|
// Copyright (c) 2019 Open Whisper Systems. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#import "ViewControllerUtils.h"
|
|
|
|
#import "PhoneNumber.h"
|
|
|
|
#import <AVFoundation/AVFoundation.h>
|
|
|
|
#import <SignalMessaging/Environment.h>
|
|
|
|
#import <SignalServiceKit/NSString+SSK.h>
|
|
|
|
#import <SignalServiceKit/PhoneNumberUtil.h>
|
|
|
|
|
|
|
|
NS_ASSUME_NONNULL_BEGIN
|
|
|
|
|
|
|
|
NSString *const TappedStatusBarNotification = @"TappedStatusBarNotification";
|
|
|
|
|
|
|
|
const NSUInteger kMin2FAPinLength = 4;
|
|
|
|
const NSUInteger kMax2FAPinLength = 16;
|
|
|
|
|
|
|
|
@implementation ViewControllerUtils
|
|
|
|
|
|
|
|
+ (void)phoneNumberTextField:(UITextField *)textField
|
|
|
|
shouldChangeCharactersInRange:(NSRange)range
|
|
|
|
replacementString:(NSString *)insertionText
|
|
|
|
countryCode:(NSString *)countryCode
|
|
|
|
{
|
|
|
|
return [self phoneNumberTextField:textField
|
|
|
|
shouldChangeCharactersInRange:range
|
|
|
|
replacementString:insertionText
|
|
|
|
countryCode:countryCode
|
|
|
|
prefix:nil];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (void)phoneNumberTextField:(UITextField *)textField
|
|
|
|
shouldChangeCharactersInRange:(NSRange)range
|
|
|
|
replacementString:(NSString *)insertionText
|
|
|
|
countryCode:(NSString *)countryCode
|
|
|
|
prefix:(nullable NSString *)prefix
|
|
|
|
{
|
|
|
|
// Phone numbers takes many forms.
|
|
|
|
//
|
|
|
|
// * We only want to let the user enter decimal digits.
|
|
|
|
// * The user shouldn't have to enter hyphen, parentheses or whitespace;
|
|
|
|
// the phone number should be formatted automatically.
|
|
|
|
// * The user should be able to copy and paste freely.
|
|
|
|
// * Invalid input should be simply ignored.
|
|
|
|
//
|
|
|
|
// We accomplish this by being permissive and trying to "take as much of the user
|
|
|
|
// input as possible".
|
|
|
|
//
|
|
|
|
// * Always accept deletes.
|
|
|
|
// * Ignore invalid input.
|
|
|
|
// * Take partial input if possible.
|
|
|
|
|
|
|
|
NSString *oldText = textField.text;
|
|
|
|
|
|
|
|
// Construct the new contents of the text field by:
|
|
|
|
// 1. Determining the "left" substring: the contents of the old text _before_ the deletion range.
|
|
|
|
// Filtering will remove non-decimal digit characters like hyphen "-".
|
|
|
|
NSString *left = [oldText substringToIndex:range.location].digitsOnly;
|
|
|
|
// 2. Determining the "right" substring: the contents of the old text _after_ the deletion range.
|
|
|
|
NSString *right = [oldText substringFromIndex:range.location + range.length].digitsOnly;
|
|
|
|
// 3. Determining the "center" substring: the contents of the new insertion text.
|
|
|
|
NSString *center = insertionText.digitsOnly;
|
|
|
|
|
|
|
|
// 3a. If user hits backspace, they should always delete a _digit_ to the
|
|
|
|
// left of the cursor, even if the text _immediately_ to the left of
|
|
|
|
// cursor is "formatting text" (e.g. whitespace, a hyphen or a
|
|
|
|
// parentheses).
|
|
|
|
bool isJustDeletion = insertionText.length == 0;
|
|
|
|
if (isJustDeletion) {
|
|
|
|
NSString *deletedText = [oldText substringWithRange:range];
|
|
|
|
BOOL didDeleteFormatting = (deletedText.length == 1 && deletedText.digitsOnly.length < 1);
|
|
|
|
if (didDeleteFormatting && left.length > 0) {
|
|
|
|
left = [left substringToIndex:left.length - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 4. Construct the "raw" new text by concatenating left, center and right.
|
|
|
|
NSString *textAfterChange = [[left stringByAppendingString:center] stringByAppendingString:right];
|
|
|
|
// 4a. Ensure we don't exceed the maximum length for a e164 phone number,
|
|
|
|
// 15 digits, per: https://en.wikipedia.org/wiki/E.164
|
|
|
|
//
|
|
|
|
// NOTE: The actual limit is 18, not 15, because of certain invalid phone numbers in Germany.
|
|
|
|
// https://github.com/googlei18n/libphonenumber/blob/master/FALSEHOODS.md
|
|
|
|
const int kMaxPhoneNumberLength = 18;
|
|
|
|
if (textAfterChange.length > kMaxPhoneNumberLength) {
|
|
|
|
textAfterChange = [textAfterChange substringToIndex:kMaxPhoneNumberLength];
|
|
|
|
}
|
|
|
|
// 5. Construct the "formatted" new text by inserting a hyphen if necessary.
|
|
|
|
// reformat the phone number, trying to keep the cursor beside the inserted or deleted digit
|
|
|
|
NSUInteger cursorPositionAfterChange = MIN(left.length + center.length, textAfterChange.length);
|
|
|
|
|
|
|
|
NSString *textToFormat = textAfterChange;
|
|
|
|
NSString *formattedText = [PhoneNumber bestEffortFormatPartialUserSpecifiedTextToLookLikeAPhoneNumber:textToFormat
|
|
|
|
withSpecifiedCountryCodeString:countryCode];
|
|
|
|
NSUInteger cursorPositionAfterReformat = [PhoneNumberUtil translateCursorPosition:cursorPositionAfterChange
|
|
|
|
from:textToFormat
|
|
|
|
to:formattedText
|
|
|
|
stickingRightward:isJustDeletion];
|
|
|
|
|
|
|
|
// PhoneNumber's formatting logic requires a calling code.
|
|
|
|
//
|
|
|
|
// If we want to edit the phone number separately from the calling code
|
|
|
|
// (e.g. in the new onboarding views), we need to temporarily prepend the
|
|
|
|
// calling code during formatting, then remove it afterward. This is
|
|
|
|
// non-trivial since the calling code itself can be affected by the
|
|
|
|
// formatting. Additionally, we need to ensure that this prepend/remove
|
|
|
|
// doesn't affect the cursor position.
|
|
|
|
BOOL hasPrefix = prefix.length > 0;
|
|
|
|
if (hasPrefix) {
|
|
|
|
// Prepend the prefix.
|
|
|
|
NSString *textToFormatWithPrefix = [prefix stringByAppendingString:textAfterChange];
|
|
|
|
// Format with the prefix.
|
|
|
|
NSString *formattedTextWithPrefix =
|
|
|
|
[PhoneNumber bestEffortFormatPartialUserSpecifiedTextToLookLikeAPhoneNumber:textToFormatWithPrefix
|
|
|
|
withSpecifiedCountryCodeString:countryCode];
|
|
|
|
// Determine the new cursor position with the prefix.
|
|
|
|
NSUInteger cursorPositionWithPrefix = [PhoneNumberUtil translateCursorPosition:cursorPositionAfterChange
|
|
|
|
from:textToFormat
|
|
|
|
to:formattedTextWithPrefix
|
|
|
|
stickingRightward:isJustDeletion];
|
|
|
|
// Try to determine how much of the formatted text is derived
|
|
|
|
// from the prefix.
|
|
|
|
NSString *_Nullable formattedPrefix =
|
|
|
|
[self findFormattedPrefixForPrefix:prefix formattedText:formattedTextWithPrefix];
|
|
|
|
if (formattedPrefix && cursorPositionWithPrefix >= formattedPrefix.length) {
|
|
|
|
// Remove the prefix from the formatted text.
|
|
|
|
formattedText = [formattedTextWithPrefix substringFromIndex:formattedPrefix.length];
|
|
|
|
// Adjust the cursor position accordingly.
|
|
|
|
cursorPositionAfterReformat = cursorPositionWithPrefix - formattedPrefix.length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
textField.text = formattedText;
|
|
|
|
UITextPosition *pos =
|
|
|
|
[textField positionFromPosition:textField.beginningOfDocument offset:(NSInteger)cursorPositionAfterReformat];
|
|
|
|
[textField setSelectedTextRange:[textField textRangeFromPosition:pos toPosition:pos]];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (nullable NSString *)findFormattedPrefixForPrefix:(NSString *)prefix formattedText:(NSString *)formattedText
|
|
|
|
{
|
|
|
|
NSCharacterSet *characterSet = [[NSCharacterSet characterSetWithCharactersInString:@"+0123456789"] invertedSet];
|
|
|
|
NSString *filteredPrefix =
|
|
|
|
[[prefix componentsSeparatedByCharactersInSet:characterSet] componentsJoinedByString:@""];
|
|
|
|
NSString *filteredText =
|
|
|
|
[[formattedText componentsSeparatedByCharactersInSet:characterSet] componentsJoinedByString:@""];
|
|
|
|
if (filteredPrefix.length < 1 || filteredText.length < 1 || ![filteredText hasPrefix:filteredPrefix]) {
|
|
|
|
OWSFailDebug(@"Invalid prefix: '%@' for formatted text: '%@'", prefix, formattedText);
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
NSString *filteredTextWithoutPrefix = [filteredText substringFromIndex:filteredPrefix.length];
|
|
|
|
// To find the "formatted prefix", try to find the shortest "tail" of formattedText
|
|
|
|
// which after being filtered is equivalent to the "filtered text" - "filter prefix".
|
|
|
|
// The "formatted prefix" is the "head" that corresponds to that "tail".
|
|
|
|
for (NSUInteger substringLength = 1; substringLength < formattedText.length - 1; substringLength++) {
|
|
|
|
NSUInteger pivot = formattedText.length - substringLength;
|
|
|
|
NSString *head = [formattedText substringToIndex:pivot];
|
|
|
|
NSString *tail = [formattedText substringFromIndex:pivot];
|
|
|
|
NSString *filteredTail =
|
|
|
|
[[tail componentsSeparatedByCharactersInSet:characterSet] componentsJoinedByString:@""];
|
|
|
|
if ([filteredTail isEqualToString:filteredTextWithoutPrefix]) {
|
|
|
|
return head;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (void)ows2FAPINTextField:(UITextField *)textField
|
|
|
|
shouldChangeCharactersInRange:(NSRange)range
|
|
|
|
replacementString:(NSString *)insertionText
|
|
|
|
{
|
|
|
|
// * We only want to let the user enter decimal digits.
|
|
|
|
// * The user should be able to copy and paste freely.
|
|
|
|
// * Invalid input should be simply ignored.
|
|
|
|
//
|
|
|
|
// We accomplish this by being permissive and trying to "take as much of the user
|
|
|
|
// input as possible".
|
|
|
|
//
|
|
|
|
// * Always accept deletes.
|
|
|
|
// * Ignore invalid input.
|
|
|
|
// * Take partial input if possible.
|
|
|
|
|
|
|
|
NSString *oldText = textField.text;
|
|
|
|
// Construct the new contents of the text field by:
|
|
|
|
// 1. Determining the "left" substring: the contents of the old text _before_ the deletion range.
|
|
|
|
// Filtering will remove non-decimal digit characters.
|
|
|
|
NSString *left = [oldText substringToIndex:range.location].digitsOnly;
|
|
|
|
// 2. Determining the "right" substring: the contents of the old text _after_ the deletion range.
|
|
|
|
NSString *right = [oldText substringFromIndex:range.location + range.length].digitsOnly;
|
|
|
|
// 3. Determining the "center" substring: the contents of the new insertion text.
|
|
|
|
NSString *center = insertionText.digitsOnly;
|
|
|
|
// 4. Construct the "raw" new text by concatenating left, center and right.
|
|
|
|
NSString *textAfterChange = [[left stringByAppendingString:center] stringByAppendingString:right];
|
|
|
|
// 5. Ensure we don't exceed the maximum length for a PIN.
|
|
|
|
if (textAfterChange.length > kMax2FAPinLength) {
|
|
|
|
textAfterChange = [textAfterChange substringToIndex:kMax2FAPinLength];
|
|
|
|
}
|
|
|
|
// 6. Construct the final text.
|
|
|
|
textField.text = textAfterChange;
|
|
|
|
NSUInteger cursorPositionAfterChange = MIN(left.length + center.length, textAfterChange.length);
|
|
|
|
UITextPosition *pos =
|
|
|
|
[textField positionFromPosition:textField.beginningOfDocument offset:(NSInteger)cursorPositionAfterChange];
|
|
|
|
[textField setSelectedTextRange:[textField textRangeFromPosition:pos toPosition:pos]];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSString *)examplePhoneNumberForCountryCode:(NSString *)countryCode callingCode:(NSString *)callingCode
|
|
|
|
{
|
|
|
|
OWSAssertDebug(countryCode.length > 0);
|
|
|
|
OWSAssertDebug(callingCode.length > 0);
|
|
|
|
|
|
|
|
NSString *examplePhoneNumber = [PhoneNumberUtil examplePhoneNumberForCountryCode:countryCode];
|
|
|
|
OWSAssertDebug(!examplePhoneNumber || [examplePhoneNumber hasPrefix:callingCode]);
|
|
|
|
if (examplePhoneNumber && [examplePhoneNumber hasPrefix:callingCode]) {
|
|
|
|
NSString *formattedPhoneNumber =
|
|
|
|
[PhoneNumber bestEffortFormatPartialUserSpecifiedTextToLookLikeAPhoneNumber:examplePhoneNumber
|
|
|
|
withSpecifiedCountryCodeString:countryCode];
|
|
|
|
if (formattedPhoneNumber.length > 0) {
|
|
|
|
examplePhoneNumber = formattedPhoneNumber;
|
|
|
|
}
|
|
|
|
|
|
|
|
return [NSString
|
|
|
|
stringWithFormat:
|
|
|
|
NSLocalizedString(@"PHONE_NUMBER_EXAMPLE_FORMAT",
|
|
|
|
@"A format for a label showing an example phone number. Embeds {{the example phone number}}."),
|
|
|
|
[examplePhoneNumber substringFromIndex:callingCode.length]];
|
|
|
|
} else {
|
|
|
|
return @"";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
NS_ASSUME_NONNULL_END
|