Permission(使用与框架使用)

目录

   一、引言
   二、效果
   三、Normal Permission和Dangerous Permission
   四、实现支持运行时权限
   五、permissionsdispatcher框架的使用

一、引言

  在Android 6.0以前(targetSdkVersion 23)以前,我们安装APP时会列出所有该APP在Manifest中列出的访问权限,而且只会出现一次,一旦我们同意安装此APP并同意所有权限,APP就可以在用户不知情的情况下访问手机中的数据,包括(通讯录、定位)等隐私,在6.0以后,APP在运行时会一个一个询问用户授予权限。


二、效果

正常允许权限:

拒绝允许权限的处理


三、Normal Permission和Dangerous Permission

Google给出的关于权限分类的官方文档,
https://developer.android.google.cn/guide/topics/security/permissions.html#normal-dangerous

Dangero Permission以分组形式给出
Permission(使用与框架使用)_第1张图片

在Android6.0(API 23) 到8.0(API 26)之间,同一组的任何权限被授权了,其他权限也自动授权,此外对于申请时弹出的提示框上面的文本说明也是对整个权限组的说明,而不是单个权限的说明。但是Android 8.0(APi 26)以后对于危险权限需要一个一个的申请。


四、实现支持运行时权限

1、首先是我们拨打电话的点击事件:

 dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                call();
            }
        });

2、这时进行权限的检查,是否拥有拨打电话的权限,如果有则进行拨号,如果没有进行权限的申请:

 private void call() {
        //检查APP是否有权限
        if(ActivityCompat.checkSelfPermission(this,Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED){
            //如果没有
            ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CALL_PHONE},PERMISSION_REQURST_CALL_PHONE);
        }else{
            callPhone();
        }
    }
    
    /**
     * 拨打电话
     */
    private void callPhone() {
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:"+ 10086);
        intent.setData(data);
        startActivity(intent);
    }

3、权限申请回调的处理:

 @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if(requestCode == PERMISSION_REQURST_CALL_PHONE){
            if(grantResults[0] != 0  && grantResults[0] == PackageManager.PERMISSION_GRANTED){
            //允许申请的权限
                callPhone();
            }else{
            //申请权限被拒绝
            Toast.makeText(this,"申请权限被拒绝",Toast.SHORTTIME).show();
                
            }
            return;
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }

4、处理“不再询问”选项
  在申请权限时,如果我们点击“拒绝”,则下一次还会继续弹出权限申请框,只不过这一次会多出来一项“不再询问”,如果我们勾选了这个选项,则下一次不会再弹出此权限申请框,用户需要去设置—》应用权限列表中手动打开权限。

Permission(使用与框架使用)_第2张图片

用户对于已经勾选“不再询问”的权限,每次调用该权限API时都会失效,用户无法使用该功能,这显然不会带来好的用户体验,这个时候我们需要使用shouldShowRequestPermissionRationale()方法向用户解释权限使用情况,并设置进入应用的设置权限所在:
Permission(使用与框架使用)_第3张图片

 @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if(requestCode == PERMISSION_REQURST_CALL_PHONE){
            if(grantResults[0] != 0  && grantResults[0] == PackageManager.PERMISSION_GRANTED){
                callPhone();
            }else{
                if(!ActivityCompat.shouldShowRequestPermissionRationale(MainActivity.this,Manifest.permission.CALL_PHONE)){
                    final AlertDialog dialog = new AlertDialog.Builder(MainActivity.this)
                            .setTitle("提示")
                            .setMessage("该功能需要访问电话的权限,否则无法使用")
                            .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                                @Override
                                public void onClick(DialogInterface dialogInterface, int i) {
                                    dialogInterface.dismiss();
                                }
                            })
                            .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                                @Override
                                public void onClick(DialogInterface dialogInterface, int i) {
                                    //跳转到当前APP的设置页面
                                    Intent intent = new Intent();
                                    intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                                    intent.addCategory(Intent.CATEGORY_DEFAULT);
                                    intent.setData(Uri.parse("package:" + getPackageName()));
                                    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                                    intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                                    intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                                    startActivity(intent);
                                }
                            }).create();
                    dialog.show();
                }
            }
            return;
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }

五、permissionsdispatcher框架的使用

1、依赖
在Project的build.gradle 中添加如下代码:

dependencies {
        ...
        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
     
    }

在app的build.gradle中添加依赖:

dependencies {
    implementation 'com.github.hotchemi:permissionsdispatcher:2.1.3'
    annotationProcessor 'com.github.hotchemi:permissionsdispatcher-processor:2.1.3'
}

