京东动态申请定位权限分析
1、安装完成后,权限状态为询问状态,此时,进入应用提示
1.1点击取消,对话框消失,Toast提示信息如下
需在手机设置中打开京东的位置信息权限。
1.2在1中点击"去打开",对话框消失,系统对话框提示
1.2.1未选中"禁止后不再询问",点击拒绝,提示信息如下
1.2.1.1点击取消,Toast提示信息和1.1一致
1.2.1.2点击"去允许",系统提示信息和1.2一致
1.2.2选中"禁止后不再询问",点击拒绝
此时点击取消显示Toast信息
点击去打开跳转到应用信息页面,在下面这个页面中打开定位权限即可
高仿京东动态申请定位权限
第一、在配置文件中声明定位权限
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
第二、检查是否获取权限(关键方法)
/**
* 第一步调用检查定位权限
*/
public void checkLocationPermission() {
Context activity = getActivity();
//未获取权限
if (ContextCompat.checkSelfPermission(activity,
Manifest.permission.ACCESS_FINE_LOCATION)
!= PackageManager.PERMISSION_GRANTED) {
//询问状态
if (ActivityCompat.shouldShowRequestPermissionRationale(getActivity(), Manifest.permission.ACCESS_FINE_LOCATION)) {
/*在Fragment中不能使用以下方式申请权限,否则,会接收不到结果
ActivityCompat.requestPermissions(getActivity(),
new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
REQUEST_PERMISSION_LOCATION);*/
new AlertDialog.Builder(getActivity())
.setMessage("为保证您正常地使用此功能,需要获取您的位置使用权限,请允许")// 禁止状态
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
showMessage("需在手机设置中打开应用的拍照权限.");
dialog.dismiss();
}
})
.setPositiveButton("去允许", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//在Fragment中申请权限
requestPermissions(new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_PERMISSION_LOCATION);
dialog.dismiss();
}
})
.show();
} else {
/*
* 禁止状态
* 注意:禁止状态调用申请权限的方法requestPermissions没反应,此时提示用户为什么要这个权限且跳转至应用信息页面,让用户开启权限
* */
new AlertDialog.Builder(getActivity())
.setMessage("未取得您的位置信息使用权限,此功能无法使用。请前往应用权限设置打开权限")// 禁止状态
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Toast.makeText(getActivity(), mCancelTip, Toast.LENGTH_SHORT);
dialog.dismiss();
}
})
.setPositiveButton("去打开", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//去应用信息页面
goToAppDetailSettingIntent(getActivity());
dialog.dismiss();
}
})
.show();
}
} else {
//已经允许
startLocation();
}
}
第三、提供用户点击始终允许后回调的方法
public abstract void startLocation();
BaseLocationFragment.java基类完整代码如下
/**
* 类描述:请求定位权限Fragment有需要定位权限功能的应用,在Actvity的布局中使用fragment标签,name设置为该
* 来的实现类即可,在回调方法中调用定位API即可
* 作者:xues
* 时间:2017年09月02日
*/
public abstract class BaseLocationFragment extends BaseFragment {
private static final int REQUEST_PERMISSION_LOCATION = 111;//定位权限请求码
private static final String mCancelTip = "需在手机设置中打开京东的位置信息权限。";//取消提示信息
/**
* 第一步调用检查定位权限
*/
public void checkLocationPermission() {
Context activity = getActivity();
//未获取权限
if (ContextCompat.checkSelfPermission(activity,
Manifest.permission.ACCESS_FINE_LOCATION)
!= PackageManager.PERMISSION_GRANTED) {
//询问状态
if (shouldShowRequestPermissionRationale(Manifest.permission.ACCESS_FINE_LOCATION)) {
/*在Fragment中不能使用以下方式申请权限,否则,会接收不到结果
ActivityCompat.requestPermissions(getActivity(),
new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
REQUEST_PERMISSION_LOCATION);*/
new AlertDialog.Builder(getActivity())
.setMessage("为保证您正常地使用此功能,需要获取您的位置使用权限,请允许")// 禁止状态
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Toast.makeText(getActivity(), mCancelTip, Toast.LENGTH_SHORT).show();
dialog.dismiss();
}
})
.setPositiveButton("去允许", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//在Fragment中申请权限
requestPermissions(new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_PERMISSION_LOCATION);
dialog.dismiss();
}
})
.show();
} else {
/*
* 禁止状态
* 注意:禁止状态调用申请权限的方法requestPermissions没反应,此时提示用户为什么要这个权限且跳转至应用信息页面,让用户开启权限
* */
new AlertDialog.Builder(getActivity())
.setMessage("未取得您的位置信息使用权限,此功能无法使用。请前往应用权限设置打开权限")// 禁止状态
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Toast.makeText(getActivity(), mCancelTip, Toast.LENGTH_SHORT);
dialog.dismiss();
}
})
.setPositiveButton("去打开", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
//去应用信息页面
goToAppDetailSettingIntent(getActivity());
dialog.dismiss();
}
})
.show();
}
} else {
//已经允许
startLocation();
}
}
/**
* 跳转到应用信息页面
*
* @param context
*/
public static void goToAppDetailSettingIntent(Context context) {
Intent localIntent = new Intent();
localIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
if (Build.VERSION.SDK_INT >= 9) {
localIntent.setAction("android.settings.APPLICATION_DETAILS_SETTINGS");
localIntent.setData(Uri.fromParts("package", context.getPackageName(), null));
} else if (Build.VERSION.SDK_INT <= 8) {
localIntent.setAction(Intent.ACTION_VIEW);
localIntent.setClassName("com.android.settings", "com.android.settings.InstalledAppDetails");
localIntent.putExtra("com.android.settings.ApplicationPkgName", context.getPackageName());
}
context.startActivity(localIntent);
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case REQUEST_PERMISSION_LOCATION: {
if (grantResults.length > 0
&& grantResults[0] == PackageManager.PERMISSION_GRANTED) {
startLocation();
}
}
}
}
/**
* 在该方法中调用定位服务
*/
public abstract void startLocation();
}
BaseLocationFragment的使用
1、继承BaseLocationFragment,让BaseLocationFragment继承你自己的BaseFragment,如果你想要在Activity中使用,建议在Activity中使用Fragment即可
2、定位前先调用checkLocationPermission检查定位服务,用户允许后会回调startLocation方法
3、在startLocation方法中,启动定位服务即可
动态申请权限时,主要有以下几点需要开发者注意的地方
1、用户选中“禁止后不再询问”再点击“拒绝” 后,代码就不做处理了,这样的结果会导致下次调用申请权限的方法时,没有任何反应(BaseLocationFragment已经处理)
2、在Fragment中申请权限时,不能使用以下方法进行申请,这样的做法会导致Fragment中的方法onRequestPermissionsResult不执行
ActivityCompat.requestPermissions(getActivity(),
new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
REQUEST_PERMISSION_LOCATION);
正确的做法是使用fragment中的方法,而不是ActivityCompat的方法
requestPermissions(new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, REQUEST_PERMISSION_LOCATION);
3、shouldShowRequestPermissionRationale,这个方法,个人的理解是:返回false表示当前权限的状态为禁止状态,即选中“禁止后不再询问”,点击拒绝后的状态
到这里,使用原生API动态申请权限的分析就结束了,有什么不正确的地方,请指示,共同进步!