Невозможно обновить настройки GPS до высокой точности с помощью FusedLocationProvider На некоторых устройствах - PullRequest
1 голос
/ 06 апреля 2019

Я использую SettingsApi и FusedLocationProvider для обновления настроек GPS и получения обновлений местоположения. Мне нужны высокоточные обновления местоположения, для которых отображается диалоговое окно «Включить GPS» с использованием SettingsApi для обновления настроек GPS до высокой точности, но в некоторые устройства (например, Mi и Gionee), даже если пользователь нажал кнопку «ОК» в диалоговом окне «Включить Gps», я получаю RESULT_CANCELED на onActivityResult, в то время как на других устройствах, таких как Motorola, Lenovo

, все работает отлично.

Когда пользователь нажимает кнопку в диалоговом окне включения GPS

  1. Если Gps выключен, он включается и переходит в режим только устройства (режим только Gps)
  2. Если Gps включен, то Gps выключен, и я получаю RESULT_CANCELED на onActivityResult в обоих случаях

Вот мой код

  1. LocationHelper
    public class LocationHelper {

        private static final String TAG = LocationHelper.class.getSimpleName();

        private long updateIntervalInMilliseconds = 10000;

        private long fastestUpdateIntervalInMilliseconds = updateIntervalInMilliseconds / 2;

        private FusedLocationProviderClient mFusedLocationClient;

        private SettingsClient mSettingsClient;

        private LocationRequest mLocationRequest;

        private LocationSettingsRequest mLocationSettingsRequest;

        private LocationCallback mLocationCallback;


        private Boolean mRequestingLocationUpdates = false;


        private int requiredGpsPriority = LocationRequest.PRIORITY_BALANCED_POWER_ACCURACY;


        public LocationHelper(Context mContext) {
            mFusedLocationClient = LocationServices.getFusedLocationProviderClient(mContext);
            mSettingsClient = LocationServices.getSettingsClient(mContext);
        }

        /**
         * Sets required gps priority
         * <p>
         * Gps Priority can be
         * <ul>
         * <li>LocationRequest.PRIORITY_HIGH_ACCURACY</li>
         * <li>LocationRequest.PRIORITY_BALANCED_POWER_ACCURACY</li>
         * <li>LocationRequest.PRIORITY_NO_POWER</li>
         * <li>LocationRequest.PRIORITY_LOW_POWER</li>
         * </ul>
         * <p>
         * default is LocationRequest.PRIORITY_BALANCED_POWER_ACCURACY
         *
         * @param requiredGpsPriority gps priority
         */
        public void setRequiredGpsPriority(int requiredGpsPriority) {
            this.requiredGpsPriority = requiredGpsPriority;
        }

        /**
         * Sets Update Interval also sets fastestUpdateIntervalInMilliseconds to half of updateIntervalInMilliseconds
         * default is 10 seconds
         *
         * @param updateIntervalInMilliseconds update Interval
         */
        public void setUpdateInterval(long updateIntervalInMilliseconds) {
            this.updateIntervalInMilliseconds = updateIntervalInMilliseconds;
            this.fastestUpdateIntervalInMilliseconds = updateIntervalInMilliseconds / 2;
        }

        /**
         * Sets fastest Update Interval
         * default is 5 seconds
         *
         * @param fastestUpdateIntervalInMilliseconds fastest update Interval
         */
        public void setFastestUpdateIntervalInMilliseconds(long fastestUpdateIntervalInMilliseconds) {
            this.fastestUpdateIntervalInMilliseconds = fastestUpdateIntervalInMilliseconds;
        }


        public void init() {
            createLocationRequest();
            buildLocationSettingsRequest();
        }


        public void setLocationCallback(LocationCallback locationCallback) {
            this.mLocationCallback = locationCallback;
        }

        private void createLocationRequest() {
            mLocationRequest = new LocationRequest();

            mLocationRequest.setInterval(updateIntervalInMilliseconds);

            mLocationRequest.setFastestInterval(fastestUpdateIntervalInMilliseconds);

            mLocationRequest.setPriority(requiredGpsPriority);
        }


        private void buildLocationSettingsRequest() {
            LocationSettingsRequest.Builder builder = new LocationSettingsRequest.Builder();
            builder.addLocationRequest(mLocationRequest);
            builder.setAlwaysShow(true);
            mLocationSettingsRequest = builder.build();
        }


        public boolean isRequestingForLocation() {
            return mRequestingLocationUpdates;
        }


        public void checkForGpsSettings(GpsSettingsCheckCallback callback) {

            if (mLocationSettingsRequest == null) {
                throw new IllegalStateException("must call init() before check for gps settings");
            }

            // Begin by checking if the device has the necessary jobLocation settings.
            mSettingsClient.checkLocationSettings(mLocationSettingsRequest)
                    .addOnSuccessListener(locationSettingsResponse -> callback.requiredGpsSettingAreAvailable())
                    .addOnFailureListener(e -> {

                        int statusCode = ((ApiException) e).getStatusCode();
                        switch (statusCode) {
                            case LocationSettingsStatusCodes.RESOLUTION_REQUIRED:
                                Log.i(TAG, "SuggestedLocation settings are not satisfied. notifying back to the requesting object ");

                                ResolvableApiException rae = (ResolvableApiException) e;
                                callback.requiredGpsSettingAreUnAvailable(rae);

                                break;

                            case LocationSettingsStatusCodes.SETTINGS_CHANGE_UNAVAILABLE:
                                Log.i(TAG, "Turn On SuggestedLocation From Settings. ");

                                callback.gpsSettingsNotAvailable();
                                break;
                        }

                    });
        }


        /**
         * Starts location updates from the FusedLocationApi.
         * <p>
         *     Consider Calling {@link #stopLocationUpdates()} when you don't want location updates it helps in saving battery
         * </p>
         */
        public void startLocationUpdates() {

            if (mLocationRequest == null) {
                throw new IllegalStateException("must call init() before requesting location updates");
            }

            if (mLocationCallback == null) {
                throw new IllegalStateException("no callback provided for delivering location updates,use setLocationCallback() for setting callback");
            }

            if (mRequestingLocationUpdates) {
                Log.d(TAG, "startLocationUpdates: already requesting location updates, no-op.");
                return;
            }

            Log.d(TAG, "startLocationUpdates: starting updates.");
            mFusedLocationClient.requestLocationUpdates(mLocationRequest, mLocationCallback, Looper.myLooper())
                    .addOnCompleteListener(task -> mRequestingLocationUpdates = true);

        }

        public void stopLocationUpdates() {
            if (!mRequestingLocationUpdates) {
                Log.d(TAG, "stopLocationUpdates: updates never requested, no-op.");
                return;
            }

            Log.d(TAG, "stopLocationUpdates: stopping location updates.");
            mFusedLocationClient.removeLocationUpdates(mLocationCallback)
                    .addOnCompleteListener(task -> mRequestingLocationUpdates = false);
        }
}
  1. GpsSettingsCheckCallback
    public interface GpsSettingsCheckCallback {


        /**
         * We don't have required Gps Settings
         * ex For High Accuracy Locations We Need Gps In High Accuracy Settings
         *
         * How To show "Turn On Gps Dialog" ?
         * 
         * From Activity :
         * <code>status.startResolutionForResult(this , REQUEST_CHECK_SETTINGS);</code>
         * 
         * From Fragment :
         * <code>
         * startIntentSenderForResult(status.getResolution().getIntentSender(), REQUEST_CHECK_SETTINGS, null, 0, 0, 0, null)
         * </code>
         */
        void requiredGpsSettingAreUnAvailable(ResolvableApiException status);


        /**
         * Everything's Good
         */
        void requiredGpsSettingAreAvailable();


        /**
         * Gps Settings Are Unavailable redirect user to settings page to turn on location
         */
        void gpsSettingsNotAvailable();
     }
  1. Код деятельности
