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);}public abstract static class RationaleHandler {private Context context;private int requestCode;private String[] permissions;protected abstract void showRationale();void showRationale(Context context, int requestCode, String[] permissions) {this.context = context;this.requestCode = requestCode;this.permissions = permissions;showRationale();}@TargetApi(Build.VERSION_CODES.M)public void requestPermissionsAgain() {((Activity) context).requestPermissions(permissions, requestCode);}}@TargetApi(Build.VERSION_CODES.M)public static void requestPermissions(Context context, int requestCode, String[] permissions, OnPermissionListener listener) {requestPermissions(context, requestCode, permissions, listener, null);}@TargetApi(Build.VERSION_CODES.M)public static void requestPermissions(Context context, int requestCode, String[] permissions, OnPermissionListener listener, RationaleHandler handler) {if (context instanceof Activity) {mRequestCode = requestCode;mOnPermissionListener = listener;String[] deniedPermissions = getDeniedPermissions(context, permissions);if (deniedPermissions.length > 0) {boolean rationale = shouldShowRequestPermissionRationale(context, deniedPermissions);if (rationale && handler != null) {handler.showRationale(context, requestCode, deniedPermissions);} else {((Activity) context).requestPermissions(deniedPermissions, requestCode);}} else {if (mOnPermissionListener != null)mOnPermissionListener.onPermissionGranted();}} else {throw new RuntimeException("Context must be an Activity");}}/*** 请求权限结果,对应Activity中onRequestPermissionsResult()方法。*/public static void onRequestPermissionsResult(Activity context, int requestCode, String[] permissions, int[]grantResults) {if (mRequestCode != -1 && requestCode == mRequestCode) {if (mOnPermissionListener != null) {String[] deniedPermissions = getDeniedPermissions(context, permissions);if (deniedPermissions.length > 0) {mOnPermissionListener.onPermissionDenied(deniedPermissions);} else {mOnPermissionListener.onPermissionGranted();}}}}/*** 获取请求权限中需要授权的权限*/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()]);}/*** 是否彻底拒绝了某项权限*/public static boolean hasAlwaysDeniedPermission(Context context, String... deniedPermissions) {for (String deniedPermission : deniedPermissions) {if (!shouldShowRequestPermissionRationale(context, deniedPermission)) {return true;}}return false;}/*** 是否有权限需要说明提示*/private static boolean shouldShowRequestPermissionRationale(Context context, 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,是本人的疏忽。因为开始我用的小米手机没有这个功能,后来发现有的手机支持有的不支持。顾名思义这个方法的意思是否需要给用户申请该权限的提示,当用户拒绝权限之后如果没有勾选不再提示,下次申请权限的时候可以加一个自定义的弹窗提示,用户点继续验证可以再次验证权限。
大致实现思路如下:

注意:
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、在基类中加上回调方法

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

3、调用工具类方法

1>拒绝后无提示

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

2>拒绝后再次申请给出提示

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

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

private void doOpenCamera() {XPermissionUtils.requestPermissions(this, RequestCode.CAMERA, new String[]{Manifest.permission.CAMERA}, new XPermissionUtils.OnPermissionListener() {@Overridepublic void onPermissionGranted() {if (PermissionHelper.isCameraEnable()) {Toast.makeText(MainActivity.this, "打开相机操作", Toast.LENGTH_LONG).show();} else {DialogUtil.showPermissionManagerDialog(MainActivity.this, "相机");}}@Overridepublic void onPermissionDenied(String[] deniedPermissions) {Toast.makeText(context, "获取相机权限失败", Toast.LENGTH_SHORT).show();if (XPermissionUtils.hasAlwaysDeniedPermission(context, deniedPermissions)) {DialogUtil.showPermissionManagerDialog(MainActivity.this, "相机");}}}, new XPermissionUtils.RationaleHandler() {@Overrideprotected void showRationale() {new AlertDialog.Builder(context).setTitle("温馨提示").setMessage("我们需要相机权限才能正常使用该功能").setNegativeButton("取消", null).setPositiveButton("验证权限", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {requestPermissionsAgain();}}).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() {@Overridepublic void onPermissionGranted() {Toast.makeText(context, "获取联系人,短信权限成功", Toast.LENGTH_SHORT).show();}@Overridepublic void onPermissionDenied(String[] deniedPermissions) {StringBuilder sBuider = new StringBuilder();for (String deniedPermission : deniedPermissions) {if (deniedPermission.equals(Manifest.permission.WRITE_CONTACTS)) {sBuider.append("联系人");sBuider.append(",");}if (deniedPermission.equals(Manifest.permission.READ_SMS)) {sBuider.append("短信");sBuider.append(",");}}if (sBuider.length() > 0) sBuider.deleteCharAt(sBuider.length() - 1);Toast.makeText(context, "获取" + sBuider.toString() + "权限失败", Toast.LENGTH_SHORT).show();if (XPermissionUtils.hasAlwaysDeniedPermission(context, deniedPermissions)) {DialogUtil.showPermissionManagerDialog(MainActivity.this, sBuider.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

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

  1. 获取权限android sync,GitHub - AndSync/XPermissionUtils: 可能是最精简的Android6.0运行时权限处理方式,支持Rationale提示...

    # XPermissionUtils 可能是最精简的Android6.0运行时权限处理方式,支持Rationale提示,只有一个类,100行代码,所有弹窗等操作由用户自行处理,在Demo中也有提供代码 ...

  2. Android6.0运行时权限解决方案

    今天在测试APP的时候发现部分手机APP定位权限使用不了.看了一下机型都是Android6.0以上版本的Android手机. 之前就听说Android6.0以上版本权限管理更严格了,没想到今天踩坑了. ...

  3. Android6.0运行时权限处理

    前言 在Android6.0版本以前,往往是应用程序需要什么权限直接在manifest.xml中直接声明,当你安装程序的时候,如果不想让该程序使用某种权限,唯一的办法只能是不装这个应用,但是我们生活中 ...

  4. Android6.0运行时权限(危险权限列表)

    从 Android 6.0(API 23)开始,用户开始在应用运行时向其授予权限,而不是在应用安装时授予.这种权限机制可以让用户更好的管理应用的权限,保障用户隐私. Android将权限分为普通权限和 ...

  5. Android6.0执行时权限解析,RxPermissions的使用,自己封装一套权限框架

    Android6.0执行时权限解析,RxPermissions的使用.自己封装一套权限框架 在Android6.0中,新添加了一个执行时的权限,我相信非常多人都已经知道了.预计也知道怎么用了,这篇博客 ...

  6. .net4.0运行时安装说明(xp,win7,win10)

    概述 现在大多windows桌面版的系统都是使用.net开发,所以我们整理了一下目前流行的.net4.0运行时的安装说明. win 10 (1511)以上(含)系统不需要安装. win7系统需要安装, ...

  7. Android8.0运行时权限策略变化和适配方案

    版权声明:转载必须注明本文转自严振杰的博客:http://blog.yanzhenjie.com Android8.0也就是Android O即将要发布了,有很多新特性,目前我们可以通过Android ...

  8. Android 8.0 运行时权限策略变化和适配方案

    Android8.0也就是Android O即将要发布了,有很多新特性,目前我们可以通过AndroidStudio3.0 Canary版本下载Android O最新的系统映像的Developer Pr ...

  9. ASP.NET 2.0运行时简要分析

    概述: 本文基于ASP.NET 2.0的源代码,对ASP.NET 2.0运行时进行了简要的分析,希望能帮助你理解ASP.NET 2.0中请求处理过程及页面编译模型. 关键字: ASP.NET 2.0运 ...

最新文章

  1. 欧洲超高速网比宽带快万倍 几秒下载一部电影
  2. Java基础看这一篇就够了
  3. pycharm 如何通过VCS快速提交代码?
  4. 操作系统-银行家算法
  5. 【译】Deep Learning with Knowledge Graphs
  6. SAP Spartacus里所有backend endpoint list
  7. C |格式化输出与变量类型
  8. Wasserstein metric的通俗解释
  9. 获取php数组最后,php获取数组最后一个值的2种方法
  10. 空间复杂度,实现从excel导出到txt文件中的java代码自动构建,逻辑条件不同实现则不同...
  11. 记录一次opper R9s Plus 从发现循环重启到解决的过程
  12. 解决Windows7下virtualbox安装ubuntu出现的0x00000000指令引用0x00000000内存,该内存不能为written问题
  13. MAC电脑实现微信多开
  14. 导入依赖失败,报错信息Failed to resolve: com.*.*:*:0.0.0<a href=““>Show in Project Structure dialog
  15. 深度报告:一文看懂通信新基建五大方向
  16. RESTful Api 身份认证总结
  17. 【SQL学习笔记】《SQL进阶教程》1.1
  18. CISCO、3COM交换机 snmp 值设置方法
  19. 竞品分析该如何写?竞品分析的维度有哪些?
  20. 测试真的会被ChatGPT代替一文告诉你

热门文章

  1. 蚂蚁p8多少股票_阿里P8收入有多高?满足大家的好奇,司龄不足10年,税前330万...
  2. [ANT] -------Ant简介
  3. Transformers北大源
  4. Android点击悬浮窗回到应用,android悬浮窗应用
  5. Android Framework(一)--对Zygote的理解
  6. 期货投资交易8个技巧
  7. 双十一提前盘点 这些必入的科大讯飞AI生产力好物
  8. 2019-2020 ICPC, NERC, Southern and Volga Russian Regional Contest B. The Feast and the Bus (经典贪心)
  9. 解决orcl中文字段内容乱码思路
  10. win7下安装Linux实现双系统全攻略