这可能是最精简的Android6.0运行时权限处理,百行代码的工具类,支持Rationale,附:各种权限详细处理

0x00:前言

对于Android6.0运行时权限的处理方式网上有很多,包括注解,RxJava等等。一直没有正面提到我关心的问题--如果我不在Activity或者Fragment里面,需要运行时权限该怎么去做?导致我开始一直以为运行时权限的处理必需要在Activity或者Fragment之中。

那么:
我有一个录音的自定义控件在很多页面需要使用怎么办?
我有一个联系人列表,要在adapter里面拨打电话怎么办?
我有一个定位的工具类要在多个页面使用怎么办?
等等...
之前我还问过一些同行,他说用回调,回调到Activity或者Fragment,我当时觉得是一种解决方案,但是却很麻烦,如果有多个页面使用,那不是要处理很多次。

直到某一天在github上看到一个分享了简单的工具类MPermissionUtils ,一下子解决了我的疑惑,虽然他也没有明确给出答案,但是我从他的使用上却恍然大悟,原来是一开始我就理解错了。我们只需要把系统回调方法onRequestPermissionsResult放到BaseActivity里面,当然你所有的用到权限的Activity必需继承自BaseActivity,将处理结果通过工具类调出来,加一个自定义的回调到请求的发起处即可。

因为你要用到运行时权限的地方总要依赖于Activity的存在,如果不再Activity里面或者当前代码获取不到Activity,那就传过去,一切的处理结果都会回到你发起请求所在的Activity。

那么一不做二不休,我们这时候有没有考虑Fragment里面的处理其实是多余的,我们可不可以都放到Activity里面来处理。于是就化繁为简产生了我的XPermissionUtils

0x01:代码实现

public class XPermissionUtils {

    private static int mRequestCode = -1;
    private static OnPermissionListener mOnPermissionListener;

    public interface OnPermissionListener {

        void onPermissionGranted();

        void onPermissionDenied(String[] deniedPermissions, boolean alwaysDenied);
    }

    @TargetApi(Build.VERSION_CODES.M)
    public static void requestPermissionsAgain(@NonNull Context context, @NonNull String[] permissions,
        @NonNull int requestCode) {
        if (context instanceof Activity) {
            ((Activity) context).requestPermissions(permissions, requestCode);
        } else {
            throw new IllegalArgumentException("Context must be an Activity");
        }
    }

    @TargetApi(Build.VERSION_CODES.M)
    public static void requestPermissions(@NonNull Context context, @NonNull int requestCode,
        @NonNull String[] permissions, OnPermissionListener listener) {
        mRequestCode = requestCode;
        mOnPermissionListener = listener;
        String[] deniedPermissions = getDeniedPermissions(context, permissions);
        if (deniedPermissions.length > 0) {
            requestPermissionsAgain(context, permissions, requestCode);
        } else {
            if (mOnPermissionListener != null) mOnPermissionListener.onPermissionGranted();
        }
    }

    /**
     * 请求权限结果,对应Activity中onRequestPermissionsResult()方法。
     */
    public static void onRequestPermissionsResult(@NonNull Activity context, int requestCode,
        @NonNull String[] permissions, int[] grantResults) {
        if (mRequestCode != -1 && requestCode == mRequestCode) {
            if (mOnPermissionListener != null) {
                String[] deniedPermissions = getDeniedPermissions(context, permissions);
                if (deniedPermissions.length > 0) {
                    boolean alwaysDenied = hasAlwaysDeniedPermission(context, permissions);
                    mOnPermissionListener.onPermissionDenied(deniedPermissions, alwaysDenied);
                } else {
                    mOnPermissionListener.onPermissionGranted();
                }
            }
        }
    }

    /**
     * 获取请求权限中需要授权的权限
     */
    private static String[] getDeniedPermissions(@NonNull Context context, @NonNull String[] permissions) {
        List<String> deniedPermissions = new ArrayList();
        for (String permission : permissions) {
            if (ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_DENIED) {
                deniedPermissions.add(permission);
            }
        }
        return deniedPermissions.toArray(new String[deniedPermissions.size()]);
    }

