1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.googlecode.android_scripting.facade;
18 
19 import android.app.Service;
20 import android.content.Context;
21 import android.location.Address;
22 import android.location.Geocoder;
23 import android.location.Location;
24 import android.location.LocationListener;
25 import android.location.LocationManager;
26 import android.os.Bundle;
27 
28 import com.google.common.collect.Maps;
29 import com.googlecode.android_scripting.jsonrpc.RpcReceiver;
30 import com.googlecode.android_scripting.rpc.Rpc;
31 import com.googlecode.android_scripting.rpc.RpcDefault;
32 import com.googlecode.android_scripting.rpc.RpcParameter;
33 import com.googlecode.android_scripting.rpc.RpcStartEvent;
34 import com.googlecode.android_scripting.rpc.RpcStopEvent;
35 
36 import java.io.IOException;
37 import java.util.HashMap;
38 import java.util.List;
39 import java.util.Map;
40 import java.util.Map.Entry;
41 
42 /**
43  * This facade exposes the LocationManager related functionality.<br>
44  * <br>
45  * <b>Overview</b><br>
46  * Once activated by 'startLocating' the LocationFacade attempts to return location data collected
47  * via GPS or the cell network. If neither are available the last known location may be retrieved.
48  * If both are available the format of the returned data is:<br>
49  * {u'network': {u'altitude': 0, u'provider': u'network', u'longitude': -0.38509020000000002,
50  * u'time': 1297079691231L, u'latitude': 52.410557300000001, u'speed': 0, u'accuracy': 75}, u'gps':
51  * {u'altitude': 51, u'provider': u'gps', u'longitude': -0.38537094593048096, u'time':
52  * 1297079709000L, u'latitude': 52.41076922416687, u'speed': 0, u'accuracy': 24}}<br>
53  * If neither are available {} is returned. <br>
54  * Example (python):<br>
55  *
56  * <pre>
57  * import android, time
58  * droid = android.Android()
59  * droid.startLocating()
60  * time.sleep(15)
61  * loc = droid.readLocation().result
62  * if loc = {}:
63  *   loc = getLastKnownLocation().result
64  * if loc != {}:
65  *   try:
66  *     n = loc['gps']
67  *   except KeyError:
68  *     n = loc['network']
69  *   la = n['latitude']
70  *   lo = n['longitude']
71  *   address = droid.geocode(la, lo).result
72  * droid.stopLocating()
73  * </pre>
74  *
75  * The address format is:<br>
76  * [{u'thoroughfare': u'Some Street', u'locality': u'Some Town', u'sub_admin_area': u'Some Borough',
77  * u'admin_area': u'Some City', u'feature_name': u'House Numbers', u'country_code': u'GB',
78  * u'country_name': u'United Kingdom', u'postal_code': u'ST1 1'}]
79  *
80  */
81 public class LocationFacade extends RpcReceiver {
82   private final EventFacade mEventFacade;
83   private final Service mService;
84   private final Map<String, Location> mLocationUpdates;
85   private final LocationManager mLocationManager;
86   private final Geocoder mGeocoder;
87 
88   private final LocationListener mLocationListener = new LocationListener() {
89     @Override
90     public synchronized void onLocationChanged(Location location) {
91       mLocationUpdates.put(location.getProvider(), location);
92       Map<String, Location> copy = Maps.newHashMap();
93       for (Entry<String, Location> entry : mLocationUpdates.entrySet()) {
94         copy.put(entry.getKey(), entry.getValue());
95       }
96       mEventFacade.postEvent("location", copy);
97     }
98 
99     @Override
100     public void onProviderDisabled(String provider) {
101     }
102 
103     @Override
104     public void onProviderEnabled(String provider) {
105     }
106 
107     @Override
108     public void onStatusChanged(String provider, int status, Bundle extras) {
109     }
110   };
111 
LocationFacade(FacadeManager manager)112   public LocationFacade(FacadeManager manager) {
113     super(manager);
114     mService = manager.getService();
115     mEventFacade = manager.getReceiver(EventFacade.class);
116     mGeocoder = new Geocoder(mService);
117     mLocationManager = (LocationManager) mService.getSystemService(Context.LOCATION_SERVICE);
118     mLocationUpdates = new HashMap<String, Location>();
119   }
120 
121   @Override
shutdown()122   public void shutdown() {
123     stopLocating();
124   }
125 
126   @Rpc(description = "Returns availables providers on the phone")
locationProviders()127   public List<String> locationProviders() {
128     return mLocationManager.getAllProviders();
129   }
130 
131   @Rpc(description = "Ask if provider is enabled")
locationProviderEnabled( @pcParametername = "provider", description = "Name of location provider") String provider)132   public boolean locationProviderEnabled(
133       @RpcParameter(name = "provider", description = "Name of location provider") String provider) {
134     return mLocationManager.isProviderEnabled(provider);
135   }
136 
137   @Rpc(description = "Starts collecting location data.")
138   @RpcStartEvent("location")
startLocating( @pcParametername = "minDistance", description = "minimum time between updates in milliseconds") @pcDefault"60000") Integer minUpdateTime, @RpcParameter(name = "minUpdateDistance", description = "minimum distance between updates in meters") @RpcDefault("30") Integer minUpdateDistance)139   public void startLocating(
140       @RpcParameter(name = "minDistance", description = "minimum time between updates in milliseconds") @RpcDefault("60000") Integer minUpdateTime,
141       @RpcParameter(name = "minUpdateDistance", description = "minimum distance between updates in meters") @RpcDefault("30") Integer minUpdateDistance) {
142     for (String provider : mLocationManager.getAllProviders()) {
143       mLocationManager.requestLocationUpdates(provider, minUpdateTime, minUpdateDistance,
144           mLocationListener, mService.getMainLooper());
145     }
146   }
147 
148   @Rpc(description = "Returns the current location as indicated by all available providers.", returns = "A map of location information by provider.")
readLocation()149   public Map<String, Location> readLocation() {
150     return mLocationUpdates;
151   }
152 
153   @Rpc(description = "Stops collecting location data.")
154   @RpcStopEvent("location")
stopLocating()155   public synchronized void stopLocating() {
156     mLocationManager.removeUpdates(mLocationListener);
157     mLocationUpdates.clear();
158   }
159 
160   @Rpc(description = "Returns the last known location of the device.", returns = "A map of location information by provider.")
getLastKnownLocation()161   public Map<String, Location> getLastKnownLocation() {
162     Map<String, Location> location = new HashMap<String, Location>();
163     for (String provider : mLocationManager.getAllProviders()) {
164       location.put(provider, mLocationManager.getLastKnownLocation(provider));
165     }
166     return location;
167   }
168 
169   @Rpc(description = "Returns a list of addresses for the given latitude and longitude.", returns = "A list of addresses.")
geocode( @pcParametername = "latitude") Double latitude, @RpcParameter(name = "longitude") Double longitude, @RpcParameter(name = "maxResults", description = "maximum number of results") @RpcDefault("1") Integer maxResults)170   public List<Address> geocode(
171       @RpcParameter(name = "latitude") Double latitude,
172       @RpcParameter(name = "longitude") Double longitude,
173       @RpcParameter(name = "maxResults", description = "maximum number of results") @RpcDefault("1") Integer maxResults)
174       throws IOException {
175     return mGeocoder.getFromLocation(latitude, longitude, maxResults);
176   }
177 }
178