项目需求:针对Android6.0以上需要动态申请权限的问题,强制让用户授予必须的权限,如果不授予则无法进入APP正常逻辑流程。
逻辑流程图
对于勾选了不再询问的处理
需要注意Android权限管理方面的机制。首先需要判断用户选择拒绝时是否勾选了以后不再询问,相关API:
public boolean shouldShowRequestPermissionRationale(@NonNull String permission)
如果返回值为false,则需要提示用户授予权限并跳转到系统设置界面,相关跳转代码如下:
Uri packageURI = Uri.parse("package:" + getPackageName());
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS, packageURI);
startActivity(intent);
系统版本判断
判断系统是否Android6.0以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
创建请求权限数组
String[] permission = new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE, ...}
//将没有赋予的权限放入下面的list中,再次请求
List<String> mPermissionList = new ArrayList<>();
请求权限函数:
private void checkPermmsion() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
for (int i = 0; i < permission.length; i++) {
if (ContextCompat.checkSelfPermission(this, permission[i]) != PackageManager.PERMISSION_GRANTED) {
mpPermissionList.add(permission[i]);
}
}
if (!mpPermissionList.isEmpty()) {
String[] permissionArr = mpPermissionList.toArray(new String[mpPermissionList.size()]);
ActivityCompat.requestPermissions(MainActivity.this, permissionArr, REQUEST_CODE);
}
}
}
请求权限的回调函数
这里有几种情况需要一一判断:
1.用户选择了同意还是拒绝
2.如果用户第二次选择了拒绝,则系统对话框会有不再提醒的复选框。这里要判断用户是否勾选过不再提醒的复选框。
回调函数的代码如下:
@RequiresApi(api = Build.VERSION_CODES.M)
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_CODE) {
for (int i = 0; i < grantResults.length; i++) {
if (grantResults[i] != PackageManager.PERMISSION_GRANTED) {
if (!shouldShowRequestPermissionRationale(permissions[i])) {
AlertDialog.Builder dialogType = new AlertDialog.Builder(MainActivity.this);
dialogType.setTitle("Warn");
dialogType.setMessage(getResources().getString(R.string.setPermissionByhand));
dialogType.setPositiveButton("sure", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Uri packageURI = Uri.parse("package:" + getPackageName());
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS, packageURI);
startActivity(intent);
}
});
dialogType.setCancelable(false);
dialogType.show();
} else {
AlertDialog.Builder dialogType = new AlertDialog.Builder(MainActivity.this);
dialogType.setTitle("Warn");
dialogType.setMessage(getResources().getString(R.string.requestPermissionAgain));
dialogType.setPositiveButton("sure", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
checkPermmsion();
}
});
dialogType.setCancelable(false);
dialogType.show();
}
}
}
}
}
如何避免用户使用退出键来回到APP主界面
使用上面的处理方式,如果用户在设置APP权限的界面点击了退出键,则可以回退到APP主界面,从而不受影响的使用APP(只有在该Activity生命周期结束,下次重新打开而执行到OnCreate函数时才会从新进行权限认证)。
要避免上面的情况产生,可以在调用系统设置界面时使用startActivityForResult这个API。
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivityForResult(intent, NOT_NOTICE);
同时复写Activity的onActivityResult方法
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if(requestCode==NOT_NOTICE){
myRequetPermission();//由于不知道是否选择了允许所以需要再次判断
}
}