    /**
     * 是否彻底拒绝了某项权限
     */
    private static boolean hasAlwaysDeniedPermission(@NonNull Context context, @NonNull String... deniedPermissions) {
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) return false;
        boolean rationale;
        for (String permission : deniedPermissions) {
            rationale = ActivityCompat.shouldShowRequestPermissionRationale((Activity) context, permission);
            if (!rationale) return true;
        }
        return false;
    }
}

0x02:实现思路

在最开始的时候本人的实现没有支持shouldShowRequestPermissionRationale,是本人的疏忽。因为开始我用的小米手机没有这个功能,后来发现有的手机支持有的不支持。顾名思义这个方法的意思是否需要给用户申请该权限的提示,当用户拒绝权限之后如果没有勾选不再提示,下次申请权限的时候可以加一个自定义的弹窗提示,用户点继续验证可以再次验证权限。
大致实现思路如下:

Flow Chart.png

注意:
1>判断是否需要提示方法shouldShowRequestPermissionRationale,只要有一个权限需要提示就返回true
2>判断是否彻底禁止权限方法hasAlwaysDeniedPermission,只要有一个彻底禁止就返回true
3>为了节省代码在发起请求与请求结果中用了同样的方法获取未授权的权限

private static String[] getDeniedPermissions(Context context, String[] permissions) {
        List<String> deniedPermissions = new ArrayList<>();
        for (String permission : permissions) {
            if (ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_DENIED) {
                deniedPermissions.add(permission);
            }
        }
        return deniedPermissions.toArray(new String[deniedPermissions.size()]);
    }

此外在请求结果的时候还可以用另外的方法获取,结果是一样的

private static String[] getDeniedPermissions(String[] permissions, int[] grantResults) {
        List<String> deniedPermissions = new ArrayList<>();
        for (int i = 0; i < permissions.length; i++) {
            if (grantResults[i] != PackageManager.PERMISSION_GRANTED) {
                deniedPermissions.add(permissions[i]);
            }
        }
        return deniedPermissions.toArray(new String[deniedPermissions.size()]);
    }

0x03:使用方式

以打开相机为例

1、首先AndroidManifest中配置必要的权限

<uses-permission android:name="android.permission.CAMERA"/>

2、在基类中加上回调方法

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

3、调用工具类方法

XPermissionUtils.requestPermissions(Context context, int requestCode, String[] permissions, OnPermissionListener listener)

这里主要注意这个Context必需是一个Activity
如果在Activity中可以传this;
如果在Fragment中传getActivity();
如果在View中传getContext();
等等.....

private void doOpenCamera() {
        XPermissionUtils.requestPermissions(this, RequestCode.CAMERA, new String[] { Manifest.permission.CAMERA },
            new XPermissionUtils.OnPermissionListener() {
                @Override
                public void onPermissionGranted() {
                    if (PermissionHelper.isCameraEnable()) {
                        Toast.makeText(MainActivity.this, "打开相机操作", Toast.LENGTH_LONG).show();
                    } else {
                        DialogUtil.showPermissionManagerDialog(MainActivity.this, "相机");
                    }
                }

                @Override
                public void onPermissionDenied(final String[] deniedPermissions, boolean alwaysDenied) {
                    Toast.makeText(context, "获取相机权限失败", Toast.LENGTH_SHORT).show();
                    if (alwaysDenied) { // 拒绝后不再询问 -> 提示跳转到设置
                        DialogUtil.showPermissionManagerDialog(MainActivity.this, "相机");
                    } else {    // 拒绝 -> 提示此公告的意义,并可再次尝试获取权限
                        new AlertDialog.Builder(context).setTitle("温馨提示")
                            .setMessage("我们需要相机权限才能正常使用该功能")
                            .setNegativeButton("取消", null)
                            .setPositiveButton("验证权限", new DialogInterface.OnClickListener() {
                                @RequiresApi(api = Build.VERSION_CODES.M)
                                @Override
                                public void onClick(DialogInterface dialog, int which) {
                                    XPermissionUtils.requestPermissionsAgain(context, deniedPermissions,
                                        RequestCode.CAMERA);
                                }
                            })
                            .show();
                    }
                }
            });
    }

