Chromium Code Reviews| Index: chrome/browser/extensions/default_apps_provider.cc |
| =================================================================== |
| --- chrome/browser/extensions/default_apps_provider.cc (revision 0) |
| +++ chrome/browser/extensions/default_apps_provider.cc (revision 0) |
| @@ -0,0 +1,186 @@ |
| +// 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/extensions/default_apps_provider.h" |
| + |
| +#include "base/command_line.h" |
| +#include "base/metrics/field_trial.h" |
| +#include "base/values.h" |
| +#include "chrome/browser/browser_process.h" |
| +#include "chrome/browser/extensions/crx_installer.h" |
| +#include "chrome/browser/extensions/default_apps_provider.h" |
| +#include "chrome/browser/extensions/default_apps_trial.h" |
| +#include "chrome/browser/extensions/extension_service.h" |
| +#include "chrome/browser/profiles/profile.h" |
| +#include "chrome/common/chrome_notification_types.h" |
| +#include "chrome/common/chrome_switches.h" |
| +#include "chrome/common/pref_names.h" |
| +#include "content/public/browser/notification_details.h" |
| +#include "content/public/browser/notification_source.h" |
| +#include "ui/base/l10n/l10n_util.h" |
| + |
| +DefaultAppsProvider::DefaultAppsProvider(VisitorInterface* service, |
| + ExternalExtensionLoader* loader, |
| + Profile* profile) |
| + : ExternalExtensionProviderImpl(service, loader, |
| + Extension::EXTERNAL_PREF, Extension::INVALID), |
| + profile_(profile) { |
| + DCHECK(profile_); |
| +} |
| + |
| +DefaultAppsProvider::~DefaultAppsProvider() { |
| +} |
| + |
| +void DefaultAppsProvider::RegisterUserPrefs(PrefService* prefs) { |
| + prefs->RegisterIntegerPref(prefs::kDefaultAppsInstallState, 0, |
| + PrefService::UNSYNCABLE_PREF); |
| +} |
| + |
| +bool DefaultAppsProvider::ShouldRegister(Profile* profile) { |
| + // We decide to install or not install default apps based on the following |
| + // criteria, from highest priority to lowest priority: |
| + // |
| + // - if this instance of chrome is participating in the default apps |
| + // field trial, then install apps based on the group |
| + // - the command line option. Tests use this option to disable installation |
| + // of default apps in some cases |
| + // - the preferences value in the profile. This value is usually set in |
| + // the master_preferences file |
| + bool install_apps = |
| + profile->GetPrefs()->GetString(prefs::kDefaultApps) == "install"; |
| + |
| + if (CommandLine::ForCurrentProcess()->HasSwitch( |
| + switches::kDisableDefaultApps)) { |
| + install_apps = false; |
| + } |
| + |
| + if (base::FieldTrialList::TrialExists(kDefaultAppsTrial_Name)) { |
| + install_apps = base::FieldTrialList::Find( |
| + kDefaultAppsTrial_Name)->group_name() != kDefaultAppsTrial_NoAppsGroup; |
| + } |
| + |
| + if (install_apps) { |
| + // Don't bother installing default apps in locales where its known that |
| + // they don't work. |
| + // TODO(rogerta): Do this check dynamically once the webstore can expose |
| + // an API. |
| + const std::string& locale = g_browser_process->GetApplicationLocale(); |
| + static const char* unsupported_locales[] = {"CN", "TR", "IR"}; |
| + for (size_t i = 0; i < arraysize(unsupported_locales); ++i) { |
| + if (EndsWith(locale, unsupported_locales[i], false)) { |
| + install_apps = false; |
| + break; |
| + } |
| + } |
| + } |
| + |
| + return install_apps; |
| +} |
| + |
| +void DefaultAppsProvider::SetPrefs(base::DictionaryValue* prefs) { |
| + CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| + |
| + if (prefs->size() > 0) { |
| + registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_INSTALLED, |
| + content::Source<Profile>(profile_)); |
| + registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_INSTALL_ERROR, |
| + NotificationService::AllSources()); |
| + } |
| + |
| + ExternalExtensionProviderImpl::SetPrefs(prefs); |
| + |
| + // If the number of invalid extension is the same as the total, then we are |
| + // done. |
| + if (prefs->size() == invalid_extensions().size()) { |
| + profile_->GetPrefs()->SetInteger(prefs::kDefaultAppsInstallState, |
| + kInstallDone); |
| + profile_->GetPrefs()->ScheduleSavePersistentPrefs(); |
| + } |
| +} |
| + |
| +void DefaultAppsProvider::ServiceShutdown() { |
| + profile_ = NULL; |
| + ExternalExtensionProviderImpl::ServiceShutdown(); |
| +} |
| + |
| +void DefaultAppsProvider::VisitRegisteredExtension() const { |
| + if (profile_) { |
| + int state = profile_->GetPrefs()->GetInteger( |
| + prefs::kDefaultAppsInstallState); |
| + switch (state) { |
| + case kInstallNotStarted: { |
| + // We never even tried to install the default apps, so try now. |
| + // Don't install default apps if the profile already has apps installed. |
| + ExtensionService* extension_service = profile_->GetExtensionService(); |
| + if (extension_service && extension_service->HasApps()) { |
| + // Keep track of the fact that we are done. |
| + profile_->GetPrefs()->SetInteger(prefs::kDefaultAppsInstallState, |
| + kInstallDone); |
| + profile_->GetPrefs()->ScheduleSavePersistentPrefs(); |
| + service()->OnExternalProviderReady(); |
| + return; |
| + } |
| + |
| + // Remember that we are now installing the default apps. |
| + profile_->GetPrefs()->SetInteger(prefs::kDefaultAppsInstallState, |
| + kInstalling); |
| + profile_->GetPrefs()->ScheduleSavePersistentPrefs(); |
| + break; |
| + } |
| + case kInstalling: |
| + // We were in the process of installing default apps into the profile |
| + // when the profile was last closed. This may happen because the the |
|
Finnur
2011/10/20 10:10:43
nit: The The? Do you mean The Who? ;)
Roger Tawa OOO till Jul 10th
2011/10/20 14:31:15
no, I really meant the the: http://en.wikipedia.or
|
| + // process takes time and is asynchronous. In this case we want to |
| + // continue the install process. |
| + break; |
| + case kInstallDone: |
| + // Defaults are installed, so don't need to try again. |
| + service()->OnExternalProviderReady(); |
| + return; |
| + } |
| + } |
| + |
| + ExternalExtensionProviderImpl::VisitRegisteredExtension(); |
|
Finnur
2011/10/20 10:10:43
nit: I'm wondering if it makes sense to comment th
Roger Tawa OOO till Jul 10th
2011/10/20 14:31:15
I think the comment in the interface is pretty goo
Finnur
2011/10/20 14:39:14
No, that's ok. Never mind...
On 2011/10/20 14:31:
|
| +} |
| + |
| +void DefaultAppsProvider::Observe(int type, |
| + const content::NotificationSource& source, |
| + const content::NotificationDetails& details) { |
| + if (!profile_) |
| + return; |
| + |
| + // If the notification is for a failed extension, remember its id so that |
| + // we don't wait forever for it to get installed. |
| + if (type == chrome::NOTIFICATION_EXTENSION_INSTALL_ERROR) { |
| + CrxInstaller* crx_installer = content::Source<CrxInstaller>(source).ptr(); |
| + install_error_extensions_.insert(crx_installer->expected_id()); |
| + } |
| + |
| + ExtensionService* extension_service = profile_->GetExtensionService(); |
| + if (!extension_service) |
| + return; |
| + |
| + for (DictionaryValue::key_iterator i = prefs()->begin_keys(); |
| + i != prefs()->end_keys(); ++i) { |
| + const std::string& extension_id = *i; |
| + |
| + // If the extension id is for one that is invalid or failed to install, |
| + // then consider it "done". |
| + if (invalid_extensions().count(extension_id) || |
| + install_error_extensions_.count(extension_id)) { |
| + continue; |
| + } |
| + |
| + if (!extension_service->GetExtensionById(extension_id, true)) { |
| + // We found an extension that is not yet known to the service. We |
| + // need to keep going. |
| + return; |
| + } |
| + } |
| + |
| + profile_->GetPrefs()->SetInteger(prefs::kDefaultAppsInstallState, |
| + kInstallDone); |
| + profile_->GetPrefs()->ScheduleSavePersistentPrefs(); |
| + registrar_.RemoveAll(); |
| +} |
| Property changes on: chrome\browser\extensions\default_apps_provider.cc |
| ___________________________________________________________________ |
| Added: svn:eol-style |
| + LF |