mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge.git
synced 2025-01-25 16:15:55 +01:00
Refactor location service
- Refactor the code from a static global instance to a lifecycle-aware service instantiated in the DeviceCommunicationService - Fix number of devices reported in the notification - Prevents leaks and properly stops when devices get disconnected
This commit is contained in:
parent
3799ffb72c
commit
500e930237
@ -106,7 +106,7 @@ import nodomain.freeyourgadget.gadgetbridge.devices.DeviceCoordinator;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.devices.DeviceManager;
|
import nodomain.freeyourgadget.gadgetbridge.devices.DeviceManager;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksContentObserver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksContentObserver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
@ -659,7 +659,7 @@ public class DebugActivity extends AbstractGBActivity {
|
|||||||
stopPhoneGpsLocationListener.setOnClickListener(new View.OnClickListener() {
|
stopPhoneGpsLocationListener.setOnClickListener(new View.OnClickListener() {
|
||||||
@Override
|
@Override
|
||||||
public void onClick(View v) {
|
public void onClick(View v) {
|
||||||
GBLocationManager.stopAll(getBaseContext());
|
GBLocationService.stop(DebugActivity.this, null);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -21,10 +21,14 @@ import android.location.LocationListener;
|
|||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.os.SystemClock;
|
import android.os.SystemClock;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.EventHandler;
|
import nodomain.freeyourgadget.gadgetbridge.devices.EventHandler;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An implementation of a {@link LocationListener} that forwards the location updates to the
|
* An implementation of a {@link LocationListener} that forwards the location updates to the
|
||||||
@ -33,18 +37,18 @@ import nodomain.freeyourgadget.gadgetbridge.devices.EventHandler;
|
|||||||
public class GBLocationListener implements LocationListener {
|
public class GBLocationListener implements LocationListener {
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(GBLocationListener.class);
|
private static final Logger LOG = LoggerFactory.getLogger(GBLocationListener.class);
|
||||||
|
|
||||||
private final EventHandler eventHandler;
|
private final GBDevice device;
|
||||||
|
|
||||||
private Location previousLocation;
|
private Location previousLocation;
|
||||||
// divide by 3.6 to get km/h to m/s
|
// divide by 3.6 to get km/h to m/s
|
||||||
private static final double SPEED_THRESHOLD = 1.0 / 3.6;
|
private static final double SPEED_THRESHOLD = 1.0 / 3.6;
|
||||||
|
|
||||||
public GBLocationListener(final EventHandler eventHandler) {
|
public GBLocationListener(final GBDevice device) {
|
||||||
this.eventHandler = eventHandler;
|
this.device = device;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onLocationChanged(final Location location) {
|
public void onLocationChanged(@NonNull final Location location) {
|
||||||
LOG.info("Location changed: {}", location);
|
LOG.info("Location changed: {}", location);
|
||||||
|
|
||||||
// Correct the location time
|
// Correct the location time
|
||||||
@ -61,16 +65,16 @@ public class GBLocationListener implements LocationListener {
|
|||||||
|
|
||||||
previousLocation = location;
|
previousLocation = location;
|
||||||
|
|
||||||
eventHandler.onSetGpsLocation(location);
|
GBApplication.deviceService(device).onSetGpsLocation(location);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onProviderDisabled(final String provider) {
|
public void onProviderDisabled(@NonNull final String provider) {
|
||||||
LOG.info("onProviderDisabled: {}", provider);
|
LOG.info("onProviderDisabled: {}", provider);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onProviderEnabled(final String provider) {
|
public void onProviderEnabled(@NonNull final String provider) {
|
||||||
LOG.info("onProviderDisabled: {}", provider);
|
LOG.info("onProviderDisabled: {}", provider);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,140 +0,0 @@
|
|||||||
/* Copyright (C) 2022-2024 halemmerich, José Rebelo, LukasEdl, Martin Boonk
|
|
||||||
|
|
||||||
This file is part of Gadgetbridge.
|
|
||||||
|
|
||||||
Gadgetbridge is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU Affero General Public License as published
|
|
||||||
by the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
Gadgetbridge is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Affero General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
|
||||||
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.location.Location;
|
|
||||||
import android.location.LocationListener;
|
|
||||||
import android.location.LocationManager;
|
|
||||||
import android.os.Bundle;
|
|
||||||
import android.os.Looper;
|
|
||||||
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.EventHandler;
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A static location manager, which keeps track of what providers are currently running. A notification is kept
|
|
||||||
* while there is at least one provider running.
|
|
||||||
*/
|
|
||||||
public class GBLocationManager {
|
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(GBLocationManager.class);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The current number of running listeners.
|
|
||||||
*/
|
|
||||||
private static Map<EventHandler, Map<LocationProviderType, AbstractLocationProvider>> providers = new HashMap<>();
|
|
||||||
|
|
||||||
public static void start(final Context context, final EventHandler eventHandler) {
|
|
||||||
GBLocationManager.start(context, eventHandler, LocationProviderType.GPS, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void start(final Context context, final EventHandler eventHandler, final LocationProviderType providerType, Integer updateInterval) {
|
|
||||||
LOG.info("Starting");
|
|
||||||
if (providers.containsKey(eventHandler) && providers.get(eventHandler).containsKey(providerType)) {
|
|
||||||
LOG.warn("EventHandler already registered");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
GB.createGpsNotification(context, providers.size());
|
|
||||||
|
|
||||||
final GBLocationListener locationListener = new GBLocationListener(eventHandler);
|
|
||||||
final AbstractLocationProvider locationProvider;
|
|
||||||
switch (providerType) {
|
|
||||||
case GPS:
|
|
||||||
LOG.info("Using gps location provider");
|
|
||||||
locationProvider = new PhoneGpsLocationProvider(locationListener);
|
|
||||||
break;
|
|
||||||
case NETWORK:
|
|
||||||
LOG.info("Using network location provider");
|
|
||||||
locationProvider = new PhoneNetworkLocationProvider(locationListener);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
LOG.info("Using default location provider: GPS");
|
|
||||||
locationProvider = new PhoneGpsLocationProvider(locationListener);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (updateInterval != null) {
|
|
||||||
locationProvider.start(context, updateInterval);
|
|
||||||
} else {
|
|
||||||
locationProvider.start(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (providers.containsKey(eventHandler)) {
|
|
||||||
providers.get(eventHandler).put(providerType, locationProvider);
|
|
||||||
} else {
|
|
||||||
Map<LocationProviderType, AbstractLocationProvider> providerMap = new HashMap<>();
|
|
||||||
providerMap.put(providerType, locationProvider);
|
|
||||||
providers.put(eventHandler, providerMap);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void stop(final Context context, final EventHandler eventHandler) {
|
|
||||||
GBLocationManager.stop(context, eventHandler, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void stop(final Context context, final EventHandler eventHandler, final LocationProviderType gpsType) {
|
|
||||||
if (!providers.containsKey(eventHandler)) return;
|
|
||||||
Map<LocationProviderType, AbstractLocationProvider> providerMap = providers.get(eventHandler);
|
|
||||||
if (gpsType == null) {
|
|
||||||
Set<LocationProviderType> toBeRemoved = new HashSet<>();
|
|
||||||
for (LocationProviderType providerType: providerMap.keySet()) {
|
|
||||||
stopProvider(context, providerMap.get(providerType));
|
|
||||||
toBeRemoved.add(providerType);
|
|
||||||
}
|
|
||||||
for (final LocationProviderType providerType : toBeRemoved) {
|
|
||||||
providerMap.remove(providerType);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
stopProvider(context, providerMap.get(gpsType));
|
|
||||||
providerMap.remove(gpsType);
|
|
||||||
}
|
|
||||||
LOG.debug("Remaining providers: " + providers.size());
|
|
||||||
if (providers.get(eventHandler).size() == 0)
|
|
||||||
providers.remove(eventHandler);
|
|
||||||
updateNotification(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void updateNotification(final Context context){
|
|
||||||
if (!providers.isEmpty()) {
|
|
||||||
GB.createGpsNotification(context, providers.size());
|
|
||||||
} else {
|
|
||||||
GB.removeGpsNotification(context);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void stopProvider(final Context context, AbstractLocationProvider locationProvider) {
|
|
||||||
if (locationProvider != null) {
|
|
||||||
locationProvider.stop(context);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void stopAll(final Context context) {
|
|
||||||
for (EventHandler eventHandler : providers.keySet()) {
|
|
||||||
stop(context, eventHandler);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -22,35 +22,30 @@ import android.location.LocationListener;
|
|||||||
/**
|
/**
|
||||||
* An abstract location provider, which periodically sends a location update to the provided {@link LocationListener}.
|
* An abstract location provider, which periodically sends a location update to the provided {@link LocationListener}.
|
||||||
*/
|
*/
|
||||||
public abstract class AbstractLocationProvider {
|
public abstract class GBLocationProvider {
|
||||||
|
private final Context context;
|
||||||
private final LocationListener locationListener;
|
private final LocationListener locationListener;
|
||||||
|
|
||||||
public AbstractLocationProvider(final LocationListener locationListener) {
|
public GBLocationProvider(final Context context, final LocationListener locationListener) {
|
||||||
|
this.context = context;
|
||||||
this.locationListener = locationListener;
|
this.locationListener = locationListener;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected final LocationListener getLocationListener() {
|
public final Context getContext() {
|
||||||
|
return this.context;
|
||||||
|
}
|
||||||
|
|
||||||
|
public final LocationListener getLocationListener() {
|
||||||
return this.locationListener;
|
return this.locationListener;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Start sending periodic location updates.
|
* Start sending periodic location updates.
|
||||||
*
|
|
||||||
* @param context the {@link Context}.
|
|
||||||
*/
|
*/
|
||||||
abstract void start(final Context context);
|
public abstract void start(final int interval);
|
||||||
|
|
||||||
/**
|
|
||||||
* Start sending periodic location updates.
|
|
||||||
*
|
|
||||||
* @param context the {@link Context}.
|
|
||||||
*/
|
|
||||||
abstract void start(final Context context, final int interval);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stop sending periodic location updates.
|
* Stop sending periodic location updates.
|
||||||
*
|
|
||||||
* @param context the {@link Context}.
|
|
||||||
*/
|
*/
|
||||||
abstract void stop(final Context context);
|
public abstract void stop();
|
||||||
}
|
}
|
@ -0,0 +1,47 @@
|
|||||||
|
/* Copyright (C) 2022-2024 LukasEdl
|
||||||
|
|
||||||
|
This file is part of Gadgetbridge.
|
||||||
|
|
||||||
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published
|
||||||
|
by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
Gadgetbridge is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.location.LocationManager;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.providers.MockLocationProvider;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.providers.PhoneLocationProvider;
|
||||||
|
|
||||||
|
public enum GBLocationProviderType {
|
||||||
|
GPS {
|
||||||
|
@Override
|
||||||
|
public GBLocationProvider newInstance(final Context context, final GBLocationListener locationListener) {
|
||||||
|
return new PhoneLocationProvider(context, locationListener, LocationManager.GPS_PROVIDER);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
NETWORK {
|
||||||
|
@Override
|
||||||
|
public GBLocationProvider newInstance(final Context context, final GBLocationListener locationListener) {
|
||||||
|
return new PhoneLocationProvider(context, locationListener, LocationManager.NETWORK_PROVIDER);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
MOCK {
|
||||||
|
@Override
|
||||||
|
public GBLocationProvider newInstance(final Context context, final GBLocationListener locationListener) {
|
||||||
|
return new MockLocationProvider(context, locationListener);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
;
|
||||||
|
|
||||||
|
public abstract GBLocationProvider newInstance(final Context context, final GBLocationListener locationListener);
|
||||||
|
}
|
@ -0,0 +1,184 @@
|
|||||||
|
/* Copyright (C) 2022-2024 halemmerich, José Rebelo, LukasEdl, Martin Boonk
|
||||||
|
|
||||||
|
This file is part of Gadgetbridge.
|
||||||
|
|
||||||
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published
|
||||||
|
by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
Gadgetbridge is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
||||||
|
|
||||||
|
import android.app.PendingIntent;
|
||||||
|
import android.content.BroadcastReceiver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.IntentFilter;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.core.app.NotificationCompat;
|
||||||
|
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.R;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.PendingIntentUtils;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A static location manager, which keeps track of what providers are currently running. A notification is kept
|
||||||
|
* while there is at least one provider running.
|
||||||
|
*/
|
||||||
|
public class GBLocationService extends BroadcastReceiver {
|
||||||
|
private static final Logger LOG = LoggerFactory.getLogger(GBLocationService.class);
|
||||||
|
|
||||||
|
public static final String ACTION_START = "GBLocationService.START";
|
||||||
|
public static final String ACTION_STOP = "GBLocationService.STOP";
|
||||||
|
public static final String ACTION_STOP_ALL = "GBLocationService.STOP_ALL";
|
||||||
|
|
||||||
|
public static final String EXTRA_TYPE = "extra_type";
|
||||||
|
public static final String EXTRA_INTERVAL = "extra_interval";
|
||||||
|
|
||||||
|
private final Context context;
|
||||||
|
private final Map<GBDevice, List<GBLocationProvider>> providersByDevice = new HashMap<>();
|
||||||
|
|
||||||
|
public GBLocationService(final Context context) {
|
||||||
|
this.context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReceive(final Context context, final Intent intent) {
|
||||||
|
if (intent.getAction() == null) {
|
||||||
|
LOG.warn("Action is null");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final GBDevice device = intent.getParcelableExtra(GBDevice.EXTRA_DEVICE);
|
||||||
|
|
||||||
|
switch (intent.getAction()) {
|
||||||
|
case ACTION_START:
|
||||||
|
if (device == null) {
|
||||||
|
LOG.error("Device is null for {}", intent.getAction());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final GBLocationProviderType providerType = GBLocationProviderType.valueOf(
|
||||||
|
intent.hasExtra(EXTRA_TYPE) ? intent.getStringExtra(EXTRA_TYPE) : "GPS"
|
||||||
|
);
|
||||||
|
final int updateInterval = intent.getIntExtra(EXTRA_INTERVAL, 1000);
|
||||||
|
|
||||||
|
LOG.debug("Starting location provider {} for {}", providerType, device.getAliasOrName());
|
||||||
|
|
||||||
|
if (!providersByDevice.containsKey(device)) {
|
||||||
|
providersByDevice.put(device, new ArrayList<>());
|
||||||
|
}
|
||||||
|
|
||||||
|
updateNotification();
|
||||||
|
|
||||||
|
final List<GBLocationProvider> existingProviders = providersByDevice.get(device);
|
||||||
|
|
||||||
|
final GBLocationListener locationListener = new GBLocationListener(device);
|
||||||
|
final GBLocationProvider locationProvider = providerType.newInstance(context, locationListener);
|
||||||
|
locationProvider.start(updateInterval);
|
||||||
|
Objects.requireNonNull(existingProviders).add(locationProvider);
|
||||||
|
return;
|
||||||
|
case ACTION_STOP:
|
||||||
|
if (device != null) {
|
||||||
|
stopDevice(device);
|
||||||
|
updateNotification();
|
||||||
|
} else {
|
||||||
|
stopAll();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
case ACTION_STOP_ALL:
|
||||||
|
stopAll();
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
LOG.warn("Unknown action {}", intent.getAction());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void stopDevice(final GBDevice device) {
|
||||||
|
LOG.debug("Stopping location providers for {}", device.getAliasOrName());
|
||||||
|
|
||||||
|
final List<GBLocationProvider> providers = providersByDevice.remove(device);
|
||||||
|
if (providers != null) {
|
||||||
|
for (final GBLocationProvider provider : providers) {
|
||||||
|
provider.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public IntentFilter buildFilter() {
|
||||||
|
final IntentFilter intentFilter = new IntentFilter();
|
||||||
|
intentFilter.addAction(ACTION_START);
|
||||||
|
intentFilter.addAction(ACTION_STOP);
|
||||||
|
return intentFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void stopAll() {
|
||||||
|
LOG.info("Stopping location service for all devices");
|
||||||
|
|
||||||
|
final List<GBDevice> gbDevices = new ArrayList<>(providersByDevice.keySet());
|
||||||
|
for (GBDevice d : gbDevices) {
|
||||||
|
stopDevice(d);
|
||||||
|
}
|
||||||
|
|
||||||
|
updateNotification();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void start(final Context context,
|
||||||
|
@NonNull final GBDevice device,
|
||||||
|
final GBLocationProviderType providerType,
|
||||||
|
final int updateInterval) {
|
||||||
|
final Intent intent = new Intent(ACTION_START);
|
||||||
|
intent.putExtra(GBDevice.EXTRA_DEVICE, device);
|
||||||
|
intent.putExtra(EXTRA_TYPE, providerType.name());
|
||||||
|
intent.putExtra(EXTRA_INTERVAL, updateInterval);
|
||||||
|
LocalBroadcastManager.getInstance(context).sendBroadcast(intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void stop(final Context context, @Nullable final GBDevice device) {
|
||||||
|
final Intent intent = new Intent(ACTION_STOP);
|
||||||
|
intent.putExtra(GBDevice.EXTRA_DEVICE, device);
|
||||||
|
LocalBroadcastManager.getInstance(context).sendBroadcast(intent);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateNotification() {
|
||||||
|
if (!providersByDevice.isEmpty()) {
|
||||||
|
final Intent notificationIntent = new Intent(context, GBLocationService.class);
|
||||||
|
notificationIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
|
||||||
|
final PendingIntent pendingIntent = PendingIntentUtils.getActivity(context, 0, notificationIntent, 0, false);
|
||||||
|
|
||||||
|
final NotificationCompat.Builder nb = new NotificationCompat.Builder(context, GB.NOTIFICATION_CHANNEL_ID_GPS)
|
||||||
|
.setTicker(context.getString(R.string.notification_gps_title))
|
||||||
|
.setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
|
||||||
|
.setContentTitle(context.getString(R.string.notification_gps_title))
|
||||||
|
.setContentText(context.getString(R.string.notification_gps_text, providersByDevice.size()))
|
||||||
|
.setContentIntent(pendingIntent)
|
||||||
|
.setSmallIcon(R.drawable.ic_gps_location)
|
||||||
|
.setOngoing(true);
|
||||||
|
|
||||||
|
GB.notify(GB.NOTIFICATION_ID_GPS, nb.build(), context);
|
||||||
|
} else {
|
||||||
|
GB.removeNotification(GB.NOTIFICATION_ID_GPS, context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,22 +0,0 @@
|
|||||||
/* Copyright (C) 2022-2024 LukasEdl
|
|
||||||
|
|
||||||
This file is part of Gadgetbridge.
|
|
||||||
|
|
||||||
Gadgetbridge is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU Affero General Public License as published
|
|
||||||
by the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
Gadgetbridge is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Affero General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
|
||||||
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
|
||||||
|
|
||||||
public enum LocationProviderType {
|
|
||||||
GPS,
|
|
||||||
NETWORK,
|
|
||||||
}
|
|
@ -1,80 +0,0 @@
|
|||||||
/* Copyright (C) 2022-2024 Lukas, LukasEdl
|
|
||||||
|
|
||||||
This file is part of Gadgetbridge.
|
|
||||||
|
|
||||||
Gadgetbridge is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU Affero General Public License as published
|
|
||||||
by the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
Gadgetbridge is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Affero General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
|
||||||
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
|
||||||
|
|
||||||
import android.Manifest;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.location.Location;
|
|
||||||
import android.location.LocationListener;
|
|
||||||
import android.location.LocationManager;
|
|
||||||
import android.os.Looper;
|
|
||||||
import android.widget.Toast;
|
|
||||||
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A location provider that uses the phone GPS, using {@link LocationManager}.
|
|
||||||
*/
|
|
||||||
public class PhoneNetworkLocationProvider extends AbstractLocationProvider {
|
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(PhoneNetworkLocationProvider.class);
|
|
||||||
|
|
||||||
private static final int INTERVAL_MIN_TIME = 1000;
|
|
||||||
private static final int INTERVAL_MIN_DISTANCE = 0;
|
|
||||||
|
|
||||||
public PhoneNetworkLocationProvider(LocationListener locationListener) {
|
|
||||||
super(locationListener);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
void start(final Context context) {
|
|
||||||
start(context, INTERVAL_MIN_TIME);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
void start(Context context, int interval) {
|
|
||||||
LOG.info("Starting phone network location provider");
|
|
||||||
|
|
||||||
if (!GB.checkPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) && !GB.checkPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION)) {
|
|
||||||
GB.toast("Location permission not granted", Toast.LENGTH_SHORT, GB.ERROR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
|
|
||||||
locationManager.removeUpdates(getLocationListener());
|
|
||||||
locationManager.requestLocationUpdates(
|
|
||||||
LocationManager.NETWORK_PROVIDER,
|
|
||||||
interval,
|
|
||||||
INTERVAL_MIN_DISTANCE,
|
|
||||||
getLocationListener(),
|
|
||||||
Looper.getMainLooper()
|
|
||||||
);
|
|
||||||
|
|
||||||
final Location lastKnownLocation = locationManager.getLastKnownLocation(LocationManager.NETWORK_PROVIDER);
|
|
||||||
LOG.debug("Last known network location: {}", lastKnownLocation);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
void stop(final Context context) {
|
|
||||||
LOG.info("Stopping phone network location provider");
|
|
||||||
|
|
||||||
final LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
|
|
||||||
locationManager.removeUpdates(getLocationListener());
|
|
||||||
}
|
|
||||||
}
|
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU Affero General Public License
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
||||||
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps.providers;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.location.Location;
|
import android.location.Location;
|
||||||
@ -26,13 +26,14 @@ import android.os.SystemClock;
|
|||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProvider;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.pebble.webview.CurrentPosition;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.pebble.webview.CurrentPosition;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A mock location provider which keeps updating the location at a constant speed, starting from the
|
* A mock location provider which keeps updating the location at a constant speed, starting from the
|
||||||
* last known location. Useful for local tests.
|
* last known location. Useful for local tests.
|
||||||
*/
|
*/
|
||||||
public class MockLocationProvider extends AbstractLocationProvider {
|
public class MockLocationProvider extends GBLocationProvider {
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(MockLocationProvider.class);
|
private static final Logger LOG = LoggerFactory.getLogger(MockLocationProvider.class);
|
||||||
|
|
||||||
private Location previousLocation = new CurrentPosition().getLastKnownLocation();
|
private Location previousLocation = new CurrentPosition().getLastKnownLocation();
|
||||||
@ -40,12 +41,12 @@ public class MockLocationProvider extends AbstractLocationProvider {
|
|||||||
/**
|
/**
|
||||||
* Interval between location updates, in milliseconds.
|
* Interval between location updates, in milliseconds.
|
||||||
*/
|
*/
|
||||||
private final int interval = 1000;
|
private static final int DEFAULT_INTERVAL = 1000;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Difference between location updates, in degrees.
|
* Difference between location updates, in degrees.
|
||||||
*/
|
*/
|
||||||
private final float coordDiff = 0.0002f;
|
private static final float COORD_DIFF = 0.0002f;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Whether the handler is running.
|
* Whether the handler is running.
|
||||||
@ -54,50 +55,40 @@ public class MockLocationProvider extends AbstractLocationProvider {
|
|||||||
|
|
||||||
private final Handler handler = new Handler(Looper.getMainLooper());
|
private final Handler handler = new Handler(Looper.getMainLooper());
|
||||||
|
|
||||||
private final Runnable locationUpdateRunnable = new Runnable() {
|
public MockLocationProvider(final Context context, final LocationListener locationListener) {
|
||||||
@Override
|
super(context, locationListener);
|
||||||
public void run() {
|
|
||||||
if (!running) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final Location newLocation = new Location(previousLocation);
|
|
||||||
newLocation.setLatitude(previousLocation.getLatitude() + coordDiff);
|
|
||||||
newLocation.setTime(System.currentTimeMillis());
|
|
||||||
newLocation.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());
|
|
||||||
|
|
||||||
getLocationListener().onLocationChanged(newLocation);
|
|
||||||
|
|
||||||
previousLocation = newLocation;
|
|
||||||
|
|
||||||
if (running) {
|
|
||||||
handler.postDelayed(this, interval);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
public MockLocationProvider(LocationListener locationListener) {
|
|
||||||
super(locationListener);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void start(final Context context) {
|
public void start(final int interval) {
|
||||||
LOG.info("Starting mock location provider");
|
LOG.info("Starting mock location provider");
|
||||||
|
|
||||||
running = true;
|
running = true;
|
||||||
handler.postDelayed(locationUpdateRunnable, interval);
|
handler.postDelayed(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
if (!running) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final Location newLocation = new Location(previousLocation);
|
||||||
|
newLocation.setLatitude(previousLocation.getLatitude() + COORD_DIFF);
|
||||||
|
newLocation.setTime(System.currentTimeMillis());
|
||||||
|
newLocation.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());
|
||||||
|
|
||||||
|
getLocationListener().onLocationChanged(newLocation);
|
||||||
|
|
||||||
|
previousLocation = newLocation;
|
||||||
|
|
||||||
|
if (running) {
|
||||||
|
handler.postDelayed(this, interval);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, interval > 0 ? interval : DEFAULT_INTERVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void start(final Context context, int minInterval) {
|
public void stop() {
|
||||||
LOG.info("Starting mock location provider");
|
|
||||||
|
|
||||||
running = true;
|
|
||||||
handler.postDelayed(locationUpdateRunnable, interval);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
void stop(final Context context) {
|
|
||||||
LOG.info("Stopping mock location provider");
|
LOG.info("Stopping mock location provider");
|
||||||
|
|
||||||
running = false;
|
running = false;
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU Affero General Public License
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
||||||
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps;
|
package nodomain.freeyourgadget.gadgetbridge.externalevents.gps.providers;
|
||||||
|
|
||||||
import android.Manifest;
|
import android.Manifest;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
@ -27,43 +27,38 @@ import android.widget.Toast;
|
|||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProvider;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A location provider that uses the phone GPS, using {@link LocationManager}.
|
* A location provider that uses the phone GPS, using {@link LocationManager}.
|
||||||
*/
|
*/
|
||||||
public class PhoneGpsLocationProvider extends AbstractLocationProvider {
|
public class PhoneLocationProvider extends GBLocationProvider {
|
||||||
private static final Logger LOG = LoggerFactory.getLogger(PhoneGpsLocationProvider.class);
|
private static final Logger LOG = LoggerFactory.getLogger(PhoneLocationProvider.class);
|
||||||
|
|
||||||
|
private final String provider;
|
||||||
|
|
||||||
private static final int INTERVAL_MIN_TIME = 1000;
|
|
||||||
private static final int INTERVAL_MIN_DISTANCE = 0;
|
private static final int INTERVAL_MIN_DISTANCE = 0;
|
||||||
|
|
||||||
public PhoneGpsLocationProvider(LocationListener locationListener) {
|
public PhoneLocationProvider(final Context context, final LocationListener locationListener, final String provider) {
|
||||||
super(locationListener);
|
super(context, locationListener);
|
||||||
}
|
this.provider = provider;
|
||||||
public PhoneGpsLocationProvider(LocationListener locationListener, int intervalTime) {
|
|
||||||
super(locationListener);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void start(final Context context) {
|
public void start(final int interval) {
|
||||||
start(context, INTERVAL_MIN_TIME);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
void start(Context context, int interval) {
|
|
||||||
LOG.info("Starting phone gps location provider");
|
LOG.info("Starting phone gps location provider");
|
||||||
|
|
||||||
if (!GB.checkPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) && !GB.checkPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION)) {
|
if (!GB.checkPermission(getContext(), Manifest.permission.ACCESS_FINE_LOCATION) && !GB.checkPermission(getContext(), Manifest.permission.ACCESS_COARSE_LOCATION)) {
|
||||||
GB.toast("Location permission not granted", Toast.LENGTH_SHORT, GB.ERROR);
|
GB.toast("Location permission not granted", Toast.LENGTH_SHORT, GB.ERROR);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
|
final LocationManager locationManager = (LocationManager) getContext().getSystemService(Context.LOCATION_SERVICE);
|
||||||
locationManager.removeUpdates(getLocationListener());
|
locationManager.removeUpdates(getLocationListener());
|
||||||
locationManager.requestLocationUpdates(
|
locationManager.requestLocationUpdates(
|
||||||
LocationManager.GPS_PROVIDER,
|
provider,
|
||||||
interval,
|
interval > 0 ? interval : 1_000,
|
||||||
INTERVAL_MIN_DISTANCE,
|
INTERVAL_MIN_DISTANCE,
|
||||||
getLocationListener(),
|
getLocationListener(),
|
||||||
Looper.getMainLooper()
|
Looper.getMainLooper()
|
||||||
@ -74,10 +69,10 @@ public class PhoneGpsLocationProvider extends AbstractLocationProvider {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void stop(final Context context) {
|
public void stop() {
|
||||||
LOG.info("Stopping phone gps location provider");
|
LOG.info("Stopping phone gps location provider");
|
||||||
|
|
||||||
final LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
|
final LocationManager locationManager = (LocationManager) getContext().getSystemService(Context.LOCATION_SERVICE);
|
||||||
locationManager.removeUpdates(getLocationListener());
|
locationManager.removeUpdates(getLocationListener());
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -82,7 +82,7 @@ import nodomain.freeyourgadget.gadgetbridge.externalevents.SMSReceiver;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.SilentModeReceiver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.SilentModeReceiver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.TimeChangeReceiver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.TimeChangeReceiver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.TinyWeatherForecastGermanyReceiver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.TinyWeatherForecastGermanyReceiver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.sleepasandroid.SleepAsAndroidAction;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.sleepasandroid.SleepAsAndroidReceiver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.sleepasandroid.SleepAsAndroidReceiver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDeviceService;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDeviceService;
|
||||||
@ -140,7 +140,7 @@ public class DeviceCommunicationService extends Service implements SharedPrefere
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private class FeatureSet{
|
private static class FeatureSet {
|
||||||
private boolean supportsWeather = false;
|
private boolean supportsWeather = false;
|
||||||
private boolean supportsActivityDataFetching = false;
|
private boolean supportsActivityDataFetching = false;
|
||||||
private boolean supportsCalendarEvents = false;
|
private boolean supportsCalendarEvents = false;
|
||||||
@ -256,7 +256,7 @@ public class DeviceCommunicationService extends Service implements SharedPrefere
|
|||||||
private AutoConnectIntervalReceiver mAutoConnectInvervalReceiver = null;
|
private AutoConnectIntervalReceiver mAutoConnectInvervalReceiver = null;
|
||||||
|
|
||||||
private AlarmReceiver mAlarmReceiver = null;
|
private AlarmReceiver mAlarmReceiver = null;
|
||||||
private List<CalendarReceiver> mCalendarReceiver = new ArrayList<>();
|
private final List<CalendarReceiver> mCalendarReceiver = new ArrayList<>();
|
||||||
private CMWeatherReceiver mCMWeatherReceiver = null;
|
private CMWeatherReceiver mCMWeatherReceiver = null;
|
||||||
private LineageOsWeatherReceiver mLineageOsWeatherReceiver = null;
|
private LineageOsWeatherReceiver mLineageOsWeatherReceiver = null;
|
||||||
private TinyWeatherForecastGermanyReceiver mTinyWeatherForecastGermanyReceiver = null;
|
private TinyWeatherForecastGermanyReceiver mTinyWeatherForecastGermanyReceiver = null;
|
||||||
@ -264,6 +264,7 @@ public class DeviceCommunicationService extends Service implements SharedPrefere
|
|||||||
private OmniJawsObserver mOmniJawsObserver = null;
|
private OmniJawsObserver mOmniJawsObserver = null;
|
||||||
private final DeviceSettingsReceiver deviceSettingsReceiver = new DeviceSettingsReceiver();
|
private final DeviceSettingsReceiver deviceSettingsReceiver = new DeviceSettingsReceiver();
|
||||||
private final IntentApiReceiver intentApiReceiver = new IntentApiReceiver();
|
private final IntentApiReceiver intentApiReceiver = new IntentApiReceiver();
|
||||||
|
private GBLocationService locationService = null;
|
||||||
|
|
||||||
private OsmandEventReceiver mOsmandAidlHelper = null;
|
private OsmandEventReceiver mOsmandAidlHelper = null;
|
||||||
|
|
||||||
@ -1343,6 +1344,11 @@ public class DeviceCommunicationService extends Service implements SharedPrefere
|
|||||||
registerReceiver(mSilentModeReceiver, filter);
|
registerReceiver(mSilentModeReceiver, filter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (locationService == null) {
|
||||||
|
locationService = new GBLocationService(this);
|
||||||
|
LocalBroadcastManager.getInstance(this).registerReceiver(locationService, locationService.buildFilter());
|
||||||
|
}
|
||||||
|
|
||||||
if (mOsmandAidlHelper == null && features.supportsNavigation()) {
|
if (mOsmandAidlHelper == null && features.supportsNavigation()) {
|
||||||
mOsmandAidlHelper = new OsmandEventReceiver(this.getApplication());
|
mOsmandAidlHelper = new OsmandEventReceiver(this.getApplication());
|
||||||
}
|
}
|
||||||
@ -1425,6 +1431,11 @@ public class DeviceCommunicationService extends Service implements SharedPrefere
|
|||||||
unregisterReceiver(mSilentModeReceiver);
|
unregisterReceiver(mSilentModeReceiver);
|
||||||
mSilentModeReceiver = null;
|
mSilentModeReceiver = null;
|
||||||
}
|
}
|
||||||
|
if (locationService != null) {
|
||||||
|
LocalBroadcastManager.getInstance(this).unregisterReceiver(locationService);
|
||||||
|
locationService.stopAll();
|
||||||
|
locationService = null;
|
||||||
|
}
|
||||||
if (mCMWeatherReceiver != null) {
|
if (mCMWeatherReceiver != null) {
|
||||||
unregisterReceiver(mCMWeatherReceiver);
|
unregisterReceiver(mCMWeatherReceiver);
|
||||||
mCMWeatherReceiver = null;
|
mCMWeatherReceiver = null;
|
||||||
|
@ -120,8 +120,8 @@ import nodomain.freeyourgadget.gadgetbridge.entities.CalendarSyncState;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.entities.CalendarSyncStateDao;
|
import nodomain.freeyourgadget.gadgetbridge.entities.CalendarSyncStateDao;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.CalendarReceiver;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.CalendarReceiver;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.LocationProviderType;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProviderType;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.Alarm;
|
import nodomain.freeyourgadget.gadgetbridge.model.Alarm;
|
||||||
@ -215,7 +215,7 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
|
|||||||
if (!gpsUpdateSetup)
|
if (!gpsUpdateSetup)
|
||||||
return;
|
return;
|
||||||
LOG.info("Stop location updates");
|
LOG.info("Stop location updates");
|
||||||
GBLocationManager.stop(getContext(), this);
|
GBLocationService.stop(getContext(), getDevice());
|
||||||
gpsUpdateSetup = false;
|
gpsUpdateSetup = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1140,14 +1140,14 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
|
|||||||
LOG.info("Using combined GPS and NETWORK based location: " + onlyUseNetworkGPS);
|
LOG.info("Using combined GPS and NETWORK based location: " + onlyUseNetworkGPS);
|
||||||
if (!onlyUseNetworkGPS) {
|
if (!onlyUseNetworkGPS) {
|
||||||
try {
|
try {
|
||||||
GBLocationManager.start(getContext(), this, LocationProviderType.GPS, intervalLength);
|
GBLocationService.start(getContext(), getDevice(), GBLocationProviderType.GPS, intervalLength);
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
LOG.warn("GPS provider could not be started", e);
|
LOG.warn("GPS provider could not be started", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
GBLocationManager.start(getContext(), this, LocationProviderType.NETWORK, intervalLength);
|
GBLocationService.start(getContext(), getDevice(), GBLocationProviderType.NETWORK, intervalLength);
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
LOG.warn("NETWORK provider could not be started", e);
|
LOG.warn("NETWORK provider could not be started", e);
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,6 @@ import android.content.pm.ApplicationInfo;
|
|||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.location.Location;
|
import android.location.Location;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.Bundle;
|
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
import android.os.Looper;
|
import android.os.Looper;
|
||||||
import android.widget.Toast;
|
import android.widget.Toast;
|
||||||
@ -117,7 +116,8 @@ import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.MiBandActivitySample;
|
import nodomain.freeyourgadget.gadgetbridge.entities.MiBandActivitySample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProviderType;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice.State;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice.State;
|
||||||
@ -2010,7 +2010,7 @@ public abstract class HuamiSupport extends AbstractBTLEDeviceSupport implements
|
|||||||
if (sendGpsToBand) {
|
if (sendGpsToBand) {
|
||||||
lastPhoneGpsSent = 0;
|
lastPhoneGpsSent = 0;
|
||||||
sendPhoneGps(HuamiPhoneGpsStatus.SEARCHING, null);
|
sendPhoneGps(HuamiPhoneGpsStatus.SEARCHING, null);
|
||||||
GBLocationManager.start(getContext(), this);
|
GBLocationService.start(getContext(), getDevice(), GBLocationProviderType.GPS, 1000);
|
||||||
} else {
|
} else {
|
||||||
sendPhoneGps(HuamiPhoneGpsStatus.DISABLED, null);
|
sendPhoneGps(HuamiPhoneGpsStatus.DISABLED, null);
|
||||||
}
|
}
|
||||||
@ -2030,7 +2030,7 @@ public abstract class HuamiSupport extends AbstractBTLEDeviceSupport implements
|
|||||||
protected void onWorkoutEnd() {
|
protected void onWorkoutEnd() {
|
||||||
final boolean startOnPhone = HuamiCoordinator.getWorkoutStartOnPhone(getDevice().getAddress());
|
final boolean startOnPhone = HuamiCoordinator.getWorkoutStartOnPhone(getDevice().getAddress());
|
||||||
|
|
||||||
GBLocationManager.stop(getContext(), this);
|
GBLocationService.stop(getContext(), getDevice());
|
||||||
|
|
||||||
if (startOnPhone) {
|
if (startOnPhone) {
|
||||||
LOG.info("Stopping OpenTracks recording");
|
LOG.info("Stopping OpenTracks recording");
|
||||||
|
@ -49,8 +49,6 @@ import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.GpsAndTime;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Menstrual;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Menstrual;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.MusicControl;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.MusicControl;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Weather;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.packets.Weather;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationListener;
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.Request;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.Request;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.GetPhoneInfoRequest;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.GetPhoneInfoRequest;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.SendMenstrualModifyTimeRequest;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huawei.requests.SendMenstrualModifyTimeRequest;
|
||||||
|
@ -44,7 +44,6 @@ import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEvent;
|
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEvent;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.DeviceCoordinator;
|
import nodomain.freeyourgadget.gadgetbridge.devices.DeviceCoordinator;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.EventHandler;
|
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiConstants;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiConstants;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiCoordinator;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiCoordinator;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiCoordinatorSupplier;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huawei.HuaweiCoordinatorSupplier;
|
||||||
@ -65,7 +64,8 @@ import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutPaceSample;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutPaceSampleDao;
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutPaceSampleDao;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutSummarySample;
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutSummarySample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutSummarySampleDao;
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuaweiWorkoutSummarySampleDao;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProviderType;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.Alarm;
|
import nodomain.freeyourgadget.gadgetbridge.entities.Alarm;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||||
@ -228,11 +228,6 @@ public class HuaweiSupportProvider {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void setGps(boolean start) {
|
public void setGps(boolean start) {
|
||||||
EventHandler handler;
|
|
||||||
if (isBLE())
|
|
||||||
handler = leSupport;
|
|
||||||
else
|
|
||||||
handler = brSupport;
|
|
||||||
if (start) {
|
if (start) {
|
||||||
if (!GBApplication.getDeviceSpecificSharedPrefs(getDevice().getAddress()).getBoolean(DeviceSettingsPreferenceConst.PREF_WORKOUT_SEND_GPS_TO_BAND, false))
|
if (!GBApplication.getDeviceSpecificSharedPrefs(getDevice().getAddress()).getBoolean(DeviceSettingsPreferenceConst.PREF_WORKOUT_SEND_GPS_TO_BAND, false))
|
||||||
return;
|
return;
|
||||||
@ -241,7 +236,7 @@ public class HuaweiSupportProvider {
|
|||||||
gpsParameterRequest.setFinalizeReq(new RequestCallback() {
|
gpsParameterRequest.setFinalizeReq(new RequestCallback() {
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
GBLocationManager.start(getContext(), handler);
|
GBLocationService.start(getContext(), getDevice(), GBLocationProviderType.GPS, 1000);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
try {
|
try {
|
||||||
@ -251,9 +246,9 @@ public class HuaweiSupportProvider {
|
|||||||
LOG.error("Failed to get GPS parameters", e);
|
LOG.error("Failed to get GPS parameters", e);
|
||||||
}
|
}
|
||||||
} else
|
} else
|
||||||
GBLocationManager.start(getContext(), handler);
|
GBLocationService.start(getContext(), getDevice(), GBLocationProviderType.GPS, 1000);
|
||||||
} else
|
} else
|
||||||
GBLocationManager.stop(getContext(), handler);
|
GBLocationService.stop(getContext(), getDevice());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setGpsParametersResponse(GpsAndTime.GpsParameters.Response response) {
|
public void setGpsParametersResponse(GpsAndTime.GpsParameters.Response response) {
|
||||||
|
@ -48,7 +48,8 @@ import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
|
import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationManager;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationProviderType;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.gps.GBLocationService;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksController;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
|
||||||
@ -664,7 +665,7 @@ public class XiaomiHealthService extends AbstractXiaomiService {
|
|||||||
if (!gpsStarted) {
|
if (!gpsStarted) {
|
||||||
gpsStarted = true;
|
gpsStarted = true;
|
||||||
gpsFixAcquired = false;
|
gpsFixAcquired = false;
|
||||||
GBLocationManager.start(getSupport().getContext(), getSupport());
|
GBLocationService.start(getSupport().getContext(), getSupport().getDevice(), GBLocationProviderType.GPS, 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
gpsTimeoutHandler.removeCallbacksAndMessages(null);
|
gpsTimeoutHandler.removeCallbacksAndMessages(null);
|
||||||
@ -673,7 +674,7 @@ public class XiaomiHealthService extends AbstractXiaomiService {
|
|||||||
LOG.debug("Timed out waiting for workout");
|
LOG.debug("Timed out waiting for workout");
|
||||||
gpsStarted = false;
|
gpsStarted = false;
|
||||||
gpsFixAcquired = false;
|
gpsFixAcquired = false;
|
||||||
GBLocationManager.stop(getSupport().getContext(), getSupport());
|
GBLocationService.stop(getSupport().getContext(), getSupport().getDevice());
|
||||||
}, 5000);
|
}, 5000);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -696,7 +697,7 @@ public class XiaomiHealthService extends AbstractXiaomiService {
|
|||||||
case WORKOUT_FINISHED:
|
case WORKOUT_FINISHED:
|
||||||
gpsStarted = false;
|
gpsStarted = false;
|
||||||
gpsFixAcquired = false;
|
gpsFixAcquired = false;
|
||||||
GBLocationManager.stop(getSupport().getContext(), getSupport());
|
GBLocationService.stop(getSupport().getContext(), getSupport().getDevice());
|
||||||
if (startOnPhone) {
|
if (startOnPhone) {
|
||||||
OpenTracksController.stopRecording(getSupport().getContext());
|
OpenTracksController.stopRecording(getSupport().getContext());
|
||||||
}
|
}
|
||||||
|
@ -500,27 +500,6 @@ public class GB {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void createGpsNotification(Context context, int numDevices) {
|
|
||||||
Intent notificationIntent = new Intent(context, ControlCenterv2.class);
|
|
||||||
notificationIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
|
|
||||||
PendingIntent pendingIntent = PendingIntentUtils.getActivity(context, 0, notificationIntent, 0, false);
|
|
||||||
|
|
||||||
NotificationCompat.Builder nb = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID_GPS)
|
|
||||||
.setTicker(context.getString(R.string.notification_gps_title))
|
|
||||||
.setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
|
|
||||||
.setContentTitle(context.getString(R.string.notification_gps_title))
|
|
||||||
.setContentText(context.getString(R.string.notification_gps_text, numDevices))
|
|
||||||
.setContentIntent(pendingIntent)
|
|
||||||
.setSmallIcon(R.drawable.ic_gps_location)
|
|
||||||
.setOngoing(true);
|
|
||||||
|
|
||||||
notify(NOTIFICATION_ID_GPS, nb.build(), context);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removeGpsNotification(Context context) {
|
|
||||||
removeNotification(NOTIFICATION_ID_GPS, context);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Notification createInstallNotification(String text, boolean ongoing,
|
private static Notification createInstallNotification(String text, boolean ongoing,
|
||||||
int percentage, Context context) {
|
int percentage, Context context) {
|
||||||
Intent notificationIntent = new Intent(context, ControlCenterv2.class);
|
Intent notificationIntent = new Intent(context, ControlCenterv2.class);
|
||||||
|
Loading…
Reference in New Issue
Block a user