Erreur "GPS nécessite ACCESS_FINE_LOCATION" d'Android, même si mon fichier manifeste contient ceci

104

Chaque fois que j'exécute l'application, mon SecurityException est levé et l'erreur du débogueur se lit comme suit:

java.lang.SecurityException: le fournisseur d'emplacement "gps" nécessite l'autorisation ACCESS_COARSE_LOCATION ou ACCESS_FINE_LOCATION.

Cela semble être une simple erreur, cependant, mon fichier manifeste est tout à fait correct. Le voici, et voici également mon code MapActivity:

<?xml version="1.0" encoding="utf-8"?>

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES" />
<uses-permission android:name="com.dev.cromer.jason.coverme.permission.MAPS_RECEIVE" />

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name=".MainActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>

    <meta-data
        android:name="com.google.android.gms.version"
        android:value="@integer/google_play_services_version" />
    <meta-data
        android:name="com.google.android.maps.v2.API_KEY"
        android:value= "@string/google_maps_key" />

    <activity
        android:name=".MapActivity"
        android:label="@string/title_activity_map" >
    </activity>
</application>

Mon activité:

    package com.dev.cromer.jason.coverme;

import android.location.Criteria;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.support.v4.app.FragmentActivity;
import android.os.Bundle;
import android.util.Log;

import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.location.LocationServices;
import com.google.android.gms.maps.CameraUpdateFactory;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.SupportMapFragment;
import com.google.android.gms.maps.model.LatLng;
import com.google.android.gms.maps.model.MarkerOptions;

public class MapActivity extends FragmentActivity implements LocationListener {

    private GoogleMap mMap; // Might be null if Google Play services APK is not available.

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

        setUpMapIfNeeded();
    }

    @Override
    protected void onResume() {
        super.onResume();
        setUpMapIfNeeded();
    }



    private void setUpMapIfNeeded() {
        // Do a null check to confirm that we have not already instantiated the map.
        if (mMap == null) {
            // Try to obtain the map from the SupportMapFragment.
            mMap = ((SupportMapFragment) getSupportFragmentManager().findFragmentById(R.id.map))
                    .getMap();

            // Check if we were successful in obtaining the map.
            if (mMap != null) {
                //mMap.setMyLocationEnabled(true);
                //mMap.setOnMyLocationChangeListener(this);
                setUpMap();
            }
        }
    }


    private void setUpMap() {
        mMap.addMarker(new MarkerOptions().position(new LatLng(0, 0)).title("Marker"));
        mMap.setMyLocationEnabled(true);

        LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);

        try {
            Location myLocation = locationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);

            if (myLocation != null) {
                Log.d("TAG", "Not null");
            }
            else {
                Log.d("TAG", "NULL");
                locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, this);
            }
        }
        catch (SecurityException se) {
            Log.d("TAG", "SE CAUGHT");
            se.printStackTrace();
        }
    }


    @Override
    public void onLocationChanged(Location location) {
        Log.d("CHANGED", "LOCATION UPDATED");

    }

    @Override
    public void onStatusChanged(String provider, int status, Bundle extras) {

    }

    @Override
    public void onProviderEnabled(String provider) {

    }

    @Override
    public void onProviderDisabled(String provider) {

    }
}
Jason Cromer
la source
Sur quelle version d'Android testez-vous cela?
CommonsWare
4
Indépendant, mais si vous demandez un bon emplacement, vous n'avez pas besoin de demander un emplacement grossier. C'est inclus.
joey_g216

Réponses:

136

ACCESS_COARSE_LOCATION, ACCESS_FINE_LOCATIONet WRITE_EXTERNAL_STORAGEfont tous partie du système d'autorisation d'exécution d'Android 6.0 . En plus de les avoir dans le manifeste comme vous le faites, vous devez également les demander à l'utilisateur au moment de l'exécution (en utilisant requestPermissions()) et voir si vous les avez (en utilisant checkSelfPermission()).

Une solution de contournement à court terme consiste à laisser tomber votre targetSdkVersionsous 23.

Mais, éventuellement, vous voudrez mettre à jour votre application pour utiliser le système d'autorisation d'exécution.

Par exemple, cette activité fonctionne avec cinq autorisations. Quatre sont des autorisations d'exécution, bien qu'il n'en gère actuellement que trois (je l'ai écrit avant d' WRITE_EXTERNAL_STORAGEêtre ajouté à la liste des autorisations d'exécution).

