// 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/ui/webui/ntp/ntp_user_data_logger.h" #include "base/metrics/histogram.h" #include "chrome/browser/search/search.h" #include "chrome/common/search_urls.h" #include "chrome/common/url_constants.h" #include "content/public/browser/navigation_details.h" #include "content/public/browser/navigation_entry.h" #include "content/public/browser/web_contents.h" namespace { // Used to track if suggestions were issued by the client or the server. enum SuggestionsType { CLIENT_SIDE = 0, SERVER_SIDE = 1, SUGGESTIONS_TYPE_COUNT = 2 }; } // namespace DEFINE_WEB_CONTENTS_USER_DATA_KEY(NTPUserDataLogger); NTPUserDataLogger::~NTPUserDataLogger() {} // static NTPUserDataLogger* NTPUserDataLogger::GetOrCreateFromWebContents( content::WebContents* content) { // Calling CreateForWebContents when an instance is already attached has no // effect, so we can do this. NTPUserDataLogger::CreateForWebContents(content); NTPUserDataLogger* logger = NTPUserDataLogger::FromWebContents(content); // We record the URL of this NTP in order to identify navigations that // originate from it. We use the NavigationController's URL since it might // differ from the WebContents URL which is usually chrome://newtab/. const content::NavigationEntry* entry = content->GetController().GetVisibleEntry(); if (entry) logger->ntp_url_ = entry->GetURL(); return logger; } void NTPUserDataLogger::EmitThumbnailErrorRate() { DCHECK_LE(number_of_thumbnail_errors_, number_of_thumbnail_attempts_); if (number_of_thumbnail_attempts_ != 0) { UMA_HISTOGRAM_PERCENTAGE( "NewTabPage.ThumbnailErrorRate", GetPercentError(number_of_thumbnail_errors_, number_of_thumbnail_attempts_)); } DCHECK_LE(number_of_fallback_thumbnails_used_, number_of_fallback_thumbnails_requested_); if (number_of_fallback_thumbnails_requested_ != 0) { UMA_HISTOGRAM_PERCENTAGE( "NewTabPage.ThumbnailFallbackRate", GetPercentError(number_of_fallback_thumbnails_used_, number_of_fallback_thumbnails_requested_)); } number_of_thumbnail_attempts_ = 0; number_of_thumbnail_errors_ = 0; number_of_fallback_thumbnails_requested_ = 0; number_of_fallback_thumbnails_used_ = 0; } void NTPUserDataLogger::EmitNtpStatistics() { UMA_HISTOGRAM_COUNTS("NewTabPage.NumberOfMouseOvers", number_of_mouseovers_); number_of_mouseovers_ = 0; UMA_HISTOGRAM_COUNTS("NewTabPage.NumberOfExternalTiles", number_of_external_tiles_); number_of_external_tiles_ = 0; UMA_HISTOGRAM_ENUMERATION( "NewTabPage.SuggestionsType", server_side_suggestions_ ? SERVER_SIDE : CLIENT_SIDE, SUGGESTIONS_TYPE_COUNT); server_side_suggestions_ = false; } void NTPUserDataLogger::LogEvent(NTPLoggingEventType event) { switch (event) { case NTP_MOUSEOVER: number_of_mouseovers_++; break; case NTP_THUMBNAIL_ATTEMPT: number_of_thumbnail_attempts_++; break; case NTP_THUMBNAIL_ERROR: number_of_thumbnail_errors_++; break; case NTP_FALLBACK_THUMBNAIL_REQUESTED: number_of_fallback_thumbnails_requested_++; break; case NTP_FALLBACK_THUMBNAIL_USED: number_of_fallback_thumbnails_used_++; break; case NTP_SERVER_SIDE_SUGGESTION: server_side_suggestions_ = true; break; case NTP_CLIENT_SIDE_SUGGESTION: // We should never get a mix of server and client side suggestions, // otherwise there could be a race condition depending on the order in // which the iframes call this method. DCHECK(!server_side_suggestions_); break; case NTP_EXTERNAL_TILE: number_of_external_tiles_++; break; default: NOTREACHED(); } } // content::WebContentsObserver override void NTPUserDataLogger::NavigationEntryCommitted( const content::LoadCommittedDetails& load_details) { if (!load_details.previous_url.is_valid()) return; if (search::MatchesOriginAndPath(ntp_url_, load_details.previous_url)) { EmitNtpStatistics(); // Only log thumbnail error rates for Instant NTP pages, as we do not have // this data for non-Instant NTPs. if (ntp_url_ != GURL(chrome::kChromeUINewTabURL)) EmitThumbnailErrorRate(); } } NTPUserDataLogger::NTPUserDataLogger(content::WebContents* contents) : content::WebContentsObserver(contents), number_of_mouseovers_(0), number_of_thumbnail_attempts_(0), number_of_thumbnail_errors_(0), number_of_fallback_thumbnails_requested_(0), number_of_fallback_thumbnails_used_(0), number_of_external_tiles_(0), server_side_suggestions_(false) { } size_t NTPUserDataLogger::GetPercentError(size_t errors, size_t events) const { return (100 * errors) / events; }