Вы не должны пытаться использовать регулярные выражения для проверки электронной почты. С постоянно меняющимися TLD ваш валидатор является либо неполным, либо неточным. Вместо этого вы должны использовать NSDataDetector
библиотеки Apple, которые возьмут строку и попытаются увидеть, есть ли какие-либо известные поля данных (электронные письма, адреса, даты и т. Д.). Apple SDK сделает все возможное, чтобы идти в ногу с TLD, и вы сможете воспользоваться их усилиями !! :)
Кроме того, если iMessage (или любое другое текстовое поле) не считает, что это электронная почта, следует ли вам рассматривать электронную почту?
Я поместил эту функцию в NSString
категорию, поэтому строка, которую вы тестируете, является self
.
- (BOOL)isValidEmail {
// Trim whitespace first
NSString *trimmedText = [self stringByTrimmingCharactersInSet:NSCharacterSet.whitespaceAndNewlineCharacterSet];
if (self && self.length > 0) return NO;
NSError *error = nil;
NSDataDetector *dataDetector = [[NSDataDetector alloc] initWithTypes:NSTextCheckingTypeLink error:&error];
if (!dataDetector) return NO;
// This string is a valid email only if iOS detects a mailto link out of the full string
NSArray<NSTextCheckingResult *> *allMatches = [dataDetector matchesInString:trimmedText options:kNilOptions range:NSMakeRange(0, trimmedText.length)];
if (error) return NO;
return (allMatches.count == 1 && [[[allMatches.firstObject URL] absoluteString] isEqual:[NSString stringWithFormat:@"mailto:%@", self]]);
}
или как быстрое String
расширение
extension String {
func isValidEmail() -> Bool {
let trimmed = self.trimmingCharacters(in: .whitespacesAndNewlines)
guard !trimmed.isEmpty, let dataDetector = try? NSDataDetector(types: NSTextCheckingResult.CheckingType.link.rawValue) else {
return false
}
let allMatches = dataDetector.matches(in: trimmed, options: [], range: NSMakeRange(0, trimmed.characters.count))
return allMatches.count == 1 && allMatches.first?.url?.absoluteString == "mailto:\(trimmed)"
}
}