/***
 Copyright (c) 2015 CommonsWare, LLC
 Licensed under the Apache License, Version 2.0 (the "License"); you may not
 use this file except in compliance with the License. You may obtain a copy
 of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
 by applicable law or agreed to in writing, software distributed under the
 License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
 OF ANY KIND, either express or implied. See the License for the specific
 language governing permissions and limitations under the License.

 From _The Busy Coder's Guide to Android Development_
 https://commonsware.com/Android
 */

package com.commonsware.android.permmonger;

import android.Manifest;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {
  private static final String[] INITIAL_PERMS={
    Manifest.permission.ACCESS_FINE_LOCATION,
    Manifest.permission.READ_CONTACTS
  };
  private static final String[] CAMERA_PERMS={
    Manifest.permission.CAMERA
  };
  private static final String[] CONTACTS_PERMS={
      Manifest.permission.READ_CONTACTS
  };
  private static final String[] LOCATION_PERMS={
      Manifest.permission.ACCESS_FINE_LOCATION
  };
  private static final int INITIAL_REQUEST=1337;
  private static final int CAMERA_REQUEST=INITIAL_REQUEST+1;
  private static final int CONTACTS_REQUEST=INITIAL_REQUEST+2;
  private static final int LOCATION_REQUEST=INITIAL_REQUEST+3;
  private TextView location;
  private TextView camera;
  private TextView internet;
  private TextView contacts;
  private TextView storage;

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

    location=(TextView)findViewById(R.id.location_value);
    camera=(TextView)findViewById(R.id.camera_value);
    internet=(TextView)findViewById(R.id.internet_value);
    contacts=(TextView)findViewById(R.id.contacts_value);
    storage=(TextView)findViewById(R.id.storage_value);

    if (!canAccessLocation() || !canAccessContacts()) {
      requestPermissions(INITIAL_PERMS, INITIAL_REQUEST);
    }
  }

  @Override
  protected void onResume() {
    super.onResume();

    updateTable();
  }

  @Override
  public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.actions, menu);

    return(super.onCreateOptionsMenu(menu));
  }

  @Override
  public boolean onOptionsItemSelected(MenuItem item) {
    switch(item.getItemId()) {
      case R.id.camera:
        if (canAccessCamera()) {
          doCameraThing();
        }
        else {
          requestPermissions(CAMERA_PERMS, CAMERA_REQUEST);
        }
        return(true);

      case R.id.contacts:
        if (canAccessContacts()) {
          doContactsThing();
        }
        else {
          requestPermissions(CONTACTS_PERMS, CONTACTS_REQUEST);
        }
        return(true);

      case R.id.location:
        if (canAccessLocation()) {
          doLocationThing();
        }
        else {
          requestPermissions(LOCATION_PERMS, LOCATION_REQUEST);
        }
        return(true);
    }

    return(super.onOptionsItemSelected(item));
  }

  @Override
  public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
    updateTable();

    switch(requestCode) {
      case CAMERA_REQUEST:
        if (canAccessCamera()) {
          doCameraThing();
        }
        else {
          bzzzt();
        }
        break;

      case CONTACTS_REQUEST:
        if (canAccessContacts()) {
          doContactsThing();
        }
        else {
          bzzzt();
        }
        break;

      case LOCATION_REQUEST:
        if (canAccessLocation()) {
          doLocationThing();
        }
        else {
          bzzzt();
        }
        break;
    }
  }

  private void updateTable() {
    location.setText(String.valueOf(canAccessLocation()));
    camera.setText(String.valueOf(canAccessCamera()));
    internet.setText(String.valueOf(hasPermission(Manifest.permission.INTERNET)));
    contacts.setText(String.valueOf(canAccessContacts()));
    storage.setText(String.valueOf(hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE)));
  }

  private boolean canAccessLocation() {
    return(hasPermission(Manifest.permission.ACCESS_FINE_LOCATION));
  }

  private boolean canAccessCamera() {
    return(hasPermission(Manifest.permission.CAMERA));
  }

  private boolean canAccessContacts() {
    return(hasPermission(Manifest.permission.READ_CONTACTS));
  }

  private boolean hasPermission(String perm) {
    return(PackageManager.PERMISSION_GRANTED==checkSelfPermission(perm));
  }

  private void bzzzt() {
    Toast.makeText(this, R.string.toast_bzzzt, Toast.LENGTH_LONG).show();
  }

  private void doCameraThing() {
    Toast.makeText(this, R.string.toast_camera, Toast.LENGTH_SHORT).show();
  }

  private void doContactsThing() {
    Toast.makeText(this, R.string.toast_contacts, Toast.LENGTH_SHORT).show();
  }

  private void doLocationThing() {
    Toast.makeText(this, R.string.toast_location, Toast.LENGTH_SHORT).show();
  }
}

