Android6.0以上动态申请权限

项目需求:针对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 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();//由于不知道是否选择了允许所以需要再次判断 
   } 
}

你可能感兴趣的:(Android6.0以上动态申请权限)