Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(23)

Side by Side Diff: content/browser/geolocation/geolocation_provider.cc

Issue 10344004: Add content API for requesting the current geolocation (Closed) Base URL: http://git.chromium.org/chromium/src.git@master
Patch Set: Removed method for mocking geolocation in unit tests for now, as requested by jam@. Created 8 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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 "content/browser/geolocation/geolocation_provider.h" 5 #include "content/browser/geolocation/geolocation_provider.h"
6 6
7 #include "base/bind.h" 7 #include "base/bind.h"
8 #include "base/bind_helpers.h" 8 #include "base/bind_helpers.h"
9 #include "base/callback.h"
10 #include "base/location.h"
11 #include "base/logging.h"
9 #include "base/memory/singleton.h" 12 #include "base/memory/singleton.h"
10 #include "base/threading/thread_restrictions.h" 13 #include "base/message_loop.h"
11 #include "content/browser/geolocation/location_arbitrator.h" 14 #include "content/browser/geolocation/location_arbitrator.h"
15 #include "content/public/browser/browser_thread.h"
12 16
13 GeolocationProvider* GeolocationProvider::GetInstance() { 17 using content::BrowserThread;
14 return Singleton<GeolocationProvider>::get();
15 }
16
17 GeolocationProvider::GeolocationProvider()
18 : base::Thread("Geolocation"),
19 client_loop_(base::MessageLoopProxy::current()),
20 is_permission_granted_(false),
21 ignore_location_updates_(false),
22 arbitrator_(NULL) {
23 }
24
25 GeolocationProvider::~GeolocationProvider() {
26 DCHECK(observers_.empty()); // observers must unregister.
27 Stop();
28 DCHECK(!arbitrator_);
29 }
30 18
31 void GeolocationProvider::AddObserver(GeolocationObserver* observer, 19 void GeolocationProvider::AddObserver(GeolocationObserver* observer,
32 const GeolocationObserverOptions& update_options) { 20 const GeolocationObserverOptions& update_options) {
33 DCHECK(OnClientThread()); 21 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
34 observers_[observer] = update_options; 22 observers_[observer] = update_options;
35 OnObserversChanged(); 23 OnClientsChanged();
36 if (position_.Validate() || 24 if (position_.Validate() ||
37 position_.error_code != content::Geoposition::ERROR_CODE_NONE) 25 position_.error_code != content::Geoposition::ERROR_CODE_NONE)
38 observer->OnLocationUpdate(position_); 26 observer->OnLocationUpdate(position_);
39 } 27 }
40 28
41 bool GeolocationProvider::RemoveObserver(GeolocationObserver* observer) { 29 bool GeolocationProvider::RemoveObserver(GeolocationObserver* observer) {
42 DCHECK(OnClientThread()); 30 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
43 size_t remove = observers_.erase(observer); 31 size_t removed = observers_.erase(observer);
44 OnObserversChanged(); 32 if (removed)
45 return remove > 0; 33 OnClientsChanged();
34 return removed > 0;
46 } 35 }
47 36
48 void GeolocationProvider::OnObserversChanged() { 37 void GeolocationProvider::RequestCallback(
49 DCHECK(OnClientThread()); 38 const content::GeolocationUpdateCallback& callback) {
39 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
40 callbacks_.push_back(callback);
41 OnClientsChanged();
42 OnPermissionGranted();
43 }
44
45 void GeolocationProvider::OnPermissionGranted() {
46 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
47 bool was_permission_granted = is_permission_granted_;
48 is_permission_granted_ = true;
jam 2012/05/03 06:17:49 this looks like a process wide security variable.
bartfab (slow) 2012/05/03 08:41:15 No. Website permissions are handled elsewhere. The
49 if (IsRunning() && !was_permission_granted)
50 InformProvidersPermissionGranted();
51 }
52
53 bool GeolocationProvider::HasPermissionBeenGranted() const {
54 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
55 return is_permission_granted_;
56 }
57
58 void GeolocationProvider::OnLocationUpdate(
59 const content::Geoposition& position) {
60 DCHECK(OnGeolocationThread());
61 // Will be true only in testing.
62 if (ignore_location_updates_)
63 return;
64 BrowserThread::PostTask(BrowserThread::IO,
65 FROM_HERE,
66 base::Bind(&GeolocationProvider::NotifyClients,
67 base::Unretained(this), position));
68 }
69
70 void GeolocationProvider::OverrideLocationForTesting(
71 const content::Geoposition& position) {
72 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
73 position_ = position;
74 ignore_location_updates_ = true;
75 NotifyClients(position);
76 }
77
78 GeolocationProvider* GeolocationProvider::GetInstance() {
79 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
80 return Singleton<GeolocationProvider>::get();
81 }
82
83 GeolocationProvider::GeolocationProvider()
84 : base::Thread("Geolocation"),
85 is_permission_granted_(false),
86 ignore_location_updates_(false),
87 arbitrator_(NULL) {
88 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
89 }
90
91 GeolocationProvider::~GeolocationProvider() {
92 // All observers should have unregistered before this singleton is destructed.
93 DCHECK(observers_.empty());
94 Stop();
95 DCHECK(!arbitrator_);
96 }
97
98 bool GeolocationProvider::OnGeolocationThread() const {
99 return MessageLoop::current() == message_loop();
100 }
101
102 void GeolocationProvider::OnClientsChanged() {
103 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
50 base::Closure task; 104 base::Closure task;
51 if (observers_.empty()) { 105 if (observers_.empty() && callbacks_.empty()) {
52 DCHECK(IsRunning()); 106 DCHECK(IsRunning());
53 task = base::Bind(&GeolocationProvider::StopProviders, 107 task = base::Bind(&GeolocationProvider::StopProviders,
54 base::Unretained(this)); 108 base::Unretained(this));
55 } else { 109 } else {
56 if (!IsRunning()) { 110 if (!IsRunning()) {
57 Start(); 111 Start();
58 if (HasPermissionBeenGranted()) 112 if (HasPermissionBeenGranted())
59 InformProvidersPermissionGranted(); 113 InformProvidersPermissionGranted();
60 } 114 }
115 // Determine a set of options that satisfies all clients.
116 GeolocationObserverOptions options =
117 GeolocationObserverOptions::Collapse(observers_);
118 // For callbacks, high accuracy position information is always requested.
119 if (!callbacks_.empty())
120 options.Collapse(GeolocationObserverOptions(true));
61 121
62 // The high accuracy requirement may have changed. 122 // Send the current options to the providers as they may have changed.
63 task = base::Bind(&GeolocationProvider::StartProviders, 123 task = base::Bind(&GeolocationProvider::StartProviders,
64 base::Unretained(this), 124 base::Unretained(this),
65 GeolocationObserverOptions::Collapse(observers_)); 125 options);
66 } 126 }
67 127
68 message_loop()->PostTask(FROM_HERE, task); 128 message_loop()->PostTask(FROM_HERE, task);
69 } 129 }
70 130
71 void GeolocationProvider::NotifyObservers( 131 void GeolocationProvider::StopProviders() {
72 const content::Geoposition& position) { 132 DCHECK(OnGeolocationThread());
73 DCHECK(OnClientThread()); 133 DCHECK(arbitrator_);
74 DCHECK(position.Validate() || 134 arbitrator_->StopProviders();
75 position.error_code != content::Geoposition::ERROR_CODE_NONE);
76 position_ = position;
77 ObserverMap::const_iterator it = observers_.begin();
78 while (it != observers_.end()) {
79 // Advance iterator before callback to guard against synchronous unregister.
80 GeolocationObserver* observer = it->first;
81 ++it;
82 observer->OnLocationUpdate(position_);
83 }
84 } 135 }
85 136
86 void GeolocationProvider::StartProviders( 137 void GeolocationProvider::StartProviders(
87 const GeolocationObserverOptions& options) { 138 const GeolocationObserverOptions& options) {
88 DCHECK(OnGeolocationThread()); 139 DCHECK(OnGeolocationThread());
89 DCHECK(arbitrator_); 140 DCHECK(arbitrator_);
90 arbitrator_->StartProviders(options); 141 arbitrator_->StartProviders(options);
91 } 142 }
92 143
93 void GeolocationProvider::StopProviders() {
94 DCHECK(OnGeolocationThread());
95 DCHECK(arbitrator_);
96 arbitrator_->StopProviders();
97 }
98
99 void GeolocationProvider::OnPermissionGranted() {
100 DCHECK(OnClientThread());
101 is_permission_granted_ = true;
102 if (IsRunning())
103 InformProvidersPermissionGranted();
104 }
105
106 void GeolocationProvider::InformProvidersPermissionGranted() { 144 void GeolocationProvider::InformProvidersPermissionGranted() {
107 DCHECK(IsRunning()); 145 DCHECK(IsRunning());
108 if (!OnGeolocationThread()) { 146 if (!OnGeolocationThread()) {
109 message_loop()->PostTask( 147 message_loop()->PostTask(
110 FROM_HERE, 148 FROM_HERE,
111 base::Bind(&GeolocationProvider::InformProvidersPermissionGranted, 149 base::Bind(&GeolocationProvider::InformProvidersPermissionGranted,
112 base::Unretained(this))); 150 base::Unretained(this)));
113 return; 151 return;
114 } 152 }
115 DCHECK(OnGeolocationThread()); 153 DCHECK(OnGeolocationThread());
116 DCHECK(arbitrator_); 154 DCHECK(arbitrator_);
117 arbitrator_->OnPermissionGranted(); 155 arbitrator_->OnPermissionGranted();
118 } 156 }
119 157
158 void GeolocationProvider::NotifyClients(const content::Geoposition& position) {
159 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
160 DCHECK(position.Validate() ||
161 position.error_code != content::Geoposition::ERROR_CODE_NONE);
162 position_ = position;
163 ObserverMap::const_iterator it = observers_.begin();
164 while (it != observers_.end()) {
165 // Advance iterator before calling the observer to guard against synchronous
166 // unregister.
167 GeolocationObserver* observer = it->first;
168 ++it;
169 observer->OnLocationUpdate(position_);
170 }
171 if (!callbacks_.empty()) {
172 for (CallbackList::iterator it = callbacks_.begin();
173 it != callbacks_.end();
174 ++it)
175 it->Run(position);
176 callbacks_.clear();
177 OnClientsChanged();
178 }
179 }
180
120 void GeolocationProvider::Init() { 181 void GeolocationProvider::Init() {
121 DCHECK(OnGeolocationThread()); 182 DCHECK(OnGeolocationThread());
122 DCHECK(!arbitrator_); 183 DCHECK(!arbitrator_);
123 arbitrator_ = GeolocationArbitrator::Create(this); 184 arbitrator_ = GeolocationArbitrator::Create(this);
124 } 185 }
125 186
126 void GeolocationProvider::CleanUp() { 187 void GeolocationProvider::CleanUp() {
127 DCHECK(OnGeolocationThread()); 188 DCHECK(OnGeolocationThread());
128 delete arbitrator_; 189 delete arbitrator_;
129 arbitrator_ = NULL; 190 arbitrator_ = NULL;
130 } 191 }
131
132 void GeolocationProvider::OnLocationUpdate(
133 const content::Geoposition& position) {
134 DCHECK(OnGeolocationThread());
135 // Will be true only in testing.
136 if (ignore_location_updates_)
137 return;
138 client_loop_->PostTask(
139 FROM_HERE,
140 base::Bind(&GeolocationProvider::NotifyObservers,
141 base::Unretained(this), position));
142 }
143
144 void GeolocationProvider::OverrideLocationForTesting(
145 const content::Geoposition& position) {
146 DCHECK(OnClientThread());
147 position_ = position;
148 ignore_location_updates_ = true;
149 NotifyObservers(position);
150 }
151
152 bool GeolocationProvider::HasPermissionBeenGranted() const {
153 DCHECK(OnClientThread());
154 return is_permission_granted_;
155 }
156
157 bool GeolocationProvider::OnClientThread() const {
158 return client_loop_->BelongsToCurrentThread();
159 }
160
161 bool GeolocationProvider::OnGeolocationThread() const {
162 return MessageLoop::current() == message_loop();
163 }
OLDNEW
« no previous file with comments | « content/browser/geolocation/geolocation_provider.h ('k') | content/browser/geolocation/geolocation_provider_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698