blob: d4a132707e5ff2519eb6099d7478f952b74efee5 [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/extensions/api/preference/preference_api.h"
#include "chrome/browser/prefs/session_startup_pref.h"
#include "chrome/browser/profiles/profile.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 "components/search_engines/search_engines_pref_names.h"
#include "components/search_engines/template_url.h"
#include "components/search_engines/template_url_prepopulate_data.h"
#include "extensions/browser/extension_prefs.h"
#include "extensions/browser/extension_prefs_factory.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/common/error_utils.h"
#include "extensions/common/manifest_constants.h"
namespace extensions {
namespace {
base::LazyInstance<BrowserContextKeyedAPIFactory<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;
std::string SubstituteInstallParam(std::string str,
const std::string& install_parameter) {
ReplaceSubstringsAfterOffset(&str, 0, "__PARAM__", install_parameter);
return str;
}
// Find the prepopulated search engine with the given id.
bool GetPrepopulatedSearchProvider(PrefService* prefs,
int prepopulated_id,
TemplateURLData* data) {
DCHECK(data);
size_t default_index;
ScopedVector<TemplateURLData> engines =
TemplateURLPrepopulateData::GetPrepopulatedEngines(prefs, &default_index);
for (ScopedVector<TemplateURLData>::iterator i = engines.begin();
i != engines.end();
++i) {
if ((*i)->prepopulate_id == prepopulated_id) {
*data = **i;
return true;
}
}
return false;
}
TemplateURLData ConvertSearchProvider(
PrefService* prefs,
const ChromeSettingsOverrides::Search_provider& search_provider,
const std::string& install_parameter) {
TemplateURLData data;
if (search_provider.prepopulated_id) {
if (!GetPrepopulatedSearchProvider(prefs, *search_provider.prepopulated_id,
&data)) {
VLOG(1) << "Settings Overrides API can't recognize prepopulated_id="
<< *search_provider.prepopulated_id;
}
}
if (search_provider.name)
data.short_name = base::UTF8ToUTF16(*search_provider.name);
if (search_provider.keyword)
data.SetKeyword(base::UTF8ToUTF16(*search_provider.keyword));
data.SetURL(SubstituteInstallParam(search_provider.search_url,
install_parameter));
if (search_provider.suggest_url) {
data.suggestions_url =
SubstituteInstallParam(*search_provider.suggest_url, install_parameter);
}
if (search_provider.instant_url) {
data.instant_url =
SubstituteInstallParam(*search_provider.instant_url, install_parameter);
}
if (search_provider.image_url) {
data.image_url =
SubstituteInstallParam(*search_provider.image_url, install_parameter);
}
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;
if (search_provider.favicon_url) {
data.favicon_url = GURL(SubstituteInstallParam(*search_provider.favicon_url,
install_parameter));
}
data.safe_for_autoreplace = false;
if (search_provider.encoding) {
data.input_encodings.clear();
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) {
data.alternate_urls.clear();
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(SubstituteInstallParam(
search_provider.alternate_urls->at(i), install_parameter));
}
}
return data;
}
} // namespace
SettingsOverridesAPI::SettingsOverridesAPI(content::BrowserContext* context)
: profile_(Profile::FromBrowserContext(context)),
url_service_(TemplateURLServiceFactory::GetForProfile(profile_)),
extension_registry_observer_(this) {
extension_registry_observer_.Add(ExtensionRegistry::Get(profile_));
}
SettingsOverridesAPI::~SettingsOverridesAPI() {
}
BrowserContextKeyedAPIFactory<SettingsOverridesAPI>*
SettingsOverridesAPI::GetFactoryInstance() {
return g_factory.Pointer();
}
void SettingsOverridesAPI::SetPref(const std::string& extension_id,
const std::string& pref_key,
base::Value* value) {
PreferenceAPI* prefs = PreferenceAPI::Get(profile_);
if (!prefs)
return; // Expected in unit tests.
prefs->SetExtensionControlledPref(extension_id,
pref_key,
kExtensionPrefsScopeRegular,
value);
}
void SettingsOverridesAPI::UnsetPref(const std::string& extension_id,
const std::string& pref_key) {
PreferenceAPI* prefs = PreferenceAPI::Get(profile_);
if (!prefs)
return; // Expected in unit tests.
prefs->RemoveExtensionControlledPref(
extension_id,
pref_key,
kExtensionPrefsScopeRegular);
}
void SettingsOverridesAPI::OnExtensionLoaded(
content::BrowserContext* browser_context,
const Extension* extension) {
const SettingsOverrides* settings = SettingsOverrides::Get(extension);
if (settings) {
std::string install_parameter =
ExtensionPrefs::Get(profile_)->GetInstallParam(extension->id());
if (settings->homepage) {
SetPref(extension->id(),
prefs::kHomePage,
new base::StringValue(SubstituteInstallParam(
settings->homepage->spec(), install_parameter)));
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<base::ListValue> url_list(new base::ListValue);
url_list->Append(new base::StringValue(SubstituteInstallParam(
settings->startup_pages[0].spec(), install_parameter)));
SetPref(
extension->id(), prefs::kURLsToRestoreOnStartup, url_list.release());
}
if (settings->search_engine) {
// Bring the preference to the correct state. Before this code set it
// to "true" for all search engines. Thus, we should overwrite it for
// all search engines.
if (settings->search_engine->is_default) {
SetPref(extension->id(),
prefs::kDefaultSearchProviderEnabled,
new base::FundamentalValue(true));
} else {
UnsetPref(extension->id(), prefs::kDefaultSearchProviderEnabled);
}
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);
}
}
}
}
void SettingsOverridesAPI::OnExtensionUnloaded(
content::BrowserContext* browser_context,
const Extension* extension,
UnloadedExtensionInfo::Reason reason) {
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(), TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION);
} else {
pending_extensions_.erase(extension);
}
}
}
}
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->get());
}
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<TemplateURL::AssociatedExtensionInfo> info(
new TemplateURL::AssociatedExtensionInfo(
TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, extension->id()));
info->wants_to_be_default_engine = settings->search_engine->is_default;
ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
info->install_time = prefs->GetInstallTime(extension->id());
std::string install_parameter = prefs->GetInstallParam(extension->id());
TemplateURLData data = ConvertSearchProvider(
profile_->GetPrefs(), *settings->search_engine, install_parameter);
data.show_in_default_list = info->wants_to_be_default_engine;
url_service_->AddExtensionControlledTURL(new TemplateURL(data), info.Pass());
}
template <>
void BrowserContextKeyedAPIFactory<
SettingsOverridesAPI>::DeclareFactoryDependencies() {
DependsOn(ExtensionPrefsFactory::GetInstance());
DependsOn(PreferenceAPI::GetFactoryInstance());
DependsOn(TemplateURLServiceFactory::GetInstance());
}
} // namespace extensions