Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 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/android/data_usage/data_use_ui_tab_model.h" | 5 #include "chrome/browser/android/data_usage/data_use_ui_tab_model.h" |
| 6 | 6 |
| 7 #include <utility> | 7 #include <utility> |
| 8 | 8 |
| 9 #include "base/logging.h" | 9 #include "base/logging.h" |
| 10 #include "base/memory/ref_counted.h" | 10 #include "base/memory/ref_counted.h" |
| 11 #include "base/single_thread_task_runner.h" | 11 #include "base/thread_task_runner_handle.h" |
| 12 #include "content/public/browser/browser_thread.h" | 12 #include "content/public/browser/browser_thread.h" |
| 13 #include "url/gurl.h" | 13 #include "url/gurl.h" |
| 14 | 14 |
| 15 namespace chrome { | 15 namespace chrome { |
| 16 | 16 |
| 17 namespace android { | 17 namespace android { |
| 18 | 18 |
| 19 // TabObserverOnIOThread registers as an observer to | |
| 20 // DataUseTabModel::TabDataUseObserver, and gets notified when tracking has | |
| 21 // started and ended on a tab. It passes these notifications to | |
| 22 // DataUseUITabModel on UI thread. TabObserverOnIOThread is not thread safe, | |
| 23 // and should only be accessed only on IO thread. | |
| 24 class DataUseUITabModel::TabObserverOnIOThread | |
| 25 : public base::RefCountedThreadSafe<TabObserverOnIOThread>, | |
| 26 public DataUseTabModel::TabDataUseObserver { | |
| 27 public: | |
| 28 // Sets the |data_use_tab_model_|, and registers as an observer to | |
| 29 // |data_use_tab_model_|. Can be called at most once. | |
| 30 TabObserverOnIOThread( | |
| 31 base::WeakPtr<DataUseUITabModel> data_use_ui_tab_model, | |
| 32 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner, | |
| 33 base::WeakPtr<DataUseTabModel> data_use_tab_model); | |
| 34 | |
| 35 private: | |
| 36 friend class DataUseUITabModelTest; | |
| 37 | |
| 38 // Ref counted classes have private destructors to avoid any code deleting | |
| 39 // the object accidentally while there are still references to it. | |
| 40 friend class base::RefCountedThreadSafe<TabObserverOnIOThread>; | |
| 41 | |
| 42 ~TabObserverOnIOThread(); | |
| 43 | |
| 44 // DataUseTabModel::Observer implementation: | |
| 45 void NotifyTrackingStarting(int32_t tab_id) override; | |
| 46 void NotifyTrackingEnding(int32_t tab_id) override; | |
| 47 | |
| 48 // |data_use_ui_tab_model_| is notified of starting and ending of tracking. | |
| 49 // |data_use_ui_tab_model_| should only be used on UI thread. | |
| 50 base::WeakPtr<DataUseUITabModel> data_use_ui_tab_model_; | |
| 51 | |
| 52 // |this| registers as an observer to |data_use_tab_model_|. | |
| 53 base::WeakPtr<DataUseTabModel> data_use_tab_model_; | |
| 54 | |
| 55 // Used to call methods on |data_use_ui_tab_model_|. | |
| 56 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner_; | |
| 57 | |
| 58 base::ThreadChecker thread_checker_; | |
| 59 | |
| 60 DISALLOW_COPY_AND_ASSIGN(TabObserverOnIOThread); | |
| 61 }; | |
| 62 | |
| 63 namespace { | |
| 64 | |
| 65 // Creates TabObserverOnIOThread object and passes |data_use_tab_model| to it on | |
| 66 // IO thread. | |
| 67 scoped_refptr<DataUseUITabModel::TabObserverOnIOThread> | |
| 68 CreateTabObserverOnIOThread( | |
| 69 base::WeakPtr<DataUseTabModel> data_use_tab_model, | |
|
sclittle
2015/11/18 21:42:03
nit: Pass in weak ptrs as const refs here, to avoi
tbansal1
2015/11/19 00:47:07
Done.
| |
| 70 base::WeakPtr<DataUseUITabModel> data_use_ui_tab_model, | |
| 71 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner) { | |
| 72 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); | |
| 73 | |
| 74 scoped_refptr<DataUseUITabModel::TabObserverOnIOThread> tab_observer( | |
| 75 new DataUseUITabModel::TabObserverOnIOThread( | |
| 76 data_use_ui_tab_model, ui_task_runner, data_use_tab_model)); | |
| 77 return tab_observer; | |
| 78 } | |
| 79 | |
| 80 } // namespace | |
| 81 | |
| 19 DataUseUITabModel::DataUseUITabModel( | 82 DataUseUITabModel::DataUseUITabModel( |
| 20 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner) | 83 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner) |
| 21 : io_task_runner_(io_task_runner) { | 84 : io_task_runner_(io_task_runner), |
| 85 ui_task_runner_(base::ThreadTaskRunnerHandle::Get()), | |
|
sclittle
2015/11/18 21:42:03
Is this |ui_task_runner_| necessary? Can't you jus
tbansal1
2015/11/19 00:47:07
Done.
| |
| 86 weak_factory_(this) { | |
| 87 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 22 DCHECK(io_task_runner_); | 88 DCHECK(io_task_runner_); |
| 89 DCHECK(ui_task_runner_); | |
| 23 } | 90 } |
| 24 | 91 |
| 25 DataUseUITabModel::~DataUseUITabModel() {} | 92 DataUseUITabModel::~DataUseUITabModel() { |
| 93 if (tab_data_use_observer_) { | |
| 94 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 95 tab_data_use_observer_->AddRef(); | |
| 96 | |
| 97 // Removes the last reference to |tab_observer|, so that it is destroyed on | |
| 98 // IO thread. | |
| 99 io_task_runner_->ReleaseSoon(FROM_HERE, tab_data_use_observer_.get()); | |
| 100 tab_data_use_observer_ = nullptr; | |
| 101 } | |
| 102 } | |
| 26 | 103 |
| 27 void DataUseUITabModel::ReportBrowserNavigation( | 104 void DataUseUITabModel::ReportBrowserNavigation( |
| 28 const GURL& gurl, | 105 const GURL& gurl, |
| 29 ui::PageTransition page_transition, | 106 ui::PageTransition page_transition, |
| 30 int32_t tab_id) const { | 107 int32_t tab_id) const { |
| 31 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 32 DCHECK(thread_checker_.CalledOnValidThread()); | 108 DCHECK(thread_checker_.CalledOnValidThread()); |
| 109 DCHECK_LE(0, tab_id); | |
| 33 | 110 |
| 34 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 111 DataUseTabModel::TransitionType transition_type; |
| 112 | |
| 113 if (ConvertTransitionType(page_transition, &transition_type)) { | |
| 114 io_task_runner_->PostTask( | |
| 115 FROM_HERE, | |
| 116 base::Bind(&DataUseTabModel::OnNavigationEvent, io_data_use_tab_model_, | |
| 117 tab_id, transition_type, gurl)); | |
| 118 } | |
| 35 } | 119 } |
| 36 | 120 |
| 37 void DataUseUITabModel::ReportTabClosure(int32_t tab_id) { | 121 void DataUseUITabModel::ReportTabClosure(int32_t tab_id) { |
| 38 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 39 DCHECK(thread_checker_.CalledOnValidThread()); | 122 DCHECK(thread_checker_.CalledOnValidThread()); |
| 123 DCHECK_LE(0, tab_id); | |
| 40 | 124 |
| 41 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 125 io_task_runner_->PostTask(FROM_HERE, |
| 126 base::Bind(&DataUseTabModel::OnTabCloseEvent, | |
| 127 io_data_use_tab_model_, tab_id)); | |
| 42 | 128 |
| 43 // Clear out local state. | 129 // Clear out local state. |
| 44 TabEvents::iterator it = tab_events_.find(tab_id); | 130 TabEvents::iterator it = tab_events_.find(tab_id); |
| 45 if (it == tab_events_.end()) | 131 if (it == tab_events_.end()) |
| 46 return; | 132 return; |
| 47 tab_events_.erase(it); | 133 tab_events_.erase(it); |
| 48 } | 134 } |
| 49 | 135 |
| 50 void DataUseUITabModel::ReportCustomTabInitialNavigation( | 136 void DataUseUITabModel::ReportCustomTabInitialNavigation( |
| 51 int32_t tab_id, | 137 int32_t tab_id, |
| 52 const std::string& url, | 138 const std::string& url, |
| 53 const std::string& package_name) { | 139 const std::string& package_name) { |
| 140 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 54 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 141 // TODO(tbansal): Post to DataUseTabModel on IO thread. |
| 55 } | 142 } |
| 56 | 143 |
| 57 void DataUseUITabModel::OnTrackingStarted(int32_t tab_id) { | 144 void DataUseUITabModel::SetIODataUseTabModel( |
| 58 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | 145 base::WeakPtr<DataUseTabModel> io_data_use_tab_model) { |
|
sclittle
2015/11/18 21:42:03
nit: pass in the WeakPtr as a const ref
tbansal1
2015/11/19 00:47:06
Done.
| |
| 146 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 147 | |
| 148 io_data_use_tab_model_ = io_data_use_tab_model; | |
| 149 | |
| 150 base::PostTaskAndReplyWithResult( | |
| 151 io_task_runner_.get(), FROM_HERE, | |
| 152 base::Bind(&CreateTabObserverOnIOThread, io_data_use_tab_model_, | |
| 153 weak_factory_.GetWeakPtr(), ui_task_runner_), | |
|
sclittle
2015/11/18 21:42:03
Could you remove |ui_task_runner_| and just call b
tbansal1
2015/11/19 00:47:06
Done.
| |
| 154 base::Bind(&DataUseUITabModel::SetTabDataUseObserver, | |
| 155 weak_factory_.GetWeakPtr())); | |
| 156 } | |
| 157 | |
| 158 void DataUseUITabModel::NotifyTrackingStarting(int32_t tab_id) { | |
| 59 DCHECK(thread_checker_.CalledOnValidThread()); | 159 DCHECK(thread_checker_.CalledOnValidThread()); |
| 60 | 160 |
| 61 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_STARTED)) | 161 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_STARTED)) |
| 62 return; | 162 return; |
| 63 // Since tracking started before the UI could indicate that it ended, it is | 163 // Since tracking started before the UI could indicate that it ended, it is |
| 64 // not useful for UI to show that it started again. | 164 // not useful for UI to show that it started again. |
| 65 RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); | 165 RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); |
| 66 } | 166 } |
| 67 | 167 |
| 68 void DataUseUITabModel::OnTrackingEnded(int32_t tab_id) { | 168 void DataUseUITabModel::NotifyTrackingEnding(int32_t tab_id) { |
| 69 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 70 DCHECK(thread_checker_.CalledOnValidThread()); | 169 DCHECK(thread_checker_.CalledOnValidThread()); |
| 71 | 170 |
| 72 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_ENDED)) | 171 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_ENDED)) |
| 73 return; | 172 return; |
| 74 // Since tracking ended before the UI could indicate that it stated, it is not | 173 // Since tracking ended before the UI could indicate that it stated, it is not |
| 75 // useful for UI to show that it ended. | 174 // useful for UI to show that it ended. |
| 76 RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); | 175 RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); |
| 77 } | 176 } |
| 78 | 177 |
| 79 bool DataUseUITabModel::HasDataUseTrackingStarted(int32_t tab_id) { | 178 bool DataUseUITabModel::HasDataUseTrackingStarted(int32_t tab_id) { |
| 80 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 81 DCHECK(thread_checker_.CalledOnValidThread()); | 179 DCHECK(thread_checker_.CalledOnValidThread()); |
| 82 | 180 |
| 83 TabEvents::iterator it = tab_events_.find(tab_id); | 181 TabEvents::iterator it = tab_events_.find(tab_id); |
| 84 if (it == tab_events_.end()) | 182 if (it == tab_events_.end()) |
| 85 return false; | 183 return false; |
| 86 | 184 |
| 87 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); | 185 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); |
| 88 } | 186 } |
| 89 | 187 |
| 90 bool DataUseUITabModel::HasDataUseTrackingEnded(int32_t tab_id) { | 188 bool DataUseUITabModel::HasDataUseTrackingEnded(int32_t tab_id) { |
| 91 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
| 92 DCHECK(thread_checker_.CalledOnValidThread()); | 189 DCHECK(thread_checker_.CalledOnValidThread()); |
| 93 | 190 |
| 94 TabEvents::iterator it = tab_events_.find(tab_id); | 191 TabEvents::iterator it = tab_events_.find(tab_id); |
| 95 if (it == tab_events_.end()) | 192 if (it == tab_events_.end()) |
| 96 return false; | 193 return false; |
| 97 | 194 |
| 98 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); | 195 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); |
| 99 } | 196 } |
| 100 | 197 |
| 198 void DataUseUITabModel::SetTabDataUseObserver( | |
| 199 scoped_refptr<TabObserverOnIOThread> tab_data_use_observer) { | |
| 200 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 201 DCHECK(!tab_data_use_observer_); | |
| 202 tab_data_use_observer_ = tab_data_use_observer; | |
| 203 DCHECK(tab_data_use_observer_); | |
| 204 } | |
| 205 | |
| 101 bool DataUseUITabModel::MaybeCreateTabEvent(int32_t tab_id, | 206 bool DataUseUITabModel::MaybeCreateTabEvent(int32_t tab_id, |
| 102 DataUseTrackingEvent event) { | 207 DataUseTrackingEvent event) { |
| 208 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 103 TabEvents::iterator it = tab_events_.find(tab_id); | 209 TabEvents::iterator it = tab_events_.find(tab_id); |
|
sclittle
2015/11/18 21:42:03
nit: this entire method could just be replaced wit
tbansal1
2015/11/19 00:47:07
Done. This is good!
| |
| 104 if (it == tab_events_.end()) { | 210 if (it == tab_events_.end()) { |
| 105 tab_events_.insert(std::make_pair(tab_id, event)); | 211 tab_events_.insert(std::make_pair(tab_id, event)); |
| 106 return true; | 212 return true; |
| 107 } | 213 } |
| 108 return false; | 214 return false; |
| 109 } | 215 } |
| 110 | 216 |
| 111 bool DataUseUITabModel::RemoveTabEvent(int32_t tab_id, | 217 bool DataUseUITabModel::RemoveTabEvent(int32_t tab_id, |
| 112 DataUseTrackingEvent event) { | 218 DataUseTrackingEvent event) { |
| 219 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 113 TabEvents::iterator it = tab_events_.find(tab_id); | 220 TabEvents::iterator it = tab_events_.find(tab_id); |
|
sclittle
2015/11/18 21:42:03
nit: this entire method could just be replaced wit
tbansal1
2015/11/19 00:47:07
Probably not, we match both the Key and Value befo
| |
| 114 DCHECK(it != tab_events_.end()); | 221 DCHECK(it != tab_events_.end()); |
| 115 if (it->second == event) { | 222 if (it->second == event) { |
| 116 tab_events_.erase(it); | 223 tab_events_.erase(it); |
| 117 return true; | 224 return true; |
| 118 } | 225 } |
| 119 return false; | 226 return false; |
| 120 } | 227 } |
| 121 | 228 |
| 229 bool DataUseUITabModel::ConvertTransitionType( | |
| 230 ui::PageTransition page_transition, | |
| 231 DataUseTabModel::TransitionType* transition_type) const { | |
| 232 if (!ui::PageTransitionIsValidType(page_transition) || | |
| 233 !ui::PageTransitionIsMainFrame(page_transition) || | |
| 234 !ui::PageTransitionIsNewNavigation(page_transition)) { | |
| 235 return false; | |
| 236 } | |
| 237 | |
| 238 const int32_t mask = 0xFFFFFFFF ^ ui::PAGE_TRANSITION_QUALIFIER_MASK; | |
| 239 | |
| 240 switch (page_transition & mask) { | |
| 241 case ui::PAGE_TRANSITION_LINK: | |
| 242 if ((page_transition & ui::PAGE_TRANSITION_FROM_API) != 0) { | |
| 243 // Clicking on bookmarks. | |
| 244 *transition_type = DataUseTabModel::TRANSITION_BOOKMARK; | |
| 245 return true; | |
| 246 } | |
| 247 return false; // Newtab, clicking on a link. | |
| 248 case ui::PAGE_TRANSITION_TYPED: | |
| 249 *transition_type = DataUseTabModel::TRANSITION_OMNIBOX_NAVIGATION; | |
| 250 return true; | |
| 251 case ui::PAGE_TRANSITION_AUTO_BOOKMARK: | |
| 252 // Auto bookmark from newtab page. | |
| 253 *transition_type = DataUseTabModel::TRANSITION_BOOKMARK; | |
| 254 return true; | |
| 255 case ui::PAGE_TRANSITION_AUTO_TOPLEVEL: | |
| 256 // History menu. | |
| 257 *transition_type = DataUseTabModel::TRANSITION_HISTORY_ITEM; | |
| 258 return true; | |
| 259 case ui::PAGE_TRANSITION_GENERATED: | |
| 260 // Omnibox search (e.g., searching for "tacos"). | |
| 261 *transition_type = DataUseTabModel::TRANSITION_OMNIBOX_SEARCH; | |
| 262 return true; | |
| 263 case ui::PAGE_TRANSITION_RELOAD: | |
| 264 // Restored tabs. | |
| 265 return false; | |
| 266 default: | |
| 267 return false; | |
| 268 } | |
| 269 } | |
| 270 | |
| 271 DataUseUITabModel::TabObserverOnIOThread::TabObserverOnIOThread( | |
| 272 base::WeakPtr<DataUseUITabModel> data_use_ui_tab_model, | |
|
sclittle
2015/11/18 21:42:03
nit: pass the WeakPtrs as const WeakPtr& to avoid
tbansal1
2015/11/19 00:47:07
Done.
| |
| 273 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner, | |
| 274 base::WeakPtr<DataUseTabModel> data_use_tab_model) | |
| 275 : data_use_ui_tab_model_(data_use_ui_tab_model), | |
| 276 data_use_tab_model_(data_use_tab_model), | |
| 277 ui_task_runner_(ui_task_runner) { | |
| 278 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); | |
| 279 DCHECK(ui_task_runner_); | |
| 280 DCHECK(data_use_tab_model_); | |
|
sclittle
2015/11/18 21:42:03
This DCHECK isn't guaranteed, the tab model could
tbansal1
2015/11/19 00:47:07
Done.
| |
| 281 | |
| 282 data_use_tab_model_->AddObserver(this); | |
| 283 } | |
| 284 | |
| 285 DataUseUITabModel::TabObserverOnIOThread::~TabObserverOnIOThread() { | |
| 286 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 287 if (data_use_tab_model_) | |
| 288 data_use_tab_model_->RemoveObserver(this); | |
| 289 } | |
| 290 | |
| 291 void DataUseUITabModel::TabObserverOnIOThread::NotifyTrackingStarting( | |
| 292 int32_t tab_id) { | |
| 293 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 294 | |
| 295 ui_task_runner_->PostTask( | |
| 296 FROM_HERE, base::Bind(&DataUseUITabModel::NotifyTrackingStarting, | |
| 297 data_use_ui_tab_model_, tab_id)); | |
| 298 } | |
| 299 | |
| 300 void DataUseUITabModel::TabObserverOnIOThread::NotifyTrackingEnding( | |
| 301 int32_t tab_id) { | |
| 302 DCHECK(thread_checker_.CalledOnValidThread()); | |
| 303 | |
| 304 ui_task_runner_->PostTask(FROM_HERE, | |
| 305 base::Bind(&DataUseUITabModel::NotifyTrackingEnding, | |
| 306 data_use_ui_tab_model_, tab_id)); | |
| 307 } | |
| 308 | |
| 122 } // namespace android | 309 } // namespace android |
| 123 | 310 |
| 124 } // namespace chrome | 311 } // namespace chrome |
| OLD | NEW |