4、一次申请多个权限

用户可能部分拒绝,因此在onPermissionDenied(String[] deniedPermissions)回调中返回了请求结果中所有被拒绝的权限,用户可用于比对判断出哪些权限被拒绝,给用户明确的提示

    private void doMorePermission() {
        XPermissionUtils.requestPermissions(this, RequestCode.MORE,
            new String[] { Manifest.permission.WRITE_CONTACTS, Manifest.permission.READ_SMS },
            new XPermissionUtils.OnPermissionListener() {
                @Override
                public void onPermissionGranted() {
                    Toast.makeText(context, "获取联系人,短信权限成功", Toast.LENGTH_SHORT).show();
                }

                @Override
                public void onPermissionDenied(String[] deniedPermissions, boolean alwaysDenied) {
                    StringBuilder sBuilder = new StringBuilder();
                    for (String deniedPermission : deniedPermissions) {
                        if (deniedPermission.equals(Manifest.permission.WRITE_CONTACTS)) {
                            sBuilder.append("联系人");
                            sBuilder.append(",");
                        }
                        if (deniedPermission.equals(Manifest.permission.READ_SMS)) {
                            sBuilder.append("短信");
                            sBuilder.append(",");
                        }
                    }
                    if (sBuilder.length() > 0) {
                        sBuilder.deleteCharAt(sBuilder.length() - 1);
                    }
                    Toast.makeText(context, "获取" + sBuilder.toString() + "权限失败", Toast.LENGTH_SHORT).show();
                    if (alwaysDenied) {
                        DialogUtil.showPermissionManagerDialog(MainActivity.this, sBuilder.toString());
                    }
                }
            });
    }

0x04:各种运行时权限处理详谈

其实在6.0之前已经存在运行时权限,只不过没有明确提出这个概念,在6.0之前,获取位置、读取通讯录、拍照、录音等都是需要在操作的时候去获取权限的。那么这些权限的区别是6.0以后需要我们去写请求获取权限的代码,而之前是当代码执行到需要权限的地方就会弹出提示框。
那么针对不同的权限可能有不同的处理方式,下面简单列举,如果需要看代码可以在源码的Demo中查看

1、拨打电话

拨打电话在某些手机上(如小米)拒绝之后是每次申请都有提示的,因为他显示的是“拒绝一次”。拨打电话其实如果不是产品要求直接拨出去可以使用调转到拨号页面实现的,这个不需要权限:

 Intent intent = new Intent(Intent.ACTION_DIAL);
        Uri data = Uri.parse("tel:10010");
        intent.setData(data);
        startActivity(intent);

2、录音

(1)录音权限在6.0之前是无法判断是否获取权限的,只能通过非常规的方法获取,详见项目Demo
(2)长按按钮录音,在第一次获取权限的时候需要特殊处理,弹出获取权限的提示框之后手指已经离开,不能进行录音的操作。

3、打开相机

相机权限在6.0之前同样也是无法判断是否获取权限的,只能通过非常规的方法获取,详见项目Demo

4、获取位置

(1)首先手机需要开启位置服务,如果没有开启,那么即使app开启获取位置权限也是获取不到的
(2)在6.0以下没有办法判断是否开启位置权限,可以根据具体使用场景进行判断。
(3)(使用系统Api)要注意在室内如果选择Gps定位会获取不到位置,这里可以参考Demo中LocationUtils的实现思路。
(4)使用百度或者高德地图可能不适用,因为他自己已经带有请求权限的处理,貌似不需要系统权限也能定位,没有深入研究。

5、获取外部存储

这个在有些手机上比较特殊,比如打开图库这样的功能,在小米手机上就不需要运行时权限,华为就需要,这个还是需要在使用的时候主动请求一下。

0x05 特别鸣谢

MPermissionUtils
PermissionGen
AndPermission

如有不足,欢迎指正。最后附上源码地址
XPermissionUtils

转载请注明出处http://www.jianshu.com/p/4a60b064a0ab

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

推荐阅读更多精彩内容