(à partir de cet exemple de projet )

Pour la fonction requestPermissions (), les paramètres doivent-ils être simplement "ACCESS_COARSE_LOCATION"? Ou devrais-je inclure le nom complet "android.permission.ACCESS_COARSE_LOCATION"?

J'utiliserais les constantes définies sur Manifest.permission, comme indiqué ci-dessus.

Aussi, quel est le code de demande?

Cela vous sera renvoyé en tant que premier paramètre à onRequestPermissionsResult(), afin que vous puissiez distinguer un requestPermissions()appel d'un autre.

CommonsWare
la source
1
Pour la fonction requestPermissions (), les paramètres doivent-ils être simplement "ACCESS_COARSE_LOCATION"? Ou devrais-je inclure le nom complet "android.permission.ACCESS_COARSE_LOCATION"?
Jason Cromer
1
Merci, cela a éliminé l'erreur. J'ai toujours des difficultés à accéder à ma position, car mon locationManager continue de renvoyer ma position comme nulle, mais cela n'est pas pertinent pour ce bogue. Merci pour votre solution!
Jason Cromer
@CommonsWare: Que voulez-vous dire par «éventuellement»? Désolé, je ne comprends pas cette partie.
theapache64
1
@ theapache64: Un jour, quelque chose va vous amener à vouloir régler votre targetSdkVersionà 23 ou plus. À ce stade, vous devrez adopter le système d'autorisation d'exécution. Jusqu'à ce que ce moment-là vienne, vous êtes invités à garder votre targetSdkVersionmoins de 23 ans et à ignorer les autorisations d'exécution.
CommonsWare
@CommonsWare: Maintenant je l'ai. :)
theapache64
39

Ma solution simple est la suivante

if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_FINE_LOCATION) ==
        PackageManager.PERMISSION_GRANTED &&
        ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_COARSE_LOCATION) ==
        PackageManager.PERMISSION_GRANTED) {
    googleMap.setMyLocationEnabled(true);
    googleMap.getUiSettings().setMyLocationButtonEnabled(true);
} else {
    Toast.makeText(this, R.string.error_permission_map, Toast.LENGTH_LONG).show();
}

ou vous pouvez ouvrir la boîte de dialogue d'autorisation dans un autre comme celui-ci

} else {
   ActivityCompat.requestPermissions(this, new String[] {
      Manifest.permission.ACCESS_FINE_LOCATION, 
      Manifest.permission.ACCESS_COARSE_LOCATION }, 
      TAG_CODE_PERMISSION_LOCATION);
}
Vasil Valchev
la source
Laways emménage dans une autre partie frère :(
Ashana.Jackol
2
ajoutez une boîte de dialogue pour ajouter des autorisations dans ce "autre", et vous êtes prêt à partir.
Vasil Valchev
C'est en fait sans aucun doute le correctif pour Android 6. Il faut noter que vous devez mettre les demandes d'autorisation dans le else.
Keith Adler du
J'ai eu cette erreur avec le SDK cible en tant que 22 et sur Android 5.1 sur l'appareil S plus (GiONEE_WBL7511). Je ne comprends pas pourquoi cet accident s'est produit. Des indices? java.lang.SecurityException: le client doit avoir l'autorisation ACCESS_FINE_LOCATION pour demander des emplacements PRIORITY_HIGH_ACCURACY.
arpitgoyal2008
5

CAUSE: "À partir d'Android 6.0 (niveau d'API 23), les utilisateurs accordent des autorisations aux applications pendant que l'application est en cours d'exécution, pas lors de l'installation de l'application." Dans ce cas, "ACCESS_FINE_LOCATION" est une "permission dangereuse et pour cette raison, vous obtenez cette 'java.lang.SecurityException: le fournisseur de localisation" gps "requiert l'autorisation ACCESS_FINE_LOCATION.' erreur ( https://developer.android.com/training/permissions/requesting.html ).

SOLUTION: implémenter le code fourni sur https://developer.android.com/training/permissions/requesting.html sous les en-têtes «Demander les autorisations dont vous avez besoin» et «Gérer la réponse à la demande d'autorisations».

Jaime Montoya
la source