most_visited_handler.cc revision d0247b1b59f9c528cb6df88b4f2b9afaf80d181e
1// Copyright (c) 2012 The Chromium Authors. All rights reserved. 2// Use of this source code is governed by a BSD-style license that can be 3// found in the LICENSE file. 4 5#include "chrome/browser/ui/webui/ntp/most_visited_handler.h" 6 7#include <set> 8 9#include "base/bind.h" 10#include "base/bind_helpers.h" 11#include "base/command_line.h" 12#include "base/md5.h" 13#include "base/memory/scoped_vector.h" 14#include "base/memory/singleton.h" 15#include "base/metrics/histogram.h" 16#include "base/prefs/pref_service.h" 17#include "base/strings/string16.h" 18#include "base/strings/string_number_conversions.h" 19#include "base/strings/utf_string_conversions.h" 20#include "base/threading/thread.h" 21#include "base/values.h" 22#include "chrome/browser/chrome_notification_types.h" 23#include "chrome/browser/history/most_visited_tiles_experiment.h" 24#include "chrome/browser/history/page_usage_data.h" 25#include "chrome/browser/history/top_sites.h" 26#include "chrome/browser/prefs/scoped_user_pref_update.h" 27#include "chrome/browser/profiles/profile.h" 28#include "chrome/browser/ui/browser.h" 29#include "chrome/browser/ui/browser_finder.h" 30#include "chrome/browser/ui/tabs/tab_strip_model.h" 31#include "chrome/browser/ui/tabs/tab_strip_model_utils.h" 32#include "chrome/browser/ui/webui/favicon_source.h" 33#include "chrome/browser/ui/webui/ntp/new_tab_ui.h" 34#include "chrome/browser/ui/webui/ntp/ntp_stats.h" 35#include "chrome/browser/ui/webui/ntp/thumbnail_source.h" 36#include "chrome/common/pref_names.h" 37#include "chrome/common/url_constants.h" 38#include "components/user_prefs/pref_registry_syncable.h" 39#include "content/public/browser/navigation_controller.h" 40#include "content/public/browser/navigation_entry.h" 41#include "content/public/browser/notification_source.h" 42#include "content/public/browser/url_data_source.h" 43#include "content/public/browser/user_metrics.h" 44#include "content/public/browser/web_contents.h" 45#include "content/public/browser/web_ui.h" 46#include "grit/chromium_strings.h" 47#include "grit/generated_resources.h" 48#include "grit/locale_settings.h" 49#include "ui/base/l10n/l10n_util.h" 50#include "url/gurl.h" 51 52using content::UserMetricsAction; 53 54MostVisitedHandler::MostVisitedHandler() 55 : weak_ptr_factory_(this), 56 got_first_most_visited_request_(false), 57 most_visited_viewed_(false), 58 user_action_logged_(false) { 59} 60 61MostVisitedHandler::~MostVisitedHandler() { 62 if (!user_action_logged_ && most_visited_viewed_) { 63 const GURL ntp_url = GURL(chrome::kChromeUINewTabURL); 64 int action_id = NTP_FOLLOW_ACTION_OTHER; 65 content::NavigationEntry* entry = 66 web_ui()->GetWebContents()->GetController().GetActiveEntry(); 67 if (entry && (entry->GetURL() != ntp_url)) { 68 action_id = 69 content::PageTransitionStripQualifier(entry->GetTransitionType()); 70 } 71 72 UMA_HISTOGRAM_ENUMERATION("NewTabPage.MostVisitedAction", action_id, 73 NUM_NTP_FOLLOW_ACTIONS); 74 } 75} 76 77void MostVisitedHandler::RegisterMessages() { 78 Profile* profile = Profile::FromWebUI(web_ui()); 79 // Set up our sources for thumbnail and favicon data. 80 ThumbnailSource* thumbnail_src_exact = new ThumbnailSource(profile, false); 81 content::URLDataSource::Add(profile, thumbnail_src_exact); 82 83 ThumbnailSource* thumbnail_src_prefix = new ThumbnailSource(profile, true); 84 content::URLDataSource::Add(profile, thumbnail_src_prefix); 85 86#if defined(OS_ANDROID) 87 // Register chrome://touch-icon as a data source for touch icons or favicons. 88 content::URLDataSource::Add(profile, 89 new FaviconSource(profile, FaviconSource::ANY)); 90#endif 91 // Register chrome://favicon as a data source for favicons. 92 content::URLDataSource::Add( 93 profile, new FaviconSource(profile, FaviconSource::FAVICON)); 94 95 history::TopSites* ts = profile->GetTopSites(); 96 if (ts) { 97 // TopSites updates itself after a delay. This is especially noticable when 98 // your profile is empty. Ask TopSites to update itself when we're about to 99 // show the new tab page. 100 ts->SyncWithHistory(); 101 102 // Register for notification when TopSites changes so that we can update 103 // ourself. 104 registrar_.Add(this, chrome::NOTIFICATION_TOP_SITES_CHANGED, 105 content::Source<history::TopSites>(ts)); 106 } 107 108 // We pre-emptively make a fetch for the most visited pages so we have the 109 // results sooner. 110 StartQueryForMostVisited(); 111 112 web_ui()->RegisterMessageCallback("getMostVisited", 113 base::Bind(&MostVisitedHandler::HandleGetMostVisited, 114 base::Unretained(this))); 115 116 // Register ourselves for any most-visited item blacklisting. 117 web_ui()->RegisterMessageCallback("blacklistURLFromMostVisited", 118 base::Bind(&MostVisitedHandler::HandleBlacklistUrl, 119 base::Unretained(this))); 120 web_ui()->RegisterMessageCallback("removeURLsFromMostVisitedBlacklist", 121 base::Bind(&MostVisitedHandler::HandleRemoveUrlsFromBlacklist, 122 base::Unretained(this))); 123 web_ui()->RegisterMessageCallback("clearMostVisitedURLsBlacklist", 124 base::Bind(&MostVisitedHandler::HandleClearBlacklist, 125 base::Unretained(this))); 126 web_ui()->RegisterMessageCallback("mostVisitedAction", 127 base::Bind(&MostVisitedHandler::HandleMostVisitedAction, 128 base::Unretained(this))); 129 web_ui()->RegisterMessageCallback("mostVisitedSelected", 130 base::Bind(&MostVisitedHandler::HandleMostVisitedSelected, 131 base::Unretained(this))); 132} 133 134void MostVisitedHandler::HandleGetMostVisited(const ListValue* args) { 135 if (!got_first_most_visited_request_) { 136 // If our initial data is already here, return it. 137 SendPagesValue(); 138 got_first_most_visited_request_ = true; 139 } else { 140 StartQueryForMostVisited(); 141 } 142} 143 144void MostVisitedHandler::SendPagesValue() { 145 if (pages_value_) { 146 Profile* profile = Profile::FromWebUI(web_ui()); 147 const DictionaryValue* url_blacklist = 148 profile->GetPrefs()->GetDictionary(prefs::kNtpMostVisitedURLsBlacklist); 149 bool has_blacklisted_urls = !url_blacklist->empty(); 150 history::TopSites* ts = profile->GetTopSites(); 151 if (ts) { 152 has_blacklisted_urls = ts->HasBlacklistedItems(); 153 154 MaybeRemovePageValues(); 155 } 156 157 base::FundamentalValue has_blacklisted_urls_value(has_blacklisted_urls); 158 web_ui()->CallJavascriptFunction("ntp.setMostVisitedPages", 159 *pages_value_, 160 has_blacklisted_urls_value); 161 pages_value_.reset(); 162 } 163} 164 165void MostVisitedHandler::StartQueryForMostVisited() { 166 history::TopSites* ts = Profile::FromWebUI(web_ui())->GetTopSites(); 167 if (ts) { 168 ts->GetMostVisitedURLs( 169 base::Bind(&MostVisitedHandler::OnMostVisitedUrlsAvailable, 170 weak_ptr_factory_.GetWeakPtr())); 171 } 172} 173 174void MostVisitedHandler::HandleBlacklistUrl(const ListValue* args) { 175 std::string url = UTF16ToUTF8(ExtractStringValue(args)); 176 BlacklistUrl(GURL(url)); 177} 178 179void MostVisitedHandler::HandleRemoveUrlsFromBlacklist(const ListValue* args) { 180 DCHECK(args->GetSize() != 0); 181 182 for (ListValue::const_iterator iter = args->begin(); 183 iter != args->end(); ++iter) { 184 std::string url; 185 bool r = (*iter)->GetAsString(&url); 186 if (!r) { 187 NOTREACHED(); 188 return; 189 } 190 content::RecordAction(UserMetricsAction("MostVisited_UrlRemoved")); 191 history::TopSites* ts = Profile::FromWebUI(web_ui())->GetTopSites(); 192 if (ts) 193 ts->RemoveBlacklistedURL(GURL(url)); 194 } 195} 196 197void MostVisitedHandler::HandleClearBlacklist(const ListValue* args) { 198 content::RecordAction(UserMetricsAction("MostVisited_BlacklistCleared")); 199 200 history::TopSites* ts = Profile::FromWebUI(web_ui())->GetTopSites(); 201 if (ts) 202 ts->ClearBlacklistedURLs(); 203} 204 205void MostVisitedHandler::HandleMostVisitedAction(const base::ListValue* args) { 206 DCHECK(args); 207 208 double action_id; 209 if (!args->GetDouble(0, &action_id)) 210 NOTREACHED(); 211 212 UMA_HISTOGRAM_ENUMERATION("NewTabPage.MostVisitedAction", 213 static_cast<int>(action_id), 214 NUM_NTP_FOLLOW_ACTIONS); 215 most_visited_viewed_ = true; 216 user_action_logged_ = true; 217} 218 219void MostVisitedHandler::HandleMostVisitedSelected( 220 const base::ListValue* args) { 221 most_visited_viewed_ = true; 222} 223 224void MostVisitedHandler::SetPagesValueFromTopSites( 225 const history::MostVisitedURLList& data) { 226 pages_value_.reset(new ListValue); 227 228 history::MostVisitedURLList top_sites(data); 229 history::MostVisitedTilesExperiment::MaybeShuffle(&top_sites); 230 231 for (size_t i = 0; i < top_sites.size(); i++) { 232 const history::MostVisitedURL& url = top_sites[i]; 233 DictionaryValue* page_value = new DictionaryValue(); 234 if (url.url.is_empty()) { 235 page_value->SetBoolean("filler", true); 236 pages_value_->Append(page_value); 237 continue; 238 } 239 240 NewTabUI::SetUrlTitleAndDirection(page_value, 241 url.title, 242 url.url); 243 pages_value_->Append(page_value); 244 } 245} 246 247void MostVisitedHandler::OnMostVisitedUrlsAvailable( 248 const history::MostVisitedURLList& data) { 249 SetPagesValueFromTopSites(data); 250 if (got_first_most_visited_request_) { 251 SendPagesValue(); 252 } 253} 254 255void MostVisitedHandler::Observe(int type, 256 const content::NotificationSource& source, 257 const content::NotificationDetails& details) { 258 DCHECK_EQ(type, chrome::NOTIFICATION_TOP_SITES_CHANGED); 259 260 // Most visited urls changed, query again. 261 StartQueryForMostVisited(); 262} 263 264void MostVisitedHandler::BlacklistUrl(const GURL& url) { 265 history::TopSites* ts = Profile::FromWebUI(web_ui())->GetTopSites(); 266 if (ts) 267 ts->AddBlacklistedURL(url); 268 content::RecordAction(UserMetricsAction("MostVisited_UrlBlacklisted")); 269} 270 271std::string MostVisitedHandler::GetDictionaryKeyForUrl(const std::string& url) { 272 return base::MD5String(url); 273} 274 275void MostVisitedHandler::MaybeRemovePageValues() { 276// The code below uses APIs not available on Android and the experiment should 277// not run there. 278#if !defined(OS_ANDROID) 279 if (!history::MostVisitedTilesExperiment::IsDontShowOpenURLsEnabled()) 280 return; 281 282 TabStripModel* tab_strip_model = chrome::FindBrowserWithWebContents( 283 web_ui()->GetWebContents())->tab_strip_model(); 284 history::TopSites* top_sites = Profile::FromWebUI(web_ui())->GetTopSites(); 285 if (!tab_strip_model || !top_sites) { 286 NOTREACHED(); 287 return; 288 } 289 290 std::set<std::string> open_urls; 291 chrome::GetOpenUrls(*tab_strip_model, *top_sites, &open_urls); 292 history::MostVisitedTilesExperiment::RemovePageValuesMatchingOpenTabs( 293 open_urls, 294 pages_value_.get()); 295#endif 296} 297 298// static 299void MostVisitedHandler::RegisterProfilePrefs( 300 user_prefs::PrefRegistrySyncable* registry) { 301 registry->RegisterDictionaryPref( 302 prefs::kNtpMostVisitedURLsBlacklist, 303 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); 304} 305