mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			160 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Objective-C
		
	
			
		
		
	
	
			160 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Objective-C
		
	
| //
 | |
| //  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
 | |
|                       callingCode:(NSString *)callingCode
 | |
| {
 | |
|     // 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:callingCode];
 | |
|     NSUInteger cursorPositionAfterReformat = [PhoneNumberUtil translateCursorPosition:cursorPositionAfterChange
 | |
|                                                                                  from:textToFormat
 | |
|                                                                                    to:formattedText
 | |
|                                                                     stickingRightward:isJustDeletion];
 | |
| 
 | |
|     textField.text = formattedText;
 | |
|     UITextPosition *pos =
 | |
|         [textField positionFromPosition:textField.beginningOfDocument offset:(NSInteger)cursorPositionAfterReformat];
 | |
|     [textField setSelectedTextRange:[textField textRangeFromPosition:pos toPosition:pos]];
 | |
| }
 | |
| 
 | |
| + (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
 |