permissionsdispatcher以注释的方式进行权限的处理,其中几个重要的注释:

注释名称 注释解释
RuntimePermissions 必要的注释,用来注册一个Activity或者一个Fragment,使他们可以处理权限
NeedsPermission 必要的注释,在需要获取权限的地方进行注释,用来获取权限
OnShowRationale 提示用户为何需要开启此权限,在用户选择拒绝后,再次需要访问该权限时调用
OnPermissionDenied 用户选择拒绝时的提示
OnNeverAskAgain 用户选择不再询问后的提示

代码如下:


import android.Manifest;
import android.content.DialogInterface;
import android.content.Intent;
import android.net.Uri;
import android.provider.Settings;
import android.support.annotation.NonNull;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import permissions.dispatcher.NeedsPermission;
import permissions.dispatcher.OnNeverAskAgain;
import permissions.dispatcher.OnPermissionDenied;
import permissions.dispatcher.OnShowRationale;
import permissions.dispatcher.PermissionRequest;
import permissions.dispatcher.RuntimePermissions;

@RuntimePermissions
public class SecondActivity extends AppCompatActivity {

    private Button dialBtn;

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

    private void initView() {
        dialBtn = (Button) findViewById(R.id.dial_btn);
        dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
              
            }
        });
    }

    @NeedsPermission(Manifest.permission.CALL_PHONE)
    //在需要获取权限的地方注释
    void call(){
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:" + 10086);
        intent.setData(data);
        startActivity(intent);
    }

    @OnShowRationale(Manifest.permission.CALL_PHONE)
    //提示用户为何要开启权限
    void showWhy(final PermissionRequest request){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("提示用户为何要开启权限")
                .setPositiveButton("知道了", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        request.proceed(); //再次执行权限请求
                    }
                })
                .show();
    }

    @OnPermissionDenied(Manifest.permission.CALL_PHONE)
    void showDenied(){
        Toast.makeText(SecondActivity.this,"用户选择拒绝时的提示",Toast.LENGTH_SHORT).show();
    }

    @OnNeverAskAgain(Manifest.permission.CALL_PHONE)
    void showNotAsk(){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("该功能需要访问电话的权限,不开启将无法使用该功能")
                .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        //跳转到当前APP的设置页面
                        Intent intent = new Intent();
                        intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                        intent.addCategory(Intent.CATEGORY_DEFAULT);
                        intent.setData(Uri.parse("package:" + getPackageName()));
                        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                        intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                        intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                        startActivity(intent);
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        dialogInterface.dismiss();
                    }
                })
                .show();
    }

}

这个时候我们编译程序,会生成一个辅助类SecondActivityPermissionsDispatcher,下面的事情由它进行解决,完整代码:

@RuntimePermissions
public class SecondActivity extends AppCompatActivity {

    private Button dialBtn;

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

    private void initView() {
        dialBtn = (Button) findViewById(R.id.dial_btn);
        dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
           //*****************重点*****************
           SecondActivityPermissionsDispatcher.callWithCheck(SecondActivity.this);
            }
        });
    }

    @NeedsPermission(Manifest.permission.CALL_PHONE)
    //在需要获取权限的地方注释
    void call(){
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:" + 10086);
        intent.setData(data);
        startActivity(intent);
    }

    @OnShowRationale(Manifest.permission.CALL_PHONE)
    //提示用户为何要开启权限
    void showWhy(final PermissionRequest request){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("提示用户为何要开启权限")
                .setPositiveButton("知道了", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        request.proceed(); //再次执行权限请求
                    }
                })
                .show();
    }

    @OnPermissionDenied(Manifest.permission.CALL_PHONE)
    void showDenied(){
        Toast.makeText(SecondActivity.this,"用户选择拒绝时的提示",Toast.LENGTH_SHORT).show();
    }

    @OnNeverAskAgain(Manifest.permission.CALL_PHONE)
    void showNotAsk(){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("该功能需要访问电话的权限,不开启将无法使用该功能")
                .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        //跳转到当前APP的设置页面
                        Intent intent = new Intent();
                        intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                        intent.addCategory(Intent.CATEGORY_DEFAULT);
                        intent.setData(Uri.parse("package:" + getPackageName()));
                        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                        intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                        intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                        startActivity(intent);
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        dialogInterface.dismiss();
                    }
                })
                .show();
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
            //*****************重点*****************
            SecondActivityPermissionsDispatcher.onRequestPermissionsResult(SecondActivity.this,requestCode,grantResults);
    }
}

OK ,这是权限里面的内容了,希望能帮到你

你可能感兴趣的:(Android进阶)