| Index: chrome/browser/search_engines/default_search_manager.cc
|
| diff --git a/chrome/browser/search_engines/default_search_manager.cc b/chrome/browser/search_engines/default_search_manager.cc
|
| index cd763d73c00cdc57f54b1b002da584a62dee2e4a..28be7d5e25995c422b77faec17372898f3665b2a 100644
|
| --- a/chrome/browser/search_engines/default_search_manager.cc
|
| +++ b/chrome/browser/search_engines/default_search_manager.cc
|
| @@ -7,8 +7,11 @@
|
| #include <algorithm>
|
| #include <utility>
|
|
|
| +#include "base/bind.h"
|
| +#include "base/bind_helpers.h"
|
| #include "base/compiler_specific.h"
|
| #include "base/i18n/case_conversion.h"
|
| +#include "base/logging.h"
|
| #include "base/prefs/pref_service.h"
|
| #include "base/prefs/pref_value_map.h"
|
| #include "base/stl_util.h"
|
| @@ -19,6 +22,7 @@
|
| #include "base/time/time.h"
|
| #include "chrome/browser/profiles/profile.h"
|
| #include "chrome/browser/search_engines/template_url.h"
|
| +#include "chrome/browser/search_engines/template_url_prepopulate_data.h"
|
| #include "chrome/browser/search_engines/template_url_service.h"
|
| #include "chrome/browser/search_engines/util.h"
|
| #include "chrome/common/pref_names.h"
|
| @@ -26,14 +30,21 @@
|
|
|
| namespace {
|
|
|
| +bool IsDefaultSearchProviderManaged(PrefService* pref_service) {
|
| + const PrefService::Preference* pref = pref_service->FindPreference(
|
| + DefaultSearchManager::kDefaultSearchProviderDataPrefName);
|
| + DCHECK(pref);
|
| + return pref->IsManaged();
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| // A dictionary to hold all data related to the Default Search Engine.
|
| // Eventually, this should replace all the data stored in the
|
| // default_search_provider.* prefs.
|
| -const char kDefaultSearchProviderData[] =
|
| +const char DefaultSearchManager::kDefaultSearchProviderDataPrefName[] =
|
| "default_search_provider_data.template_url_data";
|
|
|
| -} // namespace
|
| -
|
| const char DefaultSearchManager::kID[] = "id";
|
| const char DefaultSearchManager::kShortName[] = "short_name";
|
| const char DefaultSearchManager::kKeyword[] = "keyword";
|
| @@ -70,9 +81,25 @@ const char DefaultSearchManager::kSearchTermsReplacementKey[] =
|
| const char DefaultSearchManager::kCreatedByPolicy[] = "created_by_policy";
|
| const char DefaultSearchManager::kDisabledByPolicy[] = "disabled_by_policy";
|
|
|
| -DefaultSearchManager::DefaultSearchManager(PrefService* pref_service)
|
| - : pref_service_(pref_service) {
|
| - DCHECK(pref_service_);
|
| +DefaultSearchManager::DefaultSearchManager(
|
| + PrefService* pref_service,
|
| + const ObserverCallback& change_observer)
|
| + : pref_service_(pref_service),
|
| + change_observer_(change_observer),
|
| + default_search_controlled_by_policy_(false) {
|
| + if (pref_service_) {
|
| + pref_change_registrar_.Init(pref_service_);
|
| + pref_change_registrar_.Add(
|
| + kDefaultSearchProviderDataPrefName,
|
| + base::Bind(&DefaultSearchManager::OnDefaultSearchPrefChanged,
|
| + base::Unretained(this)));
|
| + pref_change_registrar_.Add(
|
| + prefs::kSearchProviderOverrides,
|
| + base::Bind(&DefaultSearchManager::OnOverridesPrefChanged,
|
| + base::Unretained(this)));
|
| + }
|
| + LoadPrepopulatedDefaultSearch();
|
| + LoadDefaultSearchEngineFromPrefs();
|
| }
|
|
|
| DefaultSearchManager::~DefaultSearchManager() {
|
| @@ -82,102 +109,57 @@ DefaultSearchManager::~DefaultSearchManager() {
|
| void DefaultSearchManager::RegisterProfilePrefs(
|
| user_prefs::PrefRegistrySyncable* registry) {
|
| registry->RegisterDictionaryPref(
|
| - kDefaultSearchProviderData,
|
| - user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
|
| + kDefaultSearchProviderDataPrefName,
|
| + user_prefs::PrefRegistrySyncable::SYNCABLE_PREF);
|
| }
|
|
|
| // static
|
| void DefaultSearchManager::AddPrefValueToMap(base::DictionaryValue* value,
|
| PrefValueMap* pref_value_map) {
|
| - pref_value_map->SetValue(kDefaultSearchProviderData, value);
|
| + pref_value_map->SetValue(kDefaultSearchProviderDataPrefName, value);
|
| }
|
|
|
| -bool DefaultSearchManager::GetDefaultSearchEngine(TemplateURLData* data) {
|
| - const base::DictionaryValue* url_dict =
|
| - pref_service_->GetDictionary(kDefaultSearchProviderData);
|
| -
|
| - if (url_dict->empty())
|
| - return false;
|
| -
|
| - std::string search_url;
|
| - base::string16 keyword;
|
| - url_dict->GetString(kURL, &search_url);
|
| - url_dict->GetString(kKeyword, &keyword);
|
| - if (search_url.empty())
|
| - return false;
|
| - if (keyword.empty())
|
| - keyword = TemplateURLService::GenerateKeyword(GURL(search_url));
|
| - data->SetKeyword(keyword);
|
| - data->SetURL(search_url);
|
| -
|
| - std::string id;
|
| - url_dict->GetString(kID, &id);
|
| - base::StringToInt64(id, &data->id);
|
| - url_dict->GetString(kShortName, &data->short_name);
|
| - url_dict->GetInteger(kPrepopulateID, &data->prepopulate_id);
|
| - url_dict->GetString(kSyncGUID, &data->sync_guid);
|
| -
|
| - url_dict->GetString(kSuggestionsURL, &data->suggestions_url);
|
| - url_dict->GetString(kInstantURL, &data->instant_url);
|
| - url_dict->GetString(kImageURL, &data->image_url);
|
| - url_dict->GetString(kNewTabURL, &data->new_tab_url);
|
| -
|
| - std::string favicon_url;
|
| - std::string originating_url;
|
| - url_dict->GetString(kFaviconURL, &favicon_url);
|
| - url_dict->GetString(kOriginatingURL, &originating_url);
|
| - data->favicon_url = GURL(favicon_url);
|
| - data->originating_url = GURL(originating_url);
|
| -
|
| - url_dict->GetString(kSearchURLPostParams, &data->search_url_post_params);
|
| - url_dict->GetString(kSuggestionsURLPostParams,
|
| - &data->suggestions_url_post_params);
|
| - url_dict->GetString(kInstantURLPostParams, &data->instant_url_post_params);
|
| - url_dict->GetString(kImageURLPostParams, &data->image_url_post_params);
|
| -
|
| - url_dict->GetBoolean(kSafeForAutoReplace, &data->safe_for_autoreplace);
|
| -
|
| - double date_created = 0.0;
|
| - double last_modified = 0.0;
|
| - url_dict->GetDouble(kDateCreated, &date_created);
|
| - url_dict->GetDouble(kLastModified, &last_modified);
|
| - data->date_created = base::Time::FromInternalValue(date_created);
|
| - data->last_modified = base::Time::FromInternalValue(last_modified);
|
| -
|
| - url_dict->GetInteger(kUsageCount, &data->usage_count);
|
| -
|
| - const base::ListValue* alternate_urls;
|
| - url_dict->GetList(kAlternateURLs, &alternate_urls);
|
| - data->alternate_urls.clear();
|
| - for (base::ListValue::const_iterator it = alternate_urls->begin();
|
| - it != alternate_urls->end(); ++it) {
|
| - std::string alternate_url;
|
| - if ((*it)->GetAsString(&alternate_url))
|
| - data->alternate_urls.push_back(alternate_url);
|
| +TemplateURLData* DefaultSearchManager::GetDefaultSearchEngine(
|
| + Source* source) const {
|
| + if (default_search_controlled_by_policy_) {
|
| + if (source)
|
| + *source = FROM_POLICY;
|
| + return prefs_default_search_.get();
|
| }
|
| -
|
| - const base::ListValue* encodings;
|
| - url_dict->GetList(kInputEncodings, &encodings);
|
| - data->input_encodings.clear();
|
| - for (base::ListValue::const_iterator it = encodings->begin();
|
| - it != encodings->end(); ++it) {
|
| - std::string encoding;
|
| - if ((*it)->GetAsString(&encoding))
|
| - data->input_encodings.push_back(encoding);
|
| + if (extension_default_search_) {
|
| + if (source)
|
| + *source = FROM_EXTENSION;
|
| + return extension_default_search_.get();
|
| + }
|
| + if (prefs_default_search_) {
|
| + if (source)
|
| + *source = FROM_USER;
|
| + return prefs_default_search_.get();
|
| }
|
|
|
| - url_dict->GetString(kSearchTermsReplacementKey,
|
| - &data->search_terms_replacement_key);
|
| -
|
| - url_dict->GetBoolean(kCreatedByPolicy, &data->created_by_policy);
|
| -
|
| - data->show_in_default_list = true;
|
| + if (source)
|
| + *source = FROM_FALLBACK;
|
| + return TemplateURLService::FallbackSearchEnginesDisabled()
|
| + ? NULL
|
| + : fallback_default_search_.get();
|
| +}
|
|
|
| - return true;
|
| +DefaultSearchManager::Source
|
| +DefaultSearchManager::GetDefaultSearchEngineSource() const {
|
| + Source source;
|
| + GetDefaultSearchEngine(&source);
|
| + return source;
|
| }
|
|
|
| void DefaultSearchManager::SetUserSelectedDefaultSearchEngine(
|
| const TemplateURLData& data) {
|
| + if (!pref_service_) {
|
| + prefs_default_search_.reset(new TemplateURLData(data));
|
| + MergePrefsDataWithPrepopulated();
|
| + NotifyObserver();
|
| + return;
|
| + }
|
| +
|
| base::DictionaryValue url_dict;
|
| url_dict.SetString(kID, base::Int64ToString(data.id));
|
| url_dict.SetString(kShortName, data.short_name);
|
| @@ -201,8 +183,10 @@ void DefaultSearchManager::SetUserSelectedDefaultSearchEngine(
|
|
|
| url_dict.SetBoolean(kSafeForAutoReplace, data.safe_for_autoreplace);
|
|
|
| - url_dict.SetDouble(kDateCreated, data.date_created.ToInternalValue());
|
| - url_dict.SetDouble(kLastModified, data.last_modified.ToInternalValue());
|
| + url_dict.SetString(kDateCreated,
|
| + base::Int64ToString(data.date_created.ToInternalValue()));
|
| + url_dict.SetString(kLastModified,
|
| + base::Int64ToString(data.last_modified.ToInternalValue()));
|
| url_dict.SetInteger(kUsageCount, data.usage_count);
|
|
|
| scoped_ptr<base::ListValue> alternate_urls(new base::ListValue);
|
| @@ -226,9 +210,207 @@ void DefaultSearchManager::SetUserSelectedDefaultSearchEngine(
|
|
|
| url_dict.SetBoolean(kCreatedByPolicy, data.created_by_policy);
|
|
|
| - pref_service_->Set(kDefaultSearchProviderData, url_dict);
|
| + pref_service_->Set(kDefaultSearchProviderDataPrefName, url_dict);
|
| +}
|
| +
|
| +void DefaultSearchManager::SetExtensionControlledDefaultSearchEngine(
|
| + const TemplateURLData& data) {
|
| + extension_default_search_.reset(new TemplateURLData(data));
|
| + if (GetDefaultSearchEngineSource() == FROM_EXTENSION)
|
| + NotifyObserver();
|
| +}
|
| +
|
| +void DefaultSearchManager::ClearExtensionControlledDefaultSearchEngine() {
|
| + Source old_source = GetDefaultSearchEngineSource();
|
| + extension_default_search_.reset();
|
| + if (old_source == FROM_EXTENSION)
|
| + NotifyObserver();
|
| }
|
|
|
| void DefaultSearchManager::ClearUserSelectedDefaultSearchEngine() {
|
| - pref_service_->ClearPref(kDefaultSearchProviderData);
|
| + if (pref_service_) {
|
| + pref_service_->ClearPref(kDefaultSearchProviderDataPrefName);
|
| + } else {
|
| + prefs_default_search_.reset();
|
| + NotifyObserver();
|
| + }
|
| +}
|
| +
|
| +void DefaultSearchManager::OnDefaultSearchPrefChanged() {
|
| + Source source = GetDefaultSearchEngineSource();
|
| + LoadDefaultSearchEngineFromPrefs();
|
| +
|
| + // If we were/are FROM_USER or FROM_POLICY the effective DSE may have changed.
|
| + if (source != FROM_USER && source != FROM_POLICY)
|
| + source = GetDefaultSearchEngineSource();
|
| + if (source == FROM_USER || source == FROM_POLICY)
|
| + NotifyObserver();
|
| +}
|
| +
|
| +void DefaultSearchManager::OnOverridesPrefChanged() {
|
| + LoadPrepopulatedDefaultSearch();
|
| +
|
| + TemplateURLData* effective_data = GetDefaultSearchEngine(NULL);
|
| + if (effective_data && effective_data->prepopulate_id) {
|
| + // A user-selected, policy-selected or fallback pre-populated engine is
|
| + // active and may have changed with this event.
|
| + NotifyObserver();
|
| + }
|
| +}
|
| +
|
| +void DefaultSearchManager::MergePrefsDataWithPrepopulated() {
|
| + if (!prefs_default_search_ || !prefs_default_search_->prepopulate_id)
|
| + return;
|
| +
|
| + size_t default_search_index;
|
| + ScopedVector<TemplateURLData> prepopulated_urls =
|
| + TemplateURLPrepopulateData::GetPrepopulatedEngines(pref_service_,
|
| + &default_search_index);
|
| +
|
| + for (size_t i = 0; i < prepopulated_urls.size(); ++i) {
|
| + if (prepopulated_urls[i]->prepopulate_id ==
|
| + prefs_default_search_->prepopulate_id) {
|
| + if (!prefs_default_search_->safe_for_autoreplace) {
|
| + prepopulated_urls[i]->safe_for_autoreplace = false;
|
| + prepopulated_urls[i]->SetKeyword(prefs_default_search_->keyword());
|
| + prepopulated_urls[i]->short_name = prefs_default_search_->short_name;
|
| + }
|
| + prepopulated_urls[i]->id = prefs_default_search_->id;
|
| + prepopulated_urls[i]->sync_guid = prefs_default_search_->sync_guid;
|
| + prepopulated_urls[i]->date_created = prefs_default_search_->date_created;
|
| + prepopulated_urls[i]->last_modified =
|
| + prefs_default_search_->last_modified;
|
| + prefs_default_search_.reset(prepopulated_urls[i]);
|
| + prepopulated_urls.weak_erase(prepopulated_urls.begin() + i);
|
| + return;
|
| + }
|
| + }
|
| +}
|
| +
|
| +void DefaultSearchManager::LoadDefaultSearchEngineFromPrefs() {
|
| + if (!pref_service_)
|
| + return;
|
| +
|
| + prefs_default_search_.reset();
|
| + default_search_controlled_by_policy_ =
|
| + IsDefaultSearchProviderManaged(pref_service_);
|
| +
|
| + const base::DictionaryValue* url_dict =
|
| + pref_service_->GetDictionary(kDefaultSearchProviderDataPrefName);
|
| + if (url_dict->empty())
|
| + return;
|
| +
|
| + if (default_search_controlled_by_policy_) {
|
| + bool disabled_by_policy = false;
|
| + if (url_dict->GetBoolean(kDisabledByPolicy, &disabled_by_policy) &&
|
| + disabled_by_policy)
|
| + return;
|
| + }
|
| +
|
| + std::string search_url;
|
| + base::string16 keyword;
|
| + url_dict->GetString(kURL, &search_url);
|
| + url_dict->GetString(kKeyword, &keyword);
|
| + if (search_url.empty() || keyword.empty())
|
| + return;
|
| +
|
| + prefs_default_search_.reset(new TemplateURLData);
|
| + prefs_default_search_->SetKeyword(keyword);
|
| + prefs_default_search_->SetURL(search_url);
|
| +
|
| + std::string id;
|
| + url_dict->GetString(kID, &id);
|
| + base::StringToInt64(id, &prefs_default_search_->id);
|
| + url_dict->GetString(kShortName, &prefs_default_search_->short_name);
|
| + url_dict->GetInteger(kPrepopulateID, &prefs_default_search_->prepopulate_id);
|
| + url_dict->GetString(kSyncGUID, &prefs_default_search_->sync_guid);
|
| +
|
| + url_dict->GetString(kSuggestionsURL, &prefs_default_search_->suggestions_url);
|
| + url_dict->GetString(kInstantURL, &prefs_default_search_->instant_url);
|
| + url_dict->GetString(kImageURL, &prefs_default_search_->image_url);
|
| + url_dict->GetString(kNewTabURL, &prefs_default_search_->new_tab_url);
|
| +
|
| + std::string favicon_url;
|
| + std::string originating_url;
|
| + url_dict->GetString(kFaviconURL, &favicon_url);
|
| + url_dict->GetString(kOriginatingURL, &originating_url);
|
| + prefs_default_search_->favicon_url = GURL(favicon_url);
|
| + prefs_default_search_->originating_url = GURL(originating_url);
|
| +
|
| + url_dict->GetString(kSearchURLPostParams,
|
| + &prefs_default_search_->search_url_post_params);
|
| + url_dict->GetString(kSuggestionsURLPostParams,
|
| + &prefs_default_search_->suggestions_url_post_params);
|
| + url_dict->GetString(kInstantURLPostParams,
|
| + &prefs_default_search_->instant_url_post_params);
|
| + url_dict->GetString(kImageURLPostParams,
|
| + &prefs_default_search_->image_url_post_params);
|
| +
|
| + url_dict->GetBoolean(kSafeForAutoReplace,
|
| + &prefs_default_search_->safe_for_autoreplace);
|
| +
|
| + std::string date_created_str;
|
| + std::string last_modified_str;
|
| + url_dict->GetString(kDateCreated, &date_created_str);
|
| + url_dict->GetString(kLastModified, &last_modified_str);
|
| +
|
| + int64 date_created = 0;
|
| + if (base::StringToInt64(date_created_str, &date_created)) {
|
| + prefs_default_search_->date_created =
|
| + base::Time::FromInternalValue(date_created);
|
| + }
|
| +
|
| + int64 last_modified = 0;
|
| + if (base::StringToInt64(date_created_str, &last_modified)) {
|
| + prefs_default_search_->last_modified =
|
| + base::Time::FromInternalValue(last_modified);
|
| + }
|
| +
|
| + url_dict->GetInteger(kUsageCount, &prefs_default_search_->usage_count);
|
| +
|
| + const base::ListValue* alternate_urls = NULL;
|
| + if (url_dict->GetList(kAlternateURLs, &alternate_urls)) {
|
| + for (base::ListValue::const_iterator it = alternate_urls->begin();
|
| + it != alternate_urls->end();
|
| + ++it) {
|
| + std::string alternate_url;
|
| + if ((*it)->GetAsString(&alternate_url))
|
| + prefs_default_search_->alternate_urls.push_back(alternate_url);
|
| + }
|
| + }
|
| +
|
| + const base::ListValue* encodings = NULL;
|
| + if (url_dict->GetList(kInputEncodings, &encodings)) {
|
| + for (base::ListValue::const_iterator it = encodings->begin();
|
| + it != encodings->end();
|
| + ++it) {
|
| + std::string encoding;
|
| + if ((*it)->GetAsString(&encoding))
|
| + prefs_default_search_->input_encodings.push_back(encoding);
|
| + }
|
| + }
|
| +
|
| + url_dict->GetString(kSearchTermsReplacementKey,
|
| + &prefs_default_search_->search_terms_replacement_key);
|
| +
|
| + url_dict->GetBoolean(kCreatedByPolicy,
|
| + &prefs_default_search_->created_by_policy);
|
| +
|
| + prefs_default_search_->show_in_default_list = true;
|
| + MergePrefsDataWithPrepopulated();
|
| +}
|
| +
|
| +void DefaultSearchManager::LoadPrepopulatedDefaultSearch() {
|
| + scoped_ptr<TemplateURLData> data =
|
| + TemplateURLPrepopulateData::GetPrepopulatedDefaultSearch(pref_service_);
|
| + fallback_default_search_ = data.Pass();
|
| + MergePrefsDataWithPrepopulated();
|
| +}
|
| +
|
| +void DefaultSearchManager::NotifyObserver() {
|
| + if (!change_observer_.is_null()) {
|
| + Source source = FROM_FALLBACK;
|
| + TemplateURLData* data = GetDefaultSearchEngine(&source);
|
| + change_observer_.Run(data, source);
|
| + }
|
| }
|
|
|