Permission组件解析

Android 6.0之后,对于一些敏感的权限,需要我们手动请求,本文介绍一个权限组件Permisssion,主要是用于Android6.0以后(也就是API>23)的敏感权限请求,如果项目中的targetSdkVersion大于等于23,并且某个activity要需要一个或者多个危险权限的话可以用此组件进行判断是否拥有权限以及请求权限。

组件的使用

1.在module的build.gradle 中增加: compile 'com.neteaseyx.permission:permission-lib:1.0.0',
2.创建PermissionManager对象,如果是在fragment中使用传入参数是传入 mFragment.getActivity()

 mPermissionManager = new PermissionManager(this);//创建PermissionManager对象

3.明确所需权限,可以是多个权限(哪些是危险权限都可以查得到),这些权限都是用Manifest.permission.xxx声明

 final String[] permissions = {
            Manifest.permission.CAMERA,
            Manifest.permission.WRITE_EXTERNAL_STORAGE,
            Manifest.permission.ACCESS_FINE_LOCATION
    };

4.在需要请求权限的地方使用PermissionManager.requestPermissions 方法请求权限,第一个参数是请求码,第二个参数是要请求的权限,第三个参数是请求结果的回调,PermissionCallback里面有四个回调方法,onGranted_Third是回调所有允许的权限,onExplanation是回调拒绝但是没有勾选不再询问的权限,onNormalDenied_Second是回调所有拒绝的权限,onNoMoreAskDenied_First是回调所有已经拒绝,而且勾选了不再询问的权限

mPermissionManager.requestPermissions(PERMISSION_REQ_CODE, permissions, new PermissionCallback() {

                @Override
                public void onGranted_Third(String... permissions) {    //回调所有允许的权限
                    for (String permission : permissions) {
                        addTextToResult(permission + "\n  -已批准\n", permission);
                    }
                }

                @Override
                public void onExplanation(String... permissions) {  //拒绝但没有勾选不再询问,会通过这个方法回调
                    mPermissionManager.showRequestDialog(MainActivity.this, permissions, PERMISSION_REQ_CODE);
                }

                @Override
                public void onNormalDenied_Second(String... permissions) { //回调所有拒绝的权限
                    for (String permission : permissions) {
                        addTextToResult(permission + "\n  -已拒绝\n", permission);
                    }
                }

                @Override
                public void onNoMoreAskDenied_First(String... permissions) { //拒绝,而且勾选了不再询问
                    for (String permission : permissions) {
                        addTextToResult(permission + "\n  -已永久拒绝\n", permission);
                    }
                }
            });

5.在 Activity 或者 Fragment 的 onRequestPermissionsResult 方法中调用 PermissionManager.onRequestPermissionResult, 否则不会有任何结果回调

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    mPermissionManager.onRequestPermissionResult(requestCode, permissions, grantResults);
}

组件内部实现

让我们看看组件内部是怎么实现的,首先是PermissionManager.requestPermissions 方法请求权限,它先判断是否所有的请求都被批准了,如果都被批准了,直接回调onGranted_Third(permissions),回调所有允许的权限,如果不是,就遍历权限集合,判断是否要用户选择,将需要用户选择的权限添加到permissionsToExplainList集合里,并转换成数组,如果数组不为空那么就弹出相应请求框

注意:这里callback.onExplanation(permissionsToExplain);一般情况是用户上一次选择了拒绝(没有勾选不再询问),这次又重新请求,通常在此处还需要再次弹出请求对话框,而且这里不会自动出现权限请求弹窗, 需要在用户处理完之前提到的 "解释" 之后, 手动通过 PermissionManager.showRequestDialog 显示弹窗

 /**
 * 请求权限
 *
 * @param requestCode 本次请求的唯一标识(不要过大,最好小于999)
 * @param permissions 本次需要请求的权限,可以多个
 * @param callback    本次请求在用户选择完毕后的回调
 */
public void requestPermissions(int requestCode, String[] permissions, PermissionCallback callback) {
    mCallbackMap.put(requestCode, callback);
    if (!isPermissionsGranted(permissions)) {//是否所有的请求都被批准了,如果没有被批准
        List<String> permissionsToExplainList = new ArrayList<>(3);
        for (String permission : permissions) {
            if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permission)) {//判断是否应该请求,里面做了是否是危险权限的判断
                permissionsToExplainList.add(permission);  //把应该请求的权限添加到permissionsToExplainList集合
            }
        }
        String[] permissionsToExplain = permissionsToExplainList.toArray(new String[permissionsToExplainList.size()]);//集合转化为数组
        if (permissionsToExplain.length != 0) {
            callback.onExplanation(permissionsToExplain);
        } else {
            showRequestDialog(mActivity, permissions, requestCode);
        }
    } else {
        callback.onGranted_Third(permissions);//所有的请求都被批准
    }
}

在执行完 PermissionManager.requestPermissions 后, 如果有权限请求需要用户选择, 系统会弹出相应请求框
如果没有权限需要用户选择(所有权限或是已经允许, 或是已经拒绝, 或是永久拒绝), 直接回调相应结果
接下来我们来看看回调结果

* @param requestCode  请求的唯一标识
 * @param permissions  请求的权限
 * @param grantResults 请求的结果
 */
public void onRequestPermissionResult(int requestCode, String[] permissions, int[] grantResults) {
    PermissionCallback callback = mCallbackMap.get(requestCode);
    if (callback != null) {
        List<String> grantedList = new ArrayList<>(3);//用户同意的权限
        List<String> normalDeniedList = new ArrayList<>(3);//用户拒绝的权限,但是没有勾选不再询问
        List<String> noMoreAskDeniedList = new ArrayList<>(3);//用户拒绝的权限,而且勾选了不再询问
        for (int i = 0; i < permissions.length; i++) {
            if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
                grantedList.add(permissions[i]);
            } else {
                if (!ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permissions[i])) {
                    noMoreAskDeniedList.add(permissions[i]);
                } else {
                    normalDeniedList.add(permissions[i]);
                }
            }
        }
        callback.onNoMoreAskDenied_First(noMoreAskDeniedList.toArray(new String[noMoreAskDeniedList.size()]));
        callback.onNormalDenied_Second(normalDeniedList.toArray(new String[normalDeniedList.size()]));
        callback.onGranted_Third(grantedList.toArray(new String[grantedList.size()]));
    }
}

首先定义了三个数组,以个是表示用户同意的权限,一个是用户拒绝的权限,但是没有勾选不再询问,一个是用户拒绝的权限而且勾选了不再询问,然后开始遍历权限数组,将用户选择之后的权限分成三份,放在对应的结合里,然后调用回调,传到相应的activity中,在activity中的callback里就可以知道哪些权限是用户拒绝或者同意了的,还可以对相应的权限进行操作。

注意:PermissionCallback里的各种结果的回调有个先后顺序:

  • onExplanation 如果有权限 "上一次选择了拒绝, 但没有勾选不再询问", 最开始会通过此方法回调
  • onNoMoreAskDenied_First 然后回调所有 "不再询问 且 勾选拒绝" 的权限
  • onNormalDenied_Second 然后回调所有 "拒绝" 的权限
  • onGranted_Third 最后回调所有 "允许" 的权限
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容