| Index: chrome/browser/ui/webui/sync_setup_handler2.cc
|
| diff --git a/chrome/browser/ui/webui/sync_setup_handler2.cc b/chrome/browser/ui/webui/sync_setup_handler2.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..d47e6a27d853ad2eb58752ccdbead5b930ef16cf
|
| --- /dev/null
|
| +++ b/chrome/browser/ui/webui/sync_setup_handler2.cc
|
| @@ -0,0 +1,772 @@
|
| +// Copyright (c) 2011 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.
|
| +
|
| +#include "chrome/browser/ui/webui/sync_setup_handler2.h"
|
| +
|
| +#include "base/bind.h"
|
| +#include "base/bind_helpers.h"
|
| +#include "base/json/json_reader.h"
|
| +#include "base/json/json_writer.h"
|
| +#include "base/metrics/histogram.h"
|
| +#include "base/utf_string_conversions.h"
|
| +#include "base/values.h"
|
| +#include "chrome/browser/google/google_util.h"
|
| +#include "chrome/browser/prefs/pref_service.h"
|
| +#include "chrome/browser/profiles/profile.h"
|
| +#include "chrome/browser/profiles/profile_info_cache.h"
|
| +#include "chrome/browser/profiles/profile_manager.h"
|
| +#include "chrome/browser/profiles/profile_metrics.h"
|
| +#include "chrome/browser/sync/profile_sync_service.h"
|
| +#include "chrome/browser/sync/protocol/service_constants.h"
|
| +#include "chrome/browser/sync/signin_manager.h"
|
| +#include "chrome/browser/sync/syncable/model_type.h"
|
| +#include "chrome/browser/sync/sync_setup_flow.h"
|
| +#include "chrome/browser/sync/util/oauth.h"
|
| +#include "chrome/browser/ui/browser_list.h"
|
| +#include "chrome/browser/ui/webui/sync_promo_trial.h"
|
| +#include "chrome/browser/ui/webui/sync_promo_ui.h"
|
| +#include "chrome/browser/ui/webui/user_selectable_sync_type.h"
|
| +#include "chrome/common/net/gaia/gaia_constants.h"
|
| +#include "chrome/common/pref_names.h"
|
| +#include "chrome/common/url_constants.h"
|
| +#include "content/browser/tab_contents/tab_contents.h"
|
| +#include "grit/chromium_strings.h"
|
| +#include "grit/generated_resources.h"
|
| +#include "grit/locale_settings.h"
|
| +#include "ui/base/l10n/l10n_util.h"
|
| +
|
| +using l10n_util::GetStringFUTF16;
|
| +using l10n_util::GetStringUTF16;
|
| +
|
| +namespace {
|
| +
|
| +bool GetAuthData(const std::string& json,
|
| + std::string* username,
|
| + std::string* password,
|
| + std::string* captcha,
|
| + std::string* access_code) {
|
| + scoped_ptr<Value> parsed_value(base::JSONReader::Read(json, false));
|
| + if (!parsed_value.get() || !parsed_value->IsType(Value::TYPE_DICTIONARY))
|
| + return false;
|
| +
|
| + DictionaryValue* result = static_cast<DictionaryValue*>(parsed_value.get());
|
| + if (!result->GetString("user", username) ||
|
| + !result->GetString("pass", password) ||
|
| + !result->GetString("captcha", captcha) ||
|
| + !result->GetString("access_code", access_code)) {
|
| + return false;
|
| + }
|
| + return true;
|
| +}
|
| +
|
| +bool GetConfiguration(const std::string& json, SyncConfiguration* config) {
|
| + scoped_ptr<Value> parsed_value(base::JSONReader::Read(json, false));
|
| + if (!parsed_value.get() || !parsed_value->IsType(Value::TYPE_DICTIONARY))
|
| + return false;
|
| +
|
| + DictionaryValue* result = static_cast<DictionaryValue*>(parsed_value.get());
|
| + if (!result->GetBoolean("syncAllDataTypes", &config->sync_everything))
|
| + return false;
|
| +
|
| + // These values need to be kept in sync with where they are written in
|
| + // choose_datatypes.html.
|
| + bool sync_bookmarks;
|
| + if (!result->GetBoolean("syncBookmarks", &sync_bookmarks))
|
| + return false;
|
| + if (sync_bookmarks)
|
| + config->data_types.Put(syncable::BOOKMARKS);
|
| +
|
| + bool sync_preferences;
|
| + if (!result->GetBoolean("syncPreferences", &sync_preferences))
|
| + return false;
|
| + if (sync_preferences)
|
| + config->data_types.Put(syncable::PREFERENCES);
|
| +
|
| + bool sync_themes;
|
| + if (!result->GetBoolean("syncThemes", &sync_themes))
|
| + return false;
|
| + if (sync_themes)
|
| + config->data_types.Put(syncable::THEMES);
|
| +
|
| + bool sync_passwords;
|
| + if (!result->GetBoolean("syncPasswords", &sync_passwords))
|
| + return false;
|
| + if (sync_passwords)
|
| + config->data_types.Put(syncable::PASSWORDS);
|
| +
|
| + bool sync_autofill;
|
| + if (!result->GetBoolean("syncAutofill", &sync_autofill))
|
| + return false;
|
| + if (sync_autofill)
|
| + config->data_types.Put(syncable::AUTOFILL);
|
| +
|
| + bool sync_extensions;
|
| + if (!result->GetBoolean("syncExtensions", &sync_extensions))
|
| + return false;
|
| + if (sync_extensions) {
|
| + config->data_types.Put(syncable::EXTENSIONS);
|
| + config->data_types.Put(syncable::EXTENSION_SETTINGS);
|
| + }
|
| +
|
| + bool sync_typed_urls;
|
| + if (!result->GetBoolean("syncTypedUrls", &sync_typed_urls))
|
| + return false;
|
| + if (sync_typed_urls)
|
| + config->data_types.Put(syncable::TYPED_URLS);
|
| +
|
| + bool sync_sessions;
|
| + if (!result->GetBoolean("syncSessions", &sync_sessions))
|
| + return false;
|
| + if (sync_sessions)
|
| + config->data_types.Put(syncable::SESSIONS);
|
| +
|
| + bool sync_apps;
|
| + if (!result->GetBoolean("syncApps", &sync_apps))
|
| + return false;
|
| + if (sync_apps) {
|
| + config->data_types.Put(syncable::APPS);
|
| + config->data_types.Put(syncable::APP_SETTINGS);
|
| + }
|
| +
|
| + // Encryption settings.
|
| + if (!result->GetBoolean("encryptAllData", &config->encrypt_all))
|
| + return false;
|
| +
|
| + // Passphrase settings.
|
| + bool have_passphrase;
|
| + if (!result->GetBoolean("usePassphrase", &have_passphrase))
|
| + return false;
|
| +
|
| + if (have_passphrase) {
|
| + bool is_gaia;
|
| + if (!result->GetBoolean("isGooglePassphrase", &is_gaia))
|
| + return false;
|
| + std::string passphrase;
|
| + if (!result->GetString("passphrase", &passphrase))
|
| + return false;
|
| + // The user provided a passphrase - pass it off to SyncSetupFlow as either
|
| + // the secondary or GAIA passphrase as appropriate.
|
| + if (is_gaia) {
|
| + config->set_gaia_passphrase = true;
|
| + config->gaia_passphrase = passphrase;
|
| + } else {
|
| + config->set_secondary_passphrase = true;
|
| + config->secondary_passphrase = passphrase;
|
| + }
|
| + }
|
| + return true;
|
| +}
|
| +
|
| +bool HasConfigurationChanged(const SyncConfiguration& config,
|
| + Profile* profile) {
|
| + CHECK(profile);
|
| +
|
| + // This function must be updated every time a new sync datatype is added to
|
| + // the sync preferences page.
|
| + COMPILE_ASSERT(17 == syncable::MODEL_TYPE_COUNT,
|
| + UpdateCustomConfigHistogram);
|
| +
|
| + // If service is null or if this is a first time configuration, return true.
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + if (!service || !service->HasSyncSetupCompleted())
|
| + return true;
|
| +
|
| + if ((config.set_secondary_passphrase || config.set_gaia_passphrase) &&
|
| + !service->IsUsingSecondaryPassphrase())
|
| + return true;
|
| +
|
| + if (config.encrypt_all != service->EncryptEverythingEnabled())
|
| + return true;
|
| +
|
| + PrefService* pref_service = profile->GetPrefs();
|
| + CHECK(pref_service);
|
| +
|
| + if (config.sync_everything !=
|
| + pref_service->GetBoolean(prefs::kSyncKeepEverythingSynced))
|
| + return true;
|
| +
|
| + // Only check the data types that are explicitly listed on the sync
|
| + // preferences page.
|
| + const syncable::ModelTypeSet types = config.data_types;
|
| + if (((types.Has(syncable::BOOKMARKS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncBookmarks)) ||
|
| + ((types.Has(syncable::PREFERENCES)) !=
|
| + pref_service->GetBoolean(prefs::kSyncPreferences)) ||
|
| + ((types.Has(syncable::THEMES)) !=
|
| + pref_service->GetBoolean(prefs::kSyncThemes)) ||
|
| + ((types.Has(syncable::PASSWORDS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncPasswords)) ||
|
| + ((types.Has(syncable::AUTOFILL)) !=
|
| + pref_service->GetBoolean(prefs::kSyncAutofill)) ||
|
| + ((types.Has(syncable::EXTENSIONS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncExtensions)) ||
|
| + ((types.Has(syncable::TYPED_URLS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncTypedUrls)) ||
|
| + ((types.Has(syncable::SESSIONS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncSessions)) ||
|
| + ((types.Has(syncable::APPS)) !=
|
| + pref_service->GetBoolean(prefs::kSyncApps)))
|
| + return true;
|
| +
|
| + return false;
|
| +}
|
| +
|
| +bool GetPassphrase(const std::string& json, std::string* passphrase) {
|
| + scoped_ptr<Value> parsed_value(base::JSONReader::Read(json, false));
|
| + if (!parsed_value.get() || !parsed_value->IsType(Value::TYPE_DICTIONARY))
|
| + return false;
|
| +
|
| + DictionaryValue* result = static_cast<DictionaryValue*>(parsed_value.get());
|
| + return result->GetString("passphrase", passphrase);
|
| +}
|
| +
|
| +string16 NormalizeUserName(const string16& user) {
|
| + if (user.find_first_of(ASCIIToUTF16("@")) != string16::npos)
|
| + return user;
|
| + return user + ASCIIToUTF16("@") + ASCIIToUTF16(DEFAULT_SIGNIN_DOMAIN);
|
| +}
|
| +
|
| +bool AreUserNamesEqual(const string16& user1, const string16& user2) {
|
| + return NormalizeUserName(user1) == NormalizeUserName(user2);
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +SyncSetupHandler2::SyncSetupHandler2(ProfileManager* profile_manager)
|
| + : flow_(NULL),
|
| + profile_manager_(profile_manager) {
|
| +}
|
| +
|
| +SyncSetupHandler2::~SyncSetupHandler2() {
|
| + // This case is hit when the user performs a back navigation.
|
| + if (flow_)
|
| + flow_->OnDialogClosed("");
|
| +}
|
| +
|
| +void SyncSetupHandler2::GetLocalizedValues(DictionaryValue* localized_strings) {
|
| + GetStaticLocalizedValues(localized_strings, web_ui_);
|
| +}
|
| +
|
| +void SyncSetupHandler2::GetStaticLocalizedValues(
|
| + DictionaryValue* localized_strings,
|
| + WebUI* web_ui) {
|
| + DCHECK(localized_strings);
|
| +
|
| + localized_strings->SetString(
|
| + "invalidPasswordHelpURL",
|
| + google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kInvalidPasswordHelpURL));
|
| + localized_strings->SetString(
|
| + "cannotAccessAccountURL",
|
| + google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kCanNotAccessAccountURL));
|
| + localized_strings->SetString(
|
| + "introduction",
|
| + GetStringFUTF16(IDS_SYNC_LOGIN_INTRODUCTION,
|
| + GetStringUTF16(IDS_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "chooseDataTypesInstructions",
|
| + GetStringFUTF16(IDS_SYNC_CHOOSE_DATATYPES_INSTRUCTIONS,
|
| + GetStringUTF16(IDS_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "encryptionInstructions",
|
| + GetStringFUTF16(IDS_SYNC_ENCRYPTION_INSTRUCTIONS,
|
| + GetStringUTF16(IDS_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "encryptionHelpURL",
|
| + google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kSyncEncryptionHelpURL));
|
| + localized_strings->SetString(
|
| + "passphraseEncryptionMessage",
|
| + GetStringFUTF16(IDS_SYNC_PASSPHRASE_ENCRYPTION_MESSAGE,
|
| + GetStringUTF16(IDS_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "passphraseRecover",
|
| + GetStringFUTF16(IDS_SYNC_PASSPHRASE_RECOVER,
|
| + ASCIIToUTF16(google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kSyncGoogleDashboardURL))));
|
| + localized_strings->SetString(
|
| + "promoTitle",
|
| + GetStringFUTF16(IDS_SYNC_PROMO_TITLE,
|
| + GetStringUTF16(IDS_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "promoMessageTitle",
|
| + GetStringFUTF16(IDS_SYNC_PROMO_MESSAGE_TITLE,
|
| + GetStringUTF16(IDS_SHORT_PRODUCT_NAME)));
|
| + localized_strings->SetString(
|
| + "syncEverythingHelpURL",
|
| + google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kSyncEverythingLearnMoreURL));
|
| +
|
| + // The experimental body string only appears if we are on the launch page
|
| + // version of the Sync Promo.
|
| + int message_body_resource_id = IDS_SYNC_PROMO_MESSAGE_BODY_A;
|
| + if (web_ui && SyncPromoUI::GetIsLaunchPageForSyncPromoURL(
|
| + web_ui->tab_contents()->GetURL())) {
|
| + message_body_resource_id = sync_promo_trial::GetMessageBodyResID();
|
| + }
|
| + localized_strings->SetString(
|
| + "promoMessageBody",
|
| + GetStringUTF16(message_body_resource_id));
|
| +
|
| + std::string create_account_url = google_util::StringAppendGoogleLocaleParam(
|
| + chrome::kSyncCreateNewAccountURL);
|
| + string16 create_account = GetStringUTF16(IDS_SYNC_CREATE_ACCOUNT);
|
| + create_account= UTF8ToUTF16("<a id='create-account-link' target='_blank' "
|
| + "class='account-link' href='" + create_account_url + "'>") +
|
| + create_account + UTF8ToUTF16("</a>");
|
| + localized_strings->SetString("createAccountLinkHTML",
|
| + GetStringFUTF16(IDS_SYNC_CREATE_ACCOUNT_PREFIX, create_account));
|
| +
|
| + static OptionsStringResource resources[] = {
|
| + { "syncSetupOverlayTitle", IDS_SYNC_SETUP_TITLE },
|
| + { "syncSetupConfigureTitle", IDS_SYNC_SETUP_CONFIGURE_TITLE },
|
| + { "cannotBeBlank", IDS_SYNC_CANNOT_BE_BLANK },
|
| + { "emailLabel", IDS_SYNC_LOGIN_EMAIL_NEW_LINE },
|
| + { "passwordLabel", IDS_SYNC_LOGIN_PASSWORD_NEW_LINE },
|
| + { "invalidCredentials", IDS_SYNC_INVALID_USER_CREDENTIALS },
|
| + { "signin", IDS_SYNC_SIGNIN },
|
| + { "couldNotConnect", IDS_SYNC_LOGIN_COULD_NOT_CONNECT },
|
| + { "unrecoverableError", IDS_SYNC_UNRECOVERABLE_ERROR },
|
| + { "errorLearnMore", IDS_LEARN_MORE },
|
| + { "unrecoverableErrorHelpURL", IDS_SYNC_UNRECOVERABLE_ERROR_HELP_URL },
|
| + { "cannotAccessAccount", IDS_SYNC_CANNOT_ACCESS_ACCOUNT },
|
| + { "cancel", IDS_CANCEL },
|
| + { "settingUp", IDS_SYNC_LOGIN_SETTING_UP },
|
| + { "errorSigningIn", IDS_SYNC_ERROR_SIGNING_IN },
|
| + { "signinHeader", IDS_SYNC_PROMO_SIGNIN_HEADER},
|
| + { "captchaInstructions", IDS_SYNC_GAIA_CAPTCHA_INSTRUCTIONS },
|
| + { "invalidAccessCode", IDS_SYNC_INVALID_ACCESS_CODE_LABEL },
|
| + { "enterAccessCode", IDS_SYNC_ENTER_ACCESS_CODE_LABEL },
|
| + { "getAccessCodeHelp", IDS_SYNC_ACCESS_CODE_HELP_LABEL },
|
| + { "getAccessCodeURL", IDS_SYNC_GET_ACCESS_CODE_URL },
|
| + { "syncAllDataTypes", IDS_SYNC_EVERYTHING },
|
| + { "chooseDataTypes", IDS_SYNC_CHOOSE_DATATYPES },
|
| + { "bookmarks", IDS_SYNC_DATATYPE_BOOKMARKS },
|
| + { "preferences", IDS_SYNC_DATATYPE_PREFERENCES },
|
| + { "autofill", IDS_SYNC_DATATYPE_AUTOFILL },
|
| + { "themes", IDS_SYNC_DATATYPE_THEMES },
|
| + { "passwords", IDS_SYNC_DATATYPE_PASSWORDS },
|
| + { "extensions", IDS_SYNC_DATATYPE_EXTENSIONS },
|
| + { "typedURLs", IDS_SYNC_DATATYPE_TYPED_URLS },
|
| + { "apps", IDS_SYNC_DATATYPE_APPS },
|
| + { "openTabs", IDS_SYNC_DATATYPE_TABS },
|
| + { "syncZeroDataTypesError", IDS_SYNC_ZERO_DATA_TYPES_ERROR },
|
| + { "serviceUnavailableError", IDS_SYNC_SETUP_ABORTED_BY_PENDING_CLEAR },
|
| + { "encryptAllLabel", IDS_SYNC_ENCRYPT_ALL_LABEL },
|
| + { "googleOption", IDS_SYNC_PASSPHRASE_OPT_GOOGLE },
|
| + { "explicitOption", IDS_SYNC_PASSPHRASE_OPT_EXPLICIT },
|
| + { "sectionGoogleMessage", IDS_SYNC_PASSPHRASE_MSG_GOOGLE },
|
| + { "sectionExplicitMessage", IDS_SYNC_PASSPHRASE_MSG_EXPLICIT },
|
| + { "passphraseLabel", IDS_SYNC_PASSPHRASE_LABEL },
|
| + { "confirmLabel", IDS_SYNC_CONFIRM_PASSPHRASE_LABEL },
|
| + { "emptyErrorMessage", IDS_SYNC_EMPTY_PASSPHRASE_ERROR },
|
| + { "mismatchErrorMessage", IDS_SYNC_PASSPHRASE_MISMATCH_ERROR },
|
| + { "passphraseWarning", IDS_SYNC_PASSPHRASE_WARNING },
|
| + { "customizeLinkLabel", IDS_SYNC_CUSTOMIZE_LINK_LABEL },
|
| + { "confirmSyncPreferences", IDS_SYNC_CONFIRM_SYNC_PREFERENCES },
|
| + { "syncEverything", IDS_SYNC_SYNC_EVERYTHING },
|
| + { "useDefaultSettings", IDS_SYNC_USE_DEFAULT_SETTINGS },
|
| + { "passphraseSectionTitle", IDS_SYNC_PASSPHRASE_SECTION_TITLE },
|
| + { "privacyDashboardLink", IDS_SYNC_PRIVACY_DASHBOARD_LINK_LABEL },
|
| + { "enterPassphraseTitle", IDS_SYNC_ENTER_PASSPHRASE_TITLE },
|
| + { "enterPassphraseBody", IDS_SYNC_ENTER_PASSPHRASE_BODY },
|
| + { "enterOtherPassphraseBody", IDS_SYNC_ENTER_OTHER_PASSPHRASE_BODY },
|
| + { "enterGooglePassphraseBody", IDS_SYNC_ENTER_GOOGLE_PASSPHRASE_BODY },
|
| + { "passphraseLabel", IDS_SYNC_PASSPHRASE_LABEL },
|
| + { "incorrectPassphrase", IDS_SYNC_INCORRECT_PASSPHRASE },
|
| + { "passphraseWarning", IDS_SYNC_PASSPHRASE_WARNING },
|
| + { "cancelWarningHeader", IDS_SYNC_PASSPHRASE_CANCEL_WARNING_HEADER },
|
| + { "cancelWarning", IDS_SYNC_PASSPHRASE_CANCEL_WARNING },
|
| + { "yes", IDS_SYNC_PASSPHRASE_CANCEL_YES },
|
| + { "no", IDS_SYNC_PASSPHRASE_CANCEL_NO },
|
| + { "sectionExplicitMessagePrefix", IDS_SYNC_PASSPHRASE_MSG_EXPLICIT_PREFIX },
|
| + { "sectionExplicitMessagePostfix",
|
| + IDS_SYNC_PASSPHRASE_MSG_EXPLICIT_POSTFIX },
|
| + { "encryptedDataTypesTitle", IDS_SYNC_ENCRYPTION_DATA_TYPES_TITLE },
|
| + { "encryptSensitiveOption", IDS_SYNC_ENCRYPT_SENSITIVE_DATA },
|
| + { "encryptAllOption", IDS_SYNC_ENCRYPT_ALL_DATA },
|
| + { "encryptAllOption", IDS_SYNC_ENCRYPT_ALL_DATA },
|
| + { "aspWarningText", IDS_SYNC_ASP_PASSWORD_WARNING_TEXT },
|
| + { "promoPageTitle", IDS_SYNC_PROMO_TAB_TITLE},
|
| + { "promoSkipButton", IDS_SYNC_PROMO_SKIP_BUTTON},
|
| + { "promoAdvanced", IDS_SYNC_PROMO_ADVANCED},
|
| + { "promoLearnMoreShow", IDS_SYNC_PROMO_LEARN_MORE_SHOW},
|
| + { "promoLearnMoreHide", IDS_SYNC_PROMO_LEARN_MORE_HIDE},
|
| + { "promoInformation", IDS_SYNC_PROMO_INFORMATION},
|
| + };
|
| +
|
| + RegisterStrings(localized_strings, resources, arraysize(resources));
|
| +}
|
| +
|
| +void SyncSetupHandler2::Initialize() {
|
| +}
|
| +
|
| +void SyncSetupHandler2::OnGetOAuthTokenSuccess(const std::string& oauth_token) {
|
| + flow_->OnUserSubmittedOAuth(oauth_token);
|
| +}
|
| +
|
| +void SyncSetupHandler2::OnGetOAuthTokenFailure(
|
| + const GoogleServiceAuthError& error) {
|
| + CloseSyncSetup();
|
| +}
|
| +
|
| +void SyncSetupHandler2::RegisterMessages() {
|
| + web_ui_->RegisterMessageCallback("SyncSetupDidClosePage",
|
| + base::Bind(&SyncSetupHandler2::OnDidClosePage,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupSubmitAuth",
|
| + base::Bind(&SyncSetupHandler2::HandleSubmitAuth,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupConfigure",
|
| + base::Bind(&SyncSetupHandler2::HandleConfigure,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupPassphrase",
|
| + base::Bind(&SyncSetupHandler2::HandlePassphraseEntry,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupPassphraseCancel",
|
| + base::Bind(&SyncSetupHandler2::HandlePassphraseCancel,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupAttachHandler",
|
| + base::Bind(&SyncSetupHandler2::HandleAttachHandler,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupShowErrorUI",
|
| + base::Bind(&SyncSetupHandler2::HandleShowErrorUI,
|
| + base::Unretained(this)));
|
| + web_ui_->RegisterMessageCallback("SyncSetupShowSetupUI",
|
| + base::Bind(&SyncSetupHandler2::HandleShowSetupUI,
|
| + base::Unretained(this)));
|
| +}
|
| +
|
| +// Ideal(?) solution here would be to mimic the ClientLogin overlay. Since
|
| +// this UI must render an external URL, that overlay cannot be used directly.
|
| +// The current implementation is functional, but fails asthetically.
|
| +// TODO(rickcam): Bug 90711: Update UI for OAuth sign-in flow
|
| +void SyncSetupHandler2::ShowOAuthLogin() {
|
| + DCHECK(browser_sync::IsUsingOAuth());
|
| +
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + oauth_login_.reset(new GaiaOAuthFetcher(this,
|
| + profile->GetRequestContext(),
|
| + profile,
|
| + GaiaConstants::kSyncServiceOAuth));
|
| + oauth_login_->SetAutoFetchLimit(GaiaOAuthFetcher::OAUTH1_REQUEST_TOKEN);
|
| + oauth_login_->StartGetOAuthToken();
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowGaiaLogin(const DictionaryValue& args) {
|
| + DCHECK(!browser_sync::IsUsingOAuth());
|
| + StringValue page("login");
|
| + web_ui_->CallJavascriptFunction(
|
| + "SyncSetupOverlay.showSyncSetupPage", page, args);
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowGaiaSuccessAndClose() {
|
| + web_ui_->CallJavascriptFunction("SyncSetupOverlay.showSuccessAndClose");
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowGaiaSuccessAndSettingUp() {
|
| + web_ui_->CallJavascriptFunction("SyncSetupOverlay.showSuccessAndSettingUp");
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowConfigure(const DictionaryValue& args) {
|
| + StringValue page("configure");
|
| + web_ui_->CallJavascriptFunction(
|
| + "SyncSetupOverlay.showSyncSetupPage", page, args);
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowPassphraseEntry(const DictionaryValue& args) {
|
| + StringValue page("passphrase");
|
| + web_ui_->CallJavascriptFunction(
|
| + "SyncSetupOverlay.showSyncSetupPage", page, args);
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowSettingUp() {
|
| + StringValue page("settingUp");
|
| + web_ui_->CallJavascriptFunction(
|
| + "SyncSetupOverlay.showSyncSetupPage", page);
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowSetupDone(const string16& user) {
|
| + StringValue page("done");
|
| + web_ui_->CallJavascriptFunction(
|
| + "SyncSetupOverlay.showSyncSetupPage", page);
|
| +
|
| + // Suppress the sync promo once the user signs into sync. This way the user
|
| + // doesn't see the sync promo even if they sign out of sync later on.
|
| + SyncPromoUI::SetUserSkippedSyncPromo(Profile::FromWebUI(web_ui_));
|
| +
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + if (!service->HasSyncSetupCompleted()) {
|
| + FilePath profile_file_path = profile->GetPath();
|
| + ProfileMetrics::LogProfileSyncSignIn(profile_file_path);
|
| + }
|
| +}
|
| +
|
| +void SyncSetupHandler2::SetFlow(SyncSetupFlow* flow) {
|
| + flow_ = flow;
|
| +}
|
| +
|
| +void SyncSetupHandler2::Focus() {
|
| + static_cast<RenderViewHostDelegate*>(web_ui_->tab_contents())->Activate();
|
| +}
|
| +
|
| +void SyncSetupHandler2::OnDidClosePage(const ListValue* args) {
|
| + CloseSyncSetup();
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandleSubmitAuth(const ListValue* args) {
|
| + std::string json;
|
| + if (!args->GetString(0, &json)) {
|
| + NOTREACHED() << "Could not read JSON argument";
|
| + return;
|
| + }
|
| +
|
| + if (json.empty())
|
| + return;
|
| +
|
| + std::string username, password, captcha, access_code;
|
| + if (!GetAuthData(json, &username, &password, &captcha, &access_code)) {
|
| + // The page sent us something that we didn't understand.
|
| + // This probably indicates a programming error.
|
| + NOTREACHED();
|
| + return;
|
| + }
|
| +
|
| + string16 error_message;
|
| + if (!IsLoginAuthDataValid(username, &error_message)) {
|
| + ShowLoginErrorMessage(error_message);
|
| + return;
|
| + }
|
| +
|
| + if (flow_)
|
| + flow_->OnUserSubmittedAuth(username, password, captcha, access_code);
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandleConfigure(const ListValue* args) {
|
| + std::string json;
|
| + if (!args->GetString(0, &json)) {
|
| + NOTREACHED() << "Could not read JSON argument";
|
| + return;
|
| + }
|
| + if (json.empty()) {
|
| + NOTREACHED();
|
| + return;
|
| + }
|
| +
|
| + SyncConfiguration configuration;
|
| + if (!GetConfiguration(json, &configuration)) {
|
| + // The page sent us something that we didn't understand.
|
| + // This probably indicates a programming error.
|
| + NOTREACHED();
|
| + return;
|
| + }
|
| +
|
| + // We do not do UMA logging during unit tests.
|
| + if (web_ui_) {
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + if (HasConfigurationChanged(configuration, profile)) {
|
| + UMA_HISTOGRAM_BOOLEAN("Sync.SyncEverything",
|
| + configuration.sync_everything);
|
| + if (!configuration.sync_everything) {
|
| + // Only log the data types that are explicitly listed on the sync
|
| + // preferences page.
|
| + const syncable::ModelTypeSet types = configuration.data_types;
|
| + if (types.Has(syncable::BOOKMARKS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", BOOKMARKS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::PREFERENCES))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", PREFERENCES, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::PASSWORDS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", PASSWORDS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::AUTOFILL))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", AUTOFILL, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::THEMES))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", THEMES, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::TYPED_URLS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", TYPED_URLS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::EXTENSIONS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", EXTENSIONS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::SESSIONS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", SESSIONS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + if (types.Has(syncable::APPS))
|
| + UMA_HISTOGRAM_ENUMERATION(
|
| + "Sync.CustomSync", APPS, SELECTABLE_DATATYPE_COUNT + 1);
|
| + COMPILE_ASSERT(17 == syncable::MODEL_TYPE_COUNT,
|
| + UpdateCustomConfigHistogram);
|
| + COMPILE_ASSERT(9 == SELECTABLE_DATATYPE_COUNT,
|
| + UpdateCustomConfigHistogram);
|
| + }
|
| + UMA_HISTOGRAM_BOOLEAN("Sync.EncryptAllData", configuration.encrypt_all);
|
| + UMA_HISTOGRAM_BOOLEAN("Sync.CustomPassphrase",
|
| + configuration.set_gaia_passphrase ||
|
| + configuration.set_secondary_passphrase);
|
| + }
|
| + }
|
| +
|
| + DCHECK(flow_);
|
| + flow_->OnUserConfigured(configuration);
|
| +
|
| + ProfileMetrics::LogProfileSyncInfo(ProfileMetrics::SYNC_CUSTOMIZE);
|
| + if (configuration.encrypt_all) {
|
| + ProfileMetrics::LogProfileSyncInfo(ProfileMetrics::SYNC_ENCRYPT);
|
| + }
|
| + if (configuration.set_secondary_passphrase) {
|
| + ProfileMetrics::LogProfileSyncInfo(ProfileMetrics::SYNC_PASSPHRASE);
|
| + }
|
| + if (!configuration.sync_everything) {
|
| + ProfileMetrics::LogProfileSyncInfo(ProfileMetrics::SYNC_CHOOSE);
|
| + }
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandlePassphraseEntry(const ListValue* args) {
|
| + std::string json;
|
| + if (!args->GetString(0, &json)) {
|
| + NOTREACHED() << "Could not read JSON argument";
|
| + return;
|
| + }
|
| +
|
| + if (json.empty())
|
| + return;
|
| +
|
| + std::string passphrase;
|
| + if (!GetPassphrase(json, &passphrase)) {
|
| + // Couldn't understand what the page sent. Indicates a programming error.
|
| + NOTREACHED();
|
| + return;
|
| + }
|
| +
|
| + DCHECK(flow_);
|
| + flow_->OnPassphraseEntry(passphrase);
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandlePassphraseCancel(const ListValue* args) {
|
| + DCHECK(flow_);
|
| + flow_->OnPassphraseCancel();
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandleAttachHandler(const ListValue* args) {
|
| + OpenSyncSetup();
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandleShowErrorUI(const ListValue* args) {
|
| + DCHECK(!flow_);
|
| +
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + DCHECK(service);
|
| +
|
| + service->ShowErrorUI();
|
| +}
|
| +
|
| +void SyncSetupHandler2::HandleShowSetupUI(const ListValue* args) {
|
| + DCHECK(!flow_);
|
| + if (FocusExistingWizard()) {
|
| + CloseOverlay();
|
| + return;
|
| + }
|
| + ShowSetupUI();
|
| +}
|
| +
|
| +void SyncSetupHandler2::CloseSyncSetup() {
|
| + if (flow_) {
|
| + flow_->OnDialogClosed(std::string());
|
| + flow_ = NULL;
|
| + }
|
| +}
|
| +
|
| +void SyncSetupHandler2::OpenSyncSetup() {
|
| + DCHECK(web_ui_);
|
| + DCHECK(!flow_);
|
| +
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + if (!service) {
|
| + // If there's no sync service, the user tried to manually invoke a syncSetup
|
| + // URL, but sync features are disabled. We need to close the overlay for
|
| + // this (rare) case.
|
| + CloseOverlay();
|
| + return;
|
| + }
|
| +
|
| + // If the wizard is already visible, it must be attached to another flow
|
| + // handler.
|
| + if (FocusExistingWizard()) {
|
| + CloseOverlay();
|
| + return;
|
| + }
|
| +
|
| + // Attach this as the sync setup handler, before calling ShowSetupUI().
|
| + if (!service->get_wizard().AttachSyncSetupHandler(this)) {
|
| + LOG(ERROR) << "SyncSetupHandler attach failed!";
|
| + CloseOverlay();
|
| + return;
|
| + }
|
| +
|
| + ShowSetupUI();
|
| +}
|
| +
|
| +// Private member functions.
|
| +
|
| +bool SyncSetupHandler2::FocusExistingWizard() {
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + if (!service)
|
| + return false;
|
| +
|
| + // If the wizard is already visible, focus it.
|
| + if (service->get_wizard().IsVisible()) {
|
| + service->get_wizard().Focus();
|
| + return true;
|
| + }
|
| + return false;
|
| +}
|
| +
|
| +void SyncSetupHandler2::CloseOverlay() {
|
| + web_ui_->CallJavascriptFunction("OptionsPage.closeOverlay");
|
| +}
|
| +
|
| +bool SyncSetupHandler2::IsLoginAuthDataValid(const std::string& username,
|
| + string16* error_message) {
|
| + // Happens during unit tests.
|
| + if (!web_ui_ || !profile_manager_)
|
| + return true;
|
| +
|
| + if (username.empty())
|
| + return true;
|
| +
|
| + // Check if the username is already in use by another profile.
|
| + const ProfileInfoCache& cache = profile_manager_->GetProfileInfoCache();
|
| + size_t current_profile_index = cache.GetIndexOfProfileWithPath(
|
| + Profile::FromWebUI(web_ui_)->GetPath());
|
| + string16 username_utf16 = UTF8ToUTF16(username);
|
| +
|
| + for (size_t i = 0; i < cache.GetNumberOfProfiles(); ++i) {
|
| + if (i != current_profile_index && AreUserNamesEqual(
|
| + cache.GetUserNameOfProfileAtIndex(i), username_utf16)) {
|
| + *error_message = l10n_util::GetStringUTF16(
|
| + IDS_SYNC_USER_NAME_IN_USE_ERROR);
|
| + return false;
|
| + }
|
| + }
|
| +
|
| + return true;
|
| +}
|
| +
|
| +void SyncSetupHandler2::ShowLoginErrorMessage(const string16& error_message) {
|
| + DictionaryValue args;
|
| + Profile* profile = Profile::FromWebUI(web_ui_);
|
| + ProfileSyncService* service = profile->GetProfileSyncService();
|
| + SyncSetupFlow::GetArgsForGaiaLogin(service, &args);
|
| + args.SetString("error_message", error_message);
|
| + ShowGaiaLogin(args);
|
| +}
|
|
|