1 /*
2 * Copyright (C) 2009 Apple Inc. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 * THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26 #include "config.h"
27
28 #include "modules/geolocation/GeolocationController.h"
29
30 #include "core/inspector/InspectorController.h"
31 #include "core/page/Page.h"
32 #include "modules/geolocation/GeolocationClient.h"
33 #include "modules/geolocation/GeolocationError.h"
34 #include "modules/geolocation/GeolocationInspectorAgent.h"
35 #include "modules/geolocation/GeolocationPosition.h"
36
37 namespace blink {
38
GeolocationController(LocalFrame & frame,GeolocationClient * client)39 GeolocationController::GeolocationController(LocalFrame& frame, GeolocationClient* client)
40 : PageLifecycleObserver(frame.page())
41 , m_client(client)
42 , m_hasClientForTest(false)
43 , m_isClientUpdating(false)
44 , m_inspectorAgent(nullptr)
45 {
46 // FIXME: Once GeolocationInspectorAgent is per frame, there will be a 1:1 relationship between
47 // it and this class. Until then, there's one GeolocationInspectorAgent per page that the main
48 // frame is responsible for creating.
49 if (frame.isMainFrame()) {
50 OwnPtrWillBeRawPtr<GeolocationInspectorAgent> geolocationAgent(GeolocationInspectorAgent::create());
51 m_inspectorAgent = geolocationAgent.get();
52 frame.page()->inspectorController().registerModuleAgent(geolocationAgent.release());
53 } else if (frame.page()->mainFrame()->isLocalFrame()) {
54 m_inspectorAgent = GeolocationController::from(frame.page()->deprecatedLocalMainFrame())->m_inspectorAgent;
55 }
56
57 // m_inspectorAgent is 0 for out of process iframe instantiations, since inspector is currently unable
58 // to handle that scenario.
59 if (m_inspectorAgent)
60 m_inspectorAgent->addController(this);
61
62 if (!frame.isMainFrame() && frame.page()->mainFrame()->isLocalFrame()) {
63 // internals.setGeolocationClientMock is per page.
64 GeolocationController* mainController = GeolocationController::from(frame.page()->deprecatedLocalMainFrame());
65 if (mainController->hasClientForTest())
66 setClientForTest(mainController->client());
67 }
68 }
69
startUpdatingIfNeeded()70 void GeolocationController::startUpdatingIfNeeded()
71 {
72 if (m_isClientUpdating)
73 return;
74 m_isClientUpdating = true;
75 m_client->startUpdating();
76 }
77
stopUpdatingIfNeeded()78 void GeolocationController::stopUpdatingIfNeeded()
79 {
80 if (!m_isClientUpdating)
81 return;
82 m_isClientUpdating = false;
83 m_client->stopUpdating();
84 }
85
~GeolocationController()86 GeolocationController::~GeolocationController()
87 {
88 ASSERT(m_observers.isEmpty());
89 #if !ENABLE(OILPAN)
90 if (page() && m_inspectorAgent) {
91 m_inspectorAgent->removeController(this);
92 m_inspectorAgent = nullptr;
93 }
94
95 if (m_hasClientForTest) {
96 m_client->controllerForTestRemoved(this);
97 m_hasClientForTest = false;
98 }
99 #endif
100 }
101
create(LocalFrame & frame,GeolocationClient * client)102 PassOwnPtrWillBeRawPtr<GeolocationController> GeolocationController::create(LocalFrame& frame, GeolocationClient* client)
103 {
104 return adoptPtrWillBeNoop(new GeolocationController(frame, client));
105 }
106
addObserver(Geolocation * observer,bool enableHighAccuracy)107 void GeolocationController::addObserver(Geolocation* observer, bool enableHighAccuracy)
108 {
109 // This may be called multiple times with the same observer, though removeObserver()
110 // is called only once with each.
111 bool wasEmpty = m_observers.isEmpty();
112 m_observers.add(observer);
113 if (enableHighAccuracy)
114 m_highAccuracyObservers.add(observer);
115
116 if (m_client) {
117 if (enableHighAccuracy)
118 m_client->setEnableHighAccuracy(true);
119 if (wasEmpty && page() && page()->visibilityState() == PageVisibilityStateVisible)
120 startUpdatingIfNeeded();
121 }
122 }
123
removeObserver(Geolocation * observer)124 void GeolocationController::removeObserver(Geolocation* observer)
125 {
126 if (!m_observers.contains(observer))
127 return;
128
129 m_observers.remove(observer);
130 m_highAccuracyObservers.remove(observer);
131
132 if (m_client) {
133 if (m_observers.isEmpty())
134 stopUpdatingIfNeeded();
135 else if (m_highAccuracyObservers.isEmpty())
136 m_client->setEnableHighAccuracy(false);
137 }
138 }
139
requestPermission(Geolocation * geolocation)140 void GeolocationController::requestPermission(Geolocation* geolocation)
141 {
142 if (m_client)
143 m_client->requestPermission(geolocation);
144 }
145
cancelPermissionRequest(Geolocation * geolocation)146 void GeolocationController::cancelPermissionRequest(Geolocation* geolocation)
147 {
148 if (m_client)
149 m_client->cancelPermissionRequest(geolocation);
150 }
151
positionChanged(GeolocationPosition * position)152 void GeolocationController::positionChanged(GeolocationPosition* position)
153 {
154 position = m_inspectorAgent->overrideGeolocationPosition(position);
155 if (!position) {
156 errorOccurred(GeolocationError::create(GeolocationError::PositionUnavailable, "PositionUnavailable"));
157 return;
158 }
159 m_lastPosition = position;
160 HeapVector<Member<Geolocation> > observersVector;
161 copyToVector(m_observers, observersVector);
162 for (size_t i = 0; i < observersVector.size(); ++i)
163 observersVector[i]->positionChanged();
164 }
165
errorOccurred(GeolocationError * error)166 void GeolocationController::errorOccurred(GeolocationError* error)
167 {
168 HeapVector<Member<Geolocation> > observersVector;
169 copyToVector(m_observers, observersVector);
170 for (size_t i = 0; i < observersVector.size(); ++i)
171 observersVector[i]->setError(error);
172 }
173
lastPosition()174 GeolocationPosition* GeolocationController::lastPosition()
175 {
176 if (m_lastPosition.get())
177 return m_lastPosition.get();
178
179 if (!m_client)
180 return 0;
181
182 return m_client->lastPosition();
183 }
184
setClientForTest(GeolocationClient * client)185 void GeolocationController::setClientForTest(GeolocationClient* client)
186 {
187 if (m_hasClientForTest)
188 m_client->controllerForTestRemoved(this);
189 m_client = client;
190 m_hasClientForTest = true;
191
192 client->controllerForTestAdded(this);
193 }
194
pageVisibilityChanged()195 void GeolocationController::pageVisibilityChanged()
196 {
197 if (m_observers.isEmpty() || !m_client)
198 return;
199
200 if (page() && page()->visibilityState() == PageVisibilityStateVisible)
201 startUpdatingIfNeeded();
202 else
203 stopUpdatingIfNeeded();
204 }
205
supplementName()206 const char* GeolocationController::supplementName()
207 {
208 return "GeolocationController";
209 }
210
trace(Visitor * visitor)211 void GeolocationController::trace(Visitor* visitor)
212 {
213 visitor->trace(m_client);
214 visitor->trace(m_lastPosition);
215 visitor->trace(m_observers);
216 visitor->trace(m_highAccuracyObservers);
217 visitor->trace(m_inspectorAgent);
218 WillBeHeapSupplement<LocalFrame>::trace(visitor);
219 }
220
provideGeolocationTo(LocalFrame & frame,GeolocationClient * client)221 void provideGeolocationTo(LocalFrame& frame, GeolocationClient* client)
222 {
223 WillBeHeapSupplement<LocalFrame>::provideTo(frame, GeolocationController::supplementName(), GeolocationController::create(frame, client));
224 }
225
226 } // namespace blink
227