blob: b27d983974d2e2741b76b91006c3797d045b0e9e [file] [log] [blame]
// Copyright 2013 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/api/settings_overrides/settings_overrides_api.h"
#include "base/lazy_instance.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/chrome_notification_types.h"
#include "chrome/browser/extensions/api/preference/preference_api.h"
#include "chrome/browser/extensions/extension_prefs.h"
#include "chrome/browser/prefs/session_startup_pref.h"
#include "chrome/browser/search_engines/template_url.h"
#include "chrome/browser/search_engines/template_url_service_factory.h"
#include "chrome/common/extensions/manifest_handlers/settings_overrides_handler.h"
#include "chrome/common/pref_names.h"
#include "content/public/browser/notification_details.h"
#include "content/public/browser/notification_source.h"
#include "extensions/common/error_utils.h"
#include "extensions/common/manifest_constants.h"
namespace extensions {
namespace {
base::LazyInstance<ProfileKeyedAPIFactory<SettingsOverridesAPI> >
g_factory = LAZY_INSTANCE_INITIALIZER;
const char kManyStartupPagesWarning[] = "* specifies more than 1 startup URL. "
"All but the first will be ignored.";
using api::manifest_types::ChromeSettingsOverrides;
TemplateURLData ConvertSearchProvider(
const ChromeSettingsOverrides::Search_provider& search_provider) {
TemplateURLData data;
data.short_name = UTF8ToUTF16(search_provider.name);
data.SetKeyword(UTF8ToUTF16(search_provider.keyword));
data.SetURL(search_provider.search_url);
if (search_provider.suggest_url)
data.suggestions_url = *search_provider.suggest_url;
if (search_provider.instant_url)
data.instant_url = *search_provider.instant_url;
if (search_provider.image_url)
data.image_url = *search_provider.image_url;
if (search_provider.search_url_post_params)
data.search_url_post_params = *search_provider.search_url_post_params;
if (search_provider.suggest_url_post_params)
data.suggestions_url_post_params = *search_provider.suggest_url_post_params;
if (search_provider.instant_url_post_params)
data.instant_url_post_params = *search_provider.instant_url_post_params;
if (search_provider.image_url_post_params)
data.image_url_post_params = *search_provider.image_url_post_params;
data.favicon_url = GURL(search_provider.favicon_url);
data.show_in_default_list = true;
data.safe_for_autoreplace = false;
data.input_encodings.push_back(search_provider.encoding);
data.date_created = base::Time();
data.last_modified = base::Time();
data.prepopulate_id = 0;
if (search_provider.alternate_urls) {
for (size_t i = 0; i < search_provider.alternate_urls->size(); ++i) {
if (!search_provider.alternate_urls->at(i).empty())
data.alternate_urls.push_back(search_provider.alternate_urls->at(i));
}
}
return data;
}
} // namespace
SettingsOverridesAPI::SettingsOverridesAPI(Profile* profile)
: profile_(profile),
url_service_(TemplateURLServiceFactory::GetForProfile(profile)) {
DCHECK(profile);
registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_LOADED,
content::Source<Profile>(profile));
registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_UNLOADED,
content::Source<Profile>(profile));
}
SettingsOverridesAPI::~SettingsOverridesAPI() {
}
ProfileKeyedAPIFactory<SettingsOverridesAPI>*
SettingsOverridesAPI::GetFactoryInstance() {
return &g_factory.Get();
}
void SettingsOverridesAPI::SetPref(const std::string& extension_id,
const std::string& pref_key,
base::Value* value) {
PreferenceAPI::Get(profile_)->SetExtensionControlledPref(
extension_id,
pref_key,
kExtensionPrefsScopeRegular,
value);
}
void SettingsOverridesAPI::UnsetPref(const std::string& extension_id,
const std::string& pref_key) {
PreferenceAPI::Get(profile_)->RemoveExtensionControlledPref(
extension_id,
pref_key,
kExtensionPrefsScopeRegular);
}
void SettingsOverridesAPI::Observe(
int type,
const content::NotificationSource& source,
const content::NotificationDetails& details) {
switch (type) {
case chrome::NOTIFICATION_EXTENSION_LOADED: {
const Extension* extension =
content::Details<const Extension>(details).ptr();
const SettingsOverrides* settings =
SettingsOverrides::Get(extension);
if (settings) {
if (settings->homepage) {
SetPref(extension->id(), prefs::kHomePage,
new base::StringValue(settings->homepage->spec()));
SetPref(extension->id(), prefs::kHomePageIsNewTabPage,
new base::FundamentalValue(false));
}
if (!settings->startup_pages.empty()) {
SetPref(extension->id(), prefs::kRestoreOnStartup,
new base::FundamentalValue(
SessionStartupPref::kPrefValueURLs));
if (settings->startup_pages.size() > 1) {
VLOG(1) << extensions::ErrorUtils::FormatErrorMessage(
kManyStartupPagesWarning, manifest_keys::kSettingsOverride);
}
scoped_ptr<ListValue> url_list(new ListValue);
url_list->Append(new StringValue(settings->startup_pages[0].spec()));
SetPref(extension->id(), prefs::kURLsToRestoreOnStartup,
url_list.release());
}
if (settings->search_engine) {
DCHECK(url_service_);
if (url_service_->loaded()) {
RegisterSearchProvider(extension);
} else {
if (!template_url_sub_) {
template_url_sub_ = url_service_->RegisterOnLoadedCallback(
base::Bind(&SettingsOverridesAPI::OnTemplateURLsLoaded,
base::Unretained(this)));
}
url_service_->Load();
pending_extensions_.insert(extension);
}
}
}
break;
}
case chrome::NOTIFICATION_EXTENSION_UNLOADED: {
const Extension* extension =
content::Details<UnloadedExtensionInfo>(details)->extension;
const SettingsOverrides* settings = SettingsOverrides::Get(extension);
if (settings) {
if (settings->homepage) {
UnsetPref(extension->id(), prefs::kHomePage);
UnsetPref(extension->id(), prefs::kHomePageIsNewTabPage);
}
if (!settings->startup_pages.empty()) {
UnsetPref(extension->id(), prefs::kRestoreOnStartup);
UnsetPref(extension->id(), prefs::kURLsToRestoreOnStartup);
}
if (settings->search_engine) {
DCHECK(url_service_);
if (url_service_->loaded())
url_service_->RemoveExtensionControlledTURL(extension->id());
else
pending_extensions_.erase(extension);
}
}
break;
}
default: {
NOTREACHED();
break;
}
}
}
void SettingsOverridesAPI::Shutdown() {
template_url_sub_.reset();
}
void SettingsOverridesAPI::OnTemplateURLsLoaded() {
// Register search providers for pending extensions.
template_url_sub_.reset();
for (PendingExtensions::const_iterator i(pending_extensions_.begin());
i != pending_extensions_.end(); ++i) {
RegisterSearchProvider(*i);
}
pending_extensions_.clear();
}
void SettingsOverridesAPI::RegisterSearchProvider(
const Extension* extension) const {
DCHECK(url_service_);
DCHECK(extension);
const SettingsOverrides* settings = SettingsOverrides::Get(extension);
DCHECK(settings);
DCHECK(settings->search_engine);
scoped_ptr<AssociatedExtensionInfo> info(new AssociatedExtensionInfo);
info->extension_id = extension->id();
info->wants_to_be_default_engine = settings->search_engine->is_default;
info->install_time =
ExtensionPrefs::Get(profile_)->GetInstallTime(extension->id());
TemplateURLData data = ConvertSearchProvider(*settings->search_engine);
url_service_->AddExtensionControlledTURL(new TemplateURL(profile_, data),
info.Pass());
}
} // namespace extensions