1 // Copyright (c) 2013 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/active_tab_tracker.h" 6 7 #include "chrome/browser/history/history_service.h" 8 #include "chrome/browser/history/history_service_factory.h" 9 #include "chrome/browser/profiles/profile.h" 10 #include "chrome/browser/ui/browser.h" 11 #include "chrome/browser/ui/browser_list.h" 12 #include "chrome/browser/ui/tabs/tab_strip_model.h" 13 #include "content/public/browser/navigation_entry.h" 14 #include "content/public/browser/notification_source.h" 15 #include "content/public/browser/notification_types.h" 16 #include "content/public/browser/web_contents.h" 17 18 namespace { 19 20 // Amount of time a page has to be active before we commit it. 21 const int kTimeBeforeCommitMS = 1000; 22 23 // Number of seconds input should be received before considered idle. 24 const int kIdleTimeSeconds = 30; 25 26 } // namespace 27 28 #if !defined(OS_WIN) && !defined(USE_AURA) 29 // static 30 NativeFocusTracker* NativeFocusTracker::Create(NativeFocusTrackerHost* host) { 31 return NULL; 32 } 33 #endif 34 35 ActiveTabTracker::ActiveTabTracker() 36 : browser_(NULL), 37 web_contents_(NULL), 38 idle_state_(IDLE_STATE_UNKNOWN), 39 timer_(false, false), 40 weak_ptr_factory_(this) { 41 native_focus_tracker_.reset(NativeFocusTracker::Create(this)); 42 Browser* browser = 43 BrowserList::GetInstance(chrome::GetActiveDesktop())->GetLastActive(); 44 SetBrowser(browser); 45 BrowserList::AddObserver(this); 46 } 47 48 ActiveTabTracker::~ActiveTabTracker() { 49 native_focus_tracker_.reset(); 50 SetBrowser(NULL); 51 BrowserList::RemoveObserver(this); 52 } 53 54 void ActiveTabTracker::ActiveTabChanged(content::WebContents* old_contents, 55 content::WebContents* new_contents, 56 int index, 57 int reason) { 58 SetWebContents(new_contents); 59 } 60 61 void ActiveTabTracker::TabReplacedAt(TabStripModel* tab_strip_model, 62 content::WebContents* old_contents, 63 content::WebContents* new_contents, 64 int index) { 65 if (index == tab_strip_model->selection_model().active()) 66 SetWebContents(new_contents); 67 } 68 69 void ActiveTabTracker::TabStripEmpty() { 70 SetBrowser(NULL); 71 } 72 73 void ActiveTabTracker::OnBrowserRemoved(Browser* browser) { 74 if (browser == browser_) 75 SetBrowser(NULL); 76 } 77 78 void ActiveTabTracker::Observe(int type, 79 const content::NotificationSource& source, 80 const content::NotificationDetails& details) { 81 const GURL url = GetURLFromWebContents(); 82 if (url == url_) 83 return; 84 85 CommitActiveTime(); 86 url_ = url; 87 } 88 89 void ActiveTabTracker::SetBrowser(Browser* browser) { 90 if (browser_ == browser) 91 return; 92 93 CommitActiveTime(); 94 if (browser_) 95 browser_->tab_strip_model()->RemoveObserver(this); 96 // Don't track anything for otr profiles. 97 if (browser && browser->profile()->IsOffTheRecord()) 98 browser = NULL; 99 browser_ = browser; 100 content::WebContents* web_contents = NULL; 101 if (browser_) { 102 TabStripModel* tab_strip = browser_->tab_strip_model(); 103 tab_strip->AddObserver(this); 104 web_contents = tab_strip->GetActiveWebContents(); 105 } else { 106 idle_state_ = IDLE_STATE_UNKNOWN; 107 timer_.Stop(); 108 weak_ptr_factory_.InvalidateWeakPtrs(); 109 } 110 SetWebContents(web_contents); 111 } 112 113 void ActiveTabTracker::SetWebContents(content::WebContents* web_contents) { 114 if (web_contents_ == web_contents) 115 return; 116 117 CommitActiveTime(); 118 119 active_time_ = base::TimeTicks::Now(); 120 web_contents_ = web_contents; 121 url_ = GetURLFromWebContents(); 122 registrar_.RemoveAll(); 123 // TODO(sky): this isn't quite right. We should really not include transient 124 // entries here. For that we need to make Browser::NavigationStateChanged() 125 // call through to this class. 126 if (web_contents_) { 127 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_COMMITTED, 128 content::Source<content::NavigationController>( 129 &web_contents_->GetController())); 130 QueryIdleState(); 131 } 132 } 133 134 void ActiveTabTracker::SetIdleState(IdleState idle_state) { 135 if (idle_state_ != idle_state) { 136 if (idle_state_ == IDLE_STATE_ACTIVE) 137 CommitActiveTime(); 138 else if (idle_state == IDLE_STATE_ACTIVE) 139 active_time_ = base::TimeTicks::Now(); 140 idle_state_ = idle_state; 141 } 142 if (browser_) { 143 timer_.Start(FROM_HERE, 144 base::TimeDelta::FromSeconds(kIdleTimeSeconds), 145 base::Bind(&ActiveTabTracker::QueryIdleState, 146 base::Unretained(this))); 147 } 148 } 149 150 void ActiveTabTracker::QueryIdleState() { 151 if (weak_ptr_factory_.HasWeakPtrs()) 152 return; 153 154 CalculateIdleState(kIdleTimeSeconds, 155 base::Bind(&ActiveTabTracker::SetIdleState, 156 weak_ptr_factory_.GetWeakPtr())); 157 } 158 159 GURL ActiveTabTracker::GetURLFromWebContents() const { 160 if (!web_contents_) 161 return GURL(); 162 163 // TODO: handle subframe transitions better. Maybe go back to first entry 164 // that isn't a main frame? 165 content::NavigationEntry* entry = 166 web_contents_->GetController().GetLastCommittedEntry(); 167 if (!entry || !PageTransitionIsMainFrame(entry->GetTransitionType())) 168 return GURL(); 169 return !entry->GetUserTypedURL().is_empty() ? 170 entry->GetUserTypedURL() : entry->GetURL(); 171 } 172 173 void ActiveTabTracker::CommitActiveTime() { 174 const base::TimeDelta active_delta = base::TimeTicks::Now() - active_time_; 175 active_time_ = base::TimeTicks::Now(); 176 177 if (!web_contents_ || url_.is_empty() || idle_state_ != IDLE_STATE_ACTIVE || 178 active_delta.InMilliseconds() < kTimeBeforeCommitMS) 179 return; 180 181 Profile* profile = Profile::FromBrowserContext( 182 web_contents_->GetBrowserContext()); 183 HistoryService* history = HistoryServiceFactory::GetForProfile( 184 profile, Profile::EXPLICIT_ACCESS); 185 if (history) 186 history->IncreaseSegmentDuration(url_, base::Time::Now(), active_delta); 187 } 188