| Index: chrome/browser/ui/webui/ntp/most_visited_handler.cc
 | 
| diff --git a/chrome/browser/ui/webui/ntp/most_visited_handler.cc b/chrome/browser/ui/webui/ntp/most_visited_handler.cc
 | 
| new file mode 100644
 | 
| index 0000000000000000000000000000000000000000..9deda8c1580de4d6699bf742adad7c4c4bd3b6b0
 | 
| --- /dev/null
 | 
| +++ b/chrome/browser/ui/webui/ntp/most_visited_handler.cc
 | 
| @@ -0,0 +1,294 @@
 | 
| +// Copyright (c) 2012 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/ui/webui/ntp/most_visited_handler.h"
 | 
| +
 | 
| +#include <set>
 | 
| +
 | 
| +#include "base/bind.h"
 | 
| +#include "base/bind_helpers.h"
 | 
| +#include "base/command_line.h"
 | 
| +#include "base/md5.h"
 | 
| +#include "base/memory/scoped_vector.h"
 | 
| +#include "base/memory/singleton.h"
 | 
| +#include "base/metrics/histogram.h"
 | 
| +#include "base/prefs/pref_service.h"
 | 
| +#include "base/prefs/scoped_user_pref_update.h"
 | 
| +#include "base/strings/string16.h"
 | 
| +#include "base/strings/string_number_conversions.h"
 | 
| +#include "base/strings/utf_string_conversions.h"
 | 
| +#include "base/threading/thread.h"
 | 
| +#include "base/values.h"
 | 
| +#include "chrome/browser/favicon/fallback_icon_service_factory.h"
 | 
| +#include "chrome/browser/favicon/large_icon_service_factory.h"
 | 
| +#include "chrome/browser/history/top_sites_factory.h"
 | 
| +#include "chrome/browser/profiles/profile.h"
 | 
| +#include "chrome/browser/thumbnails/thumbnail_list_source.h"
 | 
| +#include "chrome/browser/ui/browser.h"
 | 
| +#include "chrome/browser/ui/browser_finder.h"
 | 
| +#include "chrome/browser/ui/tabs/tab_strip_model.h"
 | 
| +#include "chrome/browser/ui/tabs/tab_strip_model_utils.h"
 | 
| +#include "chrome/browser/ui/webui/fallback_icon_source.h"
 | 
| +#include "chrome/browser/ui/webui/favicon_source.h"
 | 
| +#include "chrome/browser/ui/webui/large_icon_source.h"
 | 
| +#include "chrome/browser/ui/webui/ntp/new_tab_ui.h"
 | 
| +#include "chrome/browser/ui/webui/ntp/ntp_stats.h"
 | 
| +#include "chrome/browser/ui/webui/ntp/thumbnail_source.h"
 | 
| +#include "chrome/common/pref_names.h"
 | 
| +#include "chrome/common/url_constants.h"
 | 
| +#include "components/favicon/core/fallback_icon_service.h"
 | 
| +#include "components/favicon/core/large_icon_service.h"
 | 
| +#include "components/history/core/browser/page_usage_data.h"
 | 
| +#include "components/history/core/browser/top_sites.h"
 | 
| +#include "components/keyed_service/core/service_access_type.h"
 | 
| +#include "components/pref_registry/pref_registry_syncable.h"
 | 
| +#include "content/public/browser/navigation_controller.h"
 | 
| +#include "content/public/browser/navigation_entry.h"
 | 
| +#include "content/public/browser/url_data_source.h"
 | 
| +#include "content/public/browser/user_metrics.h"
 | 
| +#include "content/public/browser/web_contents.h"
 | 
| +#include "content/public/browser/web_ui.h"
 | 
| +#include "url/gurl.h"
 | 
| +
 | 
| +using base::UserMetricsAction;
 | 
| +
 | 
| +MostVisitedHandler::MostVisitedHandler()
 | 
| +    : scoped_observer_(this),
 | 
| +      got_first_most_visited_request_(false),
 | 
