Index: ios/chrome/browser/web/mailto_url_rewriter.mm |
diff --git a/ios/chrome/browser/web/mailto_url_rewriter.mm b/ios/chrome/browser/web/mailto_url_rewriter.mm |
new file mode 100644 |
index 0000000000000000000000000000000000000000..d6dc6a7acceb6174f6ee370d300cbdcabf1b273c |
--- /dev/null |
+++ b/ios/chrome/browser/web/mailto_url_rewriter.mm |
@@ -0,0 +1,184 @@ |
+// Copyright 2017 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#import "ios/chrome/browser/web/mailto_url_rewriter.h" |
+ |
+#import "base/logging.h" |
+#import "ios/chrome/browser/web/mailto_handler.h" |
+#import "ios/chrome/browser/web/mailto_handler_gmail.h" |
+#import "ios/chrome/browser/web/mailto_handler_system_mail.h" |
+ |
+#import <UIKit/UIKit.h> |
+ |
+#if !defined(__has_feature) || !__has_feature(objc_arc) |
+#error "This file requires ARC support." |
+#endif |
+ |
+namespace { |
+// The key for NSUserDefaults to store the Mail client selected to handle |
+// mailto: URL scheme. |
+NSString* const kMailtoDefaultHandlerKey = @"MailtoHandlerDefault"; |
+} // namespace |
+ |
+@interface MailtoURLRewriter () |
+ |
+// Dictionary keyed by the App Store ID of the Mail client and the value is |
+// the MailtoHandler object that can rewrite a mailto: URL. |
+@property(nonatomic, strong) |
+ NSMutableDictionary<NSString*, MailtoHandler*>* handlers; |
+ |
+// Private method for testing to clear the default state. |
++ (void)resetDefaultHandlerID; |
+ |
+// Private method to add |handlerApp| to the list of know Mail client apps. |
+- (void)addMailtoApp:(MailtoHandler*)handlerApp; |
+ |
+// Custom logic to handle the migration from Google Native App Launcher options |
+// to this simplified mailto: URL only system. This must be called after |
+// -addMailtoApp: has been called to add all the known Mail client apps. |
+// TODO(crbug.com/718601): At some point in the future when almost all users |
+// have upgraded, this method can be removed. |
+- (void)convertLegacyOptions; |
+ |
+@end |
+ |
+@implementation MailtoURLRewriter |
+@synthesize handlers = _handlers; |
+ |
++ (NSString*)systemMailApp { |
+ // This is the App Store ID for Apple Mail app. |
+ // See https://itunes.apple.com/us/app/mail/id1108187098?mt=8 |
+ return @"1108187098"; |
+} |
+ |
+- (instancetype)init { |
+ self = [super init]; |
+ if (self) { |
+ _handlers = [NSMutableDictionary dictionary]; |
+ } |
+ return self; |
+} |
+ |
+- (instancetype)initWithStandardHandlers { |
+ self = [self init]; |
+ if (self) { |
+ [self addMailtoApp:[[MailtoHandlerSystemMail alloc] init]]; |
+ [self addMailtoApp:[[MailtoHandlerGmail alloc] init]]; |
+ [self convertLegacyOptions]; |
+ } |
+ return self; |
+} |
+ |
+- (NSArray<MailtoHandler*>*)defaultHandlers { |
+ return [_handlers allValues]; |
+} |
+ |
+- (NSString*)defaultHandlerID { |
+ NSString* value = [[NSUserDefaults standardUserDefaults] |
+ stringForKey:kMailtoDefaultHandlerKey]; |
+ return value ? value : [[self class] systemMailApp]; |
+} |
+ |
+- (void)setDefaultHandlerID:(NSString*)appStoreID { |
+ DCHECK([appStoreID length]); |
+ [[NSUserDefaults standardUserDefaults] setObject:appStoreID |
+ forKey:kMailtoDefaultHandlerKey]; |
+} |
+ |
+- (NSString*)rewriteMailtoURL:(const GURL&)gURL { |
+ NSString* value = [self defaultHandlerID]; |
+ if ([value length]) { |
+ MailtoHandler* handler = _handlers[value]; |
+ if (handler) { |
+ return [handler rewriteMailtoURL:gURL]; |
+ } |
+ } |
+ return nil; |
+} |
+ |
+#pragma mark - Private |
+ |
++ (void)resetDefaultHandlerID { |
+ [[NSUserDefaults standardUserDefaults] |
+ removeObjectForKey:kMailtoDefaultHandlerKey]; |
+} |
+ |
+- (void)addMailtoApp:(MailtoHandler*)handlerApp { |
+ if ([handlerApp isAvailable]) |
+ [_handlers setObject:handlerApp forKey:[handlerApp appStoreID]]; |
+} |
+ |
+// Implements the migration logic for users of previous versions of Google |
+// Chrome which supports Google Native App Launcher. The goal is to preserve |
+// the previous behavior and support user choice of non-system provided Mail |
+// client apps. System-provided Mail client app will be referred to as |
+// "System Mail". The migration goals are: |
+// - If a user has not made a choice for preferred mailto: handler in the past, |
+// the behavior after upgrading to this version should not change. |
+// - If a user had disabled Gmail app as the preferred mailto: handler in the |
+// past, System Mail should be selected as the default mailto: handler. |
+// - If a user installs Gmail app after the installation of Chrome, Gmail app |
+// will be chosen as the default mailto: handler, preserving the previous |
+// behavior. |
+// - If a user installs another 3rd party mail client app, assuming that the |
+// 3rd party mail client app is explicitly supported in Chrome, the new 3rd |
+// party mail client app can be set as the default mailto: handler through |
+// Tools > Settings > Content Settings. |
+- (void)convertLegacyOptions { |
+ // If a default handler for mailto: has already been registered, this is not |
+ // an upgrade case that needs to take legacy options into account. |
+ NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults]; |
+ if ([defaults objectForKey:kMailtoDefaultHandlerKey]) |
+ return; |
+ |
+ // The key used in NSUserDefaults is from legacy code when Native App |
+ // Launcher was still in use. The general format is a string prefix, |
+ // underscore, then followed by the App Store ID of the application. |
+ NSString* const kGmailAppStoreID = @"422689480"; |
+ NSString* const kLegacyShouldAutoOpenKey = |
+ [NSString stringWithFormat:@"ShouldAutoOpenLinks_%@", kGmailAppStoreID]; |
+ |
+ // If Gmail app is not installed, there is nothing to migrate. |
+ MailtoHandler* gmailHandler = _handlers[kGmailAppStoreID]; |
+ if (!gmailHandler) { |
+ // There is a stale legacy key around. This may mean that the user was using |
+ // Gmail some time in the past but later uninstalled Gmail app. This clears |
+ // the stale entry to start user from a clean slate again. |
+ [defaults removeObjectForKey:kLegacyShouldAutoOpenKey]; |
+ return; |
+ } |
+ |
+ // If Gmail app is not installed or otherwise unavailable, do not make it the |
+ // default mailto: handler app. |
+ if (![gmailHandler isAvailable]) |
+ [defaults removeObjectForKey:kMailtoDefaultHandlerKey]; |
rohitrao (ping after 24h)
2017/05/08 18:57:50
In what cases could this line (and line 169) have
pkl (ping after 24h if needed)
2017/05/08 22:15:42
This and the previous if-clause can be merged to h
|
+ |
+ // User previously had a selection made for opening mailto: links with Gmail, |
+ // upgrade will set Gmail app to be the default mailto: handler. If user had |
+ // opted out to using Gmail app (even when it was installed), migrate user |
+ // to use system-provided Mail client. |
+ NSNumber* legacyValue = [defaults objectForKey:kLegacyShouldAutoOpenKey]; |
+ if (legacyValue) { |
+ switch ([legacyValue intValue]) { |
+ case 0: // kAutoOpenLinksNotSet |
+ case 2: // kAutoOpenLinksYes |
+ [defaults setObject:kGmailAppStoreID forKey:kMailtoDefaultHandlerKey]; |
+ break; |
+ case 1: // kAutoOpenLinksNo |
+ [defaults removeObjectForKey:kMailtoDefaultHandlerKey]; |
+ break; |
+ default: |
+ NOTREACHED(); |
+ break; |
+ } |
+ [defaults removeObjectForKey:kLegacyShouldAutoOpenKey]; |
+ } else { |
+ // This is an unlikely (if not impossible) case of user not having the |
+ // legacy key set at all. In this case, set the default to use Gmail app |
+ // for mailto: handling. |
+ [defaults setObject:kGmailAppStoreID forKey:kMailtoDefaultHandlerKey]; |
+ } |
+} |
+ |
+@end |