public class CheckGpsActivity extends AppCompatActivity {

    public static final String TAG = CheckGpsActivity.class.getSimpleName();
    public static final int REQUEST_LOCATION_SETTINGS_UPGRADE = 23;

    private Button turnOnLocationUpdatesBtn, turnOffLocationBtn, checkForRequredGpsSettingBtn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        LocationHelper locationHelper = new LocationHelper(this);
        locationHelper.setRequiredGpsPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
        locationHelper.init();

        locationHelper.setLocationCallback(new LocationCallback() {

            @Override
            public void onLocationResult(LocationResult locationResult) {
                super.onLocationResult(locationResult);
                Location location = locationResult.getLocations().get(0);

                if (location != null)
                    Log.d(TAG, "Gps Coords" + location.getLatitude() + "," + location.getLongitude());
            }

        });

        turnOnLocationUpdatesBtn.setOnClickListener(view -> locationHelper.startLocationUpdates());

        turnOffLocationBtn.setOnClickListener(view -> locationHelper.startLocationUpdates());

        checkForRequredGpsSettingBtn.setOnClickListener(view -> {

            locationHelper.checkForGpsSettings(new GpsSettingsCheckCallback() {
                @Override
                public void requiredGpsSettingAreUnAvailable(ResolvableApiException status) {

                    Log.d(TAG, "require gps settings upgrade");
                    try {
                        status.startResolutionForResult(CheckGpsActivity.this, REQUEST_LOCATION_SETTINGS_UPGRADE);
                    } catch (IntentSender.SendIntentException e) {
                        e.printStackTrace();
                    }

                }

                @Override
                public void requiredGpsSettingAreAvailable() {
                    Log.d(TAG, "Gps Setting are just fine");
                }

                @Override
                public void gpsSettingsNotAvailable() {
                    Log.d(TAG, "Gps Setting unavailable, redirect to settings");
                }
            });

        });

    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        //Result code I always get is 0 (RESULT_CANCELED) even if user clicked Ok in Turn On Location dialog
    }
}

1 Ответ

0 голосов
/ 24 апреля 2019

С самой библиотекой проблем нет, проблема с приложением под названием FusedLocationProvider. Перейдите в «Настройки» -> «Установленные приложения» -> «FusedLocationProvider» -> «Очистить кэш и данные», теперь ваше приложение будет нормально работать.

WorkAround # 1: Вы можете выбрать

  1. LocationRequest.PRIORITY_BALANCED_POWER_ACCURACY - Более быстрые обновления местоположения, но точность определения местоположения может быть низкой
  2. LocationRequest.PRIORITY_LOW_POWER - Медленнее или, возможно, без обновлений местоположения (если в помещенииили под землей)

WorkAround # 2: перенаправить пользователя на страницу настроек

 Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
 startActivityForResult(intent, REQUEST_UPDATE_GPS_SETTINGS_MANUALLY);
...