Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Monitoring Location Settings in a Library

I'm working on a library that can be used to monitor geofences on Android devices. I've noticed that the geofences I register with Google Play Location Services GeofencingApi class are lost after I turn Location Services off and back on in the device settings.

I have seen folks suggesting that I need to register for the android.location.PROVIDERS_CHANGED broadcast receiver in my AndroidManifest.xml file. I've done that -- but this broadcast receiver only gets called when I add and remove test providers in my test application. I don't receive it at all when toggling Location Services in the device settings.

Am I doing something wrong? Does anyone know how to reliably determine when the user toggles Location Services in the device settings? I'd like to be able to see these events in the background -- even if my app is not running.

I figure I could set up an repeating task that runs in the background and periodically checks to see if Location Services has been toggled, but it sounds gross and inefficient.

If it helps, I'm testing on an Moto G running Android 4.4.2. Everything I have done with geofences has worked fine until now.

EDIT: After doing more research I have found that the behaviour of the PROVIDERS_CHANGED broadcast is highly variable depending on phone version and model. My Nexus 5 running Android 5.1 seems to work fine actually - I am able to get the PROVIDERS_CHANGED broadcast very regularly. I also have a Moto G and Moto X phone running 4.4.x and they never produced the PROVIDERS_CHANGED broadcast for me. A Samsung Galaxy S3 on Android 4.2 would produce the broadcast for me but stop doing it after I used my Map Activity with test location providers.

Anyways, I have decide to stop pursuing this problem for now. I think Android is just being buggy.

like image 697
Rob Szumlakowski Avatar asked May 26 '15 20:05

Rob Szumlakowski


1 Answers

It's hard to tell without seeing your code if you are doing anything wrong, but I just got a simple example working and tested by using the code from here as a guide, and adapting it to have the BroadcastReceiver in a library.

One thing to note: The app using the library will need to have the receiver in its AndroidManifest.xml. See here and here, both contain information from @CommonsWare.

Adding quote here just in case the link goes bad:

Is there any way for a library project to independently register for a receiver in it's own manifest file?

Not at the present time.

So, any app that uses your library will need to have the receiveradded in its AndroidManifest.xml.

Note that this was tested on Android 4.4.4 on a Samsung S4.

I got this working using a BroadcastReceiver in a library project compiled to a aar.

Using a receiver element set up in the AndroidManifest.xml of the main app that links to the BroadcastReceiver in the library, I was able to receive the event when any change is made to the Location settings:

  • Location toggle on
  • Location toggle off
  • Change Location Mode to High Accuracy
  • Change Location Mode to Power Saving
  • Change Location Mode to GPS only

Every time a change is made in the location settings, the BroadcastReceiver in the library is triggered, even if the app is not running.

I used a simple test where the BroadcastReceiver in the library shows a Toast message every time the location settings are changed.

Here is LocationProviderChangedReceiver.java in the library project:

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.location.LocationManager;
import android.util.Log;
import android.widget.Toast;

public class LocationProviderChangedReceiver extends BroadcastReceiver {
    private final static String TAG = "LocationProviderChanged";

    boolean isGpsEnabled;
    boolean isNetworkEnabled;

    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent.getAction().matches("android.location.PROVIDERS_CHANGED"))
        {
            Log.i(TAG, "Location Providers changed");

            LocationManager locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
            isGpsEnabled = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
            isNetworkEnabled = locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER);

            Toast.makeText(context, "GPS Enabled: " + isGpsEnabled + " Network Location Enabled: " + isNetworkEnabled, Toast.LENGTH_LONG).show();
        }
    }

}

Then, in the main app, I put the compiled myLibrary.aar in the libs folder, and set up the build.gradle to compile the aar:

 repositories{
        flatDir{
            dirs 'libs'
        }
    }
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile "com.android.support:appcompat-v7:22.1.1"
    compile "com.google.android.gms:play-services:7.3.0"

    compile 'com.mylibrary.daniel:mylibrary:1.0@aar'

}

Set up the receiver in the AndroidManifest.xml of the main app:

    <receiver
        android:name="com.mylibrary.daniel.mylibrary.LocationProviderChangedReceiver"
        android:exported="false" >
        <intent-filter>
            <action android:name="android.location.PROVIDERS_CHANGED" />
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </receiver>

Then, installed the app, and modified location settings in a few different ways. The BroadcastReceiver in the library was triggered every time:

enter image description here

enter image description here

enter image description here

enter image description here

like image 179
Daniel Nugent Avatar answered Oct 12 '22 07:10

Daniel Nugent