Android 6.0 运行时权限封装框架

来源:互联网 发布:mac photoshop 编辑:程序博客网 时间:2024/05/16 10:54

1.概述


  视频讲解都已经录了好几天,但是最近实在抽不开身一直在忙Unity3D,就连光棍节都在写,上次我们只是写了Android 6.0 运行时权限处理解析,但是并未对其做代码封装,这一次我们做一个彻底的处理。
  附视频讲解地址:http://pan.baidu.com/s/1bpqqkGn
  
这里写图片描述

2.框架封装 


  2.1. 简单例子

public class MainActivity extends AppCompatActivity {    // 打电话权限申请的请求码    private static final int CALL_PHONE_REQUEST_CODE = 0x0011;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);    }    public void phoneClick(View view){        if(ContextCompat.checkSelfPermission(this, Manifest.permission.CALL_PHONE)                != PackageManager.PERMISSION_GRANTED){            Toast.makeText(this, "申请权限", Toast.LENGTH_SHORT).show();            ActivityCompat.requestPermissions(this,                    new String[]{"Manifest.permission.CALL_PHONE"}, CALL_PHONE_REQUEST_CODE);        }else {            callPhone();        }    }    /**    * 拨打电话    **/    private void callPhone() {        Intent intent = new Intent(Intent.ACTION_CALL);        Uri data = Uri.parse("tel:147****2514");        intent.setData(data);        startActivity(intent);    }    @Override    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {        super.onRequestPermissionsResult(requestCode, permissions, grantResults);        if(requestCode == CALL_PHONE_REQUEST_CODE){            if (grantResults !=null&&grantResults[0] == PackageManager.PERMISSION_GRANTED) {                // Permission Granted                callPhone();            } else {                // Permission Denied                Toast.makeText(this,"权限被拒绝了",Toast.LENGTH_SHORT).show();            }        }    }}

  这是从上一篇中复制过来的,并没有做过任何处理,如果每次申请权限的地方都需要这么做,那就比较麻烦了,那么我们肯定要对phoneClick()和onRequestPermissionsResult()里面的代码进行处理,那么要处理成什么样子?当然是所有的代码都可以优化成一行,别人都这么说的。

  2.1. 确定方式
  到底采用什么方式去封装,这个github上面有很多,中国人,日本人,韩国人各国人民都有,这个不是开完笑的真的都有,这里我才用反射+注解的方式去实现。
  
  2.2. 参数传递和传递方式
  我们肯定要新建类,至于传什么参数? 第一个参数反射的类this,第二个参数请求码用于监听反馈处理,第三个参数传请求权限的数组
  利用什么方式传递过去那么?可以在工具类里面写一个静态的方法一把塞过去。但是在这里我们可以使用链式调用这种方式我们也经常使用,比如Okhttp,Gilde,甚至是Android的AlertDialog都是使用的这个方式,可以装装B

public class PermissionHelper {    // 1. 传什么参数    // 1.1. Object Fragment or Activity  1.2. int 请求码   1.3.需要请求的权限  string[]    private Object mObject;    private int mRequestCode;    private String[] mRequestPermission;    private PermissionHelper(Object object){        this.mObject = object;    }    // 2.已什么的方式传参数    // 2.1 直接传参数    public static void  requestPermission(Activity activity,int requestCode,String[] permissions){            PermissionHelper.with(activity).requestCode(requestCode).                    requestPermission(permissions).request();    }    public static void  requestPermission(Fragment fragment,int requestCode,String[] permissions){        PermissionHelper.with(fragment).requestCode(requestCode).                requestPermission(permissions).request();    }    // 2.2 链式的方式传    // 传Activity    public static PermissionHelper with(Activity activity){        return new PermissionHelper(activity);    }    // 传Fragment    public static PermissionHelper with(Fragment fragment){        return new PermissionHelper(fragment);    }    // 添加一个请求码    public PermissionHelper requestCode(int requestCode){        this.mRequestCode = requestCode;        return this;    }    // 添加请求的权限数组    public PermissionHelper requestPermission(String... permissions){        this.mRequestPermission = permissions;        return this;    }    /**     * 3.1 真正判断和发起请求权限     */    public void request() {        // 3.2 首先判断当前的版本是不是6.0 及以上        if(!PermissionUtils.isOverMarshmallow()){            // 3.3 如果不是6.0以上  那么直接执行方法   反射获取执行方法            // 执行什么方法并不确定 那么我们只能采用注解的方式给方法打一个标记,            // 然后通过反射去执行。  注解 + 反射  执行Activity里面的callPhone            PermissionUtils.executeSucceedMethod(mObject,mRequestCode);            return;        }    }}

2.2. 处理6.0以下的版本
  对于6.0以下的版本,我们可以直接反射调用执行成功的方法,如果反射不太理解可以看一下这个视频讲解:http://pan.baidu.com/s/1bpqqkGn

