| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/prerender/prerender_tab_helper.h" | 5 #include "chrome/browser/prerender/prerender_tab_helper.h" |
| 6 | 6 |
| 7 #include "base/bind.h" | 7 #include "base/bind.h" |
| 8 #include "base/metrics/histogram.h" | 8 #include "base/metrics/histogram.h" |
| 9 #include "base/time/time.h" | 9 #include "base/time/time.h" |
| 10 #include "chrome/browser/password_manager/chrome_password_manager_client.h" | |
| 11 #include "chrome/browser/prerender/prerender_histograms.h" | 10 #include "chrome/browser/prerender/prerender_histograms.h" |
| 12 #include "chrome/browser/prerender/prerender_local_predictor.h" | |
| 13 #include "chrome/browser/prerender/prerender_manager.h" | 11 #include "chrome/browser/prerender/prerender_manager.h" |
| 14 #include "chrome/browser/prerender/prerender_manager_factory.h" | 12 #include "chrome/browser/prerender/prerender_manager_factory.h" |
| 15 #include "chrome/browser/profiles/profile.h" | 13 #include "chrome/browser/profiles/profile.h" |
| 16 #include "components/password_manager/core/browser/password_manager.h" | |
| 17 #include "content/public/browser/navigation_details.h" | |
| 18 #include "content/public/browser/navigation_entry.h" | |
| 19 #include "content/public/browser/render_frame_host.h" | 14 #include "content/public/browser/render_frame_host.h" |
| 20 #include "content/public/browser/resource_request_details.h" | 15 #include "content/public/browser/resource_request_details.h" |
| 21 #include "content/public/browser/web_contents.h" | 16 #include "content/public/browser/web_contents.h" |
| 22 #include "content/public/common/frame_navigate_params.h" | |
| 23 | 17 |
| 24 using content::WebContents; | 18 using content::WebContents; |
| 25 | 19 |
| 26 DEFINE_WEB_CONTENTS_USER_DATA_KEY(prerender::PrerenderTabHelper); | 20 DEFINE_WEB_CONTENTS_USER_DATA_KEY(prerender::PrerenderTabHelper); |
| 27 | 21 |
| 28 namespace prerender { | 22 namespace prerender { |
| 29 | 23 |
| 30 namespace { | |
| 31 | |
| 32 void ReportTabHelperURLSeenToLocalPredictor( | |
| 33 PrerenderManager* prerender_manager, | |
| 34 const GURL& url, | |
| 35 WebContents* web_contents) { | |
| 36 if (!prerender_manager) | |
| 37 return; | |
| 38 PrerenderLocalPredictor* local_predictor = | |
| 39 prerender_manager->local_predictor(); | |
| 40 if (!local_predictor) | |
| 41 return; | |
| 42 local_predictor->OnTabHelperURLSeen(url, web_contents); | |
| 43 } | |
| 44 | |
| 45 } // namespace | |
| 46 | |
| 47 PrerenderTabHelper::PrerenderTabHelper(content::WebContents* web_contents) | 24 PrerenderTabHelper::PrerenderTabHelper(content::WebContents* web_contents) |
| 48 : content::WebContentsObserver(web_contents), | 25 : content::WebContentsObserver(web_contents), |
| 49 origin_(ORIGIN_NONE), | 26 origin_(ORIGIN_NONE), |
| 50 next_load_is_control_prerender_(false), | 27 next_load_is_control_prerender_(false), |
| 51 next_load_origin_(ORIGIN_NONE), | 28 next_load_origin_(ORIGIN_NONE), |
| 52 weak_factory_(this) { | 29 weak_factory_(this) { |
| 53 ChromePasswordManagerClient* client = | |
| 54 ChromePasswordManagerClient::FromWebContents(web_contents); | |
| 55 // May be NULL during testing. | |
| 56 if (client) { | |
| 57 client->GetPasswordManager()->AddSubmissionCallback(base::Bind( | |
| 58 &PrerenderTabHelper::PasswordSubmitted, weak_factory_.GetWeakPtr())); | |
| 59 } | |
| 60 | |
| 61 // Determine if this is a prerender. | 30 // Determine if this is a prerender. |
| 62 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | 31 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); |
| 63 if (prerender_manager && | 32 if (prerender_manager && |
| 64 prerender_manager->IsWebContentsPrerendering(web_contents, &origin_)) { | 33 prerender_manager->IsWebContentsPrerendering(web_contents, &origin_)) { |
| 65 navigation_type_ = NAVIGATION_TYPE_PRERENDERED; | 34 navigation_type_ = NAVIGATION_TYPE_PRERENDERED; |
| 66 } else { | 35 } else { |
| 67 navigation_type_ = NAVIGATION_TYPE_NORMAL; | 36 navigation_type_ = NAVIGATION_TYPE_NORMAL; |
| 68 } | 37 } |
| 69 } | 38 } |
| 70 | 39 |
| 71 PrerenderTabHelper::~PrerenderTabHelper() { | 40 PrerenderTabHelper::~PrerenderTabHelper() { |
| 72 } | 41 } |
| 73 | 42 |
| 74 void PrerenderTabHelper::DidGetRedirectForResourceRequest( | 43 void PrerenderTabHelper::DidGetRedirectForResourceRequest( |
| 75 content::RenderFrameHost* render_frame_host, | 44 content::RenderFrameHost* render_frame_host, |
| 76 const content::ResourceRedirectDetails& details) { | 45 const content::ResourceRedirectDetails& details) { |
| 77 if (details.resource_type != content::RESOURCE_TYPE_MAIN_FRAME) | 46 if (details.resource_type != content::RESOURCE_TYPE_MAIN_FRAME) |
| 78 return; | 47 return; |
| 79 | 48 |
| 80 MainFrameUrlDidChange(details.new_url); | 49 MainFrameUrlDidChange(details.new_url); |
| 81 } | 50 } |
| 82 | 51 |
| 83 void PrerenderTabHelper::DidCommitProvisionalLoadForFrame( | 52 void PrerenderTabHelper::DidCommitProvisionalLoadForFrame( |
| 84 content::RenderFrameHost* render_frame_host, | 53 content::RenderFrameHost* render_frame_host, |
| 85 const GURL& validated_url, | 54 const GURL& validated_url, |
| 86 ui::PageTransition transition_type) { | 55 ui::PageTransition transition_type) { |
| 87 if (render_frame_host->GetParent()) | 56 if (render_frame_host->GetParent()) |
| 88 return; | 57 return; |
| 89 RecordEvent(EVENT_MAINFRAME_COMMIT); | |
| 90 RecordEventIfLoggedInURL(EVENT_MAINFRAME_COMMIT_DOMAIN_LOGGED_IN, | |
| 91 validated_url); | |
| 92 url_ = validated_url; | 58 url_ = validated_url; |
| 93 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | 59 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); |
| 94 if (!prerender_manager) | 60 if (!prerender_manager) |
| 95 return; | 61 return; |
| 96 if (prerender_manager->IsWebContentsPrerendering(web_contents(), NULL)) | 62 if (prerender_manager->IsWebContentsPrerendering(web_contents(), NULL)) |
| 97 return; | 63 return; |
| 98 prerender_manager->RecordNavigation(validated_url); | 64 prerender_manager->RecordNavigation(validated_url); |
| 99 ReportTabHelperURLSeenToLocalPredictor(prerender_manager, validated_url, | |
| 100 web_contents()); | |
| 101 } | 65 } |
| 102 | 66 |
| 103 void PrerenderTabHelper::DidStopLoading() { | 67 void PrerenderTabHelper::DidStopLoading() { |
| 104 // Compute the PPLT metric and report it in a histogram, if needed. If the | 68 // Compute the PPLT metric and report it in a histogram, if needed. If the |
| 105 // page is still prerendering, record the not swapped in page load time | 69 // page is still prerendering, record the not swapped in page load time |
| 106 // instead. | 70 // instead. |
| 107 if (!pplt_load_start_.is_null()) { | 71 if (!pplt_load_start_.is_null()) { |
| 108 base::TimeTicks now = base::TimeTicks::Now(); | 72 base::TimeTicks now = base::TimeTicks::Now(); |
| 109 if (IsPrerendering()) { | 73 if (IsPrerendering()) { |
| 110 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | 74 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); |
| (...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 156 origin_ = next_load_origin_; | 120 origin_ = next_load_origin_; |
| 157 next_load_is_control_prerender_ = false; | 121 next_load_is_control_prerender_ = false; |
| 158 next_load_origin_ = ORIGIN_NONE; | 122 next_load_origin_ = ORIGIN_NONE; |
| 159 } | 123 } |
| 160 | 124 |
| 161 MainFrameUrlDidChange(validated_url); | 125 MainFrameUrlDidChange(validated_url); |
| 162 } | 126 } |
| 163 | 127 |
| 164 void PrerenderTabHelper::MainFrameUrlDidChange(const GURL& url) { | 128 void PrerenderTabHelper::MainFrameUrlDidChange(const GURL& url) { |
| 165 url_ = url; | 129 url_ = url; |
| 166 RecordEvent(EVENT_MAINFRAME_CHANGE); | |
| 167 RecordEventIfLoggedInURL(EVENT_MAINFRAME_CHANGE_DOMAIN_LOGGED_IN, url); | |
| 168 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | |
| 169 if (!prerender_manager) | |
| 170 return; | |
| 171 if (prerender_manager->IsWebContentsPrerendering(web_contents(), NULL)) | |
| 172 return; | |
| 173 ReportTabHelperURLSeenToLocalPredictor(prerender_manager, url, | |
| 174 web_contents()); | |
| 175 } | |
| 176 | |
| 177 void PrerenderTabHelper::PasswordSubmitted(const autofill::PasswordForm& form) { | |
| 178 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | |
| 179 if (prerender_manager) { | |
| 180 prerender_manager->RecordLikelyLoginOnURL(form.origin); | |
| 181 RecordEvent(EVENT_LOGIN_ACTION_ADDED); | |
| 182 if (form.password_value.empty()) | |
| 183 RecordEvent(EVENT_LOGIN_ACTION_ADDED_PW_EMPTY); | |
| 184 } | |
| 185 } | 130 } |
| 186 | 131 |
| 187 PrerenderManager* PrerenderTabHelper::MaybeGetPrerenderManager() const { | 132 PrerenderManager* PrerenderTabHelper::MaybeGetPrerenderManager() const { |
| 188 return PrerenderManagerFactory::GetForProfile( | 133 return PrerenderManagerFactory::GetForProfile( |
| 189 Profile::FromBrowserContext(web_contents()->GetBrowserContext())); | 134 Profile::FromBrowserContext(web_contents()->GetBrowserContext())); |
| 190 } | 135 } |
| 191 | 136 |
| 192 bool PrerenderTabHelper::IsPrerendering() { | 137 bool PrerenderTabHelper::IsPrerendering() { |
| 193 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | 138 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); |
| 194 if (!prerender_manager) | 139 if (!prerender_manager) |
| (...skipping 15 matching lines...) Expand all Loading... |
| 210 actual_load_start_ = pplt_load_start_; | 155 actual_load_start_ = pplt_load_start_; |
| 211 pplt_load_start_ = base::TimeTicks::Now(); | 156 pplt_load_start_ = base::TimeTicks::Now(); |
| 212 } | 157 } |
| 213 } | 158 } |
| 214 | 159 |
| 215 void PrerenderTabHelper::WouldHavePrerenderedNextLoad(Origin origin) { | 160 void PrerenderTabHelper::WouldHavePrerenderedNextLoad(Origin origin) { |
| 216 next_load_is_control_prerender_ = true; | 161 next_load_is_control_prerender_ = true; |
| 217 next_load_origin_ = origin; | 162 next_load_origin_ = origin; |
| 218 } | 163 } |
| 219 | 164 |
| 220 void PrerenderTabHelper::RecordEvent(PrerenderTabHelper::Event event) const { | |
| 221 UMA_HISTOGRAM_ENUMERATION("Prerender.TabHelperEvent", | |
| 222 event, PrerenderTabHelper::EVENT_MAX_VALUE); | |
| 223 } | |
| 224 | |
| 225 void PrerenderTabHelper::RecordEventIfLoggedInURL( | |
| 226 PrerenderTabHelper::Event event, const GURL& url) { | |
| 227 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | |
| 228 if (!prerender_manager) | |
| 229 return; | |
| 230 scoped_ptr<bool> is_present(new bool); | |
| 231 scoped_ptr<bool> lookup_succeeded(new bool); | |
| 232 bool* is_present_ptr = is_present.get(); | |
| 233 bool* lookup_succeeded_ptr = lookup_succeeded.get(); | |
| 234 prerender_manager->CheckIfLikelyLoggedInOnURL( | |
| 235 url, | |
| 236 is_present_ptr, | |
| 237 lookup_succeeded_ptr, | |
| 238 base::Bind(&PrerenderTabHelper::RecordEventIfLoggedInURLResult, | |
| 239 weak_factory_.GetWeakPtr(), | |
| 240 event, | |
| 241 base::Passed(&is_present), | |
| 242 base::Passed(&lookup_succeeded))); | |
| 243 } | |
| 244 | |
| 245 void PrerenderTabHelper::RecordEventIfLoggedInURLResult( | |
| 246 PrerenderTabHelper::Event event, | |
| 247 scoped_ptr<bool> is_present, | |
| 248 scoped_ptr<bool> lookup_succeeded) { | |
| 249 if (*lookup_succeeded && *is_present) | |
| 250 RecordEvent(event); | |
| 251 } | |
| 252 | |
| 253 void PrerenderTabHelper::RecordPerceivedPageLoadTime( | 165 void PrerenderTabHelper::RecordPerceivedPageLoadTime( |
| 254 base::TimeDelta perceived_page_load_time, | 166 base::TimeDelta perceived_page_load_time, |
| 255 double fraction_plt_elapsed_at_swap_in) { | 167 double fraction_plt_elapsed_at_swap_in) { |
| 256 DCHECK(!IsPrerendering()); | 168 DCHECK(!IsPrerendering()); |
| 257 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); | 169 PrerenderManager* prerender_manager = MaybeGetPrerenderManager(); |
| 258 if (!prerender_manager) | 170 if (!prerender_manager) |
| 259 return; | 171 return; |
| 260 | 172 |
| 261 // Note: it is possible for |next_load_is_control_prerender_| to be true at | 173 // Note: it is possible for |next_load_is_control_prerender_| to be true at |
| 262 // this point. This does not affect the classification of the current load, | 174 // this point. This does not affect the classification of the current load, |
| 263 // but only the next load. (This occurs if a WOULD_HAVE_BEEN_PRERENDERED | 175 // but only the next load. (This occurs if a WOULD_HAVE_BEEN_PRERENDERED |
| 264 // navigation interrupts and aborts another navigation.) | 176 // navigation interrupts and aborts another navigation.) |
| 265 prerender_manager->RecordPerceivedPageLoadTime( | 177 prerender_manager->RecordPerceivedPageLoadTime( |
| 266 origin_, navigation_type_, perceived_page_load_time, | 178 origin_, navigation_type_, perceived_page_load_time, |
| 267 fraction_plt_elapsed_at_swap_in, url_); | 179 fraction_plt_elapsed_at_swap_in, url_); |
| 268 | 180 |
| 269 // Reset state for the next navigation. | 181 // Reset state for the next navigation. |
| 270 navigation_type_ = NAVIGATION_TYPE_NORMAL; | 182 navigation_type_ = NAVIGATION_TYPE_NORMAL; |
| 271 origin_ = ORIGIN_NONE; | 183 origin_ = ORIGIN_NONE; |
| 272 } | 184 } |
| 273 | 185 |
| 274 } // namespace prerender | 186 } // namespace prerender |
| OLD | NEW |