| +      most_visited_viewed_(false),
 | 
| +      user_action_logged_(false),
 | 
| +      weak_ptr_factory_(this) {
 | 
| +}
 | 
| +
 | 
| +MostVisitedHandler::~MostVisitedHandler() {
 | 
| +  if (!user_action_logged_ && most_visited_viewed_) {
 | 
| +    const GURL ntp_url = GURL(chrome::kChromeUINewTabURL);
 | 
| +    int action_id = NTP_FOLLOW_ACTION_OTHER;
 | 
| +    content::NavigationEntry* entry =
 | 
| +        web_ui()->GetWebContents()->GetController().GetLastCommittedEntry();
 | 
| +    if (entry && (entry->GetURL() != ntp_url)) {
 | 
| +      action_id =
 | 
| +          ui::PageTransitionStripQualifier(entry->GetTransitionType());
 | 
| +    }
 | 
| +
 | 
| +    UMA_HISTOGRAM_ENUMERATION("NewTabPage.MostVisitedAction", action_id,
 | 
| +                              NUM_NTP_FOLLOW_ACTIONS);
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::RegisterMessages() {
 | 
| +  Profile* profile = Profile::FromWebUI(web_ui());
 | 
| +  // Set up our sources for thumbnail and favicon data.
 | 
| +  content::URLDataSource::Add(profile, new ThumbnailSource(profile, false));
 | 
| +  content::URLDataSource::Add(profile, new ThumbnailSource(profile, true));
 | 
| +
 | 
| +  // Set up our sources for top-sites data.
 | 
| +  content::URLDataSource::Add(profile, new ThumbnailListSource(profile));
 | 
| +
 | 
| +  favicon::FallbackIconService* fallback_icon_service =
 | 
| +      FallbackIconServiceFactory::GetForBrowserContext(profile);
 | 
| +  favicon::LargeIconService* large_icon_service =
 | 
| +      LargeIconServiceFactory::GetForBrowserContext(profile);
 | 
| +
 | 
| +  // Register chrome://fallback-icon as a data source for fallback icons.
 | 
| +  content::URLDataSource::Add(profile,
 | 
| +                              new FallbackIconSource(fallback_icon_service));
 | 
| +
 | 
| +  // Register chrome://favicon as a data source for favicons.
 | 
| +  content::URLDataSource::Add(
 | 
| +      profile, new FaviconSource(profile, FaviconSource::FAVICON));
 | 
| +
 | 
| +  // Register chrome://large-icon as a data source for large icons.
 | 
| +  content::URLDataSource::Add(
 | 
| +      profile, new LargeIconSource(fallback_icon_service, large_icon_service));
 | 
| +
 | 
| +  scoped_refptr<history::TopSites> top_sites =
 | 
| +      TopSitesFactory::GetForProfile(profile);
 | 
| +  if (top_sites) {
 | 
| +    // TopSites updates itself after a delay. This is especially noticable when
 | 
| +    // your profile is empty. Ask TopSites to update itself when we're about to
 | 
| +    // show the new tab page.
 | 
| +    top_sites->SyncWithHistory();
 | 
| +
 | 
| +    // Register as TopSitesObserver so that we can update ourselves when the
 | 
| +    // TopSites changes.
 | 
| +    scoped_observer_.Add(top_sites.get());
 | 
| +  }
 | 
| +
 | 
| +  // We pre-emptively make a fetch for the most visited pages so we have the
 | 
| +  // results sooner.
 | 
| +  StartQueryForMostVisited();
 | 
| +
 | 
| +  web_ui()->RegisterMessageCallback("getMostVisited",
 | 
| +      base::Bind(&MostVisitedHandler::HandleGetMostVisited,
 | 
| +                 base::Unretained(this)));
 | 
| +
 | 
| +  // Register ourselves for any most-visited item blacklisting.
 | 
| +  web_ui()->RegisterMessageCallback("blacklistURLFromMostVisited",
 | 
| +      base::Bind(&MostVisitedHandler::HandleBlacklistUrl,
 | 
| +                 base::Unretained(this)));
 | 
| +  web_ui()->RegisterMessageCallback("removeURLsFromMostVisitedBlacklist",
 | 
| +      base::Bind(&MostVisitedHandler::HandleRemoveUrlsFromBlacklist,
 | 
| +                 base::Unretained(this)));
 | 
| +  web_ui()->RegisterMessageCallback("clearMostVisitedURLsBlacklist",
 | 
| +      base::Bind(&MostVisitedHandler::HandleClearBlacklist,
 | 
| +                 base::Unretained(this)));
 | 
| +  web_ui()->RegisterMessageCallback("mostVisitedAction",
 | 
| +      base::Bind(&MostVisitedHandler::HandleMostVisitedAction,
 | 
| +                 base::Unretained(this)));
 | 
| +  web_ui()->RegisterMessageCallback("mostVisitedSelected",
 | 
| +      base::Bind(&MostVisitedHandler::HandleMostVisitedSelected,
 | 
| +                 base::Unretained(this)));
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleGetMostVisited(const base::ListValue* args) {
 | 
| +  if (!got_first_most_visited_request_) {
 | 
| +    // If our initial data is already here, return it.
 | 
| +    SendPagesValue();
 | 
| +    got_first_most_visited_request_ = true;
 | 
| +  } else {
 | 
| +    StartQueryForMostVisited();
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::SendPagesValue() {
 | 
| +  if (pages_value_) {
 | 
| +    Profile* profile = Profile::FromWebUI(web_ui());
 | 
| +    const base::DictionaryValue* url_blacklist =
 | 
| +        profile->GetPrefs()->GetDictionary(prefs::kNtpMostVisitedURLsBlacklist);
 | 
| +    bool has_blacklisted_urls = !url_blacklist->empty();
 | 
| +    scoped_refptr<history::TopSites> ts =
 | 
| +        TopSitesFactory::GetForProfile(profile);
 | 
| +    if (ts)
 | 
| +      has_blacklisted_urls = ts->HasBlacklistedItems();
 | 
| +
 | 
| +    base::FundamentalValue has_blacklisted_urls_value(has_blacklisted_urls);
 | 
| +    web_ui()->CallJavascriptFunction("ntp.setMostVisitedPages",
 | 
| +                                     *pages_value_,
 | 
| +                                     has_blacklisted_urls_value);
 | 
| +    pages_value_.reset();
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::StartQueryForMostVisited() {
 | 
| +  scoped_refptr<history::TopSites> ts =
 | 
| +      TopSitesFactory::GetForProfile(Profile::FromWebUI(web_ui()));
 | 
| +  if (ts) {
 | 
| +    ts->GetMostVisitedURLs(
 | 
| +        base::Bind(&MostVisitedHandler::OnMostVisitedUrlsAvailable,
 | 
| +                   weak_ptr_factory_.GetWeakPtr()), false);
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleBlacklistUrl(const base::ListValue* args) {
 | 
| +  std::string url = base::UTF16ToUTF8(ExtractStringValue(args));
 | 
| +  BlacklistUrl(GURL(url));
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleRemoveUrlsFromBlacklist(
 | 
| +    const base::ListValue* args) {
 | 
| +  DCHECK(args->GetSize() != 0);
 | 
| +
 | 
| +  for (base::ListValue::const_iterator iter = args->begin();
 | 
| +       iter != args->end(); ++iter) {
 | 
| +    std::string url;
 | 
| +    bool r = (*iter)->GetAsString(&url);
 | 
| +    if (!r) {
 | 
| +      NOTREACHED();
 | 
| +      return;
 | 
| +    }
 | 
| +    content::RecordAction(UserMetricsAction("MostVisited_UrlRemoved"));
 | 
| +    scoped_refptr<history::TopSites> ts =
 | 
| +        TopSitesFactory::GetForProfile(Profile::FromWebUI(web_ui()));
 | 
| +    if (ts)
 | 
| +      ts->RemoveBlacklistedURL(GURL(url));
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleClearBlacklist(const base::ListValue* args) {
 | 
| +  content::RecordAction(UserMetricsAction("MostVisited_BlacklistCleared"));
 | 
| +
 | 
| +  scoped_refptr<history::TopSites> ts =
 | 
| +      TopSitesFactory::GetForProfile(Profile::FromWebUI(web_ui()));
 | 
| +  if (ts)
 | 
| +    ts->ClearBlacklistedURLs();
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleMostVisitedAction(const base::ListValue* args) {
 | 
| +  DCHECK(args);
 | 
| +
 | 
| +  double action_id;
 | 
| +  if (!args->GetDouble(0, &action_id))
 | 
| +    NOTREACHED();
 | 
| +
 | 
| +  UMA_HISTOGRAM_ENUMERATION("NewTabPage.MostVisitedAction",
 | 
| +                            static_cast<int>(action_id),
 | 
| +                            NUM_NTP_FOLLOW_ACTIONS);
 | 
| +  most_visited_viewed_ = true;
 | 
| +  user_action_logged_ = true;
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::HandleMostVisitedSelected(
 | 
| +    const base::ListValue* args) {
 | 
| +  most_visited_viewed_ = true;
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::SetPagesValueFromTopSites(
 | 
| +    const history::MostVisitedURLList& data) {
 | 
| +  pages_value_.reset(new base::ListValue);
 | 
| +
 | 
| +  history::MostVisitedURLList top_sites(data);
 | 
| +  for (size_t i = 0; i < top_sites.size(); i++) {
 | 
| +    const history::MostVisitedURL& url = top_sites[i];
 | 
| +
 | 
| +    // The items which are to be written into |page_value| are also described in
 | 
| +    // chrome/browser/resources/ntp4/new_tab.js in @typedef for PageData. Please
 | 
| +    // update it whenever you add or remove any keys here.
 | 
| +    base::DictionaryValue* page_value = new base::DictionaryValue();
 | 
| +    if (url.url.is_empty()) {
 | 
| +      page_value->SetBoolean("filler", true);
 | 
| +      pages_value_->Append(page_value);
 | 
| +      continue;
 | 
| +    }
 | 
| +
 | 
| +    NewTabUI::SetUrlTitleAndDirection(page_value,
 | 
| +                                      url.title,
 | 
| +                                      url.url);
 | 
| +    pages_value_->Append(page_value);
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::OnMostVisitedUrlsAvailable(
 | 
| +    const history::MostVisitedURLList& data) {
 | 
| +  SetPagesValueFromTopSites(data);
 | 
| +  if (got_first_most_visited_request_) {
 | 
| +    SendPagesValue();
 | 
| +  }
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::TopSitesLoaded(history::TopSites* top_sites) {
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::TopSitesChanged(history::TopSites* top_sites) {
 | 
| +  // Most visited urls changed, query again.
 | 
| +  StartQueryForMostVisited();
 | 
| +}
 | 
| +
 | 
| +void MostVisitedHandler::BlacklistUrl(const GURL& url) {
 | 
| +  scoped_refptr<history::TopSites> ts =
 | 
| +      TopSitesFactory::GetForProfile(Profile::FromWebUI(web_ui()));
 | 
| +  if (ts)
 | 
| +    ts->AddBlacklistedURL(url);
 | 
| +  content::RecordAction(UserMetricsAction("MostVisited_UrlBlacklisted"));
 | 
| +}
 | 
| +
 | 
| +std::string MostVisitedHandler::GetDictionaryKeyForUrl(const std::string& url) {
 | 
| +  return base::MD5String(url);
 | 
| +}
 | 
| +
 | 
| +// static
 | 
| +void MostVisitedHandler::RegisterProfilePrefs(
 | 
| +    user_prefs::PrefRegistrySyncable* registry) {
 | 
| +  registry->RegisterDictionaryPref(prefs::kNtpMostVisitedURLsBlacklist);
 | 
| +}
 | 
| 
 |