    /**     * 执行成功的方法     */    public static void executeSucceedMethod(Object reflectObject, int requestCode) {        // 获取class中多有的方法        Method[] methods = reflectObject.getClass().getDeclaredMethods();        // 遍历找我们打了标记的方法        for (Method method:methods){            Log.e("TAG",method+"");            // 获取该方法上面有没有打这个成功的标记            PermissionSucceed succeedMethod =  method.getAnnotation(PermissionSucceed.class);            if(succeedMethod != null){                // 代表该方法打了标记                // 并且我们的请求码必须 requestCode 一样                int methodCode = succeedMethod.requestCode();                if(methodCode == requestCode){                    // 这个就是我们要找的成功方法                    // 反射执行该方法                    Log.e("TAG","找到了该方法 :"+method);                    executeMethod(reflectObject,method);                }            }        }    }    /**     * 反射执行该方法     */    private static void executeMethod(Object reflectObject,Method method) {        // 反射执行方法  第一个是传该方法是属于哪个类   第二个参数是反射方法的参数        try {            method.setAccessible(true); // 允许执行私有方法            method.invoke(reflectObject,new Object[]{});        } catch (IllegalAccessException e) {            e.printStackTrace();        } catch (InvocationTargetException e) {            e.printStackTrace();        }    }

2.3. 处理6.0以上的版本

    /**     * 3.1 真正判断和发起请求权限     */    public void request() {        // 3.2 首先判断当前的版本是不是6.0 及以上        if(!PermissionUtils.isOverMarshmallow()){            // 3.3 如果不是6.0以上  那么直接执行方法   反射获取执行方法            // 执行什么方法并不确定 那么我们只能采用注解的方式给方法打一个标记,            // 然后通过反射去执行。  注解 + 反射  执行Activity里面的callPhone            PermissionUtils.executeSucceedMethod(mObject,mRequestCode);            return;        }        // 3.3 如果是6.0以上  那么首先需要判断权限是否授予        // 需要申请的权限中 获取没有授予过得权限        List<String> deniedPermissions = PermissionUtils.getDeniedPermissions(mObject,mRequestPermission);        // 3.3.1 如果授予了 那么我们直接执行方法   反射获取执行方法        if(deniedPermissions.size() == 0){            // 全部都是授予过的            PermissionUtils.executeSucceedMethod(mObject,mRequestCode);        }else {            // 3.3.2 如果没有授予 那么我们就申请权限  申请权限            ActivityCompat.requestPermissions(PermissionUtils.getActivity(mObject),                    deniedPermissions.toArray(new String[deniedPermissions.size()]),                    mRequestCode);        }    }

  
2.4. 处理回调
  如果用户同意或是拒绝那么会回调onRequestPermissionsResult(),我们肯定也需要对它做处理,这个方法就很简单了:

    /**     * 处理申请权限的回调     */    public static void requestPermissionsResult(Object object,int requestCode,                                                String[] permissions) {        // 再次获取没有授予的权限        List<String> deniedPermissions = PermissionUtils.getDeniedPermissions(object,permissions);        if(deniedPermissions.size() == 0){            // 权限用户都同意授予了            PermissionUtils.executeSucceedMethod(object,requestCode);        }else{            // 你申请的权限中 有用户不同意的            PermissionUtils.executeFailMethod(object,requestCode);        }    }

2.5. 最后的事例

public class MainActivity extends AppCompatActivity {    // 打电话权限申请的请求码    private static final int CALL_PHONE_REQUEST_CODE = 0x0011;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);    }   public void phoneClick(View view){        PermissionHelper.with(this).requestCode(CALL_PHONE_REQUEST_CODE)                .requestPermission(Manifest.permission.CALL_PHONE).request();    }    @PermissionSucceed(requestCode =  CALL_PHONE_REQUEST_CODE)    private void callPhone() {        Intent intent = new Intent(Intent.ACTION_CALL);        Uri data = Uri.parse("tel:147****2514");        intent.setData(data);        startActivity(intent);    }    @PermissionFail(requestCode = CALL_PHONE_REQUEST_CODE)    private void callPhoneFail(){        Toast.makeText(this,"您拒绝了拨打电话",            Toast.LENGTH_SHORT).show();    }    @Override    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {    PermissionHelper.requestPermissionsResult(this,        CALL_PHONE_REQUEST_CODE,permissions);    }}

  到最后再处理权限请求的时候会轻松很多,一行代码解决问题,反正别人都这么说,我们甚至可以把onRequestPermissionsResult()的处理写到BaseActivity中,因为权限封装我没有把它写入到基础架构部分,这里我就不处理了。

这里写图片描述
  
  项目的代码不能够发给大家,里面涉及到后台接口以及数据加密,如果大家感兴趣可以看一下我录的视频:http://pan.baidu.com/s/1bpqqkGn 。

1 0
原创粉丝点击