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( | |
sclittle
2015/11/19 21:53:24
Maybe this class isn't necessary. What if DataUseT
tbansal1
2015/11/20 21:34:05
Done.
| |
31 const base::WeakPtr<DataUseUITabModel>& data_use_ui_tab_model, | |
32 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner, | |
33 const base::WeakPtr<DataUseTabModel>& data_use_tab_model); | |
34 | |
35 private: | |
36 friend class DataUseUITabModelTest; | |
sclittle
2015/11/19 21:53:24
This friending does nothing, so remove it.
tbansal1
2015/11/20 21:34:05
Done.
| |
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 | |
19 DataUseUITabModel::DataUseUITabModel( | 63 DataUseUITabModel::DataUseUITabModel( |
20 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner) | 64 scoped_refptr<base::SingleThreadTaskRunner> io_task_runner) |
21 : io_task_runner_(io_task_runner) { | 65 : io_task_runner_(io_task_runner), weak_factory_(this) { |
66 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
22 DCHECK(io_task_runner_); | 67 DCHECK(io_task_runner_); |
23 } | 68 } |
24 | 69 |
25 DataUseUITabModel::~DataUseUITabModel() {} | 70 DataUseUITabModel::~DataUseUITabModel() { |
71 if (tab_data_use_observer_) { | |
72 DCHECK(thread_checker_.CalledOnValidThread()); | |
73 tab_data_use_observer_->AddRef(); | |
74 | |
75 // Removes the last reference to |tab_observer|, so that it is destroyed on | |
76 // IO thread. | |
77 io_task_runner_->ReleaseSoon(FROM_HERE, tab_data_use_observer_.get()); | |
sclittle
2015/11/19 21:53:24
I don't think this is safe - ReleaseSoon could fai
tbansal1
2015/11/20 21:34:05
Done.
| |
78 tab_data_use_observer_ = nullptr; | |
79 } | |
80 } | |
26 | 81 |
27 void DataUseUITabModel::ReportBrowserNavigation( | 82 void DataUseUITabModel::ReportBrowserNavigation( |
28 const GURL& gurl, | 83 const GURL& gurl, |
29 ui::PageTransition page_transition, | 84 ui::PageTransition page_transition, |
30 int32_t tab_id) const { | 85 int32_t tab_id) const { |
31 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
32 DCHECK(thread_checker_.CalledOnValidThread()); | 86 DCHECK(thread_checker_.CalledOnValidThread()); |
87 DCHECK_LE(0, tab_id); | |
33 | 88 |
34 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 89 DataUseTabModel::TransitionType transition_type; |
90 | |
91 if (ConvertTransitionType(page_transition, &transition_type)) { | |
92 io_task_runner_->PostTask( | |
93 FROM_HERE, | |
94 base::Bind(&DataUseTabModel::OnNavigationEvent, io_data_use_tab_model_, | |
95 tab_id, transition_type, gurl)); | |
96 } | |
35 } | 97 } |
36 | 98 |
37 void DataUseUITabModel::ReportTabClosure(int32_t tab_id) { | 99 void DataUseUITabModel::ReportTabClosure(int32_t tab_id) { |
38 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
39 DCHECK(thread_checker_.CalledOnValidThread()); | 100 DCHECK(thread_checker_.CalledOnValidThread()); |
101 DCHECK_LE(0, tab_id); | |
40 | 102 |
41 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 103 io_task_runner_->PostTask(FROM_HERE, |
104 base::Bind(&DataUseTabModel::OnTabCloseEvent, | |
105 io_data_use_tab_model_, tab_id)); | |
42 | 106 |
43 // Clear out local state. | 107 // Clear out local state. |
44 TabEvents::iterator it = tab_events_.find(tab_id); | 108 TabEvents::iterator it = tab_events_.find(tab_id); |
45 if (it == tab_events_.end()) | 109 if (it == tab_events_.end()) |
46 return; | 110 return; |
47 tab_events_.erase(it); | 111 tab_events_.erase(it); |
48 } | 112 } |
49 | 113 |
50 void DataUseUITabModel::ReportCustomTabInitialNavigation( | 114 void DataUseUITabModel::ReportCustomTabInitialNavigation( |
51 int32_t tab_id, | 115 int32_t tab_id, |
52 const std::string& url, | 116 const std::string& url, |
53 const std::string& package_name) { | 117 const std::string& package_name) { |
118 DCHECK(thread_checker_.CalledOnValidThread()); | |
54 // TODO(tbansal): Post to DataUseTabModel on IO thread. | 119 // TODO(tbansal): Post to DataUseTabModel on IO thread. |
55 } | 120 } |
56 | 121 |
57 void DataUseUITabModel::OnTrackingStarted(int32_t tab_id) { | 122 void DataUseUITabModel::SetIODataUseTabModel( |
58 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | 123 const base::WeakPtr<DataUseTabModel>& io_data_use_tab_model) { |
124 DCHECK(thread_checker_.CalledOnValidThread()); | |
125 DCHECK(!io_data_use_tab_model_); | |
126 | |
127 io_data_use_tab_model_ = io_data_use_tab_model; | |
128 | |
129 base::PostTaskAndReplyWithResult( | |
130 io_task_runner_.get(), FROM_HERE, | |
131 base::Bind(&CreateTabObserverOnIOThread, io_data_use_tab_model_, | |
132 weak_factory_.GetWeakPtr(), | |
133 base::ThreadTaskRunnerHandle::Get()), | |
134 base::Bind(&DataUseUITabModel::SetTabDataUseObserver, | |
135 weak_factory_.GetWeakPtr())); | |
136 } | |
137 | |
138 void DataUseUITabModel::NotifyTrackingStarting(int32_t tab_id) { | |
59 DCHECK(thread_checker_.CalledOnValidThread()); | 139 DCHECK(thread_checker_.CalledOnValidThread()); |
60 | 140 |
61 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_STARTED)) | 141 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_STARTED)) |
62 return; | 142 return; |
63 // Since tracking started before the UI could indicate that it ended, it is | 143 // Since tracking started before the UI could indicate that it ended, it is |
64 // not useful for UI to show that it started again. | 144 // not useful for UI to show that it started again. |
65 RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); | 145 RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); |
66 } | 146 } |
67 | 147 |
68 void DataUseUITabModel::OnTrackingEnded(int32_t tab_id) { | 148 void DataUseUITabModel::NotifyTrackingEnding(int32_t tab_id) { |
69 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
70 DCHECK(thread_checker_.CalledOnValidThread()); | 149 DCHECK(thread_checker_.CalledOnValidThread()); |
71 | 150 |
72 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_ENDED)) | 151 if (MaybeCreateTabEvent(tab_id, DATA_USE_TRACKING_ENDED)) |
73 return; | 152 return; |
74 // Since tracking ended before the UI could indicate that it stated, it is not | 153 // Since tracking ended before the UI could indicate that it stated, it is not |
75 // useful for UI to show that it ended. | 154 // useful for UI to show that it ended. |
76 RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); | 155 RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); |
77 } | 156 } |
78 | 157 |
79 bool DataUseUITabModel::HasDataUseTrackingStarted(int32_t tab_id) { | 158 bool DataUseUITabModel::HasDataUseTrackingStarted(int32_t tab_id) { |
80 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
81 DCHECK(thread_checker_.CalledOnValidThread()); | 159 DCHECK(thread_checker_.CalledOnValidThread()); |
82 | 160 |
83 TabEvents::iterator it = tab_events_.find(tab_id); | 161 TabEvents::iterator it = tab_events_.find(tab_id); |
84 if (it == tab_events_.end()) | 162 if (it == tab_events_.end()) |
85 return false; | 163 return false; |
86 | 164 |
87 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); | 165 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_STARTED); |
88 } | 166 } |
89 | 167 |
90 bool DataUseUITabModel::HasDataUseTrackingEnded(int32_t tab_id) { | 168 bool DataUseUITabModel::HasDataUseTrackingEnded(int32_t tab_id) { |
91 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
92 DCHECK(thread_checker_.CalledOnValidThread()); | 169 DCHECK(thread_checker_.CalledOnValidThread()); |
93 | 170 |
94 TabEvents::iterator it = tab_events_.find(tab_id); | 171 TabEvents::iterator it = tab_events_.find(tab_id); |
95 if (it == tab_events_.end()) | 172 if (it == tab_events_.end()) |
96 return false; | 173 return false; |
97 | 174 |
98 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); | 175 return RemoveTabEvent(tab_id, DATA_USE_TRACKING_ENDED); |
99 } | 176 } |
100 | 177 |
178 // static | |
179 scoped_refptr<DataUseUITabModel::TabObserverOnIOThread> | |
180 DataUseUITabModel::CreateTabObserverOnIOThread( | |
181 const base::WeakPtr<DataUseTabModel>& data_use_tab_model, | |
182 const base::WeakPtr<DataUseUITabModel>& data_use_ui_tab_model, | |
183 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner) { | |
184 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); | |
185 | |
186 scoped_refptr<DataUseUITabModel::TabObserverOnIOThread> tab_observer( | |
187 new DataUseUITabModel::TabObserverOnIOThread( | |
188 data_use_ui_tab_model, ui_task_runner, data_use_tab_model)); | |
189 return tab_observer; | |
190 } | |
191 | |
192 void DataUseUITabModel::SetTabDataUseObserver( | |
193 scoped_refptr<TabObserverOnIOThread> tab_data_use_observer) { | |
194 DCHECK(thread_checker_.CalledOnValidThread()); | |
195 DCHECK(!tab_data_use_observer_); | |
196 tab_data_use_observer_ = tab_data_use_observer; | |
197 DCHECK(tab_data_use_observer_); | |
198 } | |
199 | |
101 bool DataUseUITabModel::MaybeCreateTabEvent(int32_t tab_id, | 200 bool DataUseUITabModel::MaybeCreateTabEvent(int32_t tab_id, |
102 DataUseTrackingEvent event) { | 201 DataUseTrackingEvent event) { |
103 TabEvents::iterator it = tab_events_.find(tab_id); | 202 DCHECK(thread_checker_.CalledOnValidThread()); |
104 if (it == tab_events_.end()) { | 203 return tab_events_.insert(std::make_pair(tab_id, event)).second; |
105 tab_events_.insert(std::make_pair(tab_id, event)); | |
106 return true; | |
107 } | |
108 return false; | |
109 } | 204 } |
110 | 205 |
111 bool DataUseUITabModel::RemoveTabEvent(int32_t tab_id, | 206 bool DataUseUITabModel::RemoveTabEvent(int32_t tab_id, |
112 DataUseTrackingEvent event) { | 207 DataUseTrackingEvent event) { |
208 DCHECK(thread_checker_.CalledOnValidThread()); | |
113 TabEvents::iterator it = tab_events_.find(tab_id); | 209 TabEvents::iterator it = tab_events_.find(tab_id); |
114 DCHECK(it != tab_events_.end()); | 210 DCHECK(it != tab_events_.end()); |
115 if (it->second == event) { | 211 if (it->second == event) { |
116 tab_events_.erase(it); | 212 tab_events_.erase(it); |
117 return true; | 213 return true; |
118 } | 214 } |
119 return false; | 215 return false; |
120 } | 216 } |
121 | 217 |
218 bool DataUseUITabModel::ConvertTransitionType( | |
219 ui::PageTransition page_transition, | |
220 DataUseTabModel::TransitionType* transition_type) const { | |
221 if (!ui::PageTransitionIsValidType(page_transition) || | |
222 !ui::PageTransitionIsMainFrame(page_transition) || | |
223 !ui::PageTransitionIsNewNavigation(page_transition)) { | |
224 return false; | |
225 } | |
226 | |
227 const int32_t mask = 0xFFFFFFFF ^ ui::PAGE_TRANSITION_QUALIFIER_MASK; | |
228 | |
229 switch (page_transition & mask) { | |
230 case ui::PAGE_TRANSITION_LINK: | |
231 if ((page_transition & ui::PAGE_TRANSITION_FROM_API) != 0) { | |
232 // Clicking on bookmarks. | |
233 *transition_type = DataUseTabModel::TRANSITION_BOOKMARK; | |
234 return true; | |
235 } | |
236 return false; // Newtab, clicking on a link. | |
237 case ui::PAGE_TRANSITION_TYPED: | |
238 *transition_type = DataUseTabModel::TRANSITION_OMNIBOX_NAVIGATION; | |
239 return true; | |
240 case ui::PAGE_TRANSITION_AUTO_BOOKMARK: | |
241 // Auto bookmark from newtab page. | |
242 *transition_type = DataUseTabModel::TRANSITION_BOOKMARK; | |
243 return true; | |
244 case ui::PAGE_TRANSITION_AUTO_TOPLEVEL: | |
245 // History menu. | |
246 *transition_type = DataUseTabModel::TRANSITION_HISTORY_ITEM; | |
247 return true; | |
248 case ui::PAGE_TRANSITION_GENERATED: | |
249 // Omnibox search (e.g., searching for "tacos"). | |
250 *transition_type = DataUseTabModel::TRANSITION_OMNIBOX_SEARCH; | |
251 return true; | |
252 case ui::PAGE_TRANSITION_RELOAD: | |
253 // Restored tabs. | |
254 return false; | |
255 default: | |
256 return false; | |
257 } | |
258 } | |
259 | |
260 DataUseUITabModel::TabObserverOnIOThread::TabObserverOnIOThread( | |
261 const base::WeakPtr<DataUseUITabModel>& data_use_ui_tab_model, | |
262 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner, | |
263 const base::WeakPtr<DataUseTabModel>& data_use_tab_model) | |
264 : data_use_ui_tab_model_(data_use_ui_tab_model), | |
265 data_use_tab_model_(data_use_tab_model), | |
266 ui_task_runner_(ui_task_runner) { | |
267 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); | |
268 DCHECK(ui_task_runner_); | |
269 if (data_use_tab_model_) | |
270 data_use_tab_model_->AddObserver(this); | |
271 } | |
272 | |
273 DataUseUITabModel::TabObserverOnIOThread::~TabObserverOnIOThread() { | |
274 DCHECK(thread_checker_.CalledOnValidThread()); | |
275 if (data_use_tab_model_) | |
276 data_use_tab_model_->RemoveObserver(this); | |
277 } | |
278 | |
279 void DataUseUITabModel::TabObserverOnIOThread::NotifyTrackingStarting( | |
280 int32_t tab_id) { | |
281 DCHECK(thread_checker_.CalledOnValidThread()); | |
282 | |
283 ui_task_runner_->PostTask( | |
284 FROM_HERE, base::Bind(&DataUseUITabModel::NotifyTrackingStarting, | |
285 data_use_ui_tab_model_, tab_id)); | |
286 } | |
287 | |
288 void DataUseUITabModel::TabObserverOnIOThread::NotifyTrackingEnding( | |
289 int32_t tab_id) { | |
290 DCHECK(thread_checker_.CalledOnValidThread()); | |
291 | |
292 ui_task_runner_->PostTask(FROM_HERE, | |
293 base::Bind(&DataUseUITabModel::NotifyTrackingEnding, | |
294 data_use_ui_tab_model_, tab_id)); | |
295 } | |
296 | |
122 } // namespace android | 297 } // namespace android |
123 | 298 |
124 } // namespace chrome | 299 } // namespace chrome |
